dotnet 打包 NuGet 的配置属性大全整理

2023-04-07 09:08:46 浏览数 (1)

本文整理 dotnet 打包 CBB 组件为 NuGet 包时可以使用的配置的各个属性

本文将会持续更新,可以通过搜 《dotnet 打包 NuGet 的配置属性大全整理 林德熙》 找到我主站的博客,避免各个备份地址陈旧的内容误导

本文更新于:2023.01.31

如更新时间距离当前阅读时间过远,则表示可能你阅读的是转发的或转载的文章,推荐去到我主站的博客,了解更新的知识

基础知识

在编辑 NuGet 的打包配置属性之前,我期望你了解一些基础知识。了解这部分知识减少一些奇怪的问题和奇怪的决策

基本上使用 dotnet 打包 NuGet 包时,都是通过配置 csproj 项目文件来完成实现功能。其中 csproj 文件有多个版本,当前主力推荐使用的是 SDK 风格的 csproj 格式。可参阅此博客提供的方法将旧的 csproj 格式升级到 SDK 风格的 csproj 格式

在 csproj 项目文件里面,支持编辑内容,在 PropertyGroup 标签里面添加属性值。例如加入 TargetFramework 属性之后的 csproj 的代码大概如下

代码语言:javascript复制
<Project Sdk="Microsoft.NET.Sdk">

  <PropertyGroup>
    <OutputType>WinExe</OutputType>
  </PropertyGroup>

</Project>

更多关于 csproj 项目文件格式,请参阅 理解 C# 项目 csproj 文件格式的本质和编译流程 - walterlv

一些前置知识博客:

  • 理解 C# 项目 csproj 文件格式的本质和编译流程 - walterlv
  • 项目文件中的已知 NuGet 属性(使用这些属性,创建 NuGet 包就可以不需要 nuspec 文件啦) - walterlv
  • Roslyn 如何了解某个项目在 msbuild 中所有用到的属性以及构建过程

CSPROJ 系属性

PackageId

包的 Id 属性,这是不区分大小写的包标识符,该标识符在 nuget.org 或包所在的私有的 NuGet 源中必须是唯一的。不写默认等同于 AssemblyName 程序集名,即 $(AssemblyName) 的值。此 ID 不能包含对于URL无效的空格或字符,且通常遵循.NET命名空间规则

代码语言:javascript复制
  <PropertyGroup>
    <PackageId>Foo.Fx</PackageId>
  </PropertyGroup>

更多 Id 相关,请参阅 ID Prefix Reservation Microsoft Learn

Title

包的人类阅读友好标题,通常在UI显示中使用,如在 nuget.org 和 Visual Studio 中的包管理器上显示给开发者

默认不写等同于 PackageId 内容

代码语言:javascript复制
  <PropertyGroup>
    <Title>标题内容</Title>
  </PropertyGroup>

由于存在语言文化相关问题,如果是公开发布的包且期望国际上的朋友使用,则不建议写入中文。此标题限制为 256 个字符长度

PackageVersion

包版本号,默认不写为 1.0.0 版本号。可使用语义版本号,详细请参阅 语义版本号(Semantic Versioning) - walterlv

代码语言:javascript复制
  <PropertyGroup>
    <PackageVersion>1.0.0</PackageVersion>
  </PropertyGroup>

与此相关的还有 Version 属性,大部分情况下都采用 Version 属性。此 Version 属性将会被 PackageVersion 所使用。如果没有明确设置 PackageVersion 属性,将会使用已设置的 Version 属性

代码语言:javascript复制
  <PropertyGroup>
    <Version>1.0.0</Version>
  </PropertyGroup>

默认 dotnet 规范请参阅: NuGet 包版本引用 Microsoft Learn

如项目没有配置 AssemblyVersion 程序集版本号和 FileVersion 文件版本号,那么默认将使用此 Version 内容作为版本号

如期望自动生成版本号,请参阅 VisualStudio 2017 项目格式 自动生成版本号

Owners

此包的拥有者,可以不同于作者。大部分作用是在开源组织上,由开源组织拥有此包,然后由具体开发者作为作者。这里的拥有者是可以有多个,推荐多个之间使用分号分割。大部分情况下 Owners 拥有者将和 Company 公司相同

代码语言:javascript复制
<Project>
  <PropertyGroup>
    <Company>dotnet-campus</Company>
    <Owners>$(Company)</Owners>
  </PropertyGroup>
</Project>

Company

公司,也可以当成是组织。一般写全商标注册的公司信息。对外可以使用 Owners 写简称

Authors

作者,表示这个包由谁谁制作。作者不一定拥有此包的所有权,和 Owners 不相同。例如公司雇用你打工,你帮助公司发布的包,自然此包的 所有权 就在公司上,而你自己就是此包的作者

代码语言:javascript复制
<Project Sdk="Microsoft.NET.Sdk">
  <PropertyGroup>
    <PackageId>ClassLibDotNetStandard</PackageId>
    <Version>1.0.0</Version>
    <Authors>your_name</Authors>
    <Company>your_company</Company>
  </PropertyGroup>
</Project>

Copyright

版权信息,官方推荐的格式是 "Copyright (c) <name/company> <year> 的格式。正经的包一般都会如此遵守。年份上是可以写范围或固定某个年

代码语言:javascript复制
<Project>
  <PropertyGroup>
    <Copyright>Copyright (c) dotnet-campus 2020-2023</Copyright>
  </PropertyGroup>
</Project>

详细请参阅 Package authoring best practices Microsoft Learn

PackageIcon

包的图标,详细请看 NuGet 如何设置图标

现在推荐将图标作为文件放入到包里面,而不是使用外链图片下载地址,解决一些奇怪的地方无法拉到包或泄露隐私

GeneratePackageOnBuild

生成的时候,构建出 NuGet 包。没有开启此属性时,是需要有额外的打包过程,例如 dotnet pack 或者在 VisuslStudio 里右击打包。开启此属性之后,每次构建都会输出 NuGet 包。实际测试是开启此属性对生成的性能影响很小

代码语言:javascript复制
  <PropertyGroup>
    <GeneratePackageOnBuild>true</GeneratePackageOnBuild>
  </PropertyGroup>

IsPackable

用于设置项目是否可以被打包,默认是 true 表示项目可以打包,如果设置为 false 禁用则不打包 NuGet 包。可以用在如单元测试等项目,设置这些项目不要输出 NuGet 包

代码语言:javascript复制
  <PropertyGroup>
    <IsPackable>false</IsPackable>
  </PropertyGroup>

注:对于 ASP.NET Core 应用项目,在 SDK 里面默认设置了 IsPackable 为 false 的值

GenerateDocumentationFile

设置是否在生成的时候,同时生成注释 XML 文件。此属性设置之后,将会自动将注释 XML 文件输出到 NuGet 里

代码语言:javascript复制
  <PropertyGroup>
    <GenerateDocumentationFile>true</GenerateDocumentationFile>
  </PropertyGroup>

在 dotnet 里面,代码上的公开成员,如公开的方法公开的属性等,的注释是存放在一个和程序集同名后缀为 XML 的文件里面。开启 GenerateDocumentationFile 属性,即可在生成过程,生成注释 XML 文件。在拥有此 XML 文件,即可让 VisualStudio 等 IDE 可以自动提示引用库的代码注释,方便让开发者了解调用库的各个成员的含义。进行 NuGet 发布的时候,将注释的 XML 文件带到 NuGet 包里面,可以方便让引用此 NuGet 包的项目获取到库的代码注释

EmbedAllSources

将源代码嵌入到 PDB 文件里面,此时构建时生成的 PDB 文件里面将包含项目的所有生成相关的源代码。如此可以方便在发布给其他开发者使用时,其他开发者在调试时可以获取到只读的源代码,从而让其他开发者更好进行调试

代码语言:javascript复制
  <PropertyGroup>
    <!-- 嵌入源代码到符号文件,方便调试 -->
    <EmbedAllSources>true</EmbedAllSources>
  </PropertyGroup>

默认是 false 不将源代码嵌入到符号文件。推荐在源代码无需保护的项目,如内部开源项目或外部开源项目,以及 PDB 不对外发布的项目里,设置此属性为 true 从而将源代码嵌入到 PDB 文件里面,方便调试

详细请参阅 Roslyn 通过 EmbedAllSources 将源代码嵌入到 PDB 符号文件中方便开发者调试

AllowedOutputExtensionsInPackageBuildOutputFolder

允许哪些扩展名的输出文件带入到 NuGet 包里面

比如说最常用的是将 PDB 文件放入到 NuGet 里面,即可通过此属性设置输出文件里面的 pdb 文件需要被添加到包里面,如以下代码

代码语言:javascript复制
  <PropertyGroup>
    <!-- 输出 pdb 文件 NuGet 包 -->
    <AllowedOutputExtensionsInPackageBuildOutputFolder>$(AllowedOutputExtensionsInPackageBuildOutputFolder);.pdb</AllowedOutputExtensionsInPackageBuildOutputFolder>
  </PropertyGroup>

此属性只能决定哪些后缀名的文件会打包到 NuGet 包里面,不合适用来决定某些文件需要打包。如果需要特殊指定某些文件,请参阅 Roslyn 打包自定义的文件到 NuGet 包

虽然将 PDB 打包到 NuGet 包里面,有些版本的 VisualStudio 不会自动拷贝 PDB 文件,解决方法请看 修复 VisualStudio 构建时没有将 NuGet 的 PDB 符号文件拷贝到输出文件夹

IncludeSymbols

设置是否输出符号文件,用于制作符号包,通常和 SymbolPackageFormat 配合使用

代码语言:javascript复制
  <PropertyGroup>
    <!-- 输出符号文件 -->
    <IncludeSymbols>true</IncludeSymbols>
    <SymbolPackageFormat>snupkg</SymbolPackageFormat>
  </PropertyGroup>

SymbolPackageFormat

输出的符号文件的格式,符号文件有两个输出格式,文件名规范不相同

  • .symbols.nupkg : 默认的文件后缀。兼容性好,但是存在冲突。比如真有一个叫 Xx.Symbols 项目就凉凉。此格式已被淘汰
  • .snupkg : 专门定义的符号包格式,可以只包含符号 PDB 文件
代码语言:javascript复制
  <PropertyGroup>
    <!-- 输出符号文件 -->
    <IncludeSymbols>true</IncludeSymbols>
    <SymbolPackageFormat>snupkg</SymbolPackageFormat>
  </PropertyGroup>

官方文档: How to publish NuGet symbol packages using the new symbol package format ‘.snupkg’ Microsoft Learn

使用 .snupkg 格式对应在 .nuspec 的配置是

代码语言:javascript复制
<packageTypes>
   <packageType name="SymbolsPackage"/>
</packageTypes>

ContinuousIntegrationBuild

这个属性是比较复杂的,用于 CI 的确定性构建,默认不开。和 Roslyn 的确定性构建 使用的 Deterministic 属性是不相同的两个概念。此 ContinuousIntegrationBuild 是为了 SourceLink 的功能而引入的。此 SourceLink 功能是在 PDB 符号文件里面,嵌入源代码的下载地址,方便调试的时候获取到源代码,详细请看 dotnet 使用 SourceLink 将 NuGet 链接源代码到 GitHub 等仓库

大家都知道,在 PDB 符号文件里面包含的是源代码的绝对路径,在 CI CD 打包服务器上的绝对路径是大部分开发者所不期望的,于是才有了 ContinuousIntegrationBuild 确定性构建的存在。用来实现无论在哪台打包服务器上以及在任何时候打包都会输出相同

这个 ContinuousIntegrationBuild 属性在本机构建调试时,都不应该设置为 true 的值。否则将会丢失本地构建的绝对路径,从而难以自动跳转源代码。只有在 CI 服务器上构建才需要设置

大部分时候设置时,都需要配合设置 SourceRoot 属性

代码语言:javascript复制
  <ItemGroup>
    <SourceRoot Include="$(MSBuildThisFileDirectory)"/>
  </ItemGroup>

以上代码是推荐放在 Directory.Build.props 文件里面,详细关于 Directory.Build.props 请参阅 Roslyn 使用 Directory.Build.props 文件定义编译 和 Roslyn 使用 Directory.Build.props 管理多个项目配置 博客

例如在 GitHub 的 CI 构建时,自动设置此属性

代码语言:javascript复制
  <PropertyGroup Condition="'$(GITHUB_ACTIONS)' == 'true'">
    <ContinuousIntegrationBuild>true</ContinuousIntegrationBuild>
  </PropertyGroup>
  
  <ItemGroup>
    <SourceRoot Include="$(MSBuildThisFileDirectory)"/>
  </ItemGroup>

0 人点赞