Getting started: Continuous deployment with Azure Bicep and Azure DevOps

Photo of author

Dan Rios

11 min read

Introduction – continuous deployment with Azure Bicep

Continuous deployment with Azure Bicep using Azure DevOps provides a streamlined approach to deploying your resources. This method leverages Infrastructure as Code, harnessing the power of Azure DevOps tools such as source control, Azure Pipelines, and Azure Artifacts, all while maintaining environment control and auditing.

In this article, I aim to walk you through a reference method for the continuous deployment of your Bicep templates using Azure DevOps. I will also share insights into the decisions made during this process.

Subscription scope

Before we delve deeper into this post, I’d like to highlight that this setup is based on a Bicep deployment scoped at the subscription level. I personally prefer this approach whenever possible for resource deployment, using modules to narrow down to resource group deployments as needed.

This strategy eliminates the need to juggle different CLI cmdlet logic and deployment scopes for the pipeline (e.g., between ‘az deployment sub create’ and ‘az deployment group create’). It also allows for the deployment of a broader range of resources, such as policy, RBAC, subscription budgets, etc.

For those who may not be aware, you can use module scopes to ‘scope down’ to the level you require, keeping the deployment command consistent. For instance, refer to the ‘scope:’ section for what I’m alluding to:

module routeTable './modules/routeTable.bicep' = {
  name: 'routeTable'
  scope: resourceGroup(subID,RGName)
  params: {
    name: 'rt-${prefix}-${location}-app-${env}'
    location: location
    routes: []
    tags: {
      Pipeline: 'Yes'
      Owner: 'Dan'
      Env: env
      Created: dateTime
    }
  }
}
BICEP

Obviously this means management & tenant level resources are out of scope for this example.

Azure DevOps Setup

Prerequisites

Below are the prerequisites for following this post’s setup. The hyperlinks will direct you to the excellent Microsoft documentation, which guides you through each requirement. I won’t repeat the documentation here, as it is already very comprehensive.

Azure DevOps Organisation

Azure DevOps Project

Azure DevOps Git Repository

Create a new workload identity federation service connection

You may already have these set up, in which case you can ignore this. However, I strongly recommend using a workload identity federation service connection, as it offers a great deployment identity option with increased security and secret-less management.

Git Repository Structure

For this blog post we’re following a fairly simplistic GIT repository structure with a Bicep folder and a singular main.bicep file that contains all our resources for our application/project which is a fairly common setup for an application deployment.

This may look like this:

├── Bicep/
│   ├── modules/
│   ├── main.bicep
│   └── main.bicepparam
├── bicepconfig.json
├── .gitignore
└── README.md
YAML
Azure DevOps repository for Azure Bicep Deployments
TIP: Protect the main branch via policies by enforcing a minimum required reviewers and/or a build validation policy. You can read more here. I’ve also blogged previously about this: The ultimate Bicep validation pipeline guide: ADO Edition – Rios Engineer

If you want an example Bicep file to demo this, you can use these as an example:

main.bicep

targetScope = 'subscription'

metadata name = 'Resource Group'
metadata description = 'This template deploys a Resource Group as part of CICD - Rios Engineer'
metadata owner = '[email protected]'

@description('Resource Group region location.')
param location string

@description('Time date parameter for resource tagging.')
param dateTime string = utcNow('d')

@description('Resource Group to deploy.')
param rgName string

// Azure Resource Group tags
var tags = {
  Demo: 'ADO Bicep CICD'
  Owner: 'Dan'
  Env: 'Prod'
  Creation: dateTime
}

module resourceGroupDeploy 'br/public:avm/res/resources/resource-group:0.4.0' = {
  name: '${uniqueString(deployment().name, location)}-rg'
  params: {
    name: rgName
    location: location
    tags: tags
  }
}
BICEP

main.bicepparam

using 'main.bicep'

// Hover over the param to see their descriptions pulled from the main bicep file
param location = 'uksouth'
param rgName = 'example-rg'
BICEP

Azure DevOps Environments

A key aspect of any process is ensuring that you have the appropriate controls and safeguards in place for production deployments. A part of this process involves Azure DevOps environments. Not only will you be able to get a historical audit of each pipeline deployment to production, but you can also set specific approvals and checks in place before the stage can run.

  • In Azure DevOps, navigate to your Project. Under Pipelines, select Environments, then New environment, and finally, select ‘production’.
  • Once created, go to the ‘Approvals and checks’ tab and select ‘Approvals’ from the ‘Add your first check’ option. Here, you can add individuals or groups.
Azure DevOps Approvals

To expand on this, there are a lot of excellent checks you can introduce. Although they’re not in the scope of this particular post, you can browse them and see if any fit your use case. For example, you might want to limit deployments to your organisation’s business hours.

In addition, once you’ve started deploying using the environment (after the steps below), you’ll be able to view all historical runs. This includes changes from a source control perspective and any linked work items associated.

Azure DevOps environment audit
Continuous Deployment with Azure Bicep – Environments

Azure Pipelines – Deployment

Next up is the actual release pipeline itself. YAML has become the recent standard practice for all DevOps pipelines, so this will be no different. It allows the pipeline to be within source control, which is an added benefit, and also enables pipeline templating as you expand and standardise pipelines.

  • Create a new file in your Git repository with the extension ending in .yaml or .yml. This can be in the root or within a dedicated pipelines folder, depending on your preference.
  • Once created and committed into the main branch, you need to create a new pipeline:

Pipelines > New pipeline > Azure Repos Git > Select your repository > Existing Azure Pipelines YAML file > Save (under the down arrow on the Run button).

Azure DevOps Pipeline YAML File

For my Azure Bicep pipelines, I do tend to favour Azure CLI over built-in tasks for the most part, as a personal preference. I know many who utilise the ARM template deployment tasks and others, but I like the control and replicating how I would deploy locally in a pipeline as it makes it consistent and easier to troubleshoot issues.

Bicep Deployment YAML

Here’s the Bicep Deployment YAML example that you can copy, paste, and customise as you see fit. It’s been created as a reference for you to pick up and get started. It will work from day one and also serves as a good starting point to understand the flow. This allows you to expand it, add extra stages, steps, tasks, or conditions as you see fit for your use cases.

This is not an exhaustive method on how to structure the pipeline. These can get quite complex for different environments and use cases. For instance, you might include variables for different environments and conditions based on those. Some teams may opt to have no trigger and go for a manual pipeline with these extra conditions to control the deployment, etc.

If you don’t want to deploy at the subscription scope, simply amend the any task that has ‘az deployment sub‘ and replace with ‘az deployment group`
trigger:
branches:
include:
– main
paths:
include:
# amend folder name / structure path to suit your repository
– Bicep/*
variables:
– name: subscription
value: 'YOUR_SUB_GUID'
– name: azureServiceConnection
value: 'YOUR_SERVICE_CONNECTION_NAME'
– name: region
value: 'YOUR_AZURE_REGION'
pool:
vmImage: 'ubuntu-latest'
stages:
– stage: Build
displayName: Bicep Artifact
jobs:
– job: ArtifactDrop
displayName: 🔨 Build Bicep artifact
steps:
– task: CopyFiles@2
displayName: Copy files
inputs:
contents: '**'
targetFolder: '$(Build.ArtifactStagingDirectory)'
– publish: '$(Build.ArtifactStagingDirectory)'
displayName: 'Publish files'
artifact: drop
– stage: whatIf
dependsOn: Build
displayName: What if
jobs:
– job: whatIfBicep
displayName: '🤔 What if'
steps:
– download: current
artifact: drop
– task: AzureCLI@2
displayName: Bicep what if
inputs:
azureSubscription: $(azureServiceConnection)
scriptType: 'bash'
scriptLocation: 'inlineScript'
workingDirectory: $(Pipeline.Workspace)/drop/Bicep
inlineScript: |
name=bicepDeploy$(date +%F_%H-%M-%S)
az deployment sub what-if -l $(region) -n $name -f main.bicep -p main.bicepparam
– job: waitForValidation
dependsOn: whatIfBicep
displayName: 👍 Wait for approval
pool: server
timeoutInMinutes: 60
steps:
– task: ManualValidation@0
timeoutInMinutes: 60
inputs:
notifyUsers: |
[email protected]
instructions: 'Please validate the what-if output and resume'
onTimeout: 'resume'
– stage: Deploy
displayName: Bicep Deploy
jobs:
– deployment: deploy
environment: 'production'
displayName: 🚀 Deploy Prod
strategy:
runOnce:
deploy:
steps:
– download: current
artifact: drop
– task: AzureCLI@2
displayName: Deploy Bicep
inputs:
azureSubscription: $(azureServiceConnection)
scriptType: 'bash'
scriptLocation: 'inlineScript'
workingDirectory: $(Pipeline.Workspace)/drop/Bicep
inlineScript: |
name=bicepDeploy$(date +%F_%H-%M-%S)
az account set –subscription $(subscription)
az deployment sub create -l $(region) -n $name -f main.bicep -p main.bicepparam

Artifact stage

I prefer to construct an artifact at the moment of a merge into the main branch, allowing me to capture a snapshot of the Azure Bicep code at that specific point in time.

This approach ensures that the code you are deploying to production doesn’t inadvertently include additional merges that might occur between the actual deployment and the build phase.

Without this stage, there’s a risk that multiple merges happening in quick succession could lead to deploying several changes simultaneously, rather than managing each one individually.

The alternative would be for the pipeline to perform a repository checkout, which would bring in all current files from the main branch for deployment. These files might differ from what was planned for that particular pipeline deployment run, which is what we’re trying to avoid.

Once the build stage is completed, it will upload the contents as a pipeline artifact to a staging directory that can be used by subsequent stages. It’s worth noting that this process only copies the contents from the repository, as opposed to executing an actual ‘az bicep build’ command.

You even have the option to download the contents locally for review from the pipeline overview, by selecting ‘download artifacts’ next to the three-dot menu.

Azure DevOps Bicep Artifact

What if

Before proceeding directly to a production deployment, I prefer to execute a ‘what-if’ function. This allows you to compare the differences between the current deployment state and your changes or additions to the Bicep template.

One thing to note is that the ‘what-if’ function does have some known issues with nested resources, which can result in incorrect change log outputs aka ‘short circuit’. More can be found in this GitHub thread: Resources in nested template not shown if reference is used in the parameter values · Issue #157 · Azure/arm-template-whatif (github.com)

At this stage, I’ve included a manual stop point, allowing you to review the ‘what-if’ output and decide whether to proceed with or cancel the deployment. While some may argue this isn’t strictly necessary given that the next phase also includes a stop point, I prefer to include it as a precaution.

In my view, this approach allows technical teams to review the output. Then, you can let the actual deployment approval be a collaborative decision involving various stakeholders, some of whom may not need to be notified at this stage or fully understand the output results.

Azure DevOps Pipeline - What If stage

Before approving, you can go to the ‘Bicep what if’ task and review the output, in our case, a new Resource Group:

Azure DevOps Bicep Pipeline - What If Output

In addition to checking the output, by putting an email into the YAML we’re able to get notifications when a pipeline hits this manual validation stage:

What if manual validation email

Deployment

Finally, we reach the deployment phase. This is where the environment, set up earlier, comes into play. It will prompt you for approval to deploy our code into the production subscription. If we’re satisfied and ready to proceed, you can click on ‘Review’ > ‘Approve’. Feel free to add a comment, such as a change or incident number, for example.

Continuous Deployment with Azure Bicep: Azure DevOps

Then wait for the deployment to finish and we can review that our resource group was created as per the what-if plan.

Continuous deployment with Azure Bicep
Continuous deployment with Azure Bicep

And our resource group deployed to the subscription:

Azure Resource Group

What about dev/test deployments?

Personally, I’m a strong advocate for feature branching and frequent merging of code into the main branch. This approach offers numerous benefits, including simplicity, ease of tracking, and continuous code testing, among others.

If you have a development Azure subscription and wish to deploy into it before merging into the main branch, I would recommend maintaining a long-lived development branch. You can merge your feature branches into this development branch first.

This allows you to create a separate pipeline that triggers based on activity in the development branch and deploys into the Azure Dev subscription for testing. This strategy has the added advantage of not slowing down the development cycle as it requires fewer checks and balances – if you’re wondering why this isn’t included in the previous examples.

While this topic is beyond the scope of this post, I’d be happy to delve deeper if there’s interest. It intersects with decisions regarding DevOps branching strategies – so do let me know!

In addition, please do checkout John Lokerse’s blog post on environmental variables which is very useful for additional environment deployments within your Azure DevOps pipelines: Work with environment variables in Azure Bicep – Azure Cloud | John Lokerse. This is especially useful once you build beyond a basic deployment method and become more advanced with your CI/CD capabailities.

What about Bicep build?

As I mentioned earlier, I’m not executing ‘az bicep build’ during the deployment phase. Instead, I believe it’s best practice to run the build and linting as part of the build validation when a pull request is created. This approach allows you to catch compile issues earlier in the CI/CD lifecycle.

By shifting left as much as possible, you can identify and remediate problems before you initiate deployment runs. Typically, I would incorporate this into the Azure DevOps build policy, which would then perform a build and lint of the Bicep code during a pull request.

Expanding for multi-subscription deployments

Using the above as a reference or starting point, how can you expand to deploy more than just a single template? Perhaps your team or organisation has multiple subscriptions, and you want to enhance or build upon the above for Azure Bicep CI/CD.

In my experience and opinion, it’s always beneficial to trigger pipelines based on specific folders within the repository. This approach allows you to target pipelines based on triggers specific to these folders only. The folders can represent your subscription names, each containing their corresponding Bicep templates and configurations for that environment.

Let me try to visualise the structure a bit to help illustrate my thinking – although it is not limited to this structure, the concept can be applied to your specific use case:

.
└── bicep/  
    ├── modules/
    │   ├── keyvault.bicep
    │   └── tests/
    │       └── keyvault.tests.bicep
    ├── connectivity-sub/
    │   ├── main.bicep
    │   └── main.bicepparam
    ├── corp-sub/
    │   ├── main.bicep
    │   └── main.bicepparam
    ├── managaement-sub/
    │   ├── main.bicep
    │   └── main.bicepparam
    └── identity-sub/
        ├── main.bicep
        └── main.bicepparam
YAML

This would be similar to the pipeline example I’ve shared earlier in my GitHub gist, and it might look something like this:

trigger:
branches:
include:
– main
paths:
include:
# amend folder name / structure path to suit your repository
– bicep/connectivity-sub
variables:
– name: subscription
value: 'YOUR_SUB_GUID'
– name: azureServiceConnection
value: 'YOUR_SERVICE_CONNECTION_NAME'
– name: region
value: 'YOUR_AZURE_REGION'
pool:
vmImage: 'ubuntu-latest'
stages:
– stage: Build
displayName: Bicep Artifact
jobs:
– job: ArtifactDrop
displayName: 🔨 Build Bicep artifact
steps:
– task: CopyFiles@2
displayName: Copy files
inputs:
contents: '**'
targetFolder: '$(Build.ArtifactStagingDirectory)'
– publish: '$(Build.ArtifactStagingDirectory)'
displayName: 'Publish files'
artifact: drop
– stage: whatIf
dependsOn: Build
displayName: What if
jobs:
– job: whatIfBicep
displayName: '🤔 What if'
steps:
– download: current
artifact: drop
– task: AzureCLI@2
displayName: Bicep what if
inputs:
azureSubscription: $(azureServiceConnection)
scriptType: 'bash'
scriptLocation: 'inlineScript'
workingDirectory: $(Pipeline.Workspace)/drop/bicep/connectivity-sub
inlineScript: |
name=bicepDeploy$(date +%F_%H-%M-%S)
az deployment sub what-if -l $(region) -n $name -f main.bicep -p main.bicepparam
– stage: DeployConnectivity
dependsOn: whatif
variables:
jobs:
– deployment: prod
environment: 'production'
displayName: 'Deploy to Connectivity Subscription'
pool:
vmImage: $(vmImageName)
strategy:
runOnce:
deploy:
steps:
– download: current
artifact: drop
– task: AzureCLI@2
displayName: Deploy Connectivity Sub
inputs:
azureSubscription: $(azureServiceConnection)
scriptType: 'bash'
scriptLocation: 'inlineScript'
workingDirectory: $(Pipeline.Workspace)/drop/bicep/connectivity-sub
inlineScript: |
name=connectivityDeploy$(date +%F_%H-%M-%S)
az deployment sub what-if -l $(region) -n $name -f main.bicep -p main.bicepparam

As you can see in the example above, you can trigger a deployment based on the repository subfolder (line 79). This subfolder contains the specific subscription’s template code, performs a ‘what-if’ operation, and then deploys.

FYI: Setting the ‘azureSubscription’ ARM connection dynamically in AzureCLI and PowerShell tasks is not supported. This can make multi-env/subscription pipelines with different connection contexts more challenging, but there are workarounds. Example below:
parameters:
- name: env
  type: string
  default: 'dev'
  values:
  - dev
  - test
  - prod
  
  variables:
  - group: azure_service_connections # ADO library with each subscriptions service connection 
  - name: azureConnection
  # sets the azureConnection var to the ADO library name aka service connection depending on the parameter selected
  ${{ if eq(lower(parameters['env']), 'dev') }}:
    value: $(SERVICE_CONNECTION_NAME)
  ${{ elseif eq(lower(parameters['env']), 'test') }}:
    value: $(SERVICE_CONNECTION_NAME_TEST)
  ${{ elseif eq(lower(parameters['env']), 'prod') }}:
    value: $(SERVICE_CONNECTION_NAME_PROD)
    
  - task: AzureCLI@2
    displayName: Deploy
    inputs:
      azureSubscription: $(azureConnection) 
      scriptType: pscore
      scriptLocation: inlineScript
      inlineScript: |
YAML

Further reading & lab

If you’re interested in checking out the continuous integration side of Azure DevOps with Azure Bicep, I would recommend checking out my previous blog post on this subject: The ultimate Bicep validation pipeline guide: ADO Edition – Rios Engineer

Also Azure Bicep test files, using PSRule with a demo lab, can be found here: Bicepify/bicep-cicd-examples/module-tests-with-psrule at main · riosengineer/Bicepify (github.com) – Give it a star if you found it useful!

Conclusion

Hopefully, this was a useful walk-through for continuous deployment with Azure Bicep. There are quite a few different ways to approach this, including different Azure DevOps tasks, folder structures, and tooling options. But as a starting point, this blog should offer some insight to get running and customise for your own needs, including more complex scenarios.

Let me know in the comments how you approach this, and if there’s anything else you’d like me to cover.

6 thoughts on “Getting started: Continuous deployment with Azure Bicep and Azure DevOps”

  1. Hi Dan,

    Great blog and one I’m currently running with internally. A quick question for you from the What about dev/test deployments? part

    If you have an .env file in your branches, how would you manage that in feature branches to ensure that the long standing development and main don’t merge environment variables?

    Thanks

    Reply
    • Thanks Andrew!

      Good question, in my experience at least, I’ve tried to have a singular main file and make the bicepparam env specific so the template is agnostic as possible. Then in the pipelines I typically deploy in stages dev > test > prod where each stage grabs the env.bicepparam file and runs the main.bicep template. Of course, there are so many ways to do it, that’s just sort of how I’ve done it at a lot of gigs.

      Is this for an application deployment infra, or more like an entire landing zone?

      Dan

      Reply
  2. Ah that’s how you do it, I can see how that would work.
    It’s aimed at application deployment infra, what I have is very similar, where I have the main.bicep and one main.bicepparam file, but then within the main.bicepparam file I call the environment variables, so for example param parEnableDiagnosticSettings = bool(readEnvironmentVariable(‘ENABLE_DIAGNOSTIC_SETTINGS’))

    I just want to try and find a happy medium for this, but this is the one part I’m a little stuck on at the moment, suppose I could have different .env files which would work

    Reply
    • Got you. I guess we are both doing the same thing with a different approach. I quite like how you have done it actually! I assume you read the env var from the YAML when pipeline runs (if you do use a pipeline to deploy). So if it’s dev, it reads that and will decide if it will enable/disable that property/object/feature etc. Nice.

      I’m a big fan of merging to main little and often which helps these issues vs longer standing feature branches personally. If you have an .env in a long standing branch (feature/thing) and don’t want to overwrite when it merges to main then you probably have to have some sort of separate file. Although I can’t fully visualise the use case without seeing the repo. If you have it in GitHub (sanitised), I’m happy to check it out and bounce off with you.

      Reply
  3. That’s correct 🙂

    So in talking to you, I think I’ve got the solution. .env files in environment folders, utilise parameters on the pipeline, like you have for the Service connection and just maintain the environment config in the .env files.

    The answer for what I wanted was already in your blog 😀

    I haven’t a sanitised version at the moment, but will be updating our main repo with what i’ve got above, so would be happy to share when I get there.

    Thanks again Dan!

    Reply

Leave a comment


Skip to content