This guide describes using CMEK for Dataform and walks you through how to enable CMEK encryption of Dataform repositories.
By default, Dataform encrypts customer content at rest. Dataform handles encryption for you without any additional actions on your part. This option is called Google default encryption.
If you want to control your encryption keys, then you can use customer-managed encryption keys (CMEKs) in Cloud KMS with CMEK-integrated services including Dataform. Using Cloud KMS keys gives you control over their protection level, location, rotation schedule, usage and access permissions, and cryptographic boundaries. Using Cloud KMS also lets you track key usage, view audit logs, and control key lifecycles. Instead of Google owning and managing the symmetric key encryption keys (KEKs) that protect your data, you control and manage these keys in Cloud KMS.
After you set up your resources with CMEKs, the experience of accessing your Dataform resources is similar to using Google default encryption. For more information about your encryption options, see Customer-managed encryption keys (CMEK).
CMEK encryption of repository data
When you apply CMEK encryption to a Dataform repository, all Dataform-managed customer data in that repository is encrypted at rest using the CMEK protection key set for the repository. This data includes the following:
- Git repository content of the Dataform repository and its workspaces
- Compiled SQL queries and compilation errors
- Stored SQL queries of workflow actions
- Error details of executed workflow actions
Dataform uses CMEK protection keys in the following scenarios:
- During every operation that requires decryption of customer data stored at rest.
These operations include, but are not limited to the following:
- Responses to a user query—for example, compilationResults.query.
- Creation of Dataform resources that require previously created encrypted repository data—for example, workflow invocations.
- Git operations to update the remote repository, for example, pushing a Git commit.
 
- Responses to a user query—for example, 
- During every operation that requires storing customer data at rest.
These operations include, but are not limited to, the following:
- Responses to a user query—for example, compilationResults.create.
- Git operations to a workspace—for example, pulling a Git commit.
 
- Responses to a user query—for example, 
Dataform manages the encryption of customer data associated only with Dataform resources. Dataform does not manage encryption of customer data that is created in BigQuery through execution of Dataform workflows. To encrypt data created and stored in BigQuery, configure CMEK for BigQuery.
Supported keys
Dataform supports the following types of CMEK keys:
- Cloud KMS software keys
- Cloud Hardware Security Module (HSM) keys
- Cloud External Key Manager (Cloud EKM) keys
Key availability varies by key type and region. For more information about the geographical availability of CMEK keys, see Cloud KMS locations.
Restrictions
Dataform supports CMEK with the following restrictions:
- The maximum size of a CMEK-encrypted repository is 512 MB.
- The maximum size of a workspace in a CMEK-encrypted repository is 512 MB.
- You can't apply a CMEK protection key to a repository after the repository has been created. You can apply CMEK encryption during repository creation only.
- You can't remove a CMEK protection key from a repository.
- You can't change a CMEK protection key for a repository.
- If you set a default Dataform CMEK key for your Google Cloud project, all new repositories created in the Google Cloud project location must be encrypted with CMEK. When you create a new repository in the Google Cloud project location, you can apply the default Dataform CMEK key or a different CMEK key, but you cannot apply default encryption at rest.
- If you change the value of a default Dataform CMEK key, the previous value applies to pre-existing repositories, and the updated value applies to repositories created after the change.
- You can set only one default Dataform CMEK key per location of Google Cloud project repositories.
- CMEK organization policies are not available.
- Using Cloud HSM and Cloud EKM keys is subject to availability. For more information about availability of keys across locations, see Cloud KMS locations.
Cloud KMS quotas and Dataform
You can use Cloud HSM and Cloud EKM keys with Dataform. When you use CMEK in Dataform, your projects can consume Cloud KMS cryptographic requests quotas. For example, CMEK-encrypted Dataform repositories can consume these quotas for each change to repository contents. Encryption and decryption operations using CMEK keys affect Cloud KMS quotas only if you use hardware (Cloud HSM) or external (Cloud EKM) keys. For more information, see Cloud KMS quotas.
Managing keys
Use Cloud KMS for all key-management operations. Dataform cannot detect or act upon any key changes until they are propagated by Cloud KMS. Some operations, such as disabling or destroying a key, can take up to three hours to propagate. Changes to permissions usually propagate much faster.
After the repository is created, Dataform calls Cloud KMS to make sure that the key is still valid during each operation on encrypted repository data.
If Dataform detects that your Cloud KMS key has been disabled or destroyed, all data stored in the corresponding repository becomes inaccessible.
If calls by Dataform to Cloud KMS detect that a formerly disabled key has been re-enabled, Dataform restores access automatically.
Use external keys with Cloud EKM
As an alternative to using keys that reside on Cloud KMS, you can use keys that reside with a supported external key management partner. To do this, use Cloud External Key Manager (Cloud EKM) to create and manage external keys, which are pointers to keys that reside outside of Google Cloud. For more information, see Cloud External Key Manager.
After you create an external key with Cloud EKM, you can apply it to a new Dataform repository by providing the ID of that key when creating the repository. This procedure is the same as applying a Cloud KMS key to a new repository.
Use Dataform default CMEK keys
To encrypt multiple Dataform repositories with the same CMEK key, you can set a default Dataform CMEK key for your Google Cloud project. You must specify the location of the Google Cloud project for the default Dataform CMEK key. You can set only one default CMEK key per Google Cloud project.
After you set a default Dataform CMEK key, Dataform applies the key to all new repositories created in the Google Cloud project location by default. When you create a repository, you can use the default key, or select a different CMEK key.
How an unavailable key status is handled
In rare scenarios, such as during periods when Cloud KMS is unavailable, Dataform might be unable to retrieve the status of your key from Cloud KMS.
If your Dataform repository is protected by a key that is enabled at the time at which Dataform is unable to communicate with Cloud KMS, the encrypted repository data becomes inaccessible.
The encrypted repository data remains inaccessible until Dataform can reconnect with Cloud KMS and Cloud KMS responds that the key is active.
Conversely, if your Dataform repository is protected by a key that is disabled at the time at which Dataform is first unable to communicate with Cloud KMS, the encrypted repository data remains inaccessible until it is able to reconnect to Cloud KMS and you have re-enabled your key.
Logging
You can audit the requests that Dataform sends to Cloud KMS on your behalf in Cloud Logging, if you have enabled audit logging for the Cloud KMS API in your project. These Cloud KMS log entries are visible in Cloud Logging. For more information, see View logs.
Before you begin
- Decide whether you are going to run Dataform and Cloud KMS in different projects, or in the same project. We recommend using separate projects for greater control over permissions. For information about Google Cloud project IDs and project numbers, see Identifying projects. 
- For the Google Cloud project that runs Cloud KMS: - Enable the Cloud Key Management Service API.
- Create a key ring and a key as described in
Creating key rings and keys. Create the key ring
in a location that matches the location of your Dataform
repository:
- 
    Repositories must use matching regional keys. For example,
    a repository in region asia-northeast3must be protected with a key from a key ring located inasia-northeast3.
- 
    The globalregion can't be used with Dataform.
 
- 
    Repositories must use matching regional keys. For example,
    a repository in region 
 
Enable CMEK
Dataform can access the key on your behalf after
you grant the Cloud KMS
CryptoKey Encrypter/Decrypter (roles/cloudkms.cryptoKeyEncrypterDecrypter)
role to the default
Dataform service agent.
Your default Dataform service agent ID is in the following format:
service-YOUR_PROJECT_NUMBER@gcp-sa-dataform.iam.gserviceaccount.com
To grant the CryptoKey Encrypter/Decrypter role to the default Dataform service agent, follow these steps:
Console
- Open the Key Management page in the Google Cloud console. 
- Click the name of the key ring that contains the key. 
- Click the checkbox for the encryption key to which you want to add the role. The Permissions tab opens. 
- Click Add member. 
- Enter the email address of the service agent. - If the service agent is already on the members list, it has existing roles. Click the current role drop-down list for the service agent.
 
- Click the drop-down list for Select a role, click Cloud KMS, and then click the Cloud KMS CryptoKey Encrypter/Decrypter role. 
- Click Save to apply the role to the service account. 
gcloud
You can use the Google Cloud CLI to assign the role:
gcloud kms keys add-iam-policy-binding \
    --project=KMS_PROJECT_ID \
    --member serviceAccount:SERVICE_AGENT \
    --role roles/cloudkms.cryptoKeyEncrypterDecrypter \
    --location=KMS_KEY_LOCATION \
    --keyring=KMS_KEY_RING \
    KMS_KEY
Replace the following:
- KMS_PROJECT_ID: the ID of your Google Cloud project that is running Cloud KMS
- SERVICE_AGENT: the email address of your default Dataform service agent
- KMS_KEY_LOCATION: the location name of your Cloud KMS key
- KMS_KEY_RING: the key ring name of your Cloud KMS key
- KMS_KEY: the key name of your Cloud KMS key
Apply a CMEK organization policy
Dataform is integrated with two organization policy constraints to help ensure CMEK usage across an organization:
- constraints/gcp.restrictNonCmekServicesis used to require CMEK protection.
- constraints/gcp.restrictCmekCryptoKeyProjectsis used to limit which Cloud KMS keys are used for CMEK protection.
This integration lets you specify the following encryption compliance requirements for Dataform repositories in your organization:
Considerations when applying organization policies
Before applying any CMEK organization policies, you should be aware of the following.
Prepare for a propagation delay
After you set or update an organization policy, it can take up to 15 minutes for the new policy to take effect.
Consider existing resources
Existing resources are not subject to newly created organization policies. For example, an organization policy does not retroactively apply to existing repositories. Those resources are still accessible without a CMEK and, if applicable, are still encrypted with existing keys.
Verify required permissions to set an organization policy
The permission to set or update the organization policy might be difficult to acquire for testing purposes. You must be granted the Organization Policy Administrator role, which can only be granted at the organization level (rather than the project or folder level).
Although the role must be granted at the organization level, it is still possible to specify a policy that only applies to a specific project or folder.
Require CMEKs for all new Dataform repositories
You can use the constraints/gcp.restrictNonCmekServices constraint to require
that CMEKs be used to protect all new Dataform repositories in
an organization.
If set, this organization policy causes all resource creation requests without a specified Cloud KMS key to fail.
After you set this policy, it applies only to new repositories in the project. Any existing repositories without Cloud KMS keys applied continue to exist and are accessible without issue.
Console
- In the Google Cloud console, go to the Organization policies page. 
- Using the Filter, search for the following constraint: - constraints/gcp.restrictNonCmekServices
- In the Name column, click Restrict which services may create resources without CMEK. 
- Click Manage Policy. 
- On the Edit policy page, under Policy source, select Override parent's policy. 
- Under Rules, click Add a rule. 
- In the Policy values list, select Custom. 
- In the Policy type list, select Deny. 
- In the Custom values field, enter the following: - is:dataform.googleapis.com
- Click Done, and then click Set policy. 
gcloud
- Create a temporary file - /tmp/policy.yamlto store the policy:- name: projects/PROJECT_ID/policies/gcp.restrictNonCmekServices spec: rules: - values: deniedValues: - is:dataform.googleapis.com - Replace - PROJECT_IDwith the ID of the project where you are applying this constraint.
- Run the - org-policies set-policycommand:- gcloud org-policies set-policy /tmp/policy.yaml 
To verify that the policy is successfully applied, you can try to create a Dataform repository in the project. The process fails unless you specify a Cloud KMS key.
Restrict Cloud KMS keys for a Dataform project
You can use the constraints/gcp.restrictCmekCryptoKeyProjects constraint to
restrict the Cloud KMS keys that you can use to protect a repository in
a Dataform project.
For example, you can specify a rule similar to the following: "For all
Dataform repositories in projects/my-company-data-project,
Cloud KMS keys used in this project must come from
projects/my-company-central-keys OR projects/team-specific-keys."
Console
- In the Google Cloud console, go to the Organization policies page. 
- Using the Filter, search for the following constraint: - constraints/gcp.restrictCmekCryptoKeyProjects
- In the Name column, click Restrict which projects may supply KMS CryptoKeys for CMEK. 
- Click Manage Policy. 
- On the Edit policy page, under Policy source, select Override parent's policy. 
- Under Rules, click Add a rule. 
- In the Policy values list, select Custom. 
- In the Policy type list, select Allow. 
- In the Custom values field, enter the following: - under:projects/KMS_PROJECT_ID- Replace - KMS_PROJECT_IDwith the ID of the project where the Cloud KMS keys you want to use are located.- For example, - under:projects/my-kms-project.
- Click Done, and then click Set policy. 
gcloud
- Create a temporary file - /tmp/policy.yamlto store the policy:- name: projects/PROJECT_ID/policies/gcp.restrictCmekCryptoKeyProjects spec: rules: - values: allowedValues: - under:projects/KMS_PROJECT_ID - Replace the following - PROJECT_ID: the ID of the project where you are applying this constraint.
- KMS_PROJECT_ID: the ID of the project where the Cloud KMS keys you want to use are located.
 
- Run the - org-policies set-policycommand:- gcloud org-policies set-policy /tmp/policy.yaml 
To verify that the policy is successfully applied, you can try to create a Dataform repository using a Cloud KMS key from a different project. The process will fail.
Set a default Dataform CMEK key
Setting a default Dataform CMEK key for your Google Cloud project lets you encrypt multiple repositories with the same CMEK key. For more information, see Use a default key for Dataform repositories.
To set or edit a default CMEK key, call the Dataform API in the following request:
curl -X PATCH \
-H "Content-Type: application/json" \
-d '{"defaultKmsKeyName":"projects/PROJECT_ID/locations/PROJECT_LOCATION/keyRings/KMS_KEY_RING/cryptoKeys/KMS_KEY"}' \
https://dataform.googleapis.com/v1beta1/projects/PROJECT_ID/locations/PROJECT_LOCATION/config
Replace the following:
- KMS_KEY_RING: the key ring name of your Cloud KMS key.
- KMS_KEY: the name of your Cloud KMS key.
- PROJECT_ID: the ID of your Google Cloud project.
- PROJECT_LOCATION: the location name of your Google Cloud project.
Remove a default Dataform CMEK key
To remove a default Dataform CMEK key from your Google Cloud project, call the Dataform API in the following request:
curl -X PATCH \
-H "Content-Type: application/json" \
-d '{"defaultKmsKeyName":""}' \
https://dataform.googleapis.com/v1beta1/projects/PROJECT_ID/locations/PROJECT_LOCATION/config
Replace the following:
- PROJECT_ID: the ID of your Google Cloud project.
- PROJECT_LOCATION: the location name of your Google Cloud project where you want to unset default CMEK.
Check if a default Dataform CMEK key is set
To check if a default Dataform CMEK key is set for your, Google Cloud project, call the Dataform API in the following request:
curl -X GET \
-H "Content-Type: application/json" \
https://dataform.googleapis.com/v1beta1/projects/PROJECT_ID/locations/PROJECT_LOCATION/config
Replace the following:
- PROJECT_ID: the ID of your Google Cloud project.
- PROJECT_LOCATION: the location name of your Google Cloud project.
Apply CMEK to a repository
You can apply CMEK protection to a Dataform repository during repository creation.
To apply CMEK encryption to a Dataform repository, select encryption with the default Dataform CMEK key or specify a unique Cloud KMS key when you create the repository. For instructions, see Create a repository.
You can't change the encryption mechanism of a Dataform repository after the repository is created.
For more information, see Restrictions.
What's next
- To learn more about CMEK, see CMEK overview.
- To learn more about Cloud KMS quotas, see Cloud KMS Quotas.
- To learn more about Cloud KMS pricing, see Cloud KMS Pricing.
- To learn more about Dataform repositories, see Create a repository.