DEV Community

Ivan G
Ivan G

Posted on

Perfect .NET CI/CD with Azure Pipelines

This article was originally posted on my blog.

Every time I start a new .NET library on GitHub and want to publish it I'm struggling with CI. Most of the projects are quite simple - single or multiple libraries, a unit test library, and maybe some other trivial stuff. What I'd like to achieve from CI/CD is the following:

  1. It needs to build my code on push and on PR.
  2. It needs to run tests.
  3. It needs to package libraries in NuGet, with a proper version. I'd like to have all the properties populated, like license, package icon and so on.
  4. Push them to NuGet.org if the commit is to master branch.
  5. Create GitHub release notes.

I used to struggle with this for quite some time in previous projects, use 3rdparty plugins for Azure Pipelines, write PowerShell scripts, and so on. Then my frustration grew so much I've decided that all I do must be totally wrong, because it's such a simple and repetitive task. And I believe I've solved it with minimum steps involved. I'll go step by step explaining this.

I'm not going to explain the basics of YAML pipelines and assume you know what it is and have created a few already. I suppose you've already put azure-pipelines.yaml in the root of your GitHub repo.

Triggering Builds on Commits to Master and PR

Triggering build is simple - I need to define trigger and pr elements in yaml, this is the very beginning of my yaml:

trigger: # when to trigger a build
  batch: true # means that if there are more changes when build is running, they will be waited for instead of starting a parallel build
  branches:
    include:
      - master # trigger only on changes to master branch
  paths:
    exclude: # what to ignore for trigger - I don't need to rebuild on changes to documentation
      - README.md
      - doc/**/*

pr: # when to trigger a "PR" build.
  branches:
    include:
      - master
  paths:
    exclude:
      - README.md
      - doc/**/*
Enter fullscreen mode Exit fullscreen mode

The comments are self explanatory. You can read more about triggers in Azure Pipelines documentation. pr section is interesting - it basically says that a build should be triggered no matter where the code comes from - it can be inside your repo or someone else's, however if PR is created in GitHub, your pipeline will fetch the code automatically and build with your pipeline. Done deal โ˜•.

Building and Testing

Running builds is actually quite trivial, you can use the built-in dotnet task in your pipeline. Let's say your code is located under src folder, and there is Snappy.sln solution file, then to run build and unite tests you can simply write the following:

- task: DotNetCoreCLI@2
  displayName: 'build all'
    inputs:
    command: 'build'
    projects: 'src/Snappy.sln'
    arguments: '-c release'

- task: DotNetCoreCLI@2
  displayName: 'run tests'
    inputs:
    command: test
    projects: ./src/Snappy.sln
    arguments: '-c release'
Enter fullscreen mode Exit fullscreen mode

First, I run the build task, to check if the code actually builds. Failure to do so will terminate the whole pipeline. Then run unit tests with test command, all in release mode. However, that's never been an issue for me. The main one is how do I set the version of the library when building? Read on ๐Ÿ’จ

Packaging for NuGet

Before I go into the versioning, let me talk about versioning briefly. This is also a very simple task. On early stages of the project it's actually much easier to edit this information in project properties like so:

which will in turn modify .csproj file. This is the ProperyGroup generated:

<PropertyGroup>
    <PackageIcon>icon.png</PackageIcon>
    <PackageLicenseExpression>Apache-2.0</PackageLicenseExpression>
    <GeneratePackageOnBuild>true</GeneratePackageOnBuild>
    <Version>1.0.0</Version>
    <Description>This is a native .NET port of Google Snappy compression/decompression library. The only implementation that is stable, fast, up to date with latest Snappy improvements, and most importantly does not depend on native Snappy binaries. Works everywhere .NET Core runs.</Description>
    <PackageProjectUrl>https://github.com/aloneguid/IronSnappy</PackageProjectUrl>
    <RepositoryUrl>https://github.com/aloneguid/IronSnappy</RepositoryUrl>
    <RepositoryType>Git</RepositoryType>
    <PackageTags>google snappy compression dotnet c#</PackageTags>
    <PackageReleaseNotes>see release history - https://github.com/aloneguid/IronSnappy/releases</PackageReleaseNotes>
    <PackageRequireLicenseAcceptance>true</PackageRequireLicenseAcceptance>
    <Authors>Ivan Gavryliuk (@aloneguid)</Authors>
</PropertyGroup>

<ItemGroup>
    <None Include="icon.png" Pack="true" PackagePath="" />
</ItemGroup>
Enter fullscreen mode Exit fullscreen mode

Note that Version is the tag used by dotnet build or dotnet pack command to actually generate package version. If we could just change it on the fly, it would be awesome.

Versioning

Now that we've got proper .csproj with all the information we need, it's time to generate NuGet package. Note that the package is generated automatically, because we've set <GeneratePackageOnBuild>true</GeneratePackageOnBuild> element in .csproj. It's just it will always use version 1.0.0 (from <Version>1.0.0</Version>).

There are multiple ways to define a version for your project. You can use Azure Pipeline's built-in variables (specifically these, like Build.BuildNumber and so on) or GitVersion which can generate a number based on your Git revision history. Honestly, I've tried both and I didn't like them for many reasons. The first one will rely on Azure Pipelines project location for some variables, and GitVersion requires a third-party add-on for Pipelines, which is quite unstable.

I've ended up defining my version number explicitly in the yaml file, as I see many benefits in that:

  1. I know which version I'm working with by looking at the code.
  2. I want to explicitly set version number before the release in code, based on what I've changed, instead of relying on automatic versioning. Automatic versioning is good when it works, but I've found that I need to come back and correct it just too often.

Therefore I just declare a variable in yaml:

variables:
  v.prefix: '1.1.0'
Enter fullscreen mode Exit fullscreen mode

v.prefix is the version I should build with. How do we pass it to the .csproj? Fortunately, you can redefine any property value from .csproj from command line. For instance, to change the version number during build (Version property) I can do the following:

dotnet build src/Snappy.sln -c release /p:Version=value
Enter fullscreen mode Exit fullscreen mode

It's just a matter of passing it to the task, and it's done in the following way:

- task: DotNetCoreCLI@2
  displayName: 'build all'
    inputs:
    command: 'build'
    projects: 'src/Snappy.sln'
    arguments: '-c release  /p:Version=value'

- task: DotNetCoreCLI@2
  displayName: 'run tests'
    inputs:
    command: test
    projects: ./src/Snappy.sln
    arguments: '-c release /p:Version=value'
Enter fullscreen mode Exit fullscreen mode

Now, when running the pipeline I get the output:

 Successfully created package '/home/vsts/work/1/s/src/IronSnappy/bin/Release/IronSnappy.1.1.0.nupkg'
Enter fullscreen mode Exit fullscreen mode

That's it.

Collecting Packages

Once the build is done, NuGet packages are generated, and it's a matter of publishing them. I'm a big fan of YAML multi-stage pipelines, therefore I like to split my pipelines in multiple stages. For instance, Build stage should just build and run unit tests, nothing more. I'm putting the full stage source here:

trigger:
  batch: true
  branches:
    include:
      - master
  paths:
    exclude:
      - README.md
      - doc/**/*

pr:
  branches:
    include:
      - master
  paths:
    exclude:
      - README.md
      - doc/**/*

variables:
  v.prefix: '1.1.0'
  vmImage: 'ubuntu-latest'

stages:
- stage: Build
  jobs:
  - job: Core
    displayName: 'Core Build'
    pool:
      vmImage: $(vmImage)

    steps:

    - script: echo "##vso[build.updatebuildnumber]$(v.prefix)"

    - task: UseDotNet@2
      inputs:
        packageType: 'sdk'
        version: '3.x'

    - task: DotNetCoreCLI@2
      displayName: 'build all'
      inputs:
        command: 'build'
        projects: 'src/Snappy.sln'
        arguments: '-c release /p:Version=$(v.prefix)'

    - task: DotNetCoreCLI@2
      displayName: 'run tests'
      inputs:
        command: test
        projects: ./src/Snappy.sln
        arguments: '-c release /p:Version=$(v.prefix)'

    - task: CopyFiles@2
      displayName: 'copy generated nugets'
      inputs:
        SourceFolder: src
        Contents: '**/*.nupkg' # search for nupkg recursively
        TargetFolder: ' $(build.artifactstagingdirectory)'
        CleanTargetFolder: true
        OverWrite: true
        flattenFolders: true

    - task: PublishBuildArtifacts@1
      displayName: 'archive nugets'
      inputs:
        ArtifactName: nuget
Enter fullscreen mode Exit fullscreen mode

Notice the last two new tasks - one collects NuGet packages into $(build.artifactstagingdirectory) folder, another just publishes them out of the stage, and it's all done.

Publishing to NuGet and generating Release Notes

The next stage is to publish live and generate release notes on GitHub. This is a good place for a new stage, as it has nothing to do with build.

I'll paste the full stage definition here and explain what it does, both in comments and below.

- stage: Release
  # set dependency on Build stage, so that it runs after only
  dependsOn: ['Build']
  # set run condition:
  # 1. only run when running against master branch, we don't want to run on other branches and PR builds
  # AND
  # 2. if Build succeeds
  condition: "and( eq(variables['Build.SourceBranch'], 'refs/heads/master'), succeeded('Build') )"
  jobs:
  - deployment: NugetOrg
    displayName: 'nuget.org public feed'
    pool:
      vmImage: $(vmImage)
    environment: live
    strategy:
      runOnce:
        deploy:
          steps:
          # download artifacts - nuget packages in this case
          - task: DownloadBuildArtifacts@0
            inputs:
              buildType: 'current'
              downloadType: 'single'
              artifactName: 'nuget'
              downloadPath: '$(System.ArtifactsDirectory)'
          # push to nuget.org
          - task: NuGetCommand@2
            displayName: nuget.org
            inputs:
              command: push
              packagesToPush: '$(System.ArtifactsDirectory)/nuget/*.nupkg'
              nuGetFeedType: external
              publishFeedCredentials: 'nuget.org (aloneguid)'
          # create nice release definition in GitHub
          - task: GitHubRelease@1
            inputs:
               gitHubConnection: 'aloneguid'
               repositoryName: 'aloneguid/IronSnappy'
               action: 'create'
               target: '$(Build.SourceVersion)'
               tagSource: 'userSpecifiedTag'
               tag: '$(Build.BuildNumber)'
               changeLogCompareToRelease: 'lastFullRelease'
               changeLogType: 'commitBased'
Enter fullscreen mode Exit fullscreen mode

The first one we do is define dependency on Build stage. If you don't do that, your stage may run before build which we don't really want!

Also I'd like for it to run only against master branch, because this pipeline runs against PR branches as well, therefore I'll use stage condition expression. One of the awesome things in Azure Pipelines is that you can not just use plain conditions, but also expressions. They are a powerful way to reuse your pipeline depending on how it runs, which is missing in many other CI/CD systems and you end up just duplicating your pipeline for those different environments.

Then DownloadBuildArtifacts@0 task downloads NuGet packages from the previous stage, and NuGetCommand@2 publishes it live to NuGet.org:

And another nice feature you can add is to create release notes with GitHubRelease@1 task which allows consumers of your library and yourself to understand what were the changes in the latest release:

Saves your precious time to generate release notes yourself. In this case release notes are generated based on commits since last release, but you can customise it to your needs.

Essentially, that's it. No third-party plugins or complicated ceremony with CI/CD, all simple and effective. You can reuse this pipeline in other projects with minimum modifications as well. If you need the latest version of this pipeline and see how it works in action you can check the GitHub repo used in this post as an example.

Top comments (0)