How to deploy this example in your own Azure account(s). This project uses random suffixes used to meet globally unique names requirements for Storage Accounts and Key Vaults. For details, see example output below.
This code is NOT meant to be used for production. While great efforts were taken for code quality and best practices, certain decisions were made for convenience. For example, this demo uses resource groups to separate environments. In practice, however, Azure subscriptions are a better choice per Microsoft's Cloud Adoption Framework (CAF).
- Security Principals - User vs Service (CI/CD)
- Azure Subscription
- Azure Active Directory Tenant
- Azure DevOps Organization
- Login to Azure
- Configure Environment
- Initialize Terraform
3a) From local computer (recommended)
3b) Configure Azure Backend for Terraform (optional) - Create Deployment Plan
- Deploy Infrastructure
Make sure you read the full document because the pre-configuration of permissions is more complex. But once you've done it properly, deploying this project is simple:
export AZDO_ORG_SERVICE_URL="https://dev.azure.com/<your-demo-org-name>"
export AZDO_PERSONAL_ACCESS_TOKEN="…"
terraform init -backend=false
terraform plan -out demo.tfplan
terraform apply demo.tfplan
If you have a Visual Studio subscription, use that for this demo so that the elevated service principals required have NO access to your actual Azure environments.
-
User Principal 🙋♀️
If you want to deploy locally, you do NOT need a service principal. You still need owner permissions, but a simpleaz login
is enough to deploy the resources. Further Instructions → -
Headless Service Principal 🤖
If you are using this project sample for its Azure Pipelines, you will also need to initiate an Azure Remote Backend. Further Instructions →
For details, please read Azure Security Principals official documentation.
- User or Service Principal
with elevatedOwner
permissions, required to create Key Vault Access Policies
⚠️ Please consider carefully which Azure Active Directory (AAD) tenant you will use and read the Terraform documentation carefully about configuring the required elevated privileges. If possible, use a non-production AAD tenant.
-
Azure AD Tenant (non-production)
If you have a non-production tenant, use it because the following service principal is very privileged. -
User or Service Principal
with elevated privileges so that it can manage Azure Active Directory. Follow these steps per Terraform documentation to properly configure your Service Principal. -
Required Permissions on AD
To create AD groups and principals, there are two main ways to assign permissions to do that:-
via Directory Role
Azure AD Provider on Terraform recommends assigning theGlobal Administrator
role to the security principal (user or service) in order to manage users, groups and applications. -
via API Permissions
If instead of using a role, you can also add the grant access to the following permissions for Microsoft Graph API (AAD Graph API deprecated as of provider v2.0.0):
Application.ReadWrite.All
Directory.ReadWrite.All
See Terraform documentation on configuring service principals for full instructions.
-
-
Create "superadmins" Azure Active Directory Group (optional)
This is not applicable if you are just testing from your local computer. You will be made administrator of the Key Vaults generated by Infrastructure as Code (IaC) in this repository.If you are using CI/CD, the service principal will be made administrator. To give access to both, we can set the
superadmins_aad_object_id
variable in Terraform to include both you and the service principal. Then set the value inlocal.auto.tfvars.example
and remove the.example
extension. See file for further details.
-
DevOps Organization
Create a new organization just for this demo. -
Personal Access Token (PAT)
Create a PAT and give it full access to the newly created Azure DevOps organization.
Make sure you are authenticated to Azure.
- If you are deploying from a local machine, use
az login
- If you are deploying from a headless agent, authenticate using a service principal and client secret as described in the Terrform documentation.
The Azure DevOps Provider in Terraform expects the following environment variables to be set:
export AZDO_ORG_SERVICE_URL="https://dev.azure.com/<your-demo-org-name>"
export AZDO_PERSONAL_ACCESS_TOKEN="…"
To make this easier, create a .env
file, using the included .env.sample
as a template and fill in your values.
Then load the variables into your shell:
source ./.env
Please note that .env
file name is a common convention and some frameworks will automatically load this configuration file. However, it is meant to be used in a local development only and should never be checked into git.
First log into Azure and select the subscription you want to deploy to
az login
az account set --subscription <SUBSCRIPTION_ID>
Then override the terraform backend to be local (because repo default is remote for CI/CD) by removing .sample
from the file extension
mv _override.tf.sample _override.tf
Finally initialize Terraform
terraform init
Then continue to Create Deployment Plan →
If you're configuring for a headless CI/CD agent or just want to try using remote backend for terraform state, please follow these additional steps. It is preferred to use the Azure CLI. See ./backends for full commands.
Create an backend.hcl
file, using backend.hcl.sample
as a template, filling in the placeholders iwth your values. hcl stands for HashiCorp Language.
storage_account_name="azurestorageaccountname"
container_name="storagecontainername"
key="project.tfstate"
sas_token="?sv=2019-12-12…"
Finally run init
with our new backend config.
terraform init -backend-config=./backend.hcl
See what resources Terraform will create and save plan to a file
terraform plan -out demo.tfplan
If you are satisfied with the plan (it does not unexpectedly change or destroy resources), then deploy it with
terraform apply demo.tfplan
Please see azure-pipelines/README.md for additional technical implementation details.
If it works, you will see something like the output below, which you can also find in the Pipeline Build Results for this project.
aad_groups = [
"demo-fruits-admins-8zfj",
"demo-fruits-devs-8zfj",
"demo-fruits-team-8zfj",
"demo-infra-admins-8zfj",
"demo-infra-dev-8zfj",
"demo-infra-team-8zfj",
"demo-veggies-admins-8zfj",
"demo-veggies-devs-8zfj",
"demo-veggies-team-8zfj",
]
arm_environments = [
{
"key_vault" = "fruits-dev-8zfj-kv"
"resource_group" = "fruits-dev-8zfj-rg"
"storage_account" = "fruitsdev8zfj"
},
{
"key_vault" = "fruits-prod-8zfj-kv"
"resource_group" = "fruits-prod-8zfj-rg"
"storage_account" = "fruitsprod8zfj"
},
{
"key_vault" = "infra-shared-8zfj-kv"
"resource_group" = "infra-shared-8zfj-rg"
"storage_account" = "infrashared8zfj"
},
{
"key_vault" = "veggies-dev-8zfj-kv"
"resource_group" = "veggies-dev-8zfj-rg"
"storage_account" = "veggiesdev8zfj"
},
{
"key_vault" = "veggies-prod-8zfj-kv"
"resource_group" = "veggies-prod-8zfj-rg"
"storage_account" = "veggiesprod8zfj"
},
]
azure_devops_projects = [
{
"description" = "Demo using AAD groups"
"features" = tomap({
"artifacts" = "disabled"
"boards" = "disabled"
"pipelines" = "enabled"
"repositories" = "enabled"
"testplans" = "disabled"
})
"name" = "project-fruits"
"visibility" = "private"
},
{
"description" = "Central IT managed stuff"
"features" = tomap({
"artifacts" = "disabled"
"boards" = "disabled"
"pipelines" = "enabled"
"repositories" = "enabled"
"testplans" = "disabled"
})
"name" = "central-it"
"visibility" = "private"
},
{
"description" = "Demo using AAD groups"
"features" = tomap({
"artifacts" = "disabled"
"boards" = "disabled"
"pipelines" = "enabled"
"repositories" = "enabled"
"testplans" = "disabled"
})
"name" = "project-veggies"
"visibility" = "private"
},
]
azure_devops_service_connections = [
{
"fruits_dev" = {
"connection_name" = "fruits-dev-8zfj-rg-connection"
"scope" = tostring(null)
}
"fruits_prod" = {
"connection_name" = "fruits-prod-8zfj-rg-connection"
"scope" = tostring(null)
}
"infra_shared" = {
"connection_name" = "infra-shared-8zfj-rg-connection"
"scope" = tostring(null)
}
"veggies_dev" = {
"connection_name" = "veggies-dev-8zfj-rg-connection"
"scope" = tostring(null)
}
"veggies_prod" = {
"connection_name" = "veggies-prod-8zfj-rg-connection"
"scope" = tostring(null)
}
},
]
service_principals = [
"fruits-dev-8zfj-ci-sp",
"fruits-prod-8zfj-ci-sp",
"infra-shared-8zfj-ci-sp",
"veggies-dev-8zfj-ci-sp",
"veggies-prod-8zfj-ci-sp",
]