Skip to main content
Source

This page is generated from skills/terraform-skill/references/quick-reference.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

Quick Reference

Part of: terraform-skill Purpose: Command cheat sheets and decision flowcharts

This document provides quick lookup tables, command references, and decision flowcharts for rapid consultation during development.


Table of Contents

  1. Command Cheat Sheet
  2. Decision Flowchart
  3. Version-Specific Guidance
  4. Troubleshooting Guide
  5. Migration Paths

Command Cheat Sheet

Static Analysis

Works with both terraform and tofu commands:

# Format and validate
terraform fmt -recursive -check # or: tofu fmt -recursive -check
terraform validate # or: tofu validate

# Linting
tflint --init && tflint

# Security scanning
checkov -d .

Native Tests (1.6+)

# Run all tests
terraform test # or: tofu test

# Run tests in specific directory
terraform test -test-directory=tests/unit/

# Verbose output
terraform test -verbose

Plan Validation

# Generate and review plan
terraform plan -out tfplan # or: tofu plan -out tfplan

# Convert plan to pretty JSON
terraform show -json tfplan | jq -r '.' > tfplan.json

# Check for specific changes
terraform show tfplan | grep "will be created"

Decision Flowchart

Testing Approach Selection

Need to test Terraform/OpenTofu code?

├─ Just syntax/format?
│ └─ terraform/tofu validate + fmt

├─ Static security scan?
│ └─ trivy + checkov

├─ Terraform/OpenTofu 1.6+?
│ ├─ Simple logic test?
│ │ └─ Native terraform/tofu test
│ │
│ └─ Complex integration?
│ └─ Terratest

└─ Pre-1.6?
├─ Go team?
│ └─ Terratest

└─ Neither?
└─ Plan to upgrade Terraform/OpenTofu

Module Development Workflow

1. Plan
├─ Define inputs (variables.tf)
├─ Define outputs (outputs.tf)
└─ Document purpose (README.md)

2. Implement
├─ Create resources (main.tf)
├─ Pin versions (versions.tf)
└─ Add examples (examples/simple, examples/complete)

3. Test
├─ Static analysis (validate, fmt, lint)
├─ Unit tests (native or Terratest)
└─ Integration tests (examples/)

4. Document
├─ Update README with usage
├─ Document inputs/outputs
└─ Add CHANGELOG

5. Publish
├─ Tag version (git tag v1.0.0)
├─ Push to registry
└─ Announce changes

Version-Specific Guidance

Terraform 1.0-1.5

  • ❌ No native testing framework
  • ✅ Use Terratest
  • ✅ Focus on static analysis
  • ✅ terraform plan validation

Terraform 1.6+ / OpenTofu 1.6+

  • ✅ NEW: Native terraform test / tofu test
  • ✅ Consider migrating simple tests from Terratest
  • ✅ Keep Terratest for complex integration
  • ✅ All Terraform 1.0+ features available

Terraform 1.7+ / OpenTofu 1.7+

  • ✅ NEW: Mock providers for unit testing
  • ✅ Reduce costs with mocking
  • ✅ Use real integration tests for final validation
  • ✅ Faster test iteration

Terraform vs OpenTofu Comparison

Both Terraform and OpenTofu are fully supported by this skill. The choice depends on your requirements:

Quick Decision Matrix:

FactorTerraformOpenTofu
LicensingBusiness Source License (BSL) 1.1Mozilla Public License 2.0 (MPL 2.0)
GovernanceHashiCorp (single vendor)Linux Foundation (community-driven)
Latest Version1.14+1.11+
Native Testing1.6+1.6+
Mock Providers1.7+1.7+
Feature ParityReference implementationCompatible fork with some additions
Enterprise SupportHCP Terraform, Terraform CloudMultiple vendors
Migration PathN/ADrop-in replacement for Terraform ≤1.5

When to choose Terraform:

  • Using HashiCorp Terraform Cloud or HCP Terraform
  • Enterprise support contract with HashiCorp
  • Need absolute latest features first

When to choose OpenTofu:

  • Prefer open-source governance model
  • Want to avoid vendor lock-in concerns
  • Building on community-driven development
  • BSL 1.1 license doesn't fit your use case

For this skill:

  • Commands are shown for both: terraform and tofu
  • Most patterns work identically, though differences exist (see release notes)
  • Version-specific features noted (1.6+, 1.7+, etc.)
  • Note: Since OpenTofu 1.6, the platforms have diverged with unique features

When creating modules, Claude will ask your preference to generate appropriate commands and documentation.


Troubleshooting Guide

Issue: Tests fail in CI but pass locally

Symptoms:

  • Tests pass on your machine
  • Same tests fail in GitHub Actions/GitLab CI

Common Causes:

  1. Different Terraform/provider versions
  2. Different environment variables
  3. Different AWS credentials/permissions

Solution:

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

required_providers {
aws = {
source = "hashicorp/aws"
version = "~> 6.0" # Pin to major version
}
}
}

Issue: Parallel tests conflict

Symptoms:

  • Tests fail when run in parallel
  • Error: "ResourceAlreadyExistsException"

Cause: Resource naming collisions

Solution:

// Use unique identifiers
import "github.com/gruntwork-io/terratest/modules/random"

uniqueId := random.UniqueId()
bucketName := fmt.Sprintf("test-bucket-%s", uniqueId)

Issue: High test costs

Symptoms:

  • AWS bill increasing from tests
  • Many orphaned resources in test account

Solutions:

  1. Use mocking for unit tests (Terraform 1.7+)

    mock_provider "aws" { ... }
  2. Implement resource TTL tags

    Vars: map[string]interface{}{
    "tags": map[string]string{
    "Environment": "test",
    "TTL": "2h",
    },
    }
  3. Run integration tests only on main branch

    if: github.ref == 'refs/heads/main'
  4. Use smaller instance types

    instance_type = "t3.micro" # Not "m5.large"
  5. Share test resources when safe

    • VPCs, security groups (rarely change)
    • Don't share: instances, databases (change often)

Migration Paths

From Manual Testing → Automated

Phase 1: Static analysis

terraform validate
terraform fmt -check

Phase 2: Plan review

terraform plan -out=tfplan
# Manual review

Phase 3: Automated tests

  • Native tests (1.6+)
  • OR Terratest

Phase 4: CI/CD integration

  • GitHub Actions / GitLab CI
  • Automated apply on main branch

From Terratest → Native Tests (1.6+)

Strategy: Gradual migration

  1. Keep Terratest for:

    • Complex integration tests
    • Multi-step workflows
    • Cross-provider tests
  2. Migrate to native tests:

    • Simple unit tests
    • Logic validation
    • Mock-friendly tests
  3. During transition:

    • Maintain both frameworks
    • Gradually increase native test coverage
    • Remove Terratest tests once replaced

Example: Mixed approach

tests/
├── unit/ # Native tests
│ └── validation.tftest.hcl
└── integration/ # Terratest
└── complete_test.go

From Terraform → OpenTofu

Good news: OpenTofu is a drop-in replacement!

  1. No code changes needed

    • All Terraform syntax works
    • Same provider ecosystem
    • Compatible state files
  2. Update CI/CD:

    # Replace
    terraform init
    terraform plan
    terraform apply

    # With
    tofu init
    tofu plan
    tofu apply
  3. Update documentation:

    • README mentions OpenTofu compatibility
    • CI/CD workflows use tofu command

Pre-Commit Checklist

Formatting & Validation

Run these commands before every commit:

# Format all Terraform files
terraform fmt -recursive

# Validate configuration
terraform validate

Naming Convention Review

  • All identifiers use _ not -
  • No resource names repeat resource type (no aws_vpc.main_vpc)
  • Single-instance resources named this or descriptive name
  • Variables have plural names for lists/maps (subnet_ids not subnet_id)
  • All variables have descriptions
  • All outputs have descriptions
  • Output names follow {name}_{type}_{attribute} pattern
  • No double negatives in variable names

Code Structure Review

  • count/for_each at top of resource blocks (blank line after)
  • tags as last real argument in resources
  • depends_on after tags (if used)
  • lifecycle at end of resource (if used)
  • Variables ordered: description → type → default → sensitive → nullable → validation
  • Only # comments used (no // or /* */)

Modern Features Check

  • Using try() not element(concat())
  • Secrets use write-only arguments or external data sources (not in state)
  • nullable = false set on non-null variables
  • optional() used in object types where applicable (Terraform 1.3+)
  • Variable validation blocks added where constraints needed
  • Consider cross-variable validation for related variables (Terraform 1.9+)

Architecture Review

  • terraform.tfvars only at composition level (not in modules)
  • Remote state configured (never local state)
  • Resource modules don't hardcode values (use variables/data sources)
  • terraform_remote_state used for cross-composition dependencies
  • File structure follows standard: main.tf, variables.tf, outputs.tf, versions.tf

Documentation Check

Required documentation for all modules:

  • README.md exists with absolute links (Terraform Registry compatibility)
  • All variables documented in README with descriptions and types
  • All outputs documented in README with descriptions
  • Usage examples provided showing how to use the module
  • Version requirements specified (Terraform version, provider versions)

Version Management Quick Reference

Constraint Syntax

SyntaxMeaningUse Case
"5.0.0"Exact versionAvoid (inflexible)
"~> 5.0"Pessimistic (5.0.x)Recommended for stability
"~> 5.0.1"Pessimistic (5.0.x where x >= 1)Specific patch minimum
">= 5.0, < 6.0"RangeAny 5.x version
">= 5.0"MinimumRisky (breaking changes)

Strategy by Component

ComponentRecommendationExample
TerraformPin minor, allow patchrequired_version = "~> 1.9"
ProvidersPin major, allow minor/patchversion = "~> 5.0"
Modules (prod)Pin exact versionversion = "5.1.2"
Modules (dev)Allow patch updatesversion = "~> 5.1"

Update Workflow

# Step 1: Lock versions initially
terraform init # Creates .terraform.lock.hcl

# Step 2: Update to latest within constraints
terraform init -upgrade # Updates providers

# Step 3: Review changes
terraform plan

# Step 4: Commit lock file
git add .terraform.lock.hcl
git commit -m "Update provider versions"

Update Strategy

Security patches:

  • Update immediately
  • Test: dev → stage → prod
  • Prioritize Terraform core and provider updates

Minor versions:

  • Regular maintenance (monthly/quarterly)
  • Review changelog for breaking changes
  • Test thoroughly before production

Major versions:

  • Planned upgrade cycles
  • Dedicated testing period
  • May require code changes
  • Phased rollout: dev → stage → prod

Refactoring Quick Reference

Common Refactoring Patterns

Pattern 1: Count to For_Each Migration

When: Need stable resource addressing or items might be reordered

# Step 1: Add for_each, keep count commented
# Step 2: Add moved blocks for each resource
# Step 3: Run terraform plan (should show "moved" not "destroy/create")
# Step 4: Apply changes
# Step 5: Remove commented count

Key principle: Use moved blocks to preserve existing resources

Pattern 2: Legacy to Modern Terraform

0.12/0.13 → 1.x checklist:

  • Replace element(concat(...))try()
  • Add nullable = false where appropriate
  • Use optional() in object types (1.3+)
  • Add validation blocks
  • Migrate secrets to write-only arguments (1.11+)
  • Use moved blocks for refactoring (1.1+)
  • Add cross-variable validation (1.9+)

Pattern 3: Secrets Remediation

Goal: Move secrets out of Terraform state

# Step 1: Create secret in AWS Secrets Manager (outside Terraform)
aws secretsmanager create-secret --name prod-db-password --secret-string "..."

# Step 2: Update Terraform to use data sources
# Step 3: Use write-only argument (Terraform 1.11+)
# Step 4: Remove random_password resource or variable
# Step 5: Apply and verify secret not in state
terraform show | grep -i password # Should not appear

Refactoring Decision Tree

What are you refactoring?

├─ Resource addressing (count[0] → for_each["key"])
│ └─ Use: moved blocks + for_each conversion

├─ Secrets in state
│ └─ Use: AWS Secrets Manager + write-only arguments (1.11+)

├─ Legacy Terraform syntax (0.12/0.13)
│ └─ Use: Modern feature checklist above

└─ Module structure (rename, reorganize)
└─ Use: moved blocks to preserve resources

Migration Best Practices

Before refactoring:

  1. Backup state file
  2. Test in development first
  3. Review terraform plan carefully
  4. Document what changed and why

During refactoring:

  1. One change at a time
  2. Verify each step with terraform plan
  3. Use moved blocks, not destroy/recreate
  4. Keep git history clean with logical commits

After refactoring:

  1. Verify idempotency (plan shows no changes)
  2. Test in staging before production
  3. Update documentation
  4. Communicate changes to team

For detailed refactoring patterns, see: Code Patterns: Refactoring Patterns


Common Patterns

Resource Naming

# ✅ Good: Descriptive, contextual
resource "aws_instance" "web_server" { }
resource "aws_s3_bucket" "application_logs" { }

# ❌ Bad: Generic
resource "aws_instance" "main" { }
resource "aws_s3_bucket" "bucket" { }

Variable Naming

# ✅ Good: Context-specific
var.vpc_cidr_block
var.database_instance_class

# ❌ Bad: Generic
var.cidr
var.instance_class

File Organization

Standard module structure:
├── main.tf # Primary resources
├── variables.tf # Input variables
├── outputs.tf # Output values
├── versions.tf # Provider versions
└── README.md # Documentation

Back to: Main Skill File