Deploying Azure Policy Definitions via Azure DevOps (Part 1)
Introduction
Recently I needed to deploy a large number of Azure policy and initiative definitions at customer’s environments using Azure DevOps. These definitions needed to be deployed to different environments (different Management Group hierarchies in different Azure AD Tenants).
I faced some difficulties when working on this solution, due to the following limitations:
1. Currently templates do not support Management Groups
So I can’t use ARM templates in this case. But, I still needed to develop a solution no matter where should the definitions being deployed (either to a management group or a subscription).
2. Limitations in Azure PowerShell cmdlet New-AzPolicyDefinition and New-AzPolicySetDefinition
Definitions files are repetitive and the input files required by these commands do not contain all the required information. (more on this later)
3. Not being able to use ARM template functions such as resourceId(), reference() in policy and initiative definitions.
This makes my task extremely difficult when defining initiative definitions that contain custom definitions that are not yet deployed.
The end solution I developed was entirely based on Azure DevOps. I used:
- Azure Repo to host definition files and deployment scripts
- Azure Artifacts to host PowerShell modules (as Nuget package) used by the pipelines<
- Azure Pipeline to test and deploy the definitions to multiple management groups in multiple tenants
I won’t be able to share my experience in one blog post. I’ll cover this topic in a 3-part blog series
Part 1: Custom deployment scripts for policy and initiative definitions
Part 2: Pester-test policy and initiative definitions in the build pipeline
Part 3: Configuring build (CI) and release (CD) pipelines in Azure DevOps
NOTE: all the definitions and deployment scripts I used in this blog post series are located at:
https://github.com/tyconsulting/azurepolicy
This is the part 1 of the blog series.
Bulk Deploying Policy and Initiative Definition using PowerShell Scripts
Deploying Policy Definitions using deploy-PolicyDef.ps1
As I mentioned earlier, ARM templates is out of the picture, because they can’t be deployed to management groups at this stage. Also, I’m deploying 100+ custom policy definitions, it’s hard to put so many definitions in one or few templates. I prefer having them in individual files so it’s easier to read, develop, update and re-use.
If you take a look at the official Azure Policy GitHub repo, it’s contribution guide (https://github.com/Azure/azure-policy/tree/master/1-contribution-guide) mandates that you need to provide 3 files for each policy:
- azurepolicy.rules.json
- azurepolicy.parameters.json
- azurepolicy.json
If you look at these files closely, you will find:
- azurepolicy.rules.json contains policy rule
- azurepolicy.parameters.json contains parameter definitions (if the policy requires input parameters)
- azurepolicy.json contains
- policy rule
- parameter definitions
- name (it will become a part of of resource Id once deployed)
- display name
- description
- metadata (i.e. policy category, such as Compute, Storage, Monitoring, etc.)
When using the New-AzPolicyDefinition cmdlet to deploy a definition, the cmdlet input parameters expect:
- the policy rule definition (azurepolicy.rules.json)
- the policy parameter definitions (azurepolicy.parameter.json)
- name
- display name
- description
- metadata
So, if the azurepolicy.json already contains everything it needs to create a new definition, why can’t I just pass this file to the cmdlet instead of a mixture of input files and other string inputs? The answer is I can’t, the cmdlet does not support it!
If I adopt Microsoft’s practice according to the official repo contribution guide, I will need to duplicate the contents of some files, and the file that actually contains everything is not actually been used by the cmdlet!? if I use the New-AzPolicyDefinition cmdlet in my pipeline, not only I need to repeat it over 100 times, but also I need to store name, display name, description and metadata outside of the actual policy definition artifacts (i.e. maybe a variable group in Azure DevOps project). This can become extremely time consuming to configure and maintain the pipeline, because the artifacts doesn’t not contain all the information required.
As you can imagine, I am not happy with how the New-AzPolicyDefinition is implemented. For those who know me well, you probably have heard me telling people over and over again – Just invoke ARM REST API directly from the PowerShell scripts instead of using the official Azure PowerShell modules.
This is also too complicated. I have done it in the past, it’s hard to maintain, especially you want a group of people to look after the pipeline after initial setup.
[17/11/2019]: Stefan Stranger has pointed out it is possible to retrieve the AAD token from context token cache using Azure PowerShell task in Azure Pipelines.
This led me to create a custom deployment script (deploy-policyDef.ps1) that leverages New-AzPolicyDefinition, but the script uses azurepolicy.json since it contains all the information that’s needed for creating the policy definition. The script also supports:
- Supports both management groups and subscriptions
- Deploy multiple policy definitions if multiple file paths are passed in
- Deploy all policy definitions in a folder
- Deploy all policy definitions in a folder and its sub-folders if –recurse switch is used
- If you place policy and initiative definitions all in one folder, it can detect which ones are policy definitions, which ones are initiative definitions
- Supports running it interactively (prompt you to sign in to Azure if not already signed in, or ask you if you want to use the current login context if you’ve already signed in using Connect-AzAccount cmdlet)
- Supports running it in silent mode, which requires an existing login context and no interactive prompt. This is required so I can use the script in a pipeline
- Supports –verbose switch. Many messages have been configured to go to the verbose stream output, to help you troubleshoot issues.
For example:
Deploy a single policy definition to a subscription (interactive mode)
./deploy-policyDef.ps1 -definitionFile C:\Temp\azurepolicy.json -subscriptionId cd45c044-18c4-4abe-a908-1e0b79f45003
Deploy all policy definitions in a folder and its sub folders to a management group (silent mode, i.e. in a CI/CD pipeline):
./deploy-policyDef.ps1 -FolderPath C:\Temp -recurse -managementGroupName myMG -silent
In my repo, I structured the policy definitions in the file system as shown below:
├───policy-definition-root-folder
│ ├───target-management-group-name
│ │ ├───policy-1-folder
│ │ └───policy-2-folder
In the pipeline, all I needed is to specify the path to the target-management-group-name as the folder path, and use –recurse switch. A one-liner deploys over 100 policy definitions within couple of minutes!
Deploying Initiative Definitions using deploy-policySetDef.ps1
When working with initiative (policy set) definitions, I’m facing the exact same challenge – the samples in the official repo contribution all contain the following 3 files:
- azurepolicyset.definitions.json – contains the list of policies included in the initiative
- azurepolicyset.parameters.json – input parameters definitions
- azurepolicyset.json – contains everything
Following the same logic, the azurepolicyset.json contains everything, well almost everything! if you look at the sample initiatives, you’ll find some of the azurepolicyset.json files contain the “name” attribute, some don’t. A bit inconsistent there. The “name” is absolutely required when deploy the initiative definition. Similar to policy definitions, the New-PolicySetDefinition cmdlet from Azure PowerShell modules takes the definition and parameters from separate inputs, with some additional string inputs such as name, metadata, etc.
I wrote a similar script to deploy initiative definitions, called deploy-policySetDef.ps1. This script allows you to deploy a single initiative definition using the azurepolicyset.json file.
I also needed to address the issue that I need to be able to reference custom policy definitions in initiative definitions – the resource Ids for for these custom policy definitions are not static (depending on the target management groups or subscriptions of the deployment) and may even be unknown at the time when the initiatives are defined. The workaround I came up with is replacing a section of resource Ids for policy definitions with a string in the initiative definition, and replace the string at the time of deployment
For example, Replace
"policyDefinitionid": "/providers/Microsoft.Management/managementgroups/myMG/providers/Microsoft.Authorization/policyDefinitions/restrict-public-storageAccounts-policyDef"
with
"policyDefinitionid": "{policyLocationResourceId1}/providers/Microsoft.Authorization/policyDefinitions/restrict-public-storageAccounts-policyDef"
The {policyLocationResourceId1} represents the actual resource Id for the management group, in this case /providers/Microsoft.Management/managementgroups/myMG
The deploy-policyDef.ps1 is designed to that policy location as a hash table input parameter. i.e.
./deploy-policyDef.ps1 -definitionFile C:\Temp\azurepolicyset.json -managementGroupName myMG -PolicyLocations @{policyLocationResourceId1 = '/providers/Microsoft.Management/managementGroups/myMG'}
The -PolicyLocationId
parameter expects a hashtable input, you can list one or more replaceable strings, such as: @{policyLocationResourceId1 = ‘/providers/Microsoft.Management/managementGroups/myMG’; policyLocationResourceId2 = ‘/subscriptions/4fa56034-7d12-4ab9-8d9c-1eae722376e9’}
When -PolicyLocations
parameter is used, the deploy-policySetDefinition.ps1 script searches strings wrapped in “{}” that match each key from the input hashtable and replaces with the value associated to the key from the hashtable.
This method enables us to define the initiative definition once, and deploy it to multiple environments in a CI/CD pipeline. It also enables us to deploy the policy definitions that are members of the initiative in the same pipeline by using -PolicyLocation parameter to variableize the deployment destination.
This concludes the 1st part of this blog series. In the next instalment, I will walk through how I configured Pester tests for policy and initiative definitions.
Leave a comment