排除从 .NET Standard 中的 nuget 包构建的库
Excluding the library being built from nuget package in .NET Standard
请耐心等待 - 这是一个不寻常的场景。
我的解决方案中有 4 个项目。最顶层的项目引用了其他 3 个项目。其他 3 个项目中的 None 相互引用。所以架构是这样的:
现在,当我构建项目 A 时,我希望它生成一个包含项目 B、C 和 D 但不包含项目 A 的 nuget 包。因为这是在 .NET 标准中,我可以配置项目 A 的包选项卡以生成一个通过检查 'Generate NuGet package on build option.' 在构建时自动打包 nuget 然后,我可以通过对 A 的 csproj 文件进行以下更改来使其包含 B、C 和 D:
<ItemGroup>
<ProjectReference Include="..\B.csproj">
<PrivateAssets>all</PrivateAssets>
</ProjectReference>
<ProjectReference Include="..\C.csproj">
<PrivateAssets>all</PrivateAssets>
</ProjectReference>
<ProjectReference Include="..\D.csproj">
<PrivateAssets>all</PrivateAssets>
</ProjectReference>
</ItemGroup>
<PropertyGroup>
<TargetsForTfmSpecificBuildOutput>$(TargetsForTfmSpecificBuildOutput);CopyProjectReferencesToPackage</TargetsForTfmSpecificBuildOutput>
<Version>1.0.0-beta</Version>
<PackageId>A</PackageId>
<Company></Company>
<Product>A</Product>
<Description></Description>
<Authors></Authors>
<GeneratePackageOnBuild>true</GeneratePackageOnBuild>
</PropertyGroup>
<Target Name="CopyProjectReferencesToPackage" DependsOnTargets="ResolveReferences">
<ItemGroup>
<BuildOutputInPackage Include="@(ReferenceCopyLocalPaths->WithMetadataValue('ReferenceSourceTarget', 'ProjectReference')->WithMetadataValue('PrivateAssets', 'all'))" />
</ItemGroup>
</Target>
理想情况下,我想添加一行以从 nuget 包中删除 A.dll。这可能吗? A 是一个包装器项目,消费代码永远不需要使用。 B、C、D无法相互引用。
更新
我就是这样解决的(感谢@tom redfern)
我手动创建了一个 nuspec 文件:
<?xml version="1.0" encoding="utf-8"?>
<package xmlns="http://schemas.microsoft.com/packaging/2013/05/nuspec.xsd">
<metadata>
<id>A</id>
<version>1.0.0-beta</version>
<authors>Foo</authors>
<owners>Bar</owners>
<requireLicenseAcceptance>false</requireLicenseAcceptance>
<description>A package</description>
<dependencies>
<group targetFramework=".NETStandard2.0">
</group>
</dependencies>
</metadata>
<files>
<file src="bin\Release\netstandard2.0\B.dll" target="lib\netstandard2.0\B.dll" />
<file src="bin\Release\netstandard2.0\C.dll" target="lib\netstandard2.0\C.dll" />
<file src="bin\Release\netstandard2.0\D.dll" target="lib\netstandard2.0\D.dll" />
</files>
</package>
然后在我的 A 的 .csproj 文件中放入以下内容以在构建后自动打包它:
<Target Name="__PackNuGetPackage" AfterTargets="Build">
<Exec Command="$(NugetPackage)nuget.exe pack "A.nuspec"" />
</Target>
您可以使用 nuspec 文件来实现。当您需要绝对控制 nuget pack 进程时,请使用 nuspec。一个简单的 nuspec 文件:
<package >
<metadata>
<id>MyPackage</id>
<version>1.0</version>
<authors>Something</authors>
<owners>Something</owners>
<description>Somthing</description>
<copyright></copyright>
<dependencies>
<!-- any nuget package dependencies -->
<dependency id="AnotherPackage" version="2019.2.4.1" />
</dependencies>
</metadata>
<files>
<!-- this is where you can have complete control over which assemblies get added to your package. You can add them individually pr using wildcards. -->
<file src="..\obj\**\*.dll" target="lib" />
</files>
</package>
创建 .nuspec 文件后,将其添加到解决方案中,然后让 "Nuget Pack" 构建步骤读取 nuspec 文件而不是项目文件。
使用获得专利的 (1) elite(2) 调试技巧,我们可以弄清楚是否有可能无需手动创建和维护 nuspec 文件。
首先,让我们从NuGet's docs on creating a package with the dotnet CLI开始。它说 "msbuild -t:pack is functionality equivalent to dotnet pack"。所以,首先提示,它只是 运行ning MSBuild 目标。
所以,运行 dotnet msbuild my.csproj -pp:pp.txt
。此 "pre-processes"(评估所有 MSBuild 导入语句并将结果写入单个文件)csproj(只是一个标准的 MSBuild 文件)。然后我们搜索包目标,并向上滚动直到找到导入文件的文件名。我们看到它是 NuGet.Build.Tasks.Pack.targets
,并且由于 NuGet 在 GitHub、I can point you to the source.
上是开源的
正在 NuGet.Build.Tasks.Pack.targets
中搜索 Condition
,以查看 NuGet 团队提供了哪些可扩展性选项,我看到了 <IncludeBuildOutput Condition="'$(IncludeBuildOutput)'==''">true</IncludeBuildOutput>
。因此,在您的 csproj 中设置 <IncludeBuildOutput Condition="'$(IncludeBuildOutput)'==''">
false
</IncludeBuildOutput>
可能有效。
(1) 未获得专利
(2) 标准,但由于人们不像 C# 那样经常修改 MSBuild 文件,因此技能和工具并不为人所知
请耐心等待 - 这是一个不寻常的场景。 我的解决方案中有 4 个项目。最顶层的项目引用了其他 3 个项目。其他 3 个项目中的 None 相互引用。所以架构是这样的:
现在,当我构建项目 A 时,我希望它生成一个包含项目 B、C 和 D 但不包含项目 A 的 nuget 包。因为这是在 .NET 标准中,我可以配置项目 A 的包选项卡以生成一个通过检查 'Generate NuGet package on build option.' 在构建时自动打包 nuget 然后,我可以通过对 A 的 csproj 文件进行以下更改来使其包含 B、C 和 D:
<ItemGroup>
<ProjectReference Include="..\B.csproj">
<PrivateAssets>all</PrivateAssets>
</ProjectReference>
<ProjectReference Include="..\C.csproj">
<PrivateAssets>all</PrivateAssets>
</ProjectReference>
<ProjectReference Include="..\D.csproj">
<PrivateAssets>all</PrivateAssets>
</ProjectReference>
</ItemGroup>
<PropertyGroup>
<TargetsForTfmSpecificBuildOutput>$(TargetsForTfmSpecificBuildOutput);CopyProjectReferencesToPackage</TargetsForTfmSpecificBuildOutput>
<Version>1.0.0-beta</Version>
<PackageId>A</PackageId>
<Company></Company>
<Product>A</Product>
<Description></Description>
<Authors></Authors>
<GeneratePackageOnBuild>true</GeneratePackageOnBuild>
</PropertyGroup>
<Target Name="CopyProjectReferencesToPackage" DependsOnTargets="ResolveReferences">
<ItemGroup>
<BuildOutputInPackage Include="@(ReferenceCopyLocalPaths->WithMetadataValue('ReferenceSourceTarget', 'ProjectReference')->WithMetadataValue('PrivateAssets', 'all'))" />
</ItemGroup>
</Target>
理想情况下,我想添加一行以从 nuget 包中删除 A.dll。这可能吗? A 是一个包装器项目,消费代码永远不需要使用。 B、C、D无法相互引用。
更新 我就是这样解决的(感谢@tom redfern)
我手动创建了一个 nuspec 文件:
<?xml version="1.0" encoding="utf-8"?>
<package xmlns="http://schemas.microsoft.com/packaging/2013/05/nuspec.xsd">
<metadata>
<id>A</id>
<version>1.0.0-beta</version>
<authors>Foo</authors>
<owners>Bar</owners>
<requireLicenseAcceptance>false</requireLicenseAcceptance>
<description>A package</description>
<dependencies>
<group targetFramework=".NETStandard2.0">
</group>
</dependencies>
</metadata>
<files>
<file src="bin\Release\netstandard2.0\B.dll" target="lib\netstandard2.0\B.dll" />
<file src="bin\Release\netstandard2.0\C.dll" target="lib\netstandard2.0\C.dll" />
<file src="bin\Release\netstandard2.0\D.dll" target="lib\netstandard2.0\D.dll" />
</files>
</package>
然后在我的 A 的 .csproj 文件中放入以下内容以在构建后自动打包它:
<Target Name="__PackNuGetPackage" AfterTargets="Build">
<Exec Command="$(NugetPackage)nuget.exe pack "A.nuspec"" />
</Target>
您可以使用 nuspec 文件来实现。当您需要绝对控制 nuget pack 进程时,请使用 nuspec。一个简单的 nuspec 文件:
<package >
<metadata>
<id>MyPackage</id>
<version>1.0</version>
<authors>Something</authors>
<owners>Something</owners>
<description>Somthing</description>
<copyright></copyright>
<dependencies>
<!-- any nuget package dependencies -->
<dependency id="AnotherPackage" version="2019.2.4.1" />
</dependencies>
</metadata>
<files>
<!-- this is where you can have complete control over which assemblies get added to your package. You can add them individually pr using wildcards. -->
<file src="..\obj\**\*.dll" target="lib" />
</files>
</package>
创建 .nuspec 文件后,将其添加到解决方案中,然后让 "Nuget Pack" 构建步骤读取 nuspec 文件而不是项目文件。
使用获得专利的 (1) elite(2) 调试技巧,我们可以弄清楚是否有可能无需手动创建和维护 nuspec 文件。
首先,让我们从NuGet's docs on creating a package with the dotnet CLI开始。它说 "msbuild -t:pack is functionality equivalent to dotnet pack"。所以,首先提示,它只是 运行ning MSBuild 目标。
所以,运行 dotnet msbuild my.csproj -pp:pp.txt
。此 "pre-processes"(评估所有 MSBuild 导入语句并将结果写入单个文件)csproj(只是一个标准的 MSBuild 文件)。然后我们搜索包目标,并向上滚动直到找到导入文件的文件名。我们看到它是 NuGet.Build.Tasks.Pack.targets
,并且由于 NuGet 在 GitHub、I can point you to the source.
正在 NuGet.Build.Tasks.Pack.targets
中搜索 Condition
,以查看 NuGet 团队提供了哪些可扩展性选项,我看到了 <IncludeBuildOutput Condition="'$(IncludeBuildOutput)'==''">true</IncludeBuildOutput>
。因此,在您的 csproj 中设置 <IncludeBuildOutput Condition="'$(IncludeBuildOutput)'==''">
false
</IncludeBuildOutput>
可能有效。
(1) 未获得专利
(2) 标准,但由于人们不像 C# 那样经常修改 MSBuild 文件,因此技能和工具并不为人所知