Infrastructure as code provides many benefits, but sometimes at the expense of local developers who cannot create infrastructure from their own machines. This is in fact a huge and hidden problem, that can massively lower the productivity of software engineering teams.
This repository is a contribution on the subject of continuous deployment of infrastructure without sacrificing the developer experience. The core goals are:
- Anyone can create and destroy infrastructure from their local machine.
- A complete environment can be created from a minimum of configuration inputs.
- Infrastructure can be continously deployed.
- Infrastructure as code implies immutable and idempotent deployments.
In this project:
- Everything is CLI, everything works in bash, everything runs in containers, and every script works locally.
- Terraform's HCL provides declarative infrastructure definitions.
- Azure Pipelines continuously integrate and deploy infrastructure using custom and containerised build and release agents running as Azure Container Instances with managed identities.
In this example, we will deploy an Azure Web App with Terraform and Azure Pipelines.
- Terraform: Infrastructure definition and deployments
- Azure DevOps: Continuous Integration and Deployment
- Azure Web Apps Application host
All terraform commands can be run via bash scripts. Scripts ending in .local.sh
are intended to be run on a developer's machine. In a CI/CD pipeline, .local.sh
scripts should not be used.
docker-compose run cli
In the root directory, run the above for a local shell with terraform, and Azure CLI installed. You can see the Dockerfile for this shell at docker/Dockerfile
az login
./src/init.local.sh
You can edit src/init.local.sh
to suit your environment.
You will also need some secrets, which you can place in the .secrets/
directory.
init.local.sh
is a wrapper around init.sh
, which initialises terraform with remote state in an Azure Storage account.
./src/plan.local.sh
This script will use terraform plan. A dev.local.tfplan
file will be created, which is referenced in tfplan.local.sh
./src/tfapply.local.sh
You can edit ./src/tfapply.local.sh
as you like. Any changes made to your terraform configuration will be applied to your Azure subscription, but not to production!
./src/plan-destroy.local.sh
./src/tfapply.local.sh
We use custom build agents running on Azure Container Instances. These agents have all the default Azure DevOps tools installed, plus terraform. The agents also are created with a Managed Identity that is given contributor access to the subscription. This is how terraform gains access to the Azure Resource Manager.
Firstly, create a new agent pool in Azure DevOps called terraform-aci.
Then, get a Personal Access Token from Azure DevOps, and copy into into .secrets/vststoken.secret
To create an agent, edit and run the following script:
./agents/create-agent.sh -t "$(cat .secrets/vststoken.secret)" -g tf-agents -n vs-agent -a <YOUR_AZUREDEVOPS_ACCOUT_NAME>
Note, this can take several minutes as the image is large.
You'll need to set up a Service Connection to Azure. My connection is simply called rian
but you can call yours something different, and rename it in azure-pipelines.yml
Continuous integration means changes in source code are continuously tested then merged into the main product. This project uses Azure Pipelines yaml build definitions. You can see the build definition in the root of this project. In the build:
- bash scripts are checked for validity
- terraform files are checked for correct formatting
- a terraform plan is generated (but not used)
We're going to use Azure Pipelines to continously deploy our infrastructure.
Since all our tools are bash scripts, we are just going to run three Azure CLI tasks.
- Set the Script Path to
$(System.DefaultWorkingDirectory)/_infra/prod-drop/src/init.sh
- Set the arguments to
-g $(tfstate.resourcegroup) -a $(tfstate.storageaccountname) -e $(environment)
- Set the Script Path to
$(System.DefaultWorkingDirectory)/_infra/prod-drop/src/plan.sh
- Set the arguments to
-e $(environment) -g $(tfstate.resourcegroup) -a $(tfstate.storageaccountname) -p "$(System.DefaultWorkingDirectory)/terraform.tfplan" -f "$(System.DefaultWorkingDirectory)/tfplan-now.summary.txt"
- Set the Script Path to
$(System.DefaultWorkingDirectory)/_infra/prod-drop/src/tfapply.sh
- Set the arguments to
-p "$(System.DefaultWorkingDirectory)/terraform.tfplan"
On the agent job, make sure you choose the terraform-aci
agent pool.
Select the build (it's defined in azure-pipelines.yml
) as your artifact, and name it _infra
Set the following variables in your release pipeline
tfstate.resourcegroup
: the resource group for your terraform state storage accounttfstate.storageaccountname
: the name of the storage account for your terraform stateenvironment
: i.e. dev, prod. not necessarily used but useful
Most secrets should be pulled from Azure CLI at runtime. For secrets that cannot, create a .secrets
directory in the root of the project.
vststoken.secret
-> Personal Access Token for Azure DevOps (used by release agent)
Terraform is, in theory, awesome. However in practice it's immature, difficult and buggy. I love the idea, and don't get me wrong - the vision Hashicorp have shown in developing this tool is amazing. But issues abound, including resources not being deleted, resources accidentally being deleted, slow release of Azure features into terraform, weird bugs and more. For now I reccomend using other automation tools like Ansible or simply ARM templates and Azure CLI.