Skip to main content
Source

This page is generated from skills/terraform-skill/references/ci-cd-workflows.md. Edit the source, not this page.

Third-party skill

This skill is maintained by Anton Babenko (terraform-best-practices.com, Compliance.tf) under the Apache-2.0 license. Upstream: https://github.com/antonbabenko/terraform-skill

CI/CD Workflows for Terraform

Part of: terraform-skill Purpose: CI/CD integration patterns for Terraform/OpenTofu

This document provides detailed CI/CD workflow templates and optimization strategies for infrastructure-as-code pipelines.


Table of Contents

  1. GitHub Actions Workflow
  2. GitLab CI Template
  3. Cost Optimization
  4. Automated Cleanup
  5. Best Practices

GitHub Actions Workflow

Complete Example

# .github/workflows/terraform.yml
name: Terraform

on: [push, pull_request]

jobs:
validate:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3
- uses: hashicorp/setup-terraform@v2

- name: Terraform Format
run: terraform fmt -check -recursive

- name: Terraform Init
run: terraform init

- name: Terraform Validate
run: terraform validate

- name: TFLint
run: |
curl -s https://raw.githubusercontent.com/terraform-linters/tflint/master/install_linux.sh | bash
tflint --init
tflint

test:
needs: validate
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3

- name: Run Terraform Tests
run: terraform test

# Or for Terratest:
- name: Setup Go
uses: actions/setup-go@v4
with:
go-version: '1.21'

- name: Run Terratest
run: |
cd tests
go test -v -timeout 30m -parallel 4

plan:
needs: test
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3
- uses: hashicorp/setup-terraform@v2

- name: Terraform Init
run: terraform init

- name: Terraform Plan
run: terraform plan -out=tfplan

- name: Upload Plan
uses: actions/upload-artifact@v3
with:
name: tfplan
path: tfplan

apply:
needs: plan
runs-on: ubuntu-latest
if: github.ref == 'refs/heads/main' && github.event_name == 'push'
environment: production
steps:
- uses: actions/checkout@v3
- uses: hashicorp/setup-terraform@v2

- name: Download Plan
uses: actions/download-artifact@v3
with:
name: tfplan

- name: Terraform Apply
run: terraform apply tfplan

With Cost Estimation (Infracost)

cost-estimate:
needs: plan
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3

- name: Setup Infracost
uses: infracost/actions/setup@v2
with:
api-key: ${{ secrets.INFRACOST_API_KEY }}

- name: Generate Cost Estimate
run: |
infracost breakdown --path . \
--format json \
--out-file /tmp/infracost.json

- name: Post Cost Comment
uses: infracost/actions/comment@v1
with:
path: /tmp/infracost.json
behavior: update

GitLab CI Template

# .gitlab-ci.yml
stages:
- validate
- test
- plan
- apply

variables:
TF_ROOT: ${CI_PROJECT_DIR}

.terraform_template:
image: hashicorp/terraform:latest
before_script:
- cd ${TF_ROOT}
- terraform init

validate:
extends: .terraform_template
stage: validate
script:
- terraform fmt -check -recursive
- terraform validate

test:
extends: .terraform_template
stage: test
script:
- terraform test
only:
- merge_requests
- main

plan:
extends: .terraform_template
stage: plan
script:
- terraform plan -out=tfplan
artifacts:
paths:
- ${TF_ROOT}/tfplan
expire_in: 1 week
only:
- merge_requests
- main

apply:
extends: .terraform_template
stage: apply
script:
- terraform apply tfplan
dependencies:
- plan
only:
- main
when: manual
environment:
name: production

Cost Optimization

Strategy

  1. Use mocking for PR validation (free)
  2. Run integration tests only on main branch (controlled cost)
  3. Implement auto-cleanup (prevent orphaned resources)
  4. Tag all test resources (track spending)

Example: Conditional Test Execution

# GitHub Actions
test:
runs-on: ubuntu-latest
steps:
- name: Run Unit Tests (Mocked)
run: terraform test

- name: Run Integration Tests
if: github.ref == 'refs/heads/main'
env:
AWS_ACCESS_KEY_ID: ${{ secrets.AWS_ACCESS_KEY_ID }}
AWS_SECRET_ACCESS_KEY: ${{ secrets.AWS_SECRET_ACCESS_KEY }}
run: |
cd tests
go test -v -timeout 30m

Cost-Aware Test Tags

// In Terratest
terraformOptions := &terraform.Options{
TerraformDir: "../examples/complete",
Vars: map[string]interface{}{
"tags": map[string]string{
"Environment": "test",
"TTL": "2h",
"CreatedBy": "CI",
"JobID": os.Getenv("GITHUB_RUN_ID"),
},
},
}

Automated Cleanup

Cleanup Script (Bash)

#!/bin/bash
# cleanup-test-resources.sh

# Find and terminate instances older than 2 hours with test tag
aws resourcegroupstaggingapi get-resources \
--tag-filters Key=Environment,Values=test \
--query 'ResourceTagMappingList[?Tags[?Key==`TTL` && Value<`'$(date -u -d '2 hours ago' +%Y-%m-%dT%H:%M:%S)'`]].ResourceARN' \
--output text | \
while read arn; do
instance_id=$(echo $arn | grep -oP 'instance/\K[^/]+')
if [ ! -z "$instance_id" ]; then
echo "Terminating instance: $instance_id"
aws ec2 terminate-instances --instance-ids $instance_id
fi
done

Scheduled Cleanup (GitHub Actions)

# .github/workflows/cleanup.yml
name: Cleanup Test Resources

on:
schedule:
- cron: '0 */2 * * *' # Every 2 hours
workflow_dispatch: # Manual trigger

jobs:
cleanup:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3

- name: Configure AWS Credentials
uses: aws-actions/configure-aws-credentials@v2
with:
aws-access-key-id: ${{ secrets.AWS_ACCESS_KEY_ID }}
aws-secret-access-key: ${{ secrets.AWS_SECRET_ACCESS_KEY }}
aws-region: us-east-1

- name: Run Cleanup Script
run: ./scripts/cleanup-test-resources.sh

Best Practices

1. Separate Environments

# Different workflows for different environments
.github/workflows/
terraform-dev.yml
terraform-staging.yml
terraform-prod.yml

Or use reusable workflows:

# .github/workflows/terraform-deploy.yml (reusable)
on:
workflow_call:
inputs:
environment:
required: true
type: string

jobs:
deploy:
environment: ${{ inputs.environment }}
# ... deployment steps

2. Require Approvals for Production

apply:
environment:
name: production
# Requires manual approval in GitHub
when: manual

3. Use Remote State

# backend.tf
terraform {
backend "s3" {
bucket = "my-terraform-state"
key = "prod/terraform.tfstate"
region = "us-east-1"
dynamodb_table = "terraform-locks"
encrypt = true
}
}

4. Implement State Locking

# In CI, use -lock-timeout to handle concurrent runs
- name: Terraform Apply
run: terraform apply -lock-timeout=10m tfplan

5. Cache Terraform Plugins

# GitHub Actions
- name: Cache Terraform Plugins
uses: actions/cache@v3
with:
path: |
~/.terraform.d/plugin-cache
key: ${{ runner.os }}-terraform-${{ hashFiles('**/.terraform.lock.hcl') }}

6. Security Scanning in CI

security-scan:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3

- name: Run Trivy
uses: aquasecurity/trivy-action@master
with:
scan-type: 'config'
scan-ref: '.'

- name: Run Checkov
uses: bridgecrewio/checkov-action@master
with:
directory: .
framework: terraform

Atlantis Integration

Atlantis provides Terraform automation via pull request comments.

atlantis.yaml

version: 3
projects:
- name: production
dir: environments/prod
workspace: default
terraform_version: v1.6.0
workflow: custom

workflows:
custom:
plan:
steps:
- init
- plan:
extra_args: ["-lock", "false"]
apply:
steps:
- apply

Benefits

  • Plan results as PR comments
  • Apply via PR comments
  • Locking prevents concurrent changes
  • Integrates with VCS (GitHub, GitLab, Bitbucket)

Troubleshooting

Issue: Tests fail in CI but pass locally

Cause: Different Terraform/provider versions

Solution:

# versions.tf - Pin versions
terraform {
required_version = ">= 1.6.0"

required_providers {
aws = {
source = "hashicorp/aws"
version = "~> 5.0"
}
}
}

Issue: Parallel tests conflict

Cause: Resource naming collisions

Solution:

// Use unique identifiers
uniqueId := random.UniqueId()
bucketName := fmt.Sprintf("test-bucket-%s-%s",
os.Getenv("GITHUB_RUN_ID"),
uniqueId)

Back to: Main Skill File