Skip to content

Deploy Terraform config from GitHub via a Jenkins server in Azure using a managed identity

License

Notifications You must be signed in to change notification settings

mukundbajaj/jenkins

 
 

Repository files navigation

Jenkins on Azure VM deploying Terraform config

Jenkins is a popular CI/CD tool. This example shows the creation of a Jenkins server with a system assigned managed identity. Azure CLI and Terraform will also be installed. Jeknins is then configured to use the GitHub repo and its Jenkinsfile. The Jenkinsfile includes both Azure CLI and Terraform example steps in the pipeline.

Assumes a Bash environment with the Azure CLI, plus access to a valid subscription.

The terraform config should create a resource group, terraform-demo, and an Azure Container Instance running the inspector gadget image, but this is purely to prove that your Terraform configuration can be deployed via Jenkins using a GitHub repo and a system assigned managed identity.

If you are prefer using service principals then check the companion Service Principal readme.

Forking

Before you start, fork this repo so that you have your own.

Deploy a Jenkins server

  1. Create cloud-config.yaml

    Create a cloud-config.yaml file with the following contents.

    #cloud-config
    package_upgrade: true
    runcmd:
      - sudo apt install openjdk-11-jre ca-certificates curl apt-transport-https lsb-release gnupg gpg wget -y
      - wget -qO - https://pkg.jenkins.io/debian-stable/jenkins.io.key | sudo apt-key add -
      - wget -qO- https://apt.releases.hashicorp.com/gpg | gpg --dearmor | sudo tee /usr/share/keyrings/hashicorp-archive-keyring.gpg > /dev/null
      - wget -qO- https://packages.microsoft.com/keys/microsoft.asc | gpg --dearmor | sudo tee /usr/share/keyrings/microsoft.gpg > /dev/null
      - sh -c 'echo deb https://pkg.jenkins.io/debian-stable binary/ > /etc/apt/sources.list.d/jenkins.list'
      - sh -c 'echo "deb [arch=`dpkg --print-architecture` signed-by=/usr/share/keyrings/hashicorp-archive-keyring.gpg] https://apt.releases.hashicorp.com `lsb_release -cs` main" | sudo tee /etc/apt/sources.list.d/hashicorp.list'
      - sh -c 'echo "deb [arch=`dpkg --print-architecture` signed-by=/usr/share/keyrings/microsoft.gpg] https://packages.microsoft.com/repos/azure-cli/ `lsb_release -cs` main" | sudo tee /etc/apt/sources.list.d/azure-cli.list'
      - sudo chmod go+r /usr/share/keyrings/hashicorp-archive-keyring.gpg /usr/share/keyrings/microsoft.gpg
      - sudo apt-get update && sudo apt-get install jenkins terraform azure-cli jq -y
      - sudo service jenkins restart

    The cloud-config.yaml file will install

    • jenkins
    • terraform
    • azure-cli
    • jq
  2. Set your default region and resource group

    Set these to your preferred values.

    az config set defaults.location=uksouth defaults.group=jenkins

    Setting defaults removes the need to specify --location and --resource-group on Azure CLI commands. Defaults are stored in ~/.azure/config. Use az config unset defaults.location defaults.group to unset.

    See https://learn.microsoft.com/cli/azure/azure-cli-configuration for more.

  3. Create the resource group

    az group create --name $(az config get defaults.group --query value -otsv)
  4. Create the VM

    az vm create --name jenkins \
    --image UbuntuLTS \
    --admin-username "azureuser" \
    --generate-ssh-keys \
    --public-ip-sku Standard \
    --custom-data cloud-config.yaml \
    --assign-identity [system]
  5. Open port 8080 and 443 on the NSG

    az vm open-port --port 8080 --priority 1010 --name jenkins
    az vm open-port --port 443 --priority 1020 --name jenkins

Terraform remote state and identity RBAC

  1. Find the objectId for the managed identity

    managed_identity=$(az vm show --resource-group jenkins --name jenkins --query identity.principalId --output tsv)
  2. Create a storage account and container for the Terraform remote state

    rgId=$(az group show --name $(az config get defaults.group --query value -otsv) --query id -otsv)
    sa=terraform$(md5sum <<< $rgId | cut -c1-12)
    az storage account create --name $sa --sku Standard_LRS --allow-blob-public-access false
    az storage container create --name "tfstate" --account-name $sa --auth-mode login

    Uses md5sum to generate a predictable hash from the resource group's resource ID.

  3. Assign RBAC roles for the managed identity

    subscriptionId=/subscriptions/$(az account show --query id --output tsv)
    saId=$(az storage account show --name $sa --query id --output tsv)
    az role assignment create --assignee $managed_identity --role "Contributor" --scope $subscriptionId
    az role assignment create --assignee $managed_identity --role "Storage Blob Data Contributor" --scope $saId

SSH to your Jenkins server

  1. Grab the public IP address

    publicIp=$(az vm show --name jenkins  --show-details --query publicIps --output tsv)
    echo $publicIp
  2. SSH onto the Jenkins server

    ssh azureuser@$publicIp

Initial server config

  1. Checks

    terraform --version
    az version
    service jenkins status

    CTRL+C to break.

  2. Display the Jenkins URL and unlock password

    publicIp=$(curl -sSL -H Metadata:true --noproxy "*" http://169.254.169.254/metadata/loadbalancer?api-version=2020-10-01 \
      | jq -r .loadbalancer.publicIpAddresses[0].frontendIpAddress)
    pwdfile=/var/lib/jenkins/secrets/initialAdminPassword
    echo "Open http://${publicIp}:8080"
    sudo --user=jenkins -- test -s $pwdfile && echo "and paste in $(sudo cat $pwdfile)"

Initial UI connection

  1. Connect to the URL and paste in the initialAdminPassword

  2. Select plugins to install

  3. Filter to GitHub

  4. Check GitHub

  5. Install

    Getting Started will take a few minutes to run.

  6. Create First Admin User

    • username
    • password
    • full name
    • email address
  7. Save & Finish

  8. Start using Jenkins

Configure Jenkins Plugins

  1. Manage Jenkins | System Configuration | Manage Plugins

    1. Click on Available plugins

    2. Search for and check the following plug-ins

      • AnsiColor
      • Terraform
      • Dark Theme (optional)
    3. Check and Install without restart

      Installing additional Jenkins plugins

  2. Restart Jenkins

    Once they have all installed then check the Restart Jenkins box.

    Restarting Jenkins aftder successful install

    Alternatively:

    • Browse to http://<ip_address>:8080/restart, or
    • Run sudo service jenkins restart

Dark mode (optional)

If you installed the Dark Theme plugin:

  1. Manage Jenkins | Configure System
  2. Scroll down to Themes
  3. Select your preferred theme

Jenkins Tools

Jenkins can install tools (binaries, etc) on the fly with automatic installers. We installed Terraform via apt and will use the /usr/bin/terraform binary.

  1. Manage Jenkins | Global Tool Configuration
  2. Scroll down to the Terraform section
  3. Add Terraform
    1. Name = terraform

      For information, the name matches the second field in the Jenkinsfile tools block:

      tools {
          'org.jenkinsci.plugins.terraform.TerraformInstallation' 'terraform'
      }
    2. Install directory = /usr/bin

    3. Uncheck Install automatically

      Add Terraform as a Jenkins tool

  4. Save

Credential

  1. Display values for secrets

    rgId=$(az group show --name $(az config get defaults.group --query value -otsv) --query id -otsv)
    sa=terraform$(md5sum <<< $rgId | cut -c1-12)
    az account show --query "{tenant_id:tenantId, subscription_id:id}" --output yamlc
    az storage account show --name $sa --query "{resource_group:resourceGroup, storage_account:name}" --output yamlc
  2. Manage Jenkins | Manage Credentials

  3. System, Global credentials, + Add Credentials

  4. Kind = Secret text

    Create credentials for tenant_id, subscription_id, resource_group and storage_account, setting secret to the values shown by the commands above.

    Adding additional secrets in Jenkins

    The Jenkinsfile maps the credentials to environment variables.

Configure a pipeline

Create the Jenkins pipeline

On the Jenkins dashboard:

  1. + New Item

  2. Select *Pipeline"

  3. Enter an item name

    Create a new pipeline

  4. Click OK

    You will now be in the Configure | General screen for your pipeline.

  5. Scroll down to Build Triggers section

  6. Check GitHub hook trigger for GITScm polling

    Check the GitHub hook trigger for GITScm polling option

  7. Scroll down further, to the Pipeline section

  8. Change the Definition dropdown to Pipeline script from SCM

    Check the GitHub hook trigger for GITScm polling option

  9. SCM should be set to Git

  10. In repository URL, add in the .git path to your repo

    ⚠️ Your repo

  11. Click on Add Branch

  12. Set the specifier, e.g. */main

    Pipeline configuration for SCM in Jenkins

    Note that there is a Jenkinsfile at the root of the repo, matching the default Script Path on the configuration.

  13. Save

Trigger the initial test build

Make sure the pipeline works before creating the GitHub webhook. In Jenkins dashboard:

  1. Select your pipeline

  2. Click on Build Now

    Click on Build Now to test your pipeline

    The pipeline view will start to show in the right hand pane.

  3. Approve the deployment

    Assuming the Terraform Init, Terraform Validate and Terraform Plan stages have succeeded, you will be asked to approve the deployment. Feel free to view the log output of the Terraform Plan stage.

    Hover over the Waiting for approval stage and you will get the option to Proceed.

    Approve the deployment

    The final Terraform Apply stage should create the resource group and container instance.

Success

  1. Jenkins dashboard

    The dashboard should now show a healthy pipeline run.

    Healthy Jenkins dashboard

  2. Azure Portal

    You should now have a resource group called terraform-demo containing the example container instance.

    Showing the Inspector Gadget container instance overview in the Azure Portal

    Success!

Create the GitHub webhook

In your clone of this GitHub repo:

  1. Settings | Webhooks

  2. Set the Payload URL to http://<ip_address>:8080/github-webhook/

  3. Set Content type to application/json

    Add the webhook in GitHub

  4. Click on Add webhook

Feel free to locally clone your repo, make a change to the Jenkinsfile or Terraform files and then push that commit back up to the origin. This should trigger another build pipeline.

Next

OK, it is working. Want to harden it a little?

Resources

Resources to be explored for hardening

Configure https:

Configure secrets in Azure Key Vault:

Will also look at:

  • Using credentials to access a private GitHub repo
  • Removing the public IP and switching to Azure Bastion tunnels

Resources dismissed

There is a lot of old information out there, and a number of plugins that are moving out of Microsoft support.

Below is a list of resources that I explored and discounted as I believe that they are not on the recommended path, or they are now outdated.

About

Deploy Terraform config from GitHub via a Jenkins server in Azure using a managed identity

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • HCL 100.0%