Intro

I received feedback asking if I could do more DevOps and Bicep articles, so this is the first in a series about this topic. Like most blog series, I will start from scratch and gradually come near a production environment. It can be hard to say this is fully production-ready due to different policies and operations in each customer environment. I will show what it can be, and then it will be up to you if it will fit in your environment or with your customers.

This first article will show how to create a simple Azure App service using Bicep code. I will deploy the App Service using Azure DevOps and GitHub.

App service Bicep code

Creating an Azure App Service requires a few resources. First, you will need an App Service Plan and then the App Service itself. Application Insights is optional, so I leave it out for this first configuration.

Code for App Service Plan

param location string
param projectName string
param skuName string = 'B1'
param skuTier string = 'Basic'
param kind string = 'Linux'

resource appServicePlan 'Microsoft.Web/serverfarms@2023-12-01' = {
  name: 'asp-${projectName}'
  location: location
  sku: {
    name: skuName
    tier: skuTier
    
  }
  kind: kind
  properties: {
    reserved: (kind == 'Linux') ? true : false
  }
}

output appServicePlanId string = appServicePlan.id

Code for the App Service

param location string
param projectName string
param appServicePlanId string


resource appService 'Microsoft.Web/sites@2023-12-01' = {
  name: 'app-${projectName}'
  location: location
  properties: {
    serverFarmId: appServicePlanId    
  }
}

The code above will create the App Service Plan and App Service, but just deploying the two files requires you to grab the App Service Plan ID manually and use it as input for the App Service. Let’s instead change the code above to a module and then call the module. I will save the code below as app_service.bicep

param kind string = 'Linux'
param location string
param project_name string 
param sku_name string = 'B1'
param sku_tier string = 'Basic'

resource appServicePlan 'Microsoft.Web/serverfarms@2023-12-01' = {
  name: 'asp-${project_name}'
  location: location
  sku: {
    name: sku_name
    tier: sku_tier
    
  }
  kind: kind
  properties: {
    reserved: (kind == 'Linux') ? true : false
  }
}

resource appService 'Microsoft.Web/sites@2023-12-01' = {
  name: 'app-${project_name}'
  location: location
  properties: {
    serverFarmId: appServicePlan.id    
  }
}

To call the module, I will use a Bicep parameter file called “demo1-devops.bicepparam” and “demo1-github.bicepparam”. Below is the code for the Bicep parameter file.

using '../Modules/app_service.bicep'

param location = 'WestEurope'
param project_name = 'demo1-devops'

Azure DevOps

Repository

In Azure DevOps, I will use the default repository created for the project. I will place my module file in a " Modules " folder and the parameter file under “demo1”.

Pipeline

I will create a pipeline for continuous deployment that runs every time I change the bicep parameter file. I can also run the pipeline manually if needed.

name: Demo1

trigger:
  branches:
    include:
    - main
  paths:
    include:
      - 'demo1-devops.bicepparam'

variables:
- name: DeploymentName
  value: 'Demo1'
- name: resourceGroupName
  value: 'rg-bicep-devops-demo'
- name: ServiceConnection
  value: 'mi-bicep-devops'
- name: TemplateParameterFile
  value: 'demo1-devops.bicepparam'

pool:
  vmImage: ubuntu-latest

stages:
- stage: Validate
  jobs:
  - job: Validate
    steps:
    - task: AzurePowerShell@5
      name: Validate
      inputs:
        azureSubscription: $(ServiceConnection)
        ScriptType: 'InlineScript'
        Inline: 'Test-AzResourceGroupDeployment -ResourceGroupName $(resourceGroupName) -TemplateParameterFile $(TemplateParameterFile) -Verbose'
        azurePowerShellVersion: 'LatestVersion'
        pwsh: true
- stage: WhatIf
  jobs:
  - job: WhatIf
    steps:
    - task: AzurePowerShell@5
      name: WhatIf
      inputs:
        azureSubscription: $(ServiceConnection)
        ScriptType: 'InlineScript'
        Inline: 'New-AzResourceGroupDeployment -Name $(DeploymentName) -ResourceGroupName $(resourceGroupName) -TemplateParameterFile $(TemplateParameterFile) -WhatIf -Verbose'
        azurePowerShellVersion: 'LatestVersion'
        pwsh: true
- stage: Deploy
  dependsOn: WhatIf
  jobs:
  - deployment:
    displayName: Deploy
    environment: Production
    strategy:
     runOnce:
       deploy:
        steps:
        - checkout: self
        - task: AzurePowerShell@5
          name: Deploy
          inputs:
            azureSubscription: $(ServiceConnection)
            ScriptType: 'InlineScript'
            Inline: 'New-AzResourceGroupDeployment -Name $(DeploymentName) -ResourceGroupName $(resourceGroupName) -TemplateParameterFile $(TemplateParameterFile) -Verbose'
            azurePowerShellVersion: 'LatestVersion'
            pwsh: true

To use the pipeline, I need to either add it using the Azure DevOps CLI or create it using the file above using the portal. I will do this article in the portal, but we will revert to using the Azure DevOps CLI to manage Azure DevOps.

Adding the pipeline to Azure DevOps

First, I navigate to Pipelines, and then click “Create pipeline.”

I click on “GitHub” since I use the same repository for the pipeline and GitHub Action. The process for using the Azure DevOps repository is the same.

Next, I search for my repository and select it. In my case “mracket/bicep-DevOps.”

I select “Existing Azure Pipelines YAML file.”

I use the drop-down box to select my yaml file.

Now, I can either run my pipeline or save it. I usually save it by clicking on the small arrow next to “run” and selecting " save."

Azure DevOps uses the repository name as the pipeline name. For me, that doesn’t look good. I can click the menu at the top right and select “Rename/move.”

I name the pipeline “demo1.”

Now, I will run the pipeline, and after a bit, you will most likely see a warning saying, “This pipeline needs permission to access resources before this run can continue to Deploy.” I click on “View” to add these permissions.

I click on “Permit.”

Click on “Permit” again.

After a minute or so, the deployment is complete.

GitHub

Repository

The repository part of GitHub is the same as that of DevOps. The folder structure is the same, so only the use of GitHub service instead of Azure DevOps differs.

GitHub Action

GitHub Actions are equivalent to Azure DevOps Pipelines, and they are also very powerful and support the continuous deployment of services. The code for GitHub Actions is YAML, just like Azure DevOps. The actions used in Github are not the same as those used in Azure DevOps, but they perform the same thing, just another component. GitHub Actions have one significant advance: placing the yaml file in a special folder automatically adds it to GitHub as a workflow.

The code below will perform the same as the Azure DevOps Pipeline we created previously.

on:
  push:
    paths:
      - demo1.bicepparam
    branches: main
name: demo1
permissions:
  id-token: write
  contents: read
jobs:
  build-and-deploy:
    runs-on: ubuntu-latest
    steps:
      # Checkout code
    - uses: actions/checkout@main
    
    - name: 'Az CLI login'
      uses: azure/login@v2
      with:
       client-id: ${{ secrets.CLIENT_ID }}
       tenant-id: ${{ secrets.TENANT_ID }}
       subscription-id: ${{ secrets.SUBSCRIPTION_ID }}
    - name: Deployment
      uses: azure/bicep-deploy@v2
      with:
        type: deployment
        resource-group-name: rg-bicep-devops-demo
        operation: create
        name: demo1
        scope: resourceGroup
        subscription-id: ${{ secrets.SUBSCRIPTION_ID }}
        parameters-file: demo1.bicepparam

I can now browse to the “Actions” section in GitHub and see that my workflow is present, as well as that it has run a deployment.

App service running

To verify that my deployment worked as expected, I can check the Azure Portal for service status and click the URL for the app service. Below is a screenshot of the service in the Azure Portal, and another is where the service shows the website it is hosting.

Summary

To conclude this first post in the new series about DevOps and Bicep, deploying Bicep with either Azure DevOps or GitHub is easy. You must be comfortable with Bicep, Yaml, and Git in general. The deployments done in this post are very simple, but with this knowledge, it is possible to deploy any resource using Bicep with Azure DevOps or GitHub. In an upcoming post, I will discuss code organization, Bicep module versioning, and more advanced deployments.

You are welcome to contact me via any of my social media if you have any feedback.