Logo

dev-resources.site

for different kinds of informations.

Central Package Management in .NET - Simplify NuGet Dependencies

Published at
12/8/2024
Categories
dotnet
csharp
nuget
centralpackagemanagement
Author
milanjovanovictech
Author
18 person written this
milanjovanovictech
open
Central Package Management in .NET - Simplify NuGet Dependencies

I remember the days when managing NuGet packages across multiple projects was a real pain. You know what I mean - you open a large solution and find out every project uses a different version of the same package. Not fun!

Let me show you how Central Package Management (CPM) in .NET can fix this problem once and for all.

The Problem We Need to Solve

I often work with solutions that have lots of projects. It's not uncommon to have solutions with 30 or more projects. Each one needs similar packages like Serilog or Polly. Most test projects I create depend on xUnit. Before CPM, keeping track of package versions was a mess:

  • One project uses Serilog 4.1.0
  • Another uses Serilog 4.0.2
  • And somehow, a third one uses Serilog 3.1.1

This causes real problems. Different versions can behave differently, leading to weird bugs that are hard to track down. I've wasted many hours fixing issues caused by version mismatches.

How Central Package Management Helps

Think of CPM as a control center for all your package versions. Instead of setting versions in each project, you set them once in one place. Then, you just reference a package you want to use without specifying the version. It's that simple.

Here's what you need to use Central Package Management:

  • NuGet 6.2 or newer
  • .NET SDK 6.0.300 or newer
  • If you use Visual Studio, you need version 2022 17.2 or newer

Setting It Up

Let me show you how to set up CPM. It's easier than you might think.

  1. First, create a file called Directory.Packages.props in your solution's main folder:
<Project>
  <PropertyGroup>
    <ManagePackageVersionsCentrally>true</ManagePackageVersionsCentrally>
  </PropertyGroup>
  <ItemGroup>
    <PackageVersion Include="Newtonsoft.Json" Version="13.0.3" />
    <PackageVersion Include="Serilog" Version="4.1.0" />
    <PackageVersion Include="Polly" Version="8.5.0" />
  </ItemGroup>
</Project>
Enter fullscreen mode Exit fullscreen mode

Note the use of PackageVersion to define NuGet dependencies.

  1. In your project files, you can list the packages using PackageReference without the version component:
<ItemGroup>
  <PackageReference Include="Newtonsoft.Json" />
  <PackageReference Include="AutoMapper" />
  <PackageReference Include="Polly" />
</ItemGroup>
Enter fullscreen mode Exit fullscreen mode

That's it! Now all your projects will use the same package versions.

Cool Things You Can Do

Need a Different Version for One Project?

Sometimes you might need a specific project to use a different version. No problem! Just add this to your project file:

<PackageReference Include="Serilog" VersionOverride="3.1.1" />
Enter fullscreen mode Exit fullscreen mode

The VersionOverride property lets you define the specific version you want to use.

Want a Package in Every Project?

If you have packages that every project needs, you can make them global. Define a GlobalPackageReference in your props file:

<ItemGroup>
  <GlobalPackageReference Include="SonarAnalyzer.CSharp" Version="10.3.0.106239" />
</ItemGroup>
Enter fullscreen mode Exit fullscreen mode

Now every project gets this package automatically!

Migrating Existing Projects to Central Package Management

  1. Create the Directory.Packages.props file at the solution root
  2. Move all package versions from your .csproj files
  3. Remove version attributes from PackageReference elements
  4. Build your solution and fix any version conflicts
  5. Test thoroughly before committing

Here's a Powershell script that will list all NuGet package versions in your solution:

# Scan all .csproj files and aggregate unique package versions
$packages = Get-ChildItem -Filter *.csproj -Recurse |
    Get-Content |
    Select-String -Pattern '<PackageReference Include="([^"]+)" Version="([^"]+)"' -AllMatches |
    ForEach-Object { $_.Matches } |
    Group-Object { $_.Groups[1].Value } |
    ForEach-Object { @{
        Name = $_.Name
        Versions = $_.Group.ForEach({ $_.Groups[2].Value }) | Select-Object -Unique
    }} |
    Sort-Object { $_.Name }

# Display results
$packages | ForEach-Object {
    "$($_.Name) versions:"
    $_.Versions | ForEach-Object { " $_" }
}
Enter fullscreen mode Exit fullscreen mode

There's also a CLI tool called CentralisedPackageConverter,
which you can use to automate the migration.
It will scan for all .NET project files within that folder tree, gather all the versioned references in the projects,
remove the versions from the project files, and write the entries to the Directory.Packages.props file.

# Install the tool globally
dotnet tool install CentralisedPackageConverter --global

# Convert your solution to use Central Package Management
central-pkg-converter /PATH_TO_YOUR_SOLUTION_FOLDER
Enter fullscreen mode Exit fullscreen mode

When Should You Use CPM?

I don't see a compelling reason for not using this by default.

I recommend using CPM when:

  • You have many projects that share packages
  • You're tired of fixing version-related bugs
  • You want to make sure everyone uses the same versions

I recently added CPM to a solution with 30 projects.

Here's what happened:

  • Fewer merge conflicts
  • Caught version problems early
  • Made it easier for new team members

This was especially helpful while migrating from .NET 8 to .NET 9.

You can combine CPM with build configuration and static code analysis.

Wrapping Up

My tips for success with Central Package Management :

  1. When you add CPM to an existing solution, do it in its own change/PR
  2. If you override a version, add a comment explaining why
  3. Check your package versions regularly for updates
  4. Only make packages global if you really need them everywhere

Since I started using Central Package Management, managing NuGet packages has become much easier. It's like having a single source of truth for all your package versions.

Hope this was helpful. See you next week.


P.S. Whenever you're ready, there are 3 ways I can help you:

  1. Pragmatic Clean Architecture: Join 3,500+ students in this comprehensive course that will teach you the system I use to ship production-ready applications using Clean Architecture. Learn how to apply the best practices of modern software architecture.

  2. Modular Monolith Architecture: Join 1,500+ engineers in this in-depth course that will transform the way you build modern systems. You will learn the best practices for applying the Modular Monolith architecture in a real-world scenario.

  3. Patreon Community: Join a community of 1,050+ engineers and software architects. You will also unlock access to the source code I use in my YouTube videos, early access to future videos, and exclusive discounts for my courses.

nuget Article's
30 articles in total
Favicon
Simplifying Dependency Management with NuGet Central Package Management
Favicon
Failed to access Nuget in China
Favicon
Seu Primeiro Pacote NuGet: Um Passo a Passo Prático
Favicon
Central Package Management in .NET - Simplify NuGet Dependencies
Favicon
Custom NET8 Entity Framework Core Generic Repository
Favicon
How to publish a package on Nuget.org
Favicon
Automate Your C# Library Deployment: Publishing to NuGet and GitHub Packages with GitHub Actions
Favicon
My First NuGet Package: EmojiToText
Favicon
Publish C# Project to Nuget.org
Favicon
My Very First NuGet package
Favicon
Introducing VirtualStorageLibrary: A .NET Solution for In-Memory Tree Structures
Favicon
C# | Create Nuget Package using .NET Standard
Favicon
GitHub | Deploy .net core NuGet Packages in GitHub Packages Registry
Favicon
CSV Schema Validation
Favicon
Publishing Nuget in GitHub Packages
Favicon
.NET MAUI: Update NuGet Packages using Visual Studio Code
Favicon
How I Built a NuGet Package
Favicon
Lee's opinions on Umbraco + naming things
Favicon
Migrating the XM Cloud Introduction Repo to a new Nuget feed.
Favicon
How This NuGet Package Almost Cost Me My Job
Favicon
How to create and deploy a “Nuget Package” using Visual Studio
Favicon
How to change default Nuget packages folder on Windows
Favicon
How to Make a NuGet Package for C++ Development in Visual Studio
Favicon
Data Validation Nuget Package
Favicon
Create a NuGet-Package
Favicon
AspNetCore.VersionInfo 1.1.0 is out
Favicon
Deploying NuGet packages with Docker in GitHub actions
Favicon
Introducing TF WhatsUp, a Tool for Better Terraform Notes
Favicon
Creating a Nuget Package From a .Net 6 class library
Favicon
Elasticsearch.Net vs NEST

Featured ones: