Updating Windows and create a new Image Version for Azure Virtual Desktop automated.

Updating Windows and create a new Image Version for Azure Virtual Desktop DevOps Pipelines.

Hi Welcome back! Today I want to talk about a tricky point that always bothers me when managing and setting up an Azure Virtual Desktop in combination with a Shared Image from the Azure Compute Gallery, namely updating and running Windows Updates.

Why do I always find this a difficult point regarding maintenance? The answer is simple, it takes a lot of manual work when you have to spin up a machine, manually run the updates, sysprep it and then capture it back to the Azure Compute Gallery.

Are there no alternatives? Yes, but these also have disadvantages, for example you can use the Azure Update Manager and run updates on the machines instead of on the original image, but in the event of a disaster you must first update all machines again to bring everything up to date. The principle of one shared "golden" image no longer applies, which is why I do not use this method.

Ofcourse there are 3th party solutions that can do the same for you, but I looked for an option within my own technologies that I know can fine-tune myself.

So I would like to take you through the process of providing an image with the latest Windows updates.


Where do we begin.

We will of course start with the basics and that is of course an Azure Virtual Desktop environment with an Azure Compute Gallery, which I have already deployed in my environment. If you have not yet set this up and would like to know how to do this, I would like to refer you to one of my previous blogs. In this blog, an Azure Virtual Desktop is installed using an Azure DevOps Pipeline.

In addition to the environment that you proably already deployed, we will of course use a DevOps Pipeline, so you will also need to have a DevOps environment that makes it possible to run Pipelines.

You can use Microsoft hosted agents, which are free and allow you to run a maximum number of hours per month, for this you must create an organization within DevOps and request the free tier from Microsoft.

As an alternative you can use self-hosted agents for more capabilities, my friend Simon Lee has written a very good blog about this how to set this up.

Finally, we of course use yaml files that make it possible to set up the Pipelines in this yaml file I use:

  • Azure CLI
  • Powershell

Let's explain the process.

Let me explain the process:

  • First we deploy a virtual machine based on the latest image, which will retrieve and install the updates.
  • We will do a first check for Windows Updates and install the necessary updates.
  • We will do a second check for Windows Updates and install the necessary updates.
  • We will do a last check and output the history with installed updates and clean-up some folders.
  • It will create a snapshot of the disk.
  • When the snapshot is done it will start the sysprep process.
  • Capture the image and store it as a new version in the Azure Compute Gallery.

We need some specific powershell modules namely:

  • Az.Accounts
  • Az.Compute
  • pswindowsupdate

For this modules we need to install the package provider NuGet so that we can download and install the PSWindowsupdate module, this will be done in the yaml file, I will explain a little bit more when we dive further in the yaml file.

If you want to know more about the specific capabilities of PSWindowsUpdate, you can find more documentation about this on this GitHub site.


Let's check the yaml file.

You can find the full file in my repository, please adjust it like you want but let's break it down and view the first step:

  • We are using the "Credentials" group that is connected to the library, in this library the credentials and secrets are stored for the app registration that is connected with Azure DevOps, but also the local admin and password for the installation VM.

  • I use input parameters to start the pipeline and use this as namingconvention to get my resources in Azure.

 1name: update-windows
 2
 3trigger:
 4  branches:
 5    include:
 6      - main
 7
 8pr:
 9  branches:
10    include:
11      - main
12
13variables:
14  - group: "Credentials"
15
16parameters:
17  - name: deploymentType
18    displayName: "Select the deployment type"
19    type: string
20    default: "sub"
21    values:
22      - tenant
23      - mg
24      - sub
25      - group
26
27  - name: subscriptionId
28    displayName: "Azure Subscription"
29    type: string
30    default: ""
31
32  - name: productType
33    displayName: "Select the product type"
34    type: string
35    default: "avd"
36    values:
37      - avd
38
39  - name: environmentType
40    displayName: "Select the environment"
41    type: string
42    values:
43      - prod
44      - acc
45      - dev
46      - test
47
48  - name: location
49    displayName: "Select the location"
50    type: string
51    default: "westeurope"
52    values:
53      - westeurope
54      - northeurope
55
56  - name: vmName
57    displayName: "Name of the VM"
58    type: string
59    default: ""
60
61  - name: vmSize
62    displayName: "Size of the VM"
63    type: string
64    default: "Standard_D2s_v3"
65  
66  - name: imageVersion
67    displayName: "fill in the image version name like 2025.03.14"
68    type: string
69    default: "2025.03.14"

If we are going further in the file we are going to deploy the Virtual Machine,

  • We do this step using Azure CLI.
  • First we log in to the environment based and set the subscription az account set
  • The location will be checked based on the input variables (keep in mind that you need to check your own avd environment).
  • Virtual Machine name will be declared based on the input variables.
  • The Resource group will be declared based on the input variables.
  • The latest version will be picked in the Image Gallery based on a query.
  • NIC will be created.
  • Virtual Machine will be created.
 1stages:
 2  - stage: DeployVMs
 3    displayName: "Deploy VMs using the latest image"
 4    jobs:
 5      - job: BuildVMs
 6        displayName: "Build Virtual Machines"
 7        pool:
 8          vmImage: "ubuntu-latest"
 9        steps:
10          - task: AzureCLI@2
11            displayName: "Create VMs"
12            inputs:
13              azureSubscription: "AVDPipelines"
14              scriptType: bash
15              scriptLocation: inlineScript
16              inlineScript: |
17
18                az login --service-principal --username $(AVDPipelines) --password $(Secret) --tenant $(tenantId)
19
20                if [ $? -ne 0 ]; then
21                  echo "Azure login failed. Please check the service principal credentials."
22                  exit 1
23                fi
24                echo "Azure login successful."
25
26                az account set --subscription ${{ parameters.subscriptionId }}
27
28                locationShortCode=""
29                case "${{ parameters.location }}" in
30                  "westeurope") locationShortCode="weu" ;;
31                  "northeurope") locationShortCode="neu" ;;
32                  *) echo "Unknown location: ${{ parameters.location }}"; exit 1 ;;
33                esac
34
35                vmname="${{ parameters.vmName }}"
36                if [[ -z "$vmname" ]]; then
37                  echo "Error: VM Name cannot be empty."
38                  exit 1
39                fi
40
41                resourceGroupName="rg-${{ parameters.productType }}-${{ parameters.environmentType }}-${locationShortCode}"
42
43                
44                sigImageVersion=$(az sig image-version list \
45                  --resource-group $resourceGroupName \
46                  --gallery-name "gal${{ parameters.productType }}${{ parameters.environmentType }}${locationShortCode}" \
47                  --gallery-image-definition "img-${{ parameters.productType }}-${{ parameters.environmentType }}-${locationShortCode}" \
48                  --query "[].{Version:name}" -o tsv | sort -V | tail -n 1)
49
50                if [[ -z "$sigImageVersion" ]]; then
51                  echo "Error: Could not find the latest image version."
52                  exit 1
53                fi
54
55                username="adm_installuser"
56                nicName="nic-$vmname"
57
58                echo "NIC Name: $nicName"
59
60                
61                subnetId=$(az network vnet subnet show \
62                    --resource-group $resourceGroupName \
63                    --vnet-name "vnet-${{ parameters.productType }}-${{ parameters.environmentType }}-${locationShortCode}" \
64                    --name "snet-${{ parameters.productType }}" \
65                    --query id -o tsv)
66
67                echo "Subnet ID: $subnetId"
68
69                
70                az network nic create \
71                  --resource-group $resourceGroupName \
72                  --name $nicName \
73                  --subnet $subnetId \
74                  --accelerated-networking true
75
76               
77                az vm create \
78                  --resource-group $resourceGroupName \
79                  --name $vmname \
80                  --image "/subscriptions/${{ parameters.subscriptionId }}/resourceGroups/${resourceGroupName}/providers/Microsoft.Compute/galleries/gal${{ parameters.productType }}${{ parameters.environmentType }}${locationShortCode}/images/img-${{ parameters.productType }}-${{ parameters.environmentType }}-${locationShortCode}/versions/$sigImageVersion" \
81                  --admin-username $username \
82                  --size ${{ parameters.vmSize }} \
83                  --authentication-type password \
84                  --admin-password $(ADMIN_PASSWORD) \
85                  --nics $nicName \
86                  --security-type TrustedLaunch \
87                  --public-ip-address "" \
88                  --license-type Windows_Server \
89                  --nsg-rule None
90
91                echo "VM '$vmname' created successfully."

Next we will do the first Windows Update steps.

  • We use Powershell for these steps.
  • First we intall and import the modules to make connection with the Azure environment.
  • We connect to the Azure environment using the credentials stored in the library, if you want to call these values you need to use this format $(yourvaluehere).
  • It installs the necessary NuGet package and PSWindowsUpdate on the installation machine.
  • Get-WindowsUpdate -MicrosoftUpdate is being used to output every available update.
  • Invoke-AzVMRunCommand will be used to execute the command on the installation machine.
 1- stage: check_Windows_Updates
 2    displayName: "Check First Time for Windows Updates"
 3    jobs:
 4    - job: checkWindowsUpdates
 5      displayName: "Check First Time for Windows Updates"
 6      pool:
 7        vmImage: "windows-latest"
 8      steps:
 9        - task: PowerShell@2
10          inputs:
11            azureSubscription: "AVDPipelines"
12            targetType: 'inline'
13            script: |
14                # Ensure required Az modules are installed
15                Write-Output "Installing Az modules..."
16                Install-Module -Name Az.Accounts -Force -Scope CurrentUser -AllowClobber
17                Install-Module -Name Az.Compute -Force -Scope CurrentUser -AllowClobber
18                Import-Module Az.Accounts
19                Import-Module Az.Compute
20                Write-Output "Az modules installed successfully."
21
22                # Connect to Azure Account
23                Write-Output "Connecting to Azure..."
24                $passwd = ConvertTo-SecureString $(Secret) -AsPlainText -Force
25                $pscredential = New-Object System.Management.Automation.PSCredential('$(AVDPipelines)', $passwd)
26                Connect-AzAccount -ServicePrincipal -Credential $pscredential -Tenant $(tenantId)
27                Write-Output "Connected to Azure."
28
29                # Set the Azure subscription context
30                Write-Output "Setting Azure subscription context..."
31                Set-AzContext -Subscription "$(subscriptionId)"
32                Write-Output "Subscription context set."
33
34                # Define VM details
35                $vmname = "${{ parameters.vmName }}"
36                $locationShortCode = if ("${{ parameters.location }}" -eq "westeurope") { "weu" } else { "neu" }
37                $resourceGroupName = "rg-${{ parameters.productType }}-${{ parameters.environmentType }}-$locationShortCode"
38
39                # Run Windows Update Check
40                $script = @'
41                Write-Output "Checking for Windows Updates..."
42                Try {
43                  Install-PackageProvider -Name NuGet -Force -ErrorAction Stop
44                  Install-Module -Name PSWindowsUpdate -Force -ErrorAction Stop
45                  powershell.exe -ExecutionPolicy Bypass -Command "Import-Module PSWindowsUpdate"
46                  
47                  Write-Output "Checking for available updates..."
48                  $updates = Get-WindowsUpdate -MicrosoftUpdate -IgnoreReboot -ErrorAction Stop
49                }
50                Catch {
51                  Write-Output "Error checking updates: $_"
52                  Exit 1
53                }
54
55                if ($updates) {
56                  Write-Output "The following updates are available:"
57                  $updates | ForEach-Object { Write-Output "Title: $($_.Title) | KB: $($_.KBArticle) | Size: $($_.Size) | Severity: $($_.MsrcSeverity)" }
58                } else {
59                  Write-Output "No updates available."
60                }
61                '@
62
63                $updateResult = Invoke-AzVMRunCommand -ResourceGroupName $resourceGroupName -Name $vmname -CommandId "RunPowerShellScript" -ScriptString $script
64
65                Write-Output "Windows update check completed."
66                $updateResult.Value | ForEach-Object {
67                  Write-Output "Code          : $($_.Code)"
68                  Write-Output "Level         : $($_.Level)"
69                  Write-Output "DisplayStatus : $($_.DisplayStatus)"
70                  Write-Output "Message       : $($_.Message)"
71                  Write-Output "Time          : $($_.Time)"
72                }

Let's check the installation step.

  • Install-WindowsUpdate -MicrosoftUpdate -AcceptAll -IgnoreReboot -Confirm:$false is used to install the updates
 1- task: PowerShell@2
 2          displayName: "Install Windows Updates & Reboot VM"
 3          inputs:
 4            azureSubscription: "AVDPipelines"
 5            targetType: 'inline'
 6            script: |
 7                Write-Output "Installing Windows Updates..."
 8                 # Define VM details
 9                $vmname = "${{ parameters.vmName }}"
10                $locationShortCode = if ("${{ parameters.location }}" -eq "westeurope") { "weu" } else { "neu" }
11                $resourceGroupName = "rg-${{ parameters.productType }}-${{ parameters.environmentType }}-$locationShortCode"
12
13                $installScript = @'
14                Write-Output "Installing updates..."
15
16                Try {
17                  Install-PackageProvider -Name NuGet -Force -ErrorAction Stop
18                  Install-Module -Name PSWindowsUpdate -Force -ErrorAction Stop
19                  powershell.exe -ExecutionPolicy Bypass -Command "Import-Module PSWindowsUpdate"
20
21                  Write-Output "Starting update installation..."
22                  Install-WindowsUpdate -MicrosoftUpdate -AcceptAll -IgnoreReboot -Confirm:$false
23                  Write-Output "Updates installed successfully."
24                }
25                Catch {
26                  Write-Output "Error during update installation: $_"
27                  Exit 1
28                }
29
30                Write-Output "Rebooting VM..."
31                restart-computer -Force
32                '@
33
34                Invoke-AzVMRunCommand -ResourceGroupName $resourceGroupName -Name $vmname -CommandId "RunPowerShellScript" -ScriptString $installScript
35
36                Write-Output "VM reboot initiated."

This step will repeat itself two times, in the last step it will give a history of the installed updates.

The reason why I do it in multiple steps is because of the 60 minute running time limit per job for a Microsoft hosted agent else you can also create a loop to do all those windows updates steps in one job, or use your own self hosted devops agents.

  • Get-WUHistory is used to show the installed updates.
 1- stage: check_Windows_Updates_last_time
 2    displayName: "Check Windows Updates last time and clean up"
 3    jobs:
 4    - job: checkWindowsUpdatesLastTime
 5      displayName: "Check Windows Updates last time and clean up"
 6      pool:
 7        vmImage: "windows-latest"
 8      steps:
 9        - task: PowerShell@2
10          inputs:
11            azureSubscription: "AVDPipelines"
12            targetType: 'inline'
13            script: |
14                # Ensure required Az modules are installed
15                Write-Output "Installing Az modules..."
16                Install-Module -Name Az.Accounts -Force -Scope CurrentUser -AllowClobber
17                Install-Module -Name Az.Compute -Force -Scope CurrentUser -AllowClobber
18                Import-Module Az.Accounts
19                Import-Module Az.Compute
20                Write-Output "Az modules installed successfully."
21
22                # Connect to Azure Account
23                Write-Output "Connecting to Azure..."
24                $passwd = ConvertTo-SecureString $(Secret) -AsPlainText -Force
25                $pscredential = New-Object System.Management.Automation.PSCredential('$(AVDPipelines)', $passwd)
26                Connect-AzAccount -ServicePrincipal -Credential $pscredential -Tenant $(tenantId)
27                Write-Output "Connected to Azure."
28
29                # Set the Azure subscription context
30                Write-Output "Setting Azure subscription context..."
31                Set-AzContext -Subscription "$(subscriptionId)"
32                Write-Output "Subscription context set."
33
34                # Define VM details
35                $vmname = "${{ parameters.vmName }}"
36                $locationShortCode = if ("${{ parameters.location }}" -eq "westeurope") { "weu" } else { "neu" }
37                $resourceGroupName = "rg-${{ parameters.productType }}-${{ parameters.environmentType }}-$locationShortCode"
38
39                # Run Windows Update Check
40                $script = @'
41                Write-Output "Checking for Windows Updates..."
42                Try {
43                    Install-PackageProvider -Name NuGet -Force -ErrorAction Stop
44                    Install-Module -Name PSWindowsUpdate -Force -ErrorAction Stop
45                    powershell.exe -ExecutionPolicy Bypass -Command "Import-Module PSWindowsUpdate"
46
47                    Write-Output "Checking for available updates..."
48                    $updates = Get-WindowsUpdate -MicrosoftUpdate -IgnoreReboot -ErrorAction Stop
49                }
50                Catch {
51                    Write-Output "Error checking updates: $_"
52                    Exit 1
53                }
54
55                if ($updates) {
56                    Write-Output "The following updates are available:"
57                    $updates | ForEach-Object { Write-Output "Title: $($_.Title) | KB: $($_.KBArticle) | Size: $($_.Size) | Severity: $($_.MsrcSeverity)" }
58                } else {
59                    Write-Output "No updates available."
60                }
61                # Get Windows Update history
62                Write-Output "Fetching Windows Update history..."
63                $updateHistory = Get-WUHistory
64                if ($updateHistory) {
65                    $updateHistory | ForEach-Object { Write-Output "Date: $($_.Date) | Title: $($_.Title) | Status: $($_.ResultCode)" }
66                } else {
67                    Write-Output "No update history found."
68                }
69                '@
70
71                # Execute script on Azure VM
72                $updateResult = Invoke-AzVMRunCommand -ResourceGroupName $resourceGroupName -Name $vmname -CommandId "RunPowerShellScript" -ScriptString $script
73
74                # Output results
75                Write-Output "Windows update check completed."
76                $updateResult.Value | ForEach-Object {
77                    Write-Output "Code          : $($_.Code)"
78                    Write-Output "Level         : $($_.Level)"
79                    Write-Output "DisplayStatus : $($_.DisplayStatus)"
80                    Write-Output "Message       : $($_.Message)"
81                    Write-Output "Time          : $($_.Time)"
82                }

Windows Updates is done, now let's create a snapshot of the disk.

  • For the naming it will create a timestamp. timestamp=$(date -u +"%Y%m%dT%H%M%SZ") snapshotName="snapshot-bs-${timestamp}-${{ parameters.productType }}-${{ parameters.environmentType }}-${locationShortCode}"
 1- stage: Take_snapshot_of_the_Disk
 2    displayName: "Create a snapshot of the current disk"
 3    jobs:
 4      - job: Take_snapshot_of_the_Disk
 5        displayName: "Create a snapshot of the current disk"
 6        pool:
 7          vmImage: "ubuntu-latest"
 8        steps:
 9          - task: Bash@3
10            displayName: Azure CLI Login
11            inputs:
12              targetType: "inline"
13              script: |
14                # Install Azure CLI
15                sudo apt-get update
16                sudo apt-get install -y libicu-dev curl
17                curl -sL https://aka.ms/InstallAzureCLIDeb | sudo bash
18
19                # Login to Azure
20                az login --service-principal --username $(AVDPipelines) --password $(Secret) --tenant $(tenantId)
21
22                if [ $? -ne 0 ]; then
23                  echo "Azure login failed. Please check the service principal credentials."
24                  exit 1
25                fi
26                echo "Azure login successful."
27
28                # Confirm subscription
29                az account show
30                
31                # Define parameters
32                locationShortCode=""
33                case "${{ parameters.location }}" in
34                  "westeurope") locationShortCode="weu" ;;
35                  "northeurope") locationShortCode="neu" ;;
36                  *) echo "Unknown location: ${{ parameters.location }}"; exit 1 ;;
37                esac
38                
39                # Define variables
40                resourceGroupName="rg-${{ parameters.productType }}-${{ parameters.environmentType }}-${locationShortCode}"
41                location="${{ parameters.location }}"
42                vmName="${{ parameters.vmName }}" 
43
44                # Fetch the disk name dynamically from the VM
45                diskName=$(az vm show \
46                  --name "$vmName" \
47                  --resource-group "$resourceGroupName" \
48                  --query "storageProfile.osDisk.name" -o tsv)
49
50                if [ -z "$diskName" ]; then
51                  echo "Error: Unable to retrieve the OS disk name for VM $vmName in resource group $resourceGroupName."
52                  exit 1
53                fi
54
55                echo "Found OS Disk: $diskName"
56
57                # Define snapshot name with timestamp
58                timestamp=$(date -u +"%Y%m%dT%H%M%SZ") # Get current UTC time
59                snapshotName="snapshot-bs-${timestamp}-${{ parameters.productType }}-${{ parameters.environmentType }}-${locationShortCode}"
60
61                # Create a snapshot
62                az snapshot create \
63                  --name "$snapshotName" \
64                  --resource-group "$resourceGroupName" \
65                  --location "$location" \
66                  --source "$diskName" \
67                  --output json
68
69                if [ $? -ne 0 ]; then
70                  echo "Failed to create snapshot $snapshotName."
71                  exit 1
72                fi
73
74                echo "Snapshot $snapshotName created successfully in resource group $resourceGroupName."

It will sysprep the machine.

 1- stage: Sysprep_the_VM
 2    displayName: "Sysprep the machine"
 3    jobs:
 4      - job: Sysprep_the_VM
 5        displayName: "Sysprep the machine"
 6        pool:
 7          vmImage: "ubuntu-latest"
 8        steps:
 9          - task: Bash@3
10            displayName: Azure CLI Login
11            inputs:
12              targetType: "inline"
13              script: |
14                # Install Azure CLI
15                sudo apt-get update
16                sudo apt-get install -y libicu-dev curl
17                curl -sL https://aka.ms/InstallAzureCLIDeb | sudo bash
18
19                # Login to Azure
20                az login --service-principal --username $(AVDPipelines) --password $(Secret) --tenant $(tenantId)
21
22                if [ $? -ne 0 ]; then
23                  echo "Azure login failed. Please check the service principal credentials."
24                  exit 1
25                fi
26                echo "Azure login successful."
27
28                # Confirm subscription
29                az account show
30                
31                # Define parameters
32                locationShortCode=""
33                case "${{ parameters.location }}" in
34                  "westeurope") locationShortCode="weu" ;;
35                  "northeurope") locationShortCode="neu" ;;
36                  *) echo "Unknown location: ${{ parameters.location }}"; exit 1 ;;
37                esac
38                
39                # Define variables
40                resourceGroupName="rg-${{ parameters.productType }}-${{ parameters.environmentType }}-${locationShortCode}"
41                vmName="${{ parameters.vmName }}" 
42                version="${{ parameters.imageVersion }}"
43
44                # Sysprep the machine
45                echo "Sysprepping the machine $vmName in resource group $resourceGroupName."
46                az vm run-command invoke \
47                  --command-id RunPowerShellScript \
48                  --name "$vmName" \
49                  --resource-group "$resourceGroupName" \
50                  --scripts "Start-Process -FilePath 'C:\\Windows\\System32\\Sysprep\\Sysprep.exe' -ArgumentList '/generalize /shutdown /oobe' -Wait -NoNewWindow"
51
52                if [ $? -ne 0 ]; then
53                  echo "Failed to sysprep the VM $vmName."
54                  exit 1
55                fi
56
57                echo "Sysprep operation completed for VM $vmName."

And after the sysprep it will capture it and store it to the Azure Compute Gallery.

  • It will give the name based on the input parameter version="${{ parameters.imageVersion }}".
  1- stage: Capture_Image_Save_It_To_Gallery
  2    displayName: "Capture the image and save it to the gallery"
  3    jobs:
  4      - job: Capture_Image_Save_It_To_Gallery
  5        displayName: "Capture the image and save it to the gallery"
  6        pool:
  7          vmImage: "ubuntu-latest"
  8        steps:
  9          - task: Bash@3
 10            displayName: Azure CLI Login
 11            inputs:
 12              targetType: "inline"
 13              script: |
 14                # Install Azure CLI
 15                sudo apt-get update
 16                sudo apt-get install -y libicu-dev curl
 17                curl -sL https://aka.ms/InstallAzureCLIDeb | sudo bash
 18
 19                # Login to Azure
 20                az login --service-principal --username $(AVDPipelines) --password $(Secret) --tenant $(tenantId)
 21
 22                if [ $? -ne 0 ]; then
 23                  echo "Azure login failed. Please check the service principal credentials."
 24                  exit 1
 25                fi
 26                echo "Azure login successful."
 27
 28                # Confirm subscription
 29                az account show
 30                
 31                # Define parameters
 32                locationShortCode=""
 33                case "${{ parameters.location }}" in
 34                  "westeurope") locationShortCode="weu" ;;
 35                  "northeurope") locationShortCode="neu" ;;
 36                  *) echo "Unknown location: ${{ parameters.location }}"; exit 1 ;;
 37                esac
 38                
 39                # Define variables
 40                resourceGroupName="rg-${{ parameters.productType }}-${{ parameters.environmentType }}-${locationShortCode}"
 41                location="${{ parameters.location }}"
 42                vmName="${{ parameters.vmName }}"
 43                imageGalleryName="gal${{ parameters.productType }}${{ parameters.environmentType }}${locationShortCode}"  # Name of the image gallery
 44
 45                # Display imageGalleryName for confirmation
 46                echo "Image Gallery Name: $imageGalleryName"
 47
 48                # Generate image name dynamically with timestamp
 49                imageName="img-${{ parameters.productType }}-${{ parameters.environmentType }}-${locationShortCode}"
 50                echo "Image Name: $imageName"
 51
 52                version="${{ parameters.imageVersion }}"
 53                echo "Image Version: $version"
 54
 55                # Step 1: Deallocate the VM
 56                echo "Deallocating the VM $vmName in resource group $resourceGroupName..."
 57                az vm deallocate --resource-group "$resourceGroupName" --name "$vmName"
 58
 59                if [ $? -ne 0 ]; then
 60                  echo "Failed to deallocate VM $vmName."
 61                  exit 1
 62                fi
 63                echo "VM $vmName deallocated successfully."
 64
 65                # Step 2: Generalize the VM
 66                echo "Generalizing the VM $vmName..."
 67                az vm generalize --resource-group "$resourceGroupName" --name "$vmName"
 68
 69                if [ $? -ne 0 ]; then
 70                  echo "Failed to generalize VM $vmName."
 71                  exit 1
 72                fi
 73                echo "VM $vmName generalized successfully."
 74
 75                # Step 3: Get the VM resource ID
 76                echo "Retrieving the resource ID for the VM..."
 77                vmResourceId=$(az vm show \
 78                  --resource-group "$resourceGroupName" \
 79                  --name "$vmName" \
 80                  --query "id" \
 81                  --output tsv)
 82
 83                if [ $? -ne 0 ]; then
 84                  echo "Failed to retrieve VM resource ID."
 85                  exit 1
 86                fi
 87
 88                echo "VM Resource ID: $vmResourceId"
 89
 90                # Step 4: Create the image version in the image gallery
 91                echo "Creating image version in the gallery..."
 92                az sig image-version create \
 93                  --resource-group "$resourceGroupName" \
 94                  --gallery-name "$imageGalleryName" \
 95                  --gallery-image-definition "$imageName" \
 96                  --gallery-image-version "$version" \
 97                  --virtual-machine "$vmResourceId" \
 98                                   
 99
100                if [ $? -ne 0 ]; then
101                  echo "Failed to create image version in the gallery."
102                  exit 1
103                fi
104
105                echo "Image version created successfully in gallery $imageGalleryName."

As an final part, we are going to clean up the virtual machine in the environment .

 1- stage: Delete_Installation_Machine
 2    displayName: "Delete the installation machine"
 3    jobs:
 4      - job: Delete_Installation_Machine
 5        displayName: "Delete the installation machine"
 6        pool:
 7          vmImage: "ubuntu-latest"
 8        steps:
 9          - task: Bash@3
10            displayName: Azure CLI Login and Resource Deletion
11            inputs:
12              targetType: "inline"
13              script: |
14                # Install Azure CLI
15                sudo apt-get update
16                sudo apt-get install -y libicu-dev curl
17                curl -sL https://aka.ms/InstallAzureCLIDeb | sudo bash
18
19                # Login to Azure
20                az login --service-principal --username $(AVDPipelines) --password $(Secret) --tenant $(tenantId)
21
22                if [ $? -ne 0 ]; then
23                  echo "Azure login failed. Please check the service principal credentials."
24                  exit 1
25                fi
26                echo "Azure login successful."
27
28                # Confirm subscription
29                az account show
30
31                # Define parameters
32                locationShortCode=""
33                case "${{ parameters.location }}" in
34                  "westeurope") locationShortCode="weu" ;;
35                  "northeurope") locationShortCode="neu" ;;
36                  *) echo "Unknown location: ${{ parameters.location }}"; exit 1 ;;
37                esac
38
39                # Define variables
40                resourceGroupName="rg-${{ parameters.productType }}-${{ parameters.environmentType }}-${locationShortCode}"
41                vmName="${{ parameters.vmName }}"
42
43                echo "Deleting Virtual Machine: $vmName from Resource Group: $resourceGroupName"
44
45                # Get the attached resources
46                osDiskId=$(az vm show --name $vmName --resource-group $resourceGroupName --query "storageProfile.osDisk.managedDisk.id" -o tsv)
47                dataDiskIds=$(az vm show --name $vmName --resource-group $resourceGroupName --query "storageProfile.dataDisks[].managedDisk.id" -o tsv)
48                nicIds=$(az vm show --name $vmName --resource-group $resourceGroupName --query "networkProfile.networkInterfaces[].id" -o tsv)
49
50                # Delete the Virtual Machine
51                az vm delete --name $vmName --resource-group $resourceGroupName --yes --no-wait
52                echo "Virtual Machine $vmName deleted."
53
54                # Delete the OS Disk
55                if [ -n "$osDiskId" ]; then
56                  echo "Deleting OS Disk: $osDiskId"
57                  az disk delete --ids $osDiskId --yes --no-wait
58                fi
59
60                # Delete Data Disks
61                if [ -n "$dataDiskIds" ]; then
62                  echo "Deleting Data Disks: $dataDiskIds"
63                  for diskId in $dataDiskIds; do
64                    az disk delete --ids $diskId --yes --no-wait
65                  done
66                fi
67
68                # Delete NICs
69                if [ -n "$nicIds" ]; then
70                  echo "Deleting Network Interfaces: $nicIds"
71                  for nicId in $nicIds; do
72                    az network nic delete --ids $nicId
73                  done
74                fi
75
76                echo "Cleanup completed for VM $vmName."

How we setup the pipeline and where to find the complete file?

In an early blog I already describe how to set-up a DevOps pipeline, you can find the document here.

You can find the whole yaml file in the Bicep Automation repository.


Conclusion.

Ultimately, using this technique we have an updated image that is stored in the Azure Compute Gallery. I use this method quite often now and it helps me to get an up-to-date image in an automated way. Of course you still have to spin up new machines using the new version, but with the preview function session host configuration update this can be done very easily.

I hope you find this blog about updating an image useful. If you have any comments and/or questions, please let me know via one of my social media channels.