In order to connect to a Cloud SQL instance with a private IP address, you can use the Cloud SQL proxy. It's not possible to run this on your local machine, as the proxy requires access to the private IP address. To solve that, you can run it on a VM and use that as a jump box to connect to the Cloud SQL instance.
To make this example more realistic, we run the Cloud SQL instance in a Shared VPC, as this is normally the setup most Enterprise customers have configured. This way, we can clearly share what resources are running in which project (host and service).
To grant access to the proper IAM permissions, the module offers the ability to create a custom role. Identities have to be able to a) retrieve the instance details (compute.instances.get
), b} to update the metadata, in case they want to upload their SSH keys (compute.instances.setMetadata
) if they want to access the VM over SSH and c) access to the project details (compute.projects.get
).
As we want to apply the principle of least privilege, we apply those IAM permissions at instance level or tunnel resource level and not at project level. Otherwise, identities gain access to all the VMs in the project.
If the identity running the Terraform code doesn't have the necessary permissions, the custom role is not created and the role Instance Admin is granted on the instance itself.
To be able to run this example, you need the necessary permissions to create projects and a shared vpc.
Create a terraform.tfvars
-file in this directory, with the following variables:
Variable | Type | Description |
---|---|---|
organization_id | string | The ID of the GCP organization. |
billing_account_id | string | ID of the Billing Account. |
parent_folder_id | string | The ID of the parent folder. |
proxy_access_identities | string | Set of identities who require access to the IAP tunnel. Every identity should contain the type, in the form of user:, serviceAccount:, group:. |
prefix | string | The prefix that will be used for nearly all resources. This will be the prefix for the project names and IDs, amongst other things. |
Run the following commands to create the target environment.
# Initialise Terraform in this directory
terraform init
# Apply Terraform
terraform apply -auto-approve
This will create all the resources, depicted in the diagram. It will also create a backend.tf
-file, which contains the necessary information to copy your Terraform state to a GCS bucket. If you want that, you will have to run terraform init
again in the same directory and input yes
when asked to copy the state remotely. However, if you want to skip that, there is no need to execute that last step.
To access the environment, open two terminal windows and run the following commands:
- IAP Tunnel:
$(terraform output -json | jq -r .start_iap_tunnel.value)
- PSQL:
terraform output -json | jq -r .sql_client_command.value | pbcopy
Because of the double quotes, the command is copied to your cache and you can paste it in the terminal window. To retrieve the password for the database, run $(terraform output -json | jq -r .retrieve_db_password.value) | pbcopy
and copy/paste it where you ran the PSQL
-command (second terminal).
To destroy the environment, remove backend.tf
and re-initialise terraform again. When asked to copy the remote state to your local environment, enter yes
. Once done, run terraform destroy -auto-approve
. This will tear down the entire environment.
Variable | Type | Default | Description |
---|---|---|---|
region | string | europe-west1 | Default region for all resources. |
zone | string | europe-west1-b | Default zone for all resources. |
enable_internet_egress_traffic | bool | false | The VM only gets a private IP address, so it needs routing to the public internet to install all updated packages and the Cloud SQL proxy. Once everything is created, you can delete those resources by setting this value to true and running terraform apply -auto-approve again. |
enable_ssh_access | bool | true | By default, SSH access is blocked. If you need access to the VM, set this value to false so you can create the necessary firewall rules. |
subnet_cidr_range | string | 10.0.0.0/24 | The CIDR block for the subnet where the VM is created. |
cloud_sql_proxy_version | string | v1.21.0 | The version of the Cloud SQL proxy. |
enable_internet_egress_traffic | bool | true | Enable access to the public internet. This will create a Cloud NAT and route to the internet gateway. On the first startup, this is required as the proxy is downloaded from the internet |
enable_ssh_access | bool | false | Enable SSH access to the VM itself. For troubleshooting purposes, but should not be required in general |
create_custom_compute_get_role | bool | true | By default, a custom role is created for the identities to ensure they can both SSH into the VM and use it for IAP tunneling. If you set this to false, the more generic Compute Instance Admin role is assigned. |
Most of the outputs are commands that can be run to establish connectivity towards the Cloud SQL instance. You can run these in a shell, via $(terraform output -json | jq -r .OUTPUT_VAR_NAME.value
, replacing OUTPUT_VAR_NAME
with the name of the output.
Name | Description |
---|---|
start_ssh_tunnel | Command to start an SSH connection to the Cloud SQL proxy VM. |
sql_instance_connection_name | The connection name for the Cloud SQL instance. |
start_iap_tunnel | Command to open a TCP IAP tunnel to the VM, on port 5432. |
sql_client_command | PSQL command to connect to the database, via the proxy. |
host_network_project_id | The project ID of the host project. |
service_network_project_id | The project ID of the service project. |
retrieve_db_password | Command to retrieve the password for the database. |