Showing posts with label gitlab. Show all posts
Showing posts with label gitlab. Show all posts

Tuesday, May 31, 2022

Build and deploy .net framework / .net core to IIS with Gitlab and some awesome templates

In this blog I'll explain the way we do our builds with gitlab. It took some time to get to this, so I'm happy to share it with you. At first we set the ground-rules:
  1. You use GitLab
  2. You have GitLab runners on windows machines on powershell (core)
  3. Make use of merge requests to merge code changes to the default branch
  4. Create release tags in the form of semver '1.0.0'
  5. The default branch always gets deployed to your development IIS server
  6. After creating a release in GitLab, a production build is generated
  7. Manual trigger starts the deployment to your production IIS server
  8. There is a solution file in the root of the project
  9. There is only 1 deployable webproject in your solution
  10. The dotnet-* templates reside in a shared repository
Challenges we faced we're especially with the msdeploy command which behaves really funky under powershell (I dare you to do a duckduckgo search on this and see what you'll find). I really like the way everything is parameterized, so the core just works, but if you need a bit more, you just adjust where needed without breaking all the defaults in the build an deploy pipelines. This makes maintaining them a lot easier when environments change and forces you to use a standardized way of work. 

I also like the validation of the release tag numbering using the .pre step validate_tag job. It only runs when a tag is set and it's not in the form of a semver. You can never build a production packages without a proper number.

The .gitlab.yml is the ci file for your end project, it sets urls and can customize some configuration. The dotnet-framework.v1.gitlab-ci.yml and dotnet-core.v1.gitlab-ci.yml files are the base templates you can inherit in the .gitlab-ci.yml. 

Below are all the gists needed to get you going or just to get inspired. If you have any feedback, I'm happy to learn about your experiences. Comment on the gist or this blog!

Happy deploying,
Luuk

# This file shows how to setup the template for your projects
include:
- project: 'templates'
file: 'gitlab-templates/dotnet-framework.v1.gitlab-ci.yml'
variables:
ENABLE_UNITTESTS: 'true'
DEFAULT_IIS_SITENAME: "my-awesome-deployment.nl" # For dev deploys this gets prefixed with 'development.', for prod it gets prefixed with 'www.'
DEFAULT_DEVELOPMENT_URL: "https://my-awesome-deployment.devserver.nl/"
DEFAULT_PRODUCTION_URL: "https://www.my-awesome-deployment.nl/"
# If you want to do some additional things during the default steps,
# for example build node stuff, you can do that in the 'before_script'
build_debug:
cache:
key: npm-cache
paths:
- '$env:CI_PROJECT_DIR\\.npm\\'
before_script:
- 'npm ci --cache $env:CI_PROJECT_DIR\\.npm --prefer-offline'
- 'npx lerna bootstrap -- --cache $env:CI_PROJECT_DIR\\.npm --prefer-offline'
- 'npx lerna run test'
- 'npx lerna run build'
# Or you can overrule the variables for the seperate steps
deploy_production:
variables:
IIS_SERVER: 'overruled-prodserver.nl'
view raw .gitlab-ci.yml hosted with ❤ by GitHub
include:
- project: 'templates' # refer to the base template of dotnet-framework.v1.gitlab-ci.yml
file: 'gitlab-templates/dotnet-framework.v1.gitlab-ci.yml'
build_debug:
stage: build
script:
- 'dotnet restore'
- 'dotnet build --nologo --no-restore --verbosity quiet /p:DeployOnBuild=true /p:WebPublishMethod=Package /p:PackageAsSingleFile=true /p:PackageLocation="$CI_PROJECT_DIR\$PACKAGE_TO_DEPLOY" /p:Configuration=Debug /p:EnvironmentName=Staging'
# INSTALL https://www.nuget.org/packages/JUnitXml.TestLogger/ into the test projects
test:
script:
- 'dotnet restore'
- 'dotnet test --no-restore --test-adapter-path:. --logger:"junit;LogFilePath=..\..\artifacts\{assembly}-test-result.xml;MethodFormat=Class;FailureBodyFormat=Verbose"'
build_release:
stage: build
script:
- 'dotnet restore'
- 'dotnet build --nologo --no-restore --verbosity quiet /p:DeployOnBuild=true /p:WebPublishMethod=Package /p:PackageAsSingleFile=true /p:PackageLocation="$CI_PROJECT_DIR\$PACKAGE_TO_DEPLOY" /p:Configuration=Release /p:EnvironmentName=Production'
# This is de default build script for .net solutions
# To implement this, override the variables with [OVERRIDE THIS VARIABLE]
# Or overrule parts of the script that dont fit your needs
# See https://docs.gitlab.com/ee/ci/yaml/includes.html
# The deployment phase expects a MSDEPLOY package as a zip file in artifacts\website.zip
# To overrule the file location update the PACKAGE_TO_DEPLOY variable.
include:
- template: "Workflows/MergeRequest-Pipelines.gitlab-ci.yml"
stages:
- build
- test
- deploy
variables:
ENABLE_UNITTESTS: 'false'
DEFAULT_IIS_SITENAME: "[OVERRIDE THIS VARIABLE]"
DEFAULT_DEVELOPMENT_URL: "[OVERRIDE THIS VARIABLE]"
DEFAULT_PRODUCTION_URL: "[OVERRIDE THIS VARIABLE]"
PACKAGE_TO_DEPLOY: 'artifacts\website.zip'
NUGET_PATH: 'C:\tools\build\nuget.exe'
MSBUILD_PATH: 'C:\Program Files (x86)\Microsoft Visual Studio\2022\BuildTools\MSBuild\Current\Bin\MSBuild.exe'
MSDEPLOY_PATH: 'C:\Program Files\IIS\Microsoft Web Deploy V3\msdeploy.exe'
validate_tag:
stage: .pre
script:
- echo Release tags should follow semver numbering
- echo For example "1.0.3"
- exit 1
tags:
- windows
rules:
- if: $CI_COMMIT_TAG && $CI_COMMIT_TAG !~ /^[0-9]+[.][0-9]+[.][0-9]+$/
build_debug:
stage: build
script:
- '& "$env:NUGET_PATH" restore -Verbosity quiet'
- '& "$env:MSBUILD_PATH" /p:DeployOnBuild=true /p:DeployTarget=Package /p:PackageLocation="$env:CI_PROJECT_DIR\$env:PACKAGE_TO_DEPLOY" /p:Configuration=Debug -consoleloggerparameters:"ErrorsOnly;Summary" -verbosity:quiet -m'
artifacts:
expire_in: 5 days
paths:
- '$env:PACKAGE_TO_DEPLOY'
tags:
- windows
rules:
- if: $CI_COMMIT_TAG
when: never
- if: $CI_MERGE_REQUEST_IID
- if: $CI_COMMIT_BRANCH == $CI_DEFAULT_BRANCH
# INSTALL https://www.nuget.org/packages/JUnitXml.TestLogger/ into the test projects
test:
stage: test
script:
- '& "$env:NUGET_PATH" restore -Verbosity quiet'
- '& "$env:MSBUILD_PATH" /p:DeployOnBuild=true /p:DeployTarget=Package /p:PackageLocation="$env:CI_PROJECT_DIR\$env:PACKAGE_TO_DEPLOY" /p:Configuration=Debug -consoleloggerparameters:"ErrorsOnly;Summary" -verbosity:quiet -m'
- 'dotnet test tests\**\bin\**\*.Tests.dll --test-adapter-path:. --logger:"junit;LogFilePath=.\artifacts\{assembly}-test-result.xml;MethodFormat=Class;FailureBodyFormat=Verbose"'
artifacts:
when: always # save test results even when the task fails
expire_in: 1 week
paths:
- artifacts/*test-result.xml
reports:
junit:
- artifacts/*test-result.xml
tags:
- windows
rules:
- if: $ENABLE_UNITTESTS == 'false'
when: never
- if: $CI_COMMIT_TAG
when: never
- if: $CI_MERGE_REQUEST_IID
- if: $CI_COMMIT_BRANCH == $CI_DEFAULT_BRANCH
deploy_development:
environment:
name: development
url: $DEFAULT_DEVELOPMENT_URL
stage: deploy
variables:
IIS_SERVER: 'devserver.nl'
IIS_SITENAME: development.$DEFAULT_IIS_SITENAME
IIS_WEBDEPLOY_USERNAME: $DEVSERVER_WEBDEPLOY_USERNAME # SET BY CI/CD Protected variables
IIS_WEBDEPLOY_PASSWORD: $DEVSERVER_WEBDEPLOY_PASSWORD # SET BY CI/CD Protected variables
script:
- |
$arguments = [string[]]@(
"-verb:sync",
"-source:package='$PACKAGE_TO_DEPLOY'",
"-dest:auto,computerName='https://${IIS_SERVER}:8172/msdeploy.axd?site=$IIS_SITENAME',userName='$IIS_WEBDEPLOY_USERNAME',password='$IIS_WEBDEPLOY_PASSWORD',authtype='Basic'",
"-setParam:name='IIS Web Application Name',value='$IIS_SITENAME'",
"-allowUntrusted",
"-skip:Directory='App_Data'",
"-skip:skipaction='Delete',objectname='dirPath',absolutepath='\\media'")
- Start-Process "$env:MSDEPLOY_PATH" -ArgumentList $arguments -NoNewWindow -Wait
tags:
- windows
rules:
- if: $CI_COMMIT_TAG
when: never
- if: $CI_MERGE_REQUEST_IID
when: manual
allow_failure: true
- if: $CI_COMMIT_BRANCH == $CI_DEFAULT_BRANCH
dependencies:
- build_debug
build_release:
stage: build
script:
- '& "$env:NUGET_PATH" restore -Verbosity quiet'
- '& "$env:MSBUILD_PATH" /p:DeployOnBuild=true /p:DeployTarget=Package /p:PackageLocation="$env:CI_PROJECT_DIR\$env:PACKAGE_TO_DEPLOY" /p:Configuration=Release -consoleloggerparameters:"ErrorsOnly;Summary" -verbosity:quiet -m'
artifacts:
paths:
- '$PACKAGE_TO_DEPLOY'
tags:
- windows
rules:
- if: $CI_COMMIT_TAG
deploy_production:
environment:
name: production
url: $DEFAULT_PRODUCTION_URL
stage: deploy
variables:
IIS_SERVER: 'prodserver.nl'
IIS_SITENAME: www.$DEFAULT_IIS_SITENAME
IIS_WEBDEPLOY_USERNAME: $PRODSERVER_WEBDEPLOY_USERNAME # SET BY CI/CD Protected variables
IIS_WEBDEPLOY_PASSWORD: $PRODSERVER_WEBDEPLOY_PASSWORD # SET BY CI/CD Protected variables
script:
- |
$arguments = [string[]]@(
"-verb:sync",
"-source:package='$PACKAGE_TO_DEPLOY'",
"-dest:auto,computerName='https://${IIS_SERVER}:8172/msdeploy.axd?site=$IIS_SITENAME',userName='$IIS_WEBDEPLOY_USERNAME',password='$IIS_WEBDEPLOY_PASSWORD',authtype='Basic'",
"-setParam:name='IIS Web Application Name',value='$IIS_SITENAME'",
"-allowUntrusted",
"-skip:Directory='App_Data'",
"-skip:skipaction='Delete',objectname='dirPath',absolutepath='\\media'")
- Start-Process "$env:MSDEPLOY_PATH" -ArgumentList $arguments -NoNewWindow -Wait
tags:
- windows
rules:
- if: $CI_COMMIT_TAG
when: manual
dependencies:
- build_release