Exercise - Add a Bicep deployment action to the workflow

Completed

You've created a basic workflow, and you've configured your Azure and GitHub environments to connect. Now, you're ready to deploy your website's Bicep file to Azure from your workflow.

In this exercise, you'll:

  • Add a Bicep file to your repository.
  • Add a workflow step to download your repository source code to the runner's file system.
  • Add a workflow step to sign in to Azure.
  • Add a workflow step to deploy your Bicep file.
  • Run your workflow again and verify that it successfully deployed your website.

Add your website's Bicep file to the GitHub repository

You've already prepared your website's Bicep file, which you can use to deploy different configurations of the website resources depending on the environment and configuration. Here, you'll add your Bicep file to your repository.

  1. Open the Visual Studio Code Explorer.

  2. At the root of your repository, create a deploy folder.

  3. In the deploy folder, create a new file named main.bicep. Make sure you create the file inside the deploy folder:

    Screenshot of the Visual Studio Code Explorer, with the main dot bicep file highlighted and located in the deploy folder.

  4. Copy the following code into the main.bicep file:

    @description('The Azure region into which the resources should be deployed.')
    param location string = resourceGroup().location
    
    @description('The type of environment. This must be nonprod or prod.')
    @allowed([
      'nonprod'
      'prod'
    ])
    param environmentType string
    
    @description('A unique suffix to add to resource names that need to be globally unique.')
    @maxLength(13)
    param resourceNameSuffix string = uniqueString(resourceGroup().id)
    
    var appServiceAppName = 'toy-website-${resourceNameSuffix}'
    var appServicePlanName = 'toy-website-plan'
    var toyManualsStorageAccountName = 'toyweb${resourceNameSuffix}'
    
    // Define the SKUs for each component based on the environment type.
    var environmentConfigurationMap = {
      nonprod: {
        appServicePlan: {
          sku: {
            name: 'F1'
            capacity: 1
          }
        }
        toyManualsStorageAccount: {
          sku: {
            name: 'Standard_LRS'
          }
        }
      }
      prod: {
        appServicePlan: {
          sku: {
            name: 'S1'
            capacity: 2
          }
        }
        toyManualsStorageAccount: {
          sku: {
            name: 'Standard_ZRS'
          }
        }
      }
    }
    
    var toyManualsStorageAccountConnectionString = 'DefaultEndpointsProtocol=https;AccountName=${toyManualsStorageAccount.name};EndpointSuffix=${environment().suffixes.storage};AccountKey=${toyManualsStorageAccount.listKeys().keys[0].value}'
    
    resource appServicePlan 'Microsoft.Web/serverfarms@2023-12-01' = {
      name: appServicePlanName
      location: location
      sku: environmentConfigurationMap[environmentType].appServicePlan.sku
    }
    
    resource appServiceApp 'Microsoft.Web/sites@2023-12-01' = {
      name: appServiceAppName
      location: location
      properties: {
        serverFarmId: appServicePlan.id
        httpsOnly: true
        siteConfig: {
          appSettings: [
            {
              name: 'ToyManualsStorageAccountConnectionString'
              value: toyManualsStorageAccountConnectionString
            }
          ]
        }
      }
    }
    
    resource toyManualsStorageAccount 'Microsoft.Storage/storageAccounts@2023-05-01' = {
      name: toyManualsStorageAccountName
      location: location
      kind: 'StorageV2'
      sku: environmentConfigurationMap[environmentType].toyManualsStorageAccount.sku
    }
    
  5. Save your changes to the file.

  6. In the Visual Studio Code terminal, run this code to stage the changes, commit the changes, and push the changes to your repository:

    git add deploy/main.bicep
    git commit -m 'Add Bicep file'
    git push
    

Replace the workflow steps

Next, update your workflow definition to deploy your Bicep file to Azure.

  1. In Visual Studio Code, open the .github/workflows/workflow.yml file.

  2. At the top of the file, between on: and jobs:, add a permissions: section.

    name: deploy-toy-website
    
    on: [workflow_dispatch]
    
    permissions:
      id-token: write
      contents: read
    
    jobs:
    

    This change allows the workflow to use a workload identity.

  3. Rename the say-hello job to deploy:

    jobs:
      deploy:
        runs-on: ubuntu-latest
        steps:
    
  4. To remove the placeholder step from the workflow definition, delete the bottom two lines of the file.

  5. As a first step you'll add a task to check out the code to the runner's file system. Add a new step at the bottom of the file:

    jobs:
      deploy:
        runs-on: ubuntu-latest
        steps:
        - uses: actions/checkout@v3
    

    Note

    It's a good idea to type this code yourself instead of copying and pasting it from this module. Pay attention to the file's indentation. If your indentation isn't correct, your YAML file won't be valid. Visual Studio Code indicates errors by displaying squiggly lines.

  6. Below the step that you just added, add a task to sign in to your Azure environment. This task uses the secrets you defined earlier to sign in by using a workload identity:

    jobs:
      deploy:
        runs-on: ubuntu-latest
        steps:
        - uses: actions/checkout@v3
        - uses: azure/login@v1
          with:
            client-id: ${{ secrets.AZURE_CLIENT_ID }}
            tenant-id: ${{ secrets.AZURE_TENANT_ID }}
            subscription-id: ${{ secrets.AZURE_SUBSCRIPTION_ID }}
    
  7. Below the step that you just added, add another step to perform the Bicep deployment:

    jobs:
      deploy:
        runs-on: ubuntu-latest
        steps:
        - uses: actions/checkout@v3
        - uses: azure/login@v1
          with:
            client-id: ${{ secrets.AZURE_CLIENT_ID }}
            tenant-id: ${{ secrets.AZURE_TENANT_ID }}
            subscription-id: ${{ secrets.AZURE_SUBSCRIPTION_ID }}
        - uses: azure/arm-deploy@v1
          with:
            deploymentName: ${{ github.run_number }}
            resourceGroupName: ${{ env.AZURE_RESOURCEGROUP_NAME }}
            template: ./deploy/main.bicep
            parameters: environmentType=${{ env.ENVIRONMENT }}
    

    Notice that this task uses the github.run_number default environment variable to name the deployment in Azure. It also uses environment variables for the resource group name and for the environmentType parameter in the Bicep file.

  8. Add these variables and their values at the top of your workflow file, between permissions: and jobs:

    name: deploy-toy-website
    
    on: [workflow_dispatch]
    
    permissions:
      id-token: write
      contents: read
    
    env:
        AZURE_RESOURCEGROUP_NAME: ToyWebsite
        ENVIRONMENT: nonprod
    
    jobs:
    
  9. Save your changes to the file. Your file should look like this example:

    name: deploy-toy-website
    
    on: [workflow_dispatch]
    
    permissions:
      id-token: write
      contents: read
    
    env:
        AZURE_RESOURCEGROUP_NAME: ToyWebsite
        ENVIRONMENT: nonprod
    
    jobs:
      deploy:
        runs-on: ubuntu-latest
        steps:
        - uses: actions/checkout@v3
        - uses: azure/login@v1
          with:
            client-id: ${{ secrets.AZURE_CLIENT_ID }}
            tenant-id: ${{ secrets.AZURE_TENANT_ID }}
            subscription-id: ${{ secrets.AZURE_SUBSCRIPTION_ID }}
        - uses: azure/arm-deploy@v1
          with:
            deploymentName: ${{ github.run_number }}
            resourceGroupName: ${{ env.AZURE_RESOURCEGROUP_NAME }}
            template: ./deploy/main.bicep
            parameters: environmentType=${{ env.ENVIRONMENT }}
    
  10. In the Visual Studio Code terminal, stage your changes, commit them to your repository, and push them to Azure Repos:

    git add .
    git commit -m 'Add Azure CLI tasks to workflow'
    git push
    

Run your workflow

Now, you're ready to run your workflow!

  1. In your browser, open the workflow by selecting Actions > deploy-toy-website.

  2. Select Run workflow > Run workflow.

  3. A new run of your workflow will appear in the runs list. If it doesn't appear, refresh your browser page.

  4. Select the running workflow to view the details of the run.

    Wait for the run to finish.

  5. Select the deploy job.

    Screenshot of the GitHub interface showing the run page, with the deploy job highlighted.

  6. Select Run azure/arm-deploy@v1. This displays the task details.

  7. Select Run azure/arm-deploy@v1 in the task details.

    Screenshot of the GitHub interface showing the workflow log, with the 'environment variables' highlighted.

    Notice that this step uses the environment variables you added to the workflow file.

  8. Inspect the rest of your workflow output.

    The workflow shows a successful deployment.

Verify the deployment

  1. Go to the Azure portal.

  2. In the left menu, select Resource groups.

  3. Select ToyWebsite.

  4. In Overview, view the deployment status. You can see that one deployment succeeded.

    Screenshot of the Azure portal that shows the resource group with one successful deployment.

  5. Select the 1 Succeeded link to see the details of the deployment.

    Screenshot of the Azure portal that shows the resource group deployment history, with the deployment highlighted.

    Notice that the name of the deployment matches the workflow's run number in GitHub Actions, because you used the github.run_number environment variable to name your deployment.

  6. To see which resources were deployed, select the deployment. To expand the deployment and see more details, select Deployment details. In this case, there's a storage account, an Azure App Service plan, and an app.

    Screenshot of the Azure portal that shows the resource group deployment details, with the App Service resources highlighted.