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.
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.
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.
-
Clearly define your resource hierarchy in infra-as-code.
- For example, the org foundation example can be used to define org level components and folders. Then, the team example can be used to define projects and resources within one of the folders.
-
Break up Terraform state files into logical deployments with remote state enabled.
- For example, the
org foundation example
creates the Terraform root modules
devops
,cicd
,audit
,folders
, etc.
- For example, the
org foundation example
creates the Terraform root modules
-
Set up GCP Organization Policy Constraints for security best practices.
- For example, the org policy recipe can be used to configure and generate best practice GCP Organization Policy Constraints to help align your GCP infrastructure with HIPAA requirements.
-
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.
-
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.
-
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.
-
Install the following dependencies and add them to your PATH:
- gcloud
- Terraform 0.14.8 (consistent with the version of Terraform used by CICD)
- Go 1.14+
-
Familiarize yourself with the tools you'll use:
The infrastructure is deployed using Terraform, which is an industry standard for defining infrastructure-as-code.
-
Set up your organization or folder for Google Cloud resources.
-
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 deploymentroles/resourcemanager.folderAdmin
on the folder for folder deploymentroles/resourcemanager.projectCreator
on the org or folderroles/compute.xpnAdmin
on the org or folderroles/billing.admin
on the billing account
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.
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
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.
-
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 = { ... } }
-
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 inOUTPUT_PATH
that would be checked into source control, i.e.terraform
in the above setup. -
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. -
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.
-
-
Deploy the
devops
recipe to create administrative IAM groups and set up thedevops
project to host the Terraform state and CICD:-
Define the org/folder admins group and
devops
project owners group in thedevops
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 thedevops
project after the ownership is transferred to this group.
For example, for a folder deployment with sample prefix
gcp
and domainexample.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. -
-
Deploy the
devops
project and Terraform state bucket.cd $OUTPUT_PATH/devops terraform init terraform apply
Your
devops
project should now be ready. -
In $CONFIG_PATH, set
enable_gcs_backend
totrue
, and regenerate the Teraform configs.tfengine --config_path=$CONFIG_PATH --output_path=$OUTPUT_PATH
-
Backup the state of the
devops
project to the newly created state bucket by running the following command.terraform init -force-copy
-
-
(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.
-
Add a separate template called
groups
using theproject.hcl
recipe, and define your groups and memberships in theresources
block. Set theproject_id
to thedevops
project ID andexists
totrue
.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]"] }, ] } } }
-
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.
-
-
(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 thecicd
recipe. These groups should exist beforecicd
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.
-
Deploy org/folder infrastructure and other resources by sending a PR for local changes to the config repo.
-
Follow the instructions of all commented out blocks starting with
TODO(user)
in the config to deploy the changes. Remove the comment once done.
-
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.
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.