Terraform: String Manipulation for Default Domains


Imaging when I needed to grant access to a Google storage bucket for a list of Google service accounts, I put the list in YAML like this:

# data/gcs.yaml
google-service-accounts:
  - gaia@horizon.iam.gserviceaccount.com
  - nova@horizon.iam.gserviceaccount.com
  - cyan@horizon.iam.gserviceaccount.com
  - edi@mass-effect.iam.gserviceaccount.com
  ...

Assuming horizon is the default GCP project and mass-effect is a partner project and most GSAs are of course from the default project. So can this list be DRYed up a bit like this?

# data/gcs.yaml
google-service-accounts:
  - gaia
  - nova
  - cyan
  - edi@mass-effect
  ...
default:
  project: horizon

The terraform code to read this YAML and grant IAM members could look like:

locals {
  manifest = yamldecode(file("./data/gcs.yaml"))
}

resource "google_storage_bucket" "default" {
  ...
}

resource "google_storage_bucket_iam_binding" "default_readers" {
  bucket  = google_storage_bucket.default.name
  role    = "roles/storage.objectViewer"
  members = formatlist(
    "serviceAccount:%s",
    [for member in local.manifest['google-service-accounts'] : format("%s@%s.iam.gserviceaccount.com", split("@", member)[0], try(split("@", member)[1], local.manifest.default.project))]
  )
}

A quick explanation on how this works:

# for item `gaia`
split("@", "gaia") -> ["gaia"]
split("@", "gaia")[0] -> "gaia"
split("@", "gaia")[1] -> error # doesn't exist
try(split("@", "gaia")[1], "horizon") -> "horizon"
format("%s@%s.iam.gserviceaccount.com", ...) -> "gaia@horizon.iam.gserviceaccount.com"

# for item edi@mass-effect
split("@", "edi@mass-effect") -> ["edi", "mass-effect"]
split("@", "edi@mass-effect")[0] -> "edi"
split("@", "edi@mass-effect")[1] -> "mass-effect"
try(split("@", "edi@mass-effect")[1], "horizon") -> "mass-effect"
format("%s@%s.iam.gserviceaccount.com", ...) -> "edi@mass-effect.iam.gserviceaccount.com"

PS. yes I like the Horizon and Mass Effect games. 🙂