Implement Account Factory for Terraform (AFT) by using a bootstrap pipeline - AWS Prescriptive Guidance

Implement Account Factory for Terraform (AFT) by using a bootstrap pipeline

Created by Vinicius Elias (AWS) and Edgar Costa Filho (AWS)

Summary

Notice: AWS CodeCommit is no longer available to new customers. Existing customers of AWS CodeCommit can continue to use the service as normal. Learn more

This pattern provides a simple and secure method for deploying AWS Control Tower Account Factory for Terraform (AFT) from the management account of AWS Organizations. The core of the solution is an AWS CloudFormation template that automates the AFT configuration by creating a Terraform pipeline, which is structured to be easily adaptable for initial deployment or subsequent updates.

Security and data integrity are top priorities at AWS, so the Terraform state file, which is a critical component that tracks the state of the managed infrastructure and configurations, is securely stored in an Amazon Simple Storage Service (Amazon S3) bucket. This bucket is configured with several security measures, including server-side encryption and policies to block public access, to help ensure that your Terraform state is safeguarded against unauthorized access and data breaches.

The management account orchestrates and oversees the entire environment, so it is a critical resource in AWS Control Tower. This pattern follows AWS best practices and ensures that the deployment process is not only efficient but also aligns with security and governance standards, to offer a comprehensive, secure, and efficient way to deploy AFT in your AWS environment.

For more information about AFT, see the AWS Control Tower documentation.

Prerequisites and limitations

Prerequisites

  • A basic AWS multi-account environment with the following accounts at the minimum: management account, Log Archive account, Audit account, and one additional account for AFT management.

  • An established AWS Control Tower environment. The management account should be properly configured, because the CloudFormation template will be deployed within it.

  • The necessary permissions in the AWS management account. You'll need sufficient permissions to create and manage resources such as S3 buckets, AWS Lambda functions, AWS Identity and Access Management (IAM) roles, and AWS CodePipeline projects.

  • Familiarity with Terraform. Understanding Terraform's core concepts and workflow is important because the deployment involves generating and managing Terraform configurations.

Limitations

  • Be aware of the AWS resource quotas in your account. The deployment might create multiple resources, and encountering service quotas could impede the deployment process.

  • The template is designed for specific versions of Terraform and AWS services. Upgrading or changing versions might require template modifications.

Product versions

  • Terraform version 1.5.7 or later

  • AFT version 1.11.1 or later

Architecture

Target technology stack

  • AWS CloudFormation

  • AWS CodeBuild

  • AWS CodeCommit

  • AWS CodePipeline

  • Amazon EventBridge

  • IAM

  • AWS Lambda

  • Amazon S3

Target architecture

The following diagram illustrates the implementation discussed in this pattern.

Workflow for implementing AFT by using a bootstrap pipeline

The workflow consists of three main tasks: creating the resources, generating the content, and running the pipeline.

Creating the resources

The CloudFormation template that's provided with this pattern creates and sets up all required resources, depending on the parameters you select when you deploy the template. At the minimum, the template creates the following resources:

  • A CodeCommit repository to store the AFT Terraform bootstrap code

  • An S3 bucket to store the Terraform state file that's associated with the AFT implementation

  • A CodePipeline pipeline

  • Two CodeBuild projects to implement the Terraform plan and apply commands in different stages of the pipeline

  • IAM roles for CodeBuild and CodePipeline services

  • A second S3 bucket to store pipeline runtime artifacts

  • An EventBridge rule to capture CodeCommit repository changes on the main branch

  • Another IAM role for the EventBridge rule

Additionally, if you set the Generate AFT Files parameter in the CloudFormation template to true, the template creates these additional resources to generate the content:

  • An S3 bucket to store the generated content and to be used as the source of the CodeCommit repository

  • A Lambda function to process the given parameters and generate the appropriate content

  • An IAM function to run the Lambda function

  • A CloudFormation custom resource that runs the Lambda function when the template is deployed

Generating the content

To generate the AFT bootstrap files and their content, the solution uses a Lambda function and an S3 bucket. The function creates a folder in the bucket, and then creates two files inside the folder: main.tf and backend.tf. The function also processes the provided CloudFormation parameters and populates these files with predefined code, replacing the respective parameter values.

To view the code that's used as a template to generate the files, see the solution's GitHub repository. Basically, the files are generated as follows.

main.tf

module "aft" { source = "github.com/aws-ia/terraform-aws-control_tower_account_factory?ref=<aft_version>" # Required variables ct_management_account_id = "<ct_management_account_id>" log_archive_account_id = "<log_archive_account_id>" audit_account_id = "<audit_account_id>" aft_management_account_id = "<aft_management_account_id>" ct_home_region = "<ct_home_region>" # Optional variables tf_backend_secondary_region = "<tf_backend_secondary_region>" aft_metrics_reporting = "<false|true>" # AFT Feature flags aft_feature_cloudtrail_data_events = "<false|true>" aft_feature_enterprise_support = "<false|true>" aft_feature_delete_default_vpcs_enabled = "<false|true>" # Terraform variables terraform_version = "<terraform_version>" terraform_distribution = "<terraform_distribution>" }

backend.tf

terraform { backend "s3" { region = "<aft-main-region>" bucket = "<s3-bucket-name>" key = "aft-setup.tfstate" } }

During the CodeCommit repository creation, if you set the Generate AFT Files parameter to true, the template uses the S3 bucket with the generated content as the source of the main branch to automatically populate the repository.

Running the pipeline

After the resources have been created and the bootstrap files have been configured, the pipeline runs. The first stage (Source) fetches the source code from the main branch of the repository, and the second stage (Build) runs the Terraform plan command and generates the results to be reviewed. In the third stage (Approval), the pipeline waits for a manual action to approve or reject the last stage (Deploy). At the last stage, the pipeline runs the Terraform apply command by using the result of the previous Terraform plan command as input. Finally, a cross-account role and the permissions in the management account are used to create the AFT resources in the AFT management account.

Tools

AWS services

  • AWS CloudFormation helps you set up AWS resources, provision them quickly and consistently, and manage them throughout their lifecycle across AWS accounts and Regions.

  • AWS CodeBuild is a fully managed build service that helps you compile source code, run unit tests, and produce artifacts that are ready to deploy. 

  • AWS CodeCommit is a version control service that helps you privately store and manage Git repositories without needing to manage your own source control system.

  • AWS CodePipeline helps you quickly model and configure the different stages of a software release and automate the steps required to release software changes continuously.

  • AWS Lambda is a compute service that runs your code in response to events and automatically manages compute resources, providing a fast way to create a modern, serverless application for production.

  • AWS SDK for Python (Boto3) is a software development kit that helps you integrate your Python application, library, or script with AWS services.

Other tools

  • Terraform is an infrastructure as code (IaC) tool that lets you build, change, and version infrastructure safely and efficiently. This includes low-level components such as compute instances, storage, and networking; and high-level components such as DNS entries and SaaS features.

  • Python is an easy to learn, powerful programming language. It has efficient high-level data structures and provides a simple but effective approach to object-oriented programming.

Code repository

The code for this pattern is available in the GitHub AFT bootstrap pipeline repository.

For the official AFT repository, see AWS Control Tower Account Factory for Terraform in GitHub.

Best practices

When you deploy AFT by using the provided CloudFormation template, we recommend that you follow best practices to help ensure a secure, efficient, and successful implementation. Key guidelines and recommendations for implementing and operating the AFT include the following.

  • Thorough review of parameters: Carefully review and understand each parameter in the CloudFormation template. Accurate parameter configuration is crucial for the correct setup and functioning of AFT.

  • Regular template updates: Keep the template updated with the latest AWS features and Terraform versions. Regular updates help you take advantage of new functionality and maintain security.

  • Versioning: Pin your AFT module version and use a separate AFT deployment for testing if possible.

  • Scope: Use AFT only to deploy infrastructure guardrails and customizations. Do not use it to deploy your application.

  • Linting and validation: The AFT pipeline requires a linted and validated Terraform configuration. Run lint, validate, and test before pushing the configuration to AFT repositories.

  • Terraform modules: Build reusable Terraform code as modules, and always specify the Terraform and AWS provider versions to match your organization's requirements.

Epics

TaskDescriptionSkills required

Prepare the AWS Control Tower environment.

Set up and configure AWS Control Tower in your AWS environment to ensure centralized management and governance for your AWS accounts. For more information, see Getting started with AWS Control Tower in the AWS Control Tower documentation.

Cloud administrator

Launch the AFT management account.

Use the AWS Control Tower Account Factory to launch a new AWS account to serve as your AFT management account. For more information, see Provision accounts with AWS Service Catalog Account Factory in the AWS Control Tower documentation.

Cloud administrator
TaskDescriptionSkills required

Launch the CloudFormation template.

In this epic, you deploy the CloudFormation template provided with this solution to set up the AFT bootstrap pipeline in your AWS management account. The pipeline deploys the AFT solution in the AFT management account that you set up in the previous epic.

Step 1: Open the AWS CloudFormation console

  • Sign in to the AWS Management Console and open the AWS CloudFormation console. Make sure that you are operating within the correct AWS Control Tower main Region.

Step 2: Create a new stack

  1. Choose to create a new stack.

  2. Select the option to upload a template file, and upload the CloudFormation template that's provided with this pattern.

Step 3: Configure stack parameters

  • Repository Name: Specify the repository name for storing the AFT bootstrap module.

  • Branch Name: Specify the source repository branch.

  • CodeBuild Docker Image: Choose the file to use as the CodeBuild Docker base image.

Step 4: Decide on file generation

  • The Generate AFT Files parameter controls whether to generate default AFT deployment files. Set this parameter to:

    • true to automatically create and store AFT deployment files in the specified repository.

    • false if you want to manually handle the file creation or already have the files in place.

    If you selected false, go to step 8; otherwise, follow steps 5–7 first.

Step 5: Fill in AWS Control Tower and AFT account details

  • Input AWS Control Tower and AFT account-specific information:

    • Log Archive Account ID: The ID of the Log Archive account ID in AWS Control Tower.

    • Audit Account ID: The ID of the Audit account in AWS Control Tower.

    • AFT Management Account ID: The ID of the AFT management account that you created in the first epic.

    • AFT Main Region and AFT Secondary Region: The main and secondary AWS Regions for AFT deployment.

Step 6: Configure AFT options

  • Set up metrics reporting:

    • AFT Enable Metrics Reporting: Enable or disable AFT metrics reporting. For more information, see Operational metrics in the AWS Control Tower documentation.

  • Set AFT feature options:

    • Enable AFT CloudTrail Data Events: Enable CloudTrail data events in all AFT managed accounts. For more information, see AWS CloudTrail data events in the AWS Control Tower documentation.

    • Enable AFT Enterprise Support: Enable Enterprise Support in all AFT managed accounts. For more information, see AWS Enterprise Support plan in the AWS Control Tower documentation.

    • Enable AFT Delete Default VPC: Delete all VPCs in the AFT management account only. For more information, see Delete the AWS default VPC in the AWS Control Tower documentation.

Step 7: Specify versions

  • AFT Terraform Version: Choose the version of Terraform to use in AFT pipelines.

  • AFT Version: Define the AFT version for deployment. Keep the default setting (latest) to use the most current AFT version.

Step 8: Review and create the stack

  • Review all the parameters and settings. If everything is in order, proceed to create the stack.

Step 9: Monitor stack creation

  • AWS CloudFormation provisions and configures the resources you defined. Monitor the stack creation process on the CloudFormation console. This process might take several minutes.

Step 10: Verify the deployment

  • When the stack status shows CREATE_COMPLETE, verify that all resources have been correctly created.

  • In the Outputs section, note the TerraformBackendBucketName value.

Cloud administrator
TaskDescriptionSkills required

Populate the AFT bootstrap repository.

(Optional) After you deploy the CloudFormation template, you can populate or validate the content in the newly created AFT bootstrap repository, and test whether the pipeline has run successfully.

If you set the Generate AFT Files parameter to true, skip to the next story (validating the pipeline).

Step 1: Populate the repository

  1. Open the AWS CodeCommit console and select the newly created repository. If you kept the default name, the repository will be called aft-setup.

  2. Clone the repository to your local machine by using SSH, HTTPS, or HTTPS (GRC), and open it in an editor.

  3. Create a folder called terraform and two empty files inside it: backend.tf and main.tf.

  4. Open the backend.tf file and add this code snippet:

    terraform { backend "s3" { region = "<aft-main-region>" bucket = "<s3-bucket-name>" key = "aft-setup" } }

    In the file:

    • Replace <aft-main-region> with the main AFT Region. This should match the AWS Control Tower main Region.

    • Replace <s3-bucket-name> with the name of the Terraform backend bucket. You can find this in the TerraformBackendBucketName output generated by the CloudFormation template you deployed earlier.

  5. Open the main.tf file and use one of the examples available in the AFT repository to deploy AFT. For example, you can work with your preferred version control system (VCS) provider (CodeCommit, GitHub, or Bitbucket) or customize the AFT VPC. For more AFT input options, see the README file in the AFT repository.

Step 2: Commit and push your changes

  • After you have created and populated the folder and files, confirm your changes, and upload the code to the repository. The pipeline starts automatically, runs through the Source and Build stages, and then waits for an Approval action before the Deploy stage.

Cloud administrator

Validate the AFT bootstrap pipeline.

Step 1: View the pipeline

  • Open the CodePipeline console and check whether the aft-bootstrap-pipeline pipeline was started successfully. It should be running a Terraform plan or waiting for a manual approval action.

Step 2: Approve the Terraform plan results

  • You can review the results of the Terraform plan by looking at the execution logs of the Build stage, and then approve or reject the execution on the Approval stage. If you approve, the pipeline starts deploying AFT resources in the provided AFT management account.

Step 3: Wait for the deployment

  • Wait for the pipeline to run successfully. This should take around 30 minutes. Any failures you might encounter are often caused by API quotas. In these cases, you can rerun the pipeline to continue the deployment.

Step 4: Check created resources

  • Access the AFT management account and confirm that the resources have been created.

Cloud administrator

Troubleshooting

IssueSolution

The custom Lambda function included in the CloudFormation template fails during deployment.

Check the Amazon CloudWatch logs for the Lambda function to identify the error. The logs provide detailed information and can help pinpoint the specific issue. Confirm that the Lambda function has the necessary permissions and that the environment variables have been set correctly.

You encounter failures in resource creation or management caused by inadequate permissions.

Review the IAM roles and policies that are attached to the Lambda function, CodeBuild, and other services involved in the deployment. Confirm that they have the necessary permissions. If there are permission issues, adjust the IAM policies to grant the required access.

You’re using an outdated version of the CloudFormation template with newer AWS services or Terraform versions.

Regularly update the CloudFormation template to be compatible with the latest AWS and Terraform releases. Check the release notes or documentation for any version-specific changes or requirements.

You reach AWS service quotas during deployment.

Before you deploy the pipeline, check AWS service quotas for resources such as S3 buckets, IAM roles, and Lambda functions. Request increases if necessary. For more information, see AWS service quotas on the AWS website.

You encounter errors due to incorrect input parameters in the CloudFormation template.

Double-check all input parameters for typos or incorrect values. Confirm that resource identifiers, such as account IDs and Region names, are accurate.

Related resources

To implement this pattern successfully, review the following resources. These resources provide additional information and guidance that can be invaluable in setting up and managing AFT by using AWS CloudFormation.

AWS documentation:

IAM policies and best practices:

Terraform on AWS:

AWS service quotas:

  • AWS service quotas provides information about how to view AWS service quotas and how to request increases.