排除从 .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-&gt;WithMetadataValue('ReferenceSourceTarget', 'ProjectReference')-&gt;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 &quot;A.nuspec&quot;" />
  </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 文件,因此技能和工具并不为人所知