diff --git a/README.md b/README.md index 7d4b6b9..861aa71 100644 --- a/README.md +++ b/README.md @@ -83,6 +83,7 @@ See [`main.tf`](https://github.com/runatlantis/terraform-gce-atlantis/tree/maste - Provisioning the Google Cloud Managed SSL certificate can take up to 25 minutes after the `terraform apply` has finished. - If you bring your own Docker image (*not using any Atlantis image as base image*), be sure to create an Atlantis user using a uid (user ID) of 100. +- As per Docker spec, the base image's `CMD` will be overridden when you define a new `ENTRYPOINT` through the `command` variable: ### After it's successfully deployed @@ -232,7 +233,9 @@ You can check the status of the certificate in the Google Cloud Console. | Name | Description | Type | Default | Required | |------|-------------|------|---------|:--------:| +| [args](#input\_args) | Arguments to override the container image default command (CMD). | `list(string)` | `null` | no | | [block\_project\_ssh\_keys\_enabled](#input\_block\_project\_ssh\_keys\_enabled) | Blocks the use of project-wide publich SSH keys | `bool` | `false` | no | +| [command](#input\_command) | Command to override the container image ENTRYPOINT | `list(string)` | `null` | no | | [default\_backend\_security\_policy](#input\_default\_backend\_security\_policy) | Name of the security policy to apply to the default backend service | `string` | `null` | no | | [disk\_kms\_key\_self\_link](#input\_disk\_kms\_key\_self\_link) | The self link of the encryption key that is stored in Google Cloud KMS | `string` | `null` | no | | [domain](#input\_domain) | Domain to associate Atlantis with and to request a managed SSL certificate for. Without `https://` | `string` | n/a | yes | diff --git a/examples/secure-env-vars/.github/workflows/docker-gcp-secrets.yaml b/examples/secure-env-vars/.github/workflows/docker-gcp-secrets.yaml deleted file mode 100644 index 5576f2e..0000000 --- a/examples/secure-env-vars/.github/workflows/docker-gcp-secrets.yaml +++ /dev/null @@ -1,53 +0,0 @@ -name: Release -on: - push: - branches: - - main - -jobs: - docker-release: - name: Tagged Docker release to Google Artifact Registry - runs-on: ubuntu-latest - - permissions: - contents: 'read' - id-token: 'write' - - steps: - - uses: "actions/checkout@v3" - - - id: "auth" - name: "Authenticate to Google Cloud" - uses: "google-github-actions/auth@v0" - with: - token_format: access_token - workload_identity_provider: "projects//locations/global/workloadIdentityPools//providers/" - service_account: "@.iam.gserviceaccount.com" - access_token_lifetime: 300s - - - id: "secrets" - uses: "google-github-actions/get-secretmanager-secrets@v1" - with: - secrets: |- - atlantis_gh_user:/ - atlantis_gh_token:/ - atlantis_gh_webhook_secret:/ - - - name: Login to Artifact Registry - uses: docker/login-action@v1 - with: - registry: europe-docker.pkg.dev/ - username: oauth2accesstoken - password: ${{ steps.auth.outputs.access_token }} - - - id: docker-push-tagged - name: Tag Atlantis Docker image and push to Google Artifact Registry - uses: docker/build-push-action@v2 - with: - push: true - build-args: | - ATLANTIS_GH_USER=${{ steps.secrets.outputs.atlantis_gh_user }} - ATLANTIS_GH_TOKEN=${{ steps.secrets.outputs.atlantis_gh_token }} - ATLANTIS_GH_WEBHOOK_SECRET=${{ steps.secrets.outputs.atlantis_gh_webhook_secret }} - tags: | - -docker.pkg.dev///: diff --git a/examples/secure-env-vars/.github/workflows/docker-github-secrets.yaml b/examples/secure-env-vars/.github/workflows/docker-github-secrets.yaml deleted file mode 100644 index 55d2b33..0000000 --- a/examples/secure-env-vars/.github/workflows/docker-github-secrets.yaml +++ /dev/null @@ -1,46 +0,0 @@ -name: Release -on: - push: - branches: - - main - -jobs: - docker-release: - name: Tagged Docker release to Google Artifact Registry - runs-on: ubuntu-latest - - permissions: - contents: 'read' - id-token: 'write' - - steps: - - uses: "actions/checkout@v3" - - - id: "auth" - name: "Authenticate to Google Cloud" - uses: "google-github-actions/auth@v0" - with: - token_format: access_token - workload_identity_provider: "projects//locations/global/workloadIdentityPools//providers/" - service_account: "@.iam.gserviceaccount.com" - access_token_lifetime: 300s - - - name: Login to Artifact Registry - uses: docker/login-action@v1 - with: - registry: europe-docker.pkg.dev/ - username: oauth2accesstoken - password: ${{ steps.auth.outputs.access_token }} - - - id: docker-push-tagged - name: Tag Atlantis Docker image and push to Google Artifact Registry - uses: docker/build-push-action@v2 - with: - push: true - build-args: | - ATLANTIS_GH_USER=${{ secrets.ATLANTIS_GH_USER }} - ATLANTIS_GH_TOKEN=${{ secrets.ATLANTIS_GH_TOKEN }} - ATLANTIS_GH_WEBHOOK_SECRET=${{ secrets.ATLANTIS_GH_WEBHOOK_SECRET }} - tags: | - -docker.pkg.dev///: - diff --git a/examples/secure-env-vars/Dockerfile b/examples/secure-env-vars/Dockerfile deleted file mode 100644 index 5b3a00d..0000000 --- a/examples/secure-env-vars/Dockerfile +++ /dev/null @@ -1,9 +0,0 @@ -FROM ghcr.io/runatlantis/atlantis:latest - -ARG ATLANTIS_GH_USER -ARG ATLANTIS_GH_TOKEN -ARG ATLANTIS_GH_WEBHOOK_SECRET - -ENV ATLANTIS_GH_USER $ATLANTIS_GH_USER -ENV ATLANTIS_GH_TOKEN $ATLANTIS_GH_TOKEN -ENV ATLANTIS_GH_WEBHOOK_SECRET $ATLANTIS_GH_WEBHOOK_SECRET diff --git a/examples/secure-env-vars/README.md b/examples/secure-env-vars/README.md index cb42c7a..fb7099b 100644 --- a/examples/secure-env-vars/README.md +++ b/examples/secure-env-vars/README.md @@ -2,9 +2,11 @@ This guide explains how to secure environment variables when using the Atlantis module on Google Cloud Platform. For more information on using this module, see the [`basic example`](https://github.com/runatlantis/terraform-gce-atlantis/tree/master/examples/basic). +Additionally, this example uses a GitHub App for authentication, and a custom image entrypoint to set environment variables at container startup time. + - [Prerequisites](#prerequisites) - [How to deploy](#how-to-deploy) - - [Important](#Important) + - [Important](#important) - [Setting sensitive environment variables](#setting-sensitive-environment-variables) - [Setting non sensitive environment variables](#setting-non-sensitive-environment-variables) @@ -12,8 +14,12 @@ This guide explains how to secure environment variables when using the Atlantis You should already have the following resources: -- An Artifact or Container Registry in Google Cloud. -- A CI/CD system with a secret manager integration (such as GitHub, Gitlab, Jenkins, or Cloud Build). +- Google network, subnetwork and a Cloud NAT +- Service account, [specifics can be found here](../../README.md#service-account) +- Domain, [specifics can be found here](../../README.md#dns-record) +- The secrets for the GitHub app id, secret, and webhook secret. + +If you prefer an example that includes the above resources, see [`complete example`](https://github.com/bschaatsbergen/atlantis-on-gcp-vm/tree/master/examples/complete) ## How to deploy @@ -22,6 +28,7 @@ To deploy the Atlantis module, see [`Dockerfile`](https://github.com/runatlantis ### Important - If you bring your own Docker image (not using any Atlantis image as base image), be sure to create an Atlantis user using a uid (user ID) of 100. +- As per Docker spec, the base image's `CMD` will be overridden when you define a new `ENTRYPOINT` through the `command` variable: ## Configuring Atlantis @@ -35,6 +42,9 @@ Use a wrapper Atlantis Docker image to set environment variables that contain se - [**GitHub Actions**: pull secrets from Google Secret Manager](https://github.com/runatlantis/terraform-gce-atlantis/tree/master/examples/secure-env-vars/.github/workflows/docker-gcp-secrets.yaml) - [**GitHub Actions**: use GitHub secrets](https://github.com/runatlantis/terraform-gce-atlantis/tree/master/examples/secure-env-vars/.github/workflows/docker-github-secrets.yaml) +You can export sensitive values in the [`custom-entrypoint.sh.tftpl`](custom-entrypoint.sh.tftpl) script, which will be executed as the container entrypoint. +This strategy allow us to use the base Atlantis image, and to export environment variables that do not appear in the Console (e.g. ATLANTIS_GH_WEBHOOK_SECRET). + ### Setting non-sensitive environment variables Use the `var.env_vars` variable to set non-sensitive environment variables. diff --git a/examples/secure-env-vars/cloudbuild.yaml b/examples/secure-env-vars/cloudbuild.yaml deleted file mode 100644 index 83a1dc5..0000000 --- a/examples/secure-env-vars/cloudbuild.yaml +++ /dev/null @@ -1,23 +0,0 @@ -steps: - - name: "gcr.io/cloud-builders/docker" - entrypoint: "bash" - args: - [ - "-c", - "docker build -t ${_IMAGE} . --build-arg ATLANTIS_GH_USER=$$ATLANTIS_GH_USER --build-arg ATLANTIS_GH_TOKEN=$$ATLANTIS_GH_TOKEN --build-arg ATLANTIS_GH_WEBHOOK_SECRET=$$ATLANTIS_GH_WEBHOOK_SECRET", - ] - dir: "atlantis/gcp-secrets" - secretEnv: - ["ATLANTIS_GH_USER", "ATLANTIS_GH_TOKEN", "ATLANTIS_GH_WEBHOOK_SECRET"] -availableSecrets: - secretManager: - - versionName: projects/$PROJECT_ID/secrets/atlantis-gh-user/versions/latest - env: "ATLANTIS_GH_USER" - - versionName: projects/$PROJECT_ID/secrets/atlantis-gh-token/versions/latest - env: "ATLANTIS_GH_TOKEN" - - versionName: projects/$PROJECT_ID/secrets/atlantis-gh-webhook-secret/versions/latest - env: "ATLANTIS_GH_WEBHOOK_SECRET" - -images: ["${_IMAGE}"] -substitutions: - _IMAGE: "europe-docker.pkg.dev///:" diff --git a/examples/secure-env-vars/custom-entrypoint.sh.tftpl b/examples/secure-env-vars/custom-entrypoint.sh.tftpl new file mode 100644 index 0000000..352efa9 --- /dev/null +++ b/examples/secure-env-vars/custom-entrypoint.sh.tftpl @@ -0,0 +1,43 @@ +#!/bin/bash + +set -e + +mkdir -p ${mount_folder} +chown 100 ${mount_folder} +cat <<'EOF' > "${mount_folder}/${entrypoint_filename}" +#!/bin/bash + +set -e + +ARCH="x86_64" +apk --no-cache upgrade && apk --no-cache add \ + curl \ + python3 \ + py3-crcmod \ + py3-openssl \ + bash \ + libc6-compat \ + openssh-client \ + git \ + gnupg \ + && curl -O https://dl.google.com/dl/cloudsdk/channels/rapid/downloads/google-cloud-cli-${cloud_sdk_version}-linux-$${ARCH}.tar.gz && \ + tar xzf google-cloud-cli-${cloud_sdk_version}-linux-$${ARCH}.tar.gz && \ + rm google-cloud-cli-${cloud_sdk_version}-linux-$${ARCH}.tar.gz + +export PATH=$PATH:/google-cloud-sdk/bin +gcloud config set core/disable_usage_reporting true +gcloud config set component_manager/disable_update_check true +gcloud config set metrics/environment github_docker_image +gcloud --version + +gcloud secrets versions access latest --secret="${app_key_secret_name}" > "${key_file_path}" && chmod 400 "${key_file_path}" && chown 100 "${key_file_path}" + +export ATLANTIS_GH_APP_ID=$(gcloud secrets versions access latest --secret="${app_id_secret_name}") +export ATLANTIS_GH_APP_KEY_FILE="${key_file_path}" +export ATLANTIS_GH_WEBHOOK_SECRET=$(gcloud secrets versions access latest --secret="${webhook_secret_secret_name}") + +# Call original atlantis entrypoint, passing along all arguments +docker-entrypoint.sh "$@" +EOF + +chmod 0755 "${mount_folder}/${entrypoint_filename}" diff --git a/examples/secure-env-vars/main.tf b/examples/secure-env-vars/main.tf index 516e15d..e72520c 100644 --- a/examples/secure-env-vars/main.tf +++ b/examples/secure-env-vars/main.tf @@ -4,11 +4,16 @@ locals { subnetwork = "" region = "" zone = "" - image = "" domain = "" managed_zone = "" github_repo_allow_list = "github.com/example/*" + + secret_names = { + app_id = "" + app_key = "" + webhook = "" + } } # Create a service account and attach the required Cloud Logging permissions to it. @@ -33,7 +38,6 @@ resource "google_project_iam_member" "atlantis_metric_writer" { module "atlantis" { source = "runatlantis/atlantis/gce" name = "atlantis" - image = local.image # Your wrapper Atlantis Docker image network = local.network subnetwork = local.subnetwork region = local.region @@ -42,14 +46,29 @@ module "atlantis" { email = google_service_account.atlantis.email scopes = ["cloud-platform"] } - # Declare the non-sensitive environment variables here - # The sensitive environment variables are set in the Dockerfile! + env_vars = { - ATLANTIS_REPO_ALLOWLIST = local.github_repo_allow_list - ATLANTIS_ATLANTIS_URL = "https://${local.domain}" + ATLANTIS_REPO_ALLOWLIST = local.github_repo_allow_list + ATLANTIS_ATLANTIS_URL = "https://${local.domain}" + ATLANTIS_REPO_CONFIG_JSON = jsonencode(yamldecode(file("${path.module}/server-atlantis.yaml"))) + ATLANTIS_WRITE_GIT_CREDS = "true" } domain = local.domain project = local.project_id + + image = "ghcr.io/runatlantis/atlantis:latest" + command = ["/home/atlantis/custom-entrypoint.sh"] + args = ["server"] + + startup_script = templatefile("${path.module}/custom-entrypoint.sh.tftpl", { + cloud_sdk_version = "455.0.0" + app_key_secret_name = local.secret_names.app_key + app_id_secret_name = local.secret_names.app_id + webhook_secret_secret_name = local.secret_names.webhook + key_file_path = "/home/atlantis/gh_app_key.pem" + mount_folder = "/mnt/disks/gce-containers-mounts/gce-persistent-disks/atlantis-disk-0" + entrypoint_filename = "custom-entrypoint.sh" + }) } # As your DNS records might be managed at another registrar's site, we create the DNS record outside of the module. diff --git a/examples/secure-env-vars/server-atlantis.yaml b/examples/secure-env-vars/server-atlantis.yaml new file mode 100644 index 0000000..2005c3a --- /dev/null +++ b/examples/secure-env-vars/server-atlantis.yaml @@ -0,0 +1,7 @@ +repos: +- id: /.*/ + apply_requirements: [mergeable] + allowed_overrides: [apply_requirements, workflow] + allow_custom_workflows: true + delete_source_branch_on_merge: true + \ No newline at end of file