Skip to content

Latest commit

 

History

History
449 lines (344 loc) · 19.4 KB

File metadata and controls

449 lines (344 loc) · 19.4 KB

Terraform Engine

The Terraform Engine is a tool to generate complete end-to-end Terraform deployments for Google Cloud with security, compliance, and best practices baked in.

Watch our tutorial video and use example configs to quickly get started.

Note that YAML-formatted configs were used at the time when the Tutorial video was made. The config format has been changed to HCL.

Terminology

The Terraform engine introduces the concept of "templates". Templates can be used to generate Terraform root modules specific to your Google Cloud organization and structure.

Users typically pick from a set of recipes which implement a template for one core piece of GCP infrastructure. See the recipe docs for individual recipe schemas.

Why

Users hosting sensitive data on Google Cloud typically complete common and repetitive processes such as setting up devops (Remote state, CICD), auditing, and monitoring. By using out-of-the-box end-to-end configs that implement these steps for you, you can quickly set up a secure and compliant environment and focus on the parts of the infrastructure that drive your business.

This tool helps you follow Google Cloud, Terraform and security best practices.

Infra-as-code best practices

Google Cloud best practices

  • Set up GCP Organization Policy Constraints for security best practices.

  • Define many security sensitive resources such as VPC networks and storage resources.

    • For example, the project recipe can be used to create projects and resources within projects.
  • Benefit from per-service best practices through use of the Cloud Foundation Toolkit.

    • Our recipes use Cloud Foundation Toolkit modules wherever they make sense. When there are multiple options, we choose the most secure option. For example, creating a GKE cluster through our project recipe will utilize the safer GKE cluster module from Cloud Foundation Toolkit.
  • Promote centralized network control pattern.

    • For example, the team example configures deployment to use a VPC host project to manage networks and subnets in a centralized way (enabling network administration to be separated from project administration). Resources in different projects communicate securely with internal IPs.

DevOps best practices

  • Configure Continuous Integration and Continuous Deployment (CICD) pipelines to reduce human access to the org infrastructure. Promote coding and version control best practices.

    • For example, the CICD recipe sets up a pipeline that is run by Cloud Build service accounts. Through integration with Github, changes to infrastructure can be made via pull requests. The hooks we set up will automatically display the latest Terraform plan so users can be confident in their changes. The changes can be automatically applied when the pull request gets merged.
  • Allow logical folders within your hierarchy to be managed independently, thus reducing org-wide broad access to single service account and chances of cascading errors.

    • For example, the devops recipe can be used on different folders to setup a separate CICD pipeline and service account to manage projects and resources within the folder. The service accounts of other CICD pipelines cannot access these projects.
  • Delegate responsibility through groups and service accounts:

    • For example, the group component can be used to create and manage Cloud Identity groups and memberships. IAM roles should only be assigned to these groups so that individuals obtain permissions through groups rather than direct IAM roles. See the multi-envs example for how to create and configure groups in devops and project recipes.

HIPAA alignment

  • Work towards alignment with HIPAA and compliance requirements for auditing and monitoring.

    • For example, the audit recipe creates a dedicated project to host audit logs and creates logs routers to export all audit logs to BigQuery (for 1 year) and to GCS (for 7 years). These configurations help align with HIPAA audit log requirements.

Prerequisites

  1. Install the following dependencies and add them to your PATH:

  2. Familiarize yourself with the tools you'll use:

    The infrastructure is deployed using Terraform, which is an industry standard for defining infrastructure-as-code.

  3. Set up your organization or folder for Google Cloud resources.

  4. The running user will need to be a Google Workspace Super Admin and have the following Cloud IAM roles:

    • roles/resourcemanager.organizationAdmin on the org for org deployment
    • roles/resourcemanager.folderAdmin on the folder for folder deployment
    • roles/resourcemanager.projectCreator on the org or folder
    • roles/compute.xpnAdmin on the org or folder
    • roles/billing.admin on the billing account

Defining Architecture

Plan out your org architecture and map them to engine configs. Our configs can help you setup a wide variety of architectures that suit your needs.

As a rule of thumb, we recommend having one org config that sets up core security and compliance infrastructure such as auditing, monitoring and org policies. It should define a devops project to manage Terraform state and a CICD pipeline. You can also use this config to define the org's folder hierarchy. We recommend creating a folder for each team and major application in the org.

To define projects for a team or application, we recommend having a separate config. These configs should set the root parent folder to be one of the folders setup by the org config. Each config should define a devops project to manage Terraform state and a CICD pipeline.

You may wish to replicate the configs across multiple environments (dev, staging, etc).

All configs and the generated Terraform code should be checked into version control. You can use a separate repo for each CICD pipeline. Note if using GitHub you can share a single repo but with different paths within the repo. For Cloud Source Repository, you must use a repo within each individual devops project.

Installation

Download a pre-built tfengine binary:

VERSION=v0.8.0
wget -O /usr/local/bin/tfengine https://github.com/GoogleCloudPlatform/healthcare-data-protection-suite/releases/download/${VERSION}/tfengine_${VERSION}_linux-amd64
chmod +x /usr/local/bin/tfengine

or build it yourself:

git clone https://github.com/GoogleCloudPlatform/healthcare-data-protection-suite
cd healthcare-data-protection-suite
go install ./cmd/tfengine

Usage

The engine takes a path to an input config and a path to output the generated Terraform configs. For details on fields for the input schemas, see the schemas directory. The input config can use Go templates to programmatically express the resources being deployed. After the output has been generated, there is no longer a dependency on the engine and the user can directly use the terraform binary to deploy the infrastructure.

  1. Replace the values in a suitable example with values for your infrastructure. Save it as a separate file called config.hcl and copy it to your local configs repo root directory.

    Project and Bucket names are globally unique and must be changed from defaults. You will get an error if you have chosen a name that is already taken.

    TIP: Prefer to remotely fetch recipes from a release which can be more stable than using the HEAD recipes.

    template "devops" {
      recipe_path = "git://github.com/GoogleCloudPlatform/healthcare-data-protection-suite//templates/tfengine/recipes/devops.hcl?ref=templates-v0.4.0"
      output_path = "./devops"
      data = {
        ...
      }
    }
  2. To set up helper environment vars, run the following commands:

    GIT_ROOT=/path/to/your/local/configs/repo
    CONFIG_PATH=$GIT_ROOT/config.hcl
    OUTPUT_PATH=$GIT_ROOT/terraform

    NOTE: If you plan to set up CICD, the terraform_root variable set in the CICD recipe should correspond to the parent directory in OUTPUT_PATH that would be checked into source control, i.e. terraform in the above setup.

  3. Login to your Google account by following instructions here. Make sure to login using gcloud auth application-default login since Terraform uses Application Default Credentials.

  4. Run the engine to generate your Terraform configs:

    tfengine --config_path=$CONFIG_PATH --output_path=$OUTPUT_PATH

    NOTE: For usage information on all supported flags, please refer to main.go.

    Each generated folder typically contains the following files:

    • main.tf: This file defines the Terraform resources and modules to manage.

    • variables.tf: This file defines any input variables that the deployment can take.

    • outputs.tf: This file defines any outputs from this deployment. These values can be used by other deployments.

    • terraform.tfvars: This file defines values for the input variables.

  5. Deploy the devops recipe to create administrative IAM groups and set up the devops project to host the Terraform state and CICD:

    1. Define the org/folder admins group and devops project owners group in the devops recipe. It is recommended to use the following naming conventions:

      • {PREFIX}-{org|folder}-admins@{DOMAIN}: Members of this group get administrative access to the org or folder. This group can be used in break-glass situations to give humans access to the org or folder to make changes.

      • {PREFIX}-devops-owners@{DOMAIN}: Members of this group get owners access to the devops project to make changes to the CICD project or to make changes to the Terraform state. Make sure to include yourself as an owner of this group. Otherwise, you might lose access to the devops project after the ownership is transferred to this group.

      For example, for a folder deployment with sample prefix gcp and domain example.com, the admins group should be named as [email protected].

      WARNING: The best practice is to always deploy changes using CICD. These privileged groups should remain empty and only have humans added for emergency situations or when investigation is required.

      NOTE: The underlying Terraform module and resources used in our recipes to manage groups and memberships have these known limitations. It is currently recommended to only create the groups (so they can be used for IAM role assignment in other recipes seamlessly) and set initial owners of the groups through Terraform and make further memberships modifications through the Google Workspace Admin console.

      Alternatively, create these groups manually and use them as existing groups in the devops recipe.

    2. Deploy the devops project and Terraform state bucket.

      cd $OUTPUT_PATH/devops
      terraform init
      terraform apply

      Your devops project should now be ready.

    3. In $CONFIG_PATH, set enable_gcs_backend to true, and regenerate the Teraform configs.

      tfengine --config_path=$CONFIG_PATH --output_path=$OUTPUT_PATH
    4. Backup the state of the devops project to the newly created state bucket by running the following command.

      terraform init -force-copy
  6. (Optional) Create Cloud Identity Groups for coming IAM role assignments.

    It is recommended to delegate responsibility through groups and service accounts so that individuals only obtain permissions through groups rather than direct IAM roles.

    1. Add a separate template called groups using the project.hcl recipe, and define your groups and memberships in the resources block. Set the project_id to the devops project ID and exists to true.

      template "groups" {
        recipe_path = "git://github.com/GoogleCloudPlatform/healthcare-data-protection-suite//templates/tfengine/recipes/project.hcl"
        output_path = "./groups"
        data = {
          project = {
            project_id = "example-devops"
            exists     = true
          }
          resources = {
            groups = [
              {
                id = "[email protected]"
                customer_id = "c12345678"
                owners = ["[email protected]"]
                members = ["[email protected]"]
              },
            ]
          }
        }
      }
    2. Create groups and initial memberships. You must be at least Google Workspace Group Editor to be able to do so.

      cd $OUTPUT_PATH/groups
      terraform init
      terraform apply

    NOTE: The underlying Terraform module and resources used in our recipes to manage groups and memberships have these known limitations. It is currently recommended to only create the groups (so they can be used for IAM role assignment in other recipes seamlessly) and set initial owners of the groups through Terraform and make further memberships modifications through the Google Workspace Admin console.

  7. (Optional) To deploy Continuous Integration (CI) and Continuous Deployment (CD) resources, follow the instructions here or equivalently, $OUTPUT_PATH/cicd/README.md in the generated Terraform configs directory.

    Note that the above groups template must be deployed manually first if you also use it to create groups to be used in the cicd recipe. These groups should exist before cicd recipe can be deployed.

    Your devops project and CICD pipelines are ready. The following changes should be made as Pull Requests (PRs) and go though code reviews. After approval is granted and CI tests pass, merge the PR. The CD job automatically deploys the change to your Google Cloud infra.

  8. Deploy org/folder infrastructure and other resources by sending a PR for local changes to the config repo.

  9. Follow the instructions of all commented out blocks starting with TODO(user) in the config to deploy the changes. Remove the comment once done.

Tips

  • If you plan on using the engine again, do not manually modify any generated file as it will be overwritten the next time the engine runs. Instead, prefer to change the input config, recipe, or component for the generated files and add new resources under new deployments that are not generated by the engine.

  • Always back up any engine configs as well as the generated Terraform configs and any modifications to the Terraform configs to version control.

  • Any component file with the extension .tmpl will have it removed when generated.

Continuous integration (CI) and continuous deployment (CD)

See CICD specific documentations here.

If you are setting up a multi environment workflow (dev, staging, prod), see multi_envs.hcl for an example config.