Libraries can be packaged in one or more of the following ways: as part of the .NETCore shared framework, part of a transport package, or as a NuGet package.
To add a library to the .NETCoreApp shared framework, that library's AssemblyName
should be added to NetCoreAppLibrary.props
The library should have both a ref
and src
project. Its reference assembly will be included in the targeting pack (also called ref pack) for the Microsoft.NETCore.App shared framework, and its implementation assembly will be included in the runtime pack.
Including a library in the shared framework only includes the best applicable TargetFramework build of that library: $(NetCoreAppCurrent)
if it exists, but possibly netstandard2.1
or another if that is best. If a library has builds for other frameworks those will only be shipped if the library also produces a Nuget package.
In some occasions we may want to include a library in the shared framework, but not expose it publicly. The library should be named in a way to discourage use at runtime, for example using the System.Private
prefix. We should avoid hiding arbitrary public libraries as it complicates deployment and servicing.
Libraries included in the shared framework should ensure all direct and transitive assembly references are also included in the shared framework. This will be validated as part of the build and errors raised if any dependencies are unsatisfied.
Source generators and analyzers can be included in the shared framework by adding their project name into the NetCoreAppLibrary.props file under the NetCoreAppLibraryGenerator
section. These projects should specify AnalyzerLanguage
as mentioned below.
Removing a library from the shared framework is a breaking change and should be avoided.
It's beneficial to avoid project references to libraries that are in the shared framework because it makes the package graph smaller which reduces the number of packages that require servicing and the number of libraries that end up being copied into the application directory.
If a dependency is part of the shared framework a project/package reference is never required on the latest version (NetCoreAppCurrent
). A reference is required for previous .NET versions even if the dependency is part of the shared framework if the project you are building targets .NETStandard and references the project there. You may completely avoid a package dependency on .NETStandard and .NET if it's not needed for .NETStandard (for example - if it is an implementation only dependency and you're building a PNSE assembly for .NETStandard).
Warning NETPKG0001 is emitted when you have an unnecessary reference to a library that is part of the shared framework. To avoid this warning, make sure your ProjectReference is conditioned so that it doesn't apply on NetCoreAppCurrent
.
Transport packages are non-shipping packages that dotnet/runtime produces in order to share binaries with other repositories.
Such transport packages represent the set of libraries which are produced in dotnet/runtime and ship in target repo's shared framework (i.e. Microsoft.AspNetCore.App and Microsoft.WindowsDesktop.App). We produce a transport package so that we can easily share reference, implementation and analyzer assemblies that might not be present in NuGet packages that also ship.
To add a library to the target's shared framework, that library should be listed in the AspNetCoreAppLibrary
or WindowsDesktopAppLibrary
section in NetCoreAppLibrary.props
.
Source generators and analyzers can be included in the package by adding them to the Microsoft.Internal.Runtime.**TARGET**.Transport.proj
as a ProjectReference with the ReferenceOutputAssembly=false
and PackAsAnalyzer=true
metadata set. The analyzer projects should specify AnalyzerLanguage
as mentioned below.
Libraries included in this transport package should ensure all direct and transitive assembly references are also included in either the target's shared framework or the Microsoft.NETCore.App shared framework. This is not validated in dotnet/runtime at the moment: dotnet#52562
Removing a library from this transport package is a breaking change and should be avoided.
Libraries to be packaged must set IsPackable
to true. By default, all libraries/*/src
projects are considered for packaging.
Package versions and shipping state should be controlled using the properties defined by the Arcade SDK. Typically libraries should not need to explicitly set any of these properties.
Most metadata for packages is controlled centrally in the repository and individual projects may not need to make any changes to these. One property is required to be set in each project: PackageDescription
. This should be set to a descriptive summary of the purpose of the package. Example:
<PackageDescription>Logging abstractions for Microsoft.Extensions.Logging.</PackageDescription>
Package content can be defined using any of the publicly defined Pack inputs: https://learn.microsoft.com/nuget/reference/msbuild-targets
Packages can include a Markdown Readme file with a short usage documentation. To include a package Readme, create a PACKAGE.md
file in the library src
directory. The file will be automatically included in the package and used as a Readme if its name matches this convention.
The package Readme is displayed on the package details page on NuGet gallery. You can include the following content in it:
- A description of the package purpose.
- A list of package key features
- A code example that demostrates how to use the package.
- Information when package should be used. For example, if the library is included in the shared framework in .NET, but needs to be installed via NuGet on .NET Framework, it should be mentioned.
- A list of common entry-point types for the package, with links to their API docs under .NET API Browser.
- Links to related documentation.
- Information about how to provide feedback on the package and contribute to it.
Use the following Markdown template for a package Readme:
## About
<!-- A description of the package and where one can find more documentation -->
## Key Features
<!-- The key features of this package -->
## How to Use
<!-- A compelling example on how to use this package with code, as well as any specific guidelines for when to use the package -->
## Main Types
<!-- The main types provided in this library -->
## Additional Documentation
* [Conceptual documentation](...)
* [API documentation](...)
## Related Packages
<!-- The related packages associated with this package -->
## Feedback & Contributing
<!-- How to provide feedback on this package and contribute to it -->
ExamplePackage is released as open source under the [MIT license](https://licenses.nuget.org/MIT). Bug reports and contributions are welcome at [the GitHub repository](https://github.com/dotnet/runtime).
For a list of supported Markdown features, see NuGet documentation.
Build props and targets may be needed in NuGet packages. To define these, author a build folder in your src project and place the necessary props/targets in this subfolder. You can then add items to include these in the package by defining Content
items and setting PackagePath
as follows:
<ItemGroup>
<Content Include="build\netstandard2.0\$(MSBuildProjectName).props" PackagePath="%(Identity)" />
<Content Include="build\netstandard2.0\$(MSBuildProjectName).targets" PackagePath="%(Identity)" />
</ItemGroup>
Some packages may wish to include a companion analyzer or source-generator with their library. Analyzers are much different from normal library contributors: their dependencies shouldn't be treated as nuget package dependencies, their TargetFramework isn't applicable to the project they are consumed in (since they run in the compiler). To facilitate this, we've defined some common infrastructure for packaging Analyzers.
To include an analyzer in a package, simply add a ProjectReference
item to the project that produces the package that should contain the analyzer and set the ReferenceOutputAssembly
metadata to false and the PackAsAnalyzer
metadata to true. If you also want to consume the analyzer, set the OutputItemType
metadata to Analyzer
.
<ItemGroup>
<!-- Includes the analyzer in the package without consuming it. -->
<ProjectReference Include="..\gen\System.Banana.Generators.csproj" ReferenceOutputAssembly="false" PackAsAnalyzer="true" />
<!-- Includes the analyzer in the package and consumes it. -->
<ProjectReference Include="..\gen\System.Banana.Generators.csproj" ReferenceOutputAssembly="false" OutputItemType="Analyzer" PackAsAnalyzer="true" />
</ItemGroup>
In the analyzer project make sure to do the following. Ensure it only targets netstandard2.0
since this is a requirement of the compiler. Enable localization by setting UsingToolXliff
. Set the AnalyzerLanguage
property to either cs
or vb
if the analyzer is specific to that language. By default the analyzer will be packaged as language-agnostic. Avoid any dependencies in Analyzer projects that aren't already provided by the compiler.
<PropertyGroup>
<TargetFramework>netstandard2.0</TargetFramework>
<UsingToolXliff>true</UsingToolXliff>
<AnalyzerLanguage>cs</AnalyzerLanguage>
</PropertyGroup>
In order to mitigate design-time/build-time performance issues with source generators, we generate build logic to allow the end user to disable the source generator from the package. By default, the MSBuild property an end user can set is named Disable{PackageId}SourceGenerator
. If a package needs a custom property name, this can be overridden by setting the following property in the project that produces the package
<PropertyGroup>
<DisableSourceGeneratorPropertyName>CustomPropertyName</DisableSourceGeneratorPropertyName>
</PropertyGroup>
For libraries that support .NETStandard, the .NETStandard Compatibility packaging infrastructure makes sure that out-of-support target frameworks like netcoreapp3.1 or net461 are unsupported by the produced package. That enables library authors to support .NETStandard but explicitly not support unsupported .NETStandard compatible target frameworks.
The infrastructure generates a targets file that throws a user readable Error when msbuild invokes a project with an unsupported target framework. In addition to the targets file, placeholder files _._
are placed into the minimum supported .NETStandard compatible target framework's package folder (as time of writing net8.0
and net462
), so that the generated targets files don't apply for that and any newer/compatible target framework. Example:
buildTransitive\net461\Microsoft.Extensions.Configuration.UserSecrets.targets <- This file is generated and throws an Error
buildTransitive\net462\_._
buildTransitive\netcoreapp2.0\Microsoft.Extensions.Configuration.UserSecrets.targets <- This file is generated and throws an Error
buildTransitive\net8.0\_._
Whenever a library wants to author their own set of props and targets files (i.e. for source generators) and the above mentioned infrastructure kicks in (because the library targets .NETStandard), such files must be included not only for the .NETStandard target framework but also for the specific minimum supported target frameworks. The .NETStandard Compatibility packaging infrastructure then omits the otherwise necessary placeholder files. Example:
buildTransitive\netstandard2.0\Microsoft.Extensions.Configuration.UserSecrets.targets <- This file is hand authored and doesn't throw an error
buildTransitive\net461\Microsoft.Extensions.Configuration.UserSecrets.targets <- This file is generated and throws an Error
buildTransitive\net462\Microsoft.Extensions.Configuration.UserSecrets.targets <- This file is hand authored and doesn't throw an error
buildTransitive\netcoreapp2.0\Microsoft.Extensions.Configuration.UserSecrets.targets <- This file is generated and throws an Error
buildTransitive\net8.0\Microsoft.Extensions.Configuration.UserSecrets.targets <- This file is hand authored and doesn't throw an error
The above layout is achieved via the following item declaration in the project file. In that case, the hand authored msbuild props and/or targets files are located in a buildTransitive folder in the project tree. Note that the trailing directory separators are required.
<ItemGroup>
<Content Include="buildTransitive\$(MSBuildProjectName).*"
PackagePath="buildTransitive\netstandard2.0\;
buildTransitive\$(NetFrameworkMinimum)\;
buildTransitive\$(NetCoreAppMinimum)\" />
</ItemGroup>