name: render-env description: Add a new Terraform Cloud variable for the Render-hosted backend across production, sandbox, and test. Declares the tfe_variable in terraform/global/{production,sandbox,test}.tf and the matching variable {} block in terraform/{production,sandbox,test}/variables.tf, then reminds the user to wire it into render.tf / the render_service module. user-invocable: true allowed-tools: Read Edit Write Bash Grep Glob
Add a Render env variable to Terraform
This skill declares a new Terraform Cloud variable so a value can be set via the TFC UI and consumed by the Render backend services. It only does the plumbing — it does not wire the variable into a Render env group. Wiring requires picking the right *_config / *_secrets object in terraform/modules/render_service/, which is task-specific and the user should drive.
Inputs
The skill takes two positional args from the invocation: /render-env <name> <description>.
${name}— the Terraform variable name in snake_case. Used verbatim as the TFC variablekey, thetfe_variableresource suffix, and thevariableblock name. Example:stripe_climate_api_key.${description}— a short human-readable description. Used in thedescriptionattribute and (with the env appended) in the per-env tfe_variable description. Example:Stripe Climate API key.
If either arg is missing, ask the user before doing anything. Also ask:
- Sensitive? Default to
true(almost everything in these files is sensitive). Only setfalsefor non-secret config strings (cf.slo_report_slack_channel,customer_portal_url_overrides). - Which environments? Default to all three (
production,sandbox,test). The user may want to skip one —testin particular often omits variables that aren't exercised there.
Do not ask about lifecycle { ignore_changes = [value] }. New variables here have no value baked into the Terraform code, so there's nothing for Terraform to overwrite — TFC just holds whatever's typed into the UI, and ignore_changes would be a no-op. The handful of existing blocks that include it (e.g. polar_organization_id, customer_portal_url_overrides) seed a default value in code and want UI overrides to stick; that's a different shape from a fresh secret and the user will tell you up-front if they want it.
Naming convention
Follow the modern bare-key pattern that recent additions use (e.g. polar_access_token, tinybird_api_token, customer_portal_url_overrides):
- The TFC
keyis the bare name:key = "${name}"— no_production/_sandbox/_testsuffix on the key. Each variable set is per-workspace so the key doesn't need to be globally unique. - The
tfe_variableresource label does get the env suffix:resource "tfe_variable" "${name}_${env}". - The matching
variableblock interraform/${env}/variables.tfuses the bare name:variable "${name}". This letsrender.tfreference it uniformly asvar.${name}across all envs.
A handful of older variables (e.g. google_client_id_production, backend_secret_production) use a _production/_sandbox-suffixed key and matching variable. Don't replicate that pattern for new additions — it's legacy.
Step 1: Add the tfe_variable to each global/{env}.tf
For each selected ${env} in production, sandbox, test, append a block to terraform/global/${env}.tf (after the existing tfe_variable resources, before the file ends):
resource "tfe_variable" "${name}_${env}" {
key = "${name}"
category = "terraform"
description = "${description} for ${env}"
sensitive = ${sensitive}
variable_set_id = tfe_variable_set.${env}.id
}
Only add a lifecycle { ignore_changes = [value] } block or a value = "..." line if the user explicitly asks for one (rare — usually let TFC hold the value).
Use Edit with enough surrounding context that the insertion lands at the correct spot. Prefer appending after the last existing tfe_variable resource in the file rather than rewriting the file.
Step 2: Add the variable {} block to each {env}/variables.tf
For each selected ${env}, append a block to terraform/${env}/variables.tf:
variable "${name}" {
description = "${description}"
type = string
sensitive = ${sensitive}
}
Drop the sensitive = true line when ${sensitive} is false. Drop nothing else.
Step 3: Format
Run:
terraform fmt -recursive terraform
from the repo root. If terraform isn't on PATH, note it and skip — the formatting is a nicety, not required.
Step 4: Hand off
Report to the user:
- The six files touched (or fewer if they skipped an env).
- That the variable is now declared but not yet consumed. To consume it, they need to:
- Add a field to the relevant config/secrets object in
terraform/modules/render_service/variables.tf. Pick by purpose:backend_config— non-sensitive backend env vars (URLs, flags, log level, tax processor list).backend_secrets— sensitive backend env vars (API keys, tokens, signing secrets).- Themed
render_env_groupblocks (stripe,github,logfire,tinybird,aws_s3,apple,prometheus,slo_report,google,openai, etc.) each have their own object — use the matching one when the var belongs to a clear bucket. - Use
optional(string, "<default>")if you want a module-level default; otherwise plainstring.
- Wire the field into the matching
render_env_groupblock interraform/modules/render_service/main.tfasPOLAR_${NAME_UPPER} = { value = var.<object>.<field> }. Two backend groups exist:render_env_group "backend"— applied to every environment.render_env_group "backend_production"— production-only values (e.g.POLAR_BACKOFFICE_HOST,POLAR_PLAIN_TOKEN). Put a var here when sandbox/test should not see it.
- Pass the value in from each
terraform/${env}/render.tfmodule call, e.g.backend_secrets = { ... ${field} = var.${name} ... }. Sandbox and test won't have this line if the var is production-only. - Set the actual value in TFC under the matching variable set (Production / Sandbox / Test).
- If this is a
POLAR_*env var, also add the field to theSettingsclass inserver/polar/config.py(PydanticBaseSettingswithenv_prefix="polar_"; the env var name isPOLAR_<FIELD_NAME>).
- Add a field to the relevant config/secrets object in
Hardcoded string vs tfe_variable
Choose the right shape up-front:
- Hardcoded in
render.tf(e.g.tax_processors = "[\"stripe\"]"): use when the value is static and you're fine editing + PR'ing terraform to change it. tfe_variablevia this skill: use when the value is a secret or needs to be editable from the TFC UI without a code deploy. Don't hardcode a"{}"/""default inrender.tffor something that's supposed to be UI-tunable — it defeats the point.
Don't
- Don't write the variable into
terraform/global/main.tf(the cross-org "Global Settings" set). Per-env sets inglobal/{env}.tfshadow it, so an entry inmain.tfis dead weight when there's already a per-env one. - Don't hardcode a
value = "..."unless the user asks. The point of atfe_variableis that it can be set in the TFC UI. - Don't try to wire the variable into
render.tfor therender_servicemodule yourself — that's a structural decision (which secrets object? new object?) the user should make. - Don't
git addor commit. Leave the changes staged for the user to review.