KEMBAR78
GitHub - OP5dev/TF-via-PR: Plan and apply Terraform/OpenTofu via PR automation, using best practices for secure and scalable IaC workflows.
Skip to content

Plan and apply Terraform/OpenTofu via PR automation, using best practices for secure and scalable IaC workflows.

License

Notifications You must be signed in to change notification settings

OP5dev/TF-via-PR

Use this GitHub action with your project
Add this Action to an existing workflow or create a new one
View on Marketplace

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 

Terraform Compatible OpenTofu Compatible * GitHub license GitHub release tag * GitHub repository stargazers

Terraform/OpenTofu via Pull Request (TF-via-PR)

What does it do?

Who is it for?

  • Plan and apply changes with CLI arguments and encrypted plan file to avoid configuration drift.
  • Outline diff within up-to-date PR comment and matrix-friendly workflow summary, complete with log.
  • DevOps and Platform engineers wanting to empower their teams to self-service scalably.
  • Maintainers looking to secure their pipeline without the overhead of containers or VMs.

PR comment of plan output with "Diff of changes" section expanded.


Usage Examples

How to get started?

on:
  pull_request:
  push:
    branches: [main]

jobs:
  provision:
    runs-on: ubuntu-latest

    permissions:
      actions: read        # Required to identify workflow run.
      checks: write        # Required to add status summary.
      contents: read       # Required to checkout repository.
      pull-requests: write # Required to add PR comment.

    steps:
      - uses: actions/checkout@v5

      - uses: hashicorp/setup-terraform@v3
        with:
          terraform_wrapper: false

      # Run plan by default, or apply on merge.
      - uses: op5dev/tf-via-pr@v13
        with:
          working-directory: path/to/directory
          command: ${{ github.event_name == 'push' && 'apply' || 'plan' }}
          arg-lock: ${{ github.event_name == 'push' }}
          arg-backend-config: env/dev.tfbackend
          arg-var-file: env/dev.tfvars
          arg-workspace: dev-use1
          plan-encrypt: ${{ secrets.PASSPHRASE }}

Tip

  • All supported arguments (e.g., -backend-config, -destroy, -parallelism, etc.) are listed below.
  • Environment variables can be passed in for cloud platform authentication (e.g., configure-aws-credentials for short-lived credentials via OIDC).
  • Recommend setting terraform_wrapper/tofu_wrapper to false in order to output the detailed exit code for better error handling.

Where to find more examples?

The following workflows showcase common use cases, while a comprehensive list of inputs is documented below.

Runs on pull_request (plan) and push (apply) events with Terraform, AWS authentication and cache.


Runs on pull_request (plan) and merge_group (apply) events with OpenTofu in matrix strategy.


Runs on pull_request (plan) and push (apply) events with fmt/validate checks and TFLint.


Runs on pull_request (plan) and push (apply) events with conditional jobs based on plan file.


Runs on labeled and workflow_dispatch manual events on GitHub Enterprise (GHE) self-hosted runner.


Runs on schedule cron event with -refresh-only to open an issue on configuration drift.



Are there online references?

  1. Medium - Enhance Terraform/Tofu PR Automation with GitHub Action
  2. Medium - Secure cloud provisioning pipeline with GitHub automation
  3. Medium - The Case for Terraform/Tofu Merge Queue
  4. Terraform Weekly - Issue #207
  5. Awesome OpenTofu - CI Tools
  6. LinkedIn - GitOps with Terraform

How does encryption work?

Before the workflow uploads the plan file as an artifact, it can be encrypted-at-rest with a passphrase using plan-encrypt input to prevent exposure of sensitive data (e.g., ${{ secrets.PASSPHRASE }}). This is done with OpenSSL's symmetric stream counter mode (256 bit AES in CTR) encryption with salt and pbkdf2.

In order to decrypt the plan file locally, use the following commands after downloading the artifact (adding a whitespace before openssl to prevent recording the command in shell history):

unzip <tfplan.zip>
openssl enc -d -aes-256-ctr -pbkdf2 -salt \
  -in   tfplan.encrypted \
  -out  tfplan.decrypted \
  -pass pass:"<passphrase>"
<tf.tool> show tfplan.decrypted

Inputs

All supported CLI argument inputs are listed below with accompanying options, while workflow configuration inputs are listed here.

Configuration

Type Name Description
CLI working-directory Specify the working directory of TF code, alias of arg-chdir.
Example: path/to/directory
CLI command Command to run between: plan or apply.1
Example: plan
CLI tool Provisioning tool to use between: terraform or tofu.
Default: terraform
CLI plan-file Supply existing plan file path instead of the auto-generated one.
Example: path/to/file.tfplan
CLI pr-number Specify PR number in case of unsupported workflow trigger.
Example: 123
Check format Check format of TF code.
Default: false
Check validate Check validation of TF code.
Default: false
Check plan-parity Replace plan file if it matches a newly-generated one to prevent stale apply.2
Default: false
Security plan-encrypt Encrypt plan file artifact with the given input.3
Example: ${{ secrets.PASSPHRASE }}
Security preserve-plan Preserve plan file "tfplan" in the given working directory after workflow execution.
Default: false
Security upload-plan Upload plan file as GitHub workflow artifact.
Default: true
Security retention-days Duration after which plan file artifact will expire in days.
Example: 90
Security token Specify a GitHub token.
Default: ${{ github.token }}
UI expand-diff Expand the collapsible diff section.
Default: false
UI expand-summary Expand the collapsible summary section.
Default: false
UI comment-pr Add a PR comment: always, on-diff, or never.4
Default: always
UI comment-method PR comment by: update existing comment or recreate and delete previous one.5
Default: update
UI tag-actor Tag the workflow triggering actor: always, on-diff, or never.4
Default: always
UI hide-args Hide comma-separated list of CLI arguments from the command input.6
Default: detailed-exitcode,parallelism,lock,out,var=
UI show-args Show comma-separated list of CLI arguments in the command input.6
Default: workspace

  1. Both command: plan and command: apply include: init, fmt (with format: true), validate (with validate: true), and workspace (with arg-workspace) commands rolled into it automatically.
    To separately run checks and/or generate outputs only, command: init can be used.

  2. Originally intended for merge_group event trigger, plan-parity: true input helps to prevent stale apply within a series of workflow runs when merging multiple PRs.

  3. The secret string input for plan-encrypt can be of any length, as long as it's consistent between encryption (plan) and decryption (apply).

  4. The on-diff option is true when the exit code of the last TF command is non-zero (ensure terraform_wrapper/tofu_wrapper is set to false).

  5. The default behavior of comment-method is to update the existing PR comment with the latest plan/apply output, making it easy to track changes over time through the comment's revision history.

    PR comment revision history comparing plan and apply outputs.

  6. It can be desirable to hide certain arguments from the last run command input to prevent exposure in the PR comment (e.g., sensitive arg-var values). Conversely, it can be desirable to show other arguments even if they are not in last run command input (e.g., arg-workspace or arg-backend-config selection).

Arguments

Note

  • Arguments are passed to the appropriate TF command(s) automatically, whether that's fmt, init, validate, plan, or apply.
  • For repeated arguments like arg-var, arg-var-file, arg-backend-config, arg-replace and arg-target, use commas to separate multiple values (e.g., arg-var: key1=value1,key2=value2).

Applicable to respective "plan" and "apply" command inputs ("init" included).

Name CLI Argument
arg-auto-approve -auto-approve
arg-backend-config -backend-config
arg-backend -backend
arg-backup -backup
arg-chdir -chdir
Alias: working-directory
arg-compact-warnings -compact-warnings
arg-concise -concise
arg-destroy -destroy
arg-detailed-exitcode -detailed-exitcode
Default: true
arg-force-copy -force-copy
arg-from-module -from-module
arg-generate-config-out -generate-config-out
arg-get -get
arg-lock-timeout -lock-timeout
arg-lock -lock
arg-lockfile -lockfile
arg-migrate-state -migrate-state
arg-parallelism -parallelism
arg-plugin-dir -plugin-dir
arg-reconfigure -reconfigure
arg-refresh-only -refresh-only
arg-refresh -refresh
arg-replace -replace
arg-state-out -state-out
arg-state -state
arg-target -target
arg-upgrade -upgrade
arg-var-file -var-file
arg-var -var
arg-workspace -workspace
Alias: TF_WORKSPACE

Applicable only when format: true.

Name CLI Argument
arg-check -check
Default: true
arg-diff -diff
Default: true
arg-list -list
arg-recursive -recursive
Default: true
arg-write -write

Applicable only when validate: true.

Name CLI Argument
arg-no-tests -no-tests
arg-test-directory -test-directory

Outputs

Type Name Description
Artifact plan-id ID of the plan file artifact.
Artifact plan-url URL of the plan file artifact.
CLI command Input of the last TF command.
CLI diff Diff of changes, if present (truncated).
CLI exitcode Exit code of the last TF command.
CLI result Result of the last TF command (truncated).
CLI summary Summary of the last TF command.
Workflow check-id ID of the check run.
Workflow comment-body Body of the PR comment.
Workflow comment-id ID of the PR comment.
Workflow job-id ID of the workflow job.
Workflow run-url URL of the workflow run.
Workflow identifier Unique name of the workflow run and artifact.

Security

View security policy and reporting instructions. Integrating security in your CI/CD pipeline is critical to practicing DevSecOps. This action aims to be secure by default, and it should be complemented with your own review to ensure it meets your (organization's) security requirements.

Tip

Pin your GitHub Action to a commit SHA to harden your CI/CD pipeline security against supply chain attacks.


Changelog

View all notable changes to this project in Keep a Changelog format, which adheres to Semantic Versioning.

Tip

All forms of contribution are very welcome and deeply appreciated for fostering open-source projects.


To-Do

  • Handling of inputs which contain space(s) (e.g., working-directory: path to/directory).
  • Handling of comma-separated inputs which contain comma(s) (e.g., arg-var: token=1,2,3)—use TF_CLI_ARGS workaround.

License

About

Plan and apply Terraform/OpenTofu via PR automation, using best practices for secure and scalable IaC workflows.

Topics

Resources

License

Code of conduct

Contributing

Security policy

Stars

Watchers

Forks

Sponsor this project

  •  
  •  

Contributors 10