Skip to content
🎉 Terragrunt v1.0 is here! Read the announcement to learn more.

Explicit Stacks

For an alternate approach (that is more flexible, but not necessarily always the better solution), you can define explicit stacks using terragrunt.stack.hcl files. These are blueprints that programmatically generate units at runtime.

A terragrunt.stack.hcl file is a blueprint that defines how to generate Terragrunt configurations programmatically. It tells Terragrunt:

  • What units to create.
  • Where to get their configurations from.
  • Where to place them in the directory structure.
  • What values to pass to each unit.

unit blocks - Define Individual Infrastructure Components

Section titled “unit blocks - Define Individual Infrastructure Components”
  • Purpose: Define a single, deployable piece of infrastructure.
  • Use case: When you want to create a single piece of isolated infrastructure (e.g. a specific VPC, database, or application).
  • Result: Generates a directory with a single terragrunt.hcl file in the specified path from the specified source.

stack blocks - Define Reusable Infrastructure Patterns

Section titled “stack blocks - Define Reusable Infrastructure Patterns”
  • Purpose: Define a stack of units to be deployed together.
  • Use case: When you have a common, multi-unit pattern (like “dev environment” or “three-tier web application”) that you want to deploy multiple times.
  • Result: Generates a directory with another terragrunt.stack.hcl file in the specified path from the specified source.
terragrunt.stack.hcl
unit "vpc" {
source = "git::git@github.com:acme/infrastructure-catalog.git//units/vpc?ref=v0.0.1"
path = "vpc"
values = {
vpc_name = "main"
cidr = "10.0.0.0/16"
}
}
unit "database" {
source = "git::git@github.com:acme/infrastructure-catalog.git//units/database?ref=v0.0.1"
path = "database"
values = {
engine = "postgres"
version = "13"
vpc_path = "../vpc"
}
}

Running terragrunt stack generate in the directory containing that terragrunt.stack.hcl file generates:

  • Directory.terragrunt-stack
    • Directoryvpc
      • terragrunt.hcl
      • terragrunt.values.hcl
    • Directorydatabase
      • terragrunt.hcl
      • terragrunt.values.hcl

Example: Nested Stack with Reusable Patterns

Section titled “Example: Nested Stack with Reusable Patterns”
terragrunt.stack.hcl
stack "dev" {
source = "git::git@github.com:acme/infrastructure-catalog.git//stacks/environment?ref=v0.0.1"
path = "dev"
values = {
environment = "development"
cidr = "10.0.0.0/16"
}
}
stack "prod" {
source = "git::git@github.com:acme/infrastructure-catalog.git//stacks/environment?ref=v0.0.1"
path = "prod"
values = {
environment = "production"
cidr = "10.1.0.0/16"
}
}

The referenced stack might contain:

stacks/environment/terragrunt.stack.hcl
unit "vpc" {
source = "git::git@github.com:acme/infrastructure-catalog.git//units/vpc?ref=v0.0.1"
path = "vpc"
values = {
vpc_name = values.environment
cidr = values.cidr
}
}
unit "database" {
source = "git::git@github.com:acme/infrastructure-catalog.git//units/database?ref=v0.0.1"
path = "database"
values = {
environment = values.environment
vpc_path = "../vpc"
}
}

Running terragrunt stack generate in the directory containing that terragrunt.stack.hcl file generates:

  • Directory.terragrunt-stack
    • Directorydev
      • terragrunt.stack.hcl
      • Directory.terragrunt-stack
        • Directoryvpc
          • terragrunt.hcl
          • terragrunt.values.hcl
        • Directorydatabase
          • terragrunt.hcl
          • terragrunt.values.hcl
    • Directoryprod
      • terragrunt.stack.hcl
      • Directory.terragrunt-stack
        • Directoryvpc
          • terragrunt.hcl
          • terragrunt.values.hcl
        • Directorydatabase
          • terragrunt.hcl
          • terragrunt.values.hcl
Terminal window
# Generate units from the `terragrunt.stack.hcl` file in the current
# working directory (and all stacks in child directories).
terragrunt stack generate
# Deploy all generated units defined using the `terragrunt.stack.hcl` file
# in the current working directory (and any units generated by stacks in this file).
#
# Note that this will also automatically generate the stack if it is not already generated.
terragrunt stack run apply
  • Reusability: Define patterns once, reuse them across environments.
  • Consistency: Ensure all environments follow the same structure.
  • Version Control: Version collections of infrastructure patterns alongside the units of infrastructure that make them up.
  • Automation: Generate complex infrastructure from simple blueprints.
  • Flexibility: Easy to create variations with different values.
  • Complexity: Requires understanding another configuration file.
  • Generation Overhead: Units must be generated before use.
  • Debugging: Generated files can be harder to debug if you accidentally generate files that are not what you intended.

When defining a stack using a terragrunt.stack.hcl file, you also have the ability to interact with the aggregated outputs of all the units in the stack from the CLI.

To do this, use the stack output command (not the stack run output command).

Terminal window
$ terragrunt stack output
backend_app = {
domain = "backend-app.example.com"
}
frontend_app = {
domain = "frontend-app.example.com"
}
mysql = {
endpoint = "terraform-20250504140737772400000001.abcdefghijkl.us-east-1.rds.amazonaws.com"
}
valkey = {
endpoint = "serverless-valkey-01.amazonaws.com"
}
vpc = {
vpc_id = "vpc-1234567890"
}

This returns a single aggregated HCL object aggregating all the outputs for all the units within the stack.

When using Explicit Stacks, you might want to use local state files instead of remote state for development, testing, or specific use cases. However, this presents a challenge because the generated .terragrunt-stack directory can be safely deleted and regenerated using terragrunt stack clean && terragrunt stack generate, which would normally cause local state files to be lost.

To solve this problem, you can configure your stack to store state files outside of the .terragrunt-stack directory, in a persistent location that survives stack regeneration.

Here’s how to configure local state that persists across stack regeneration:

1. Create a root.hcl file with local backend configuration:

root.hcl
remote_state {
backend = "local"
generate = {
path = "backend.tf"
if_exists = "overwrite_terragrunt"
}
config = {
path = "${get_parent_terragrunt_dir()}/.terragrunt-local-state/${path_relative_to_include()}/tofu.tfstate"
}
}

2. Create your stack definition:

live/terragrunt.stack.hcl
unit "vpc" {
source = "${find_in_parent_folders("units/vpc")}"
path = "vpc"
}
unit "database" {
source = "${find_in_parent_folders("units/database")}"
path = "database"
}
unit "app" {
source = "${find_in_parent_folders("units/app")}"
path = "app"
}

3. Configure your units to include the root configuration:

units/vpc/terragrunt.hcl
include "root" {
path = find_in_parent_folders("root.hcl")
}
terraform {
source = "."
}

4. Add a .gitignore file to exclude state files from version control:

.gitignore
.terragrunt-local-state

Important: Local state files should never be committed to version control as they may contain sensitive information and can cause conflicts when multiple developers work on the same infrastructure.

The key insight is using path_relative_to_include() in the state path configuration. This function returns the relative path from each unit to the root.hcl file, creating unique state file paths like:

.terragrunt-local-state/live/.terragrunt-stack/vpc/tofu.tfstate
.terragrunt-local-state/live/.terragrunt-stack/database/tofu.tfstate
.terragrunt-local-state/live/.terragrunt-stack/app/tofu.tfstate

Since these state files are stored in .terragrunt-local-state/ (outside of .terragrunt-stack/), they persist when you run:

Terminal window
terragrunt stack clean && terragrunt stack generate

After running the stack, your directory structure will look like this:

  • Directory.
    • root.hcl
    • .gitignore (Excludes .terragrunt-local-state)
    • Directory.terragrunt-local-state/ (Persistent state files - ignored by git)
      • Directorylive/
        • Directory.terragrunt-stack/
          • Directoryvpc/
            • tofu.tfstate
          • Directorydatabase/
            • tofu.tfstate
          • Directoryapp/
            • tofu.tfstate
    • Directorylive/
      • terragrunt.stack.hcl
      • Directory.terragrunt-stack/ (Generated stack - can be deleted)
        • Directoryvpc/
          • terragrunt.hcl
          • main.tf
        • Directorydatabase/
          • terragrunt.hcl
          • main.tf
        • Directoryapp/
          • terragrunt.hcl
          • main.tf
    • Directoryunits/ (Reusable unit definitions)
      • Directoryvpc/
      • Directorydatabase/
      • Directoryapp/

When authoring stacks in a catalog, you can use the update_source_with_cas attribute on nested blocks to replace remote Git URLs with relative paths.

Without CAS, catalog authors need to plumb remote Git URLs through values expressions so that generated files reference the correct remote sources:

stacks/service/terragrunt.stack.hcl
unit "service" {
source = "git::git@github.com:acme/catalog.git//units/service?ref=${values.version}"
path = "service"
values = {
version = values.version
}
}

This is error-prone and creates unnecessary diffs when the version changes. It also requires a network fetch for every nested source during generation.

With update_source_with_cas, you can use relative paths instead:

stacks/service/terragrunt.stack.hcl
unit "service" {
source = "../..//units/service"
update_source_with_cas = true
path = "service"
}

The // separator works the same as with any Terragrunt source. The part before // is the base path, and the part after // is the subdirectory.

The attribute can also be used in terragrunt.hcl files within the terraform block:

units/service/terragrunt.hcl
terraform {
source = "../..//modules/service"
update_source_with_cas = true
}

When update_source_with_cas = true is set:

  1. Terragrunt clones the catalog repository once via CAS and stores the content.
  2. It follows relative source paths within the cloned repository.
  3. At each level, it rewrites the source attribute to a cas:: reference (e.g., cas::sha1:abc123...) that points to stored CAS content.
  4. Generated .terragrunt-stack files contain deterministic CAS references instead of remote URLs.
  5. Subsequent generation waves resolve cas:: references directly from the CAS store, with no network access required.

Consumers do not set update_source_with_cas themselves. When the cas experiment is enabled and the source is remote, Terragrunt uses the CAS path automatically. The attribute only has effect inside catalog files, where it flags nested source attributes for rewriting.

Given the catalog files above, and an ordinary consumer stack like:

live/terragrunt.stack.hcl
stack "service" {
source = "git::git@github.com:acme/catalog.git//stacks/service?ref=v1.0.0"
path = "service"
}

Running terragrunt --experiment cas stack generate produces:

  • Directorylive/
    • terragrunt.stack.hcl
    • Directory.terragrunt-stack/
      • Directoryservice/
        • terragrunt.stack.hcl (sources rewritten to cas:: references)
        • Directory.terragrunt-stack/
          • Directoryservice/
            • terragrunt.hcl (terraform source rewritten to cas:: reference)

The generated stack file at .terragrunt-stack/service/terragrunt.stack.hcl contains the rewritten source:

.terragrunt-stack/service/terragrunt.stack.hcl
unit "service" {
source = "cas::sha1:a1b2c3d4..."
update_source_with_cas = true
path = "service"
}

And the generated unit file at .terragrunt-stack/service/.terragrunt-stack/service/terragrunt.hcl contains:

.terragrunt-stack/service/.terragrunt-stack/service/terragrunt.hcl
terraform {
source = "cas::sha1:e5f6a7b8...//modules/service"
update_source_with_cas = true
}

The first cas:: reference points to a synthetic tree in the CAS that contains the exact files needed for the unit. The second points to the full repository tree, with //modules/service telling Terragrunt which subdirectory to use as the OpenTofu/Terraform module source.

Requirements:

  • The cas experiment must be enabled (--experiment cas).
  • The --no-cas flag must not be set.
  • Sources using update_source_with_cas must be relative paths within the same repository.

The consumer stack’s source can be a local filesystem path in addition to a remote Git URL. Both go through the same CAS-backed rewrite pipeline:

live/terragrunt.stack.hcl
stack "service" {
source = "../catalog//stacks/service"
path = "service"
}

Terragrunt copies the referenced directory into a temporary directory, computes a content-addressed root hash over (relative path, mode, file-content hash) triples using SHA-256, and rewrites nested source attributes against that copy. The original directory is never modified, so the catalog on disk stays clean even though the rewritten source values need somewhere to live.

This is useful when iterating on a catalog before tagging a release: the same update_source_with_cas = true attributes in the catalog’s terragrunt.stack.hcl and terragrunt.hcl files apply unchanged, whether consumers pull the catalog over Git or point at a local checkout.

There are currently some known limitations with explicit stacks that you should be aware of as you start to adopt them.

The dependency block cannot set the value of the config_path attribute to that of a stack. This is functionality that is planned for the future, but is not currently supported.

As such, if you currently have multiple stacks that need to depend on each other, or on units within each other’s stacks, you will need to either use implicit stacks, or work around this limitation by setting the config_path attribute to the path of the unit within the stack, and carefully ensuring that all stacks are generated before any units are run.

Deeply nested stack generation can be slow

Section titled “Deeply nested stack generation can be slow”

Every generation of a stack from a terragrunt.stack.hcl file can potentially result in network traffic to fetch the source for the stack and filesystem traffic to copy the generated units to the .terragrunt-stack directory. This can result in slow stack generation if you have very deeply nested stacks.

To mitigate this, enable the cas experiment and use update_source_with_cas = true in your catalog’s stack and unit files. This deduplicates repository clones and resolves content from the local CAS store instead of fetching from the network on every generation. See CAS Integration for details.

Includes are not supported in terragrunt.stack.hcl files

Section titled “Includes are not supported in terragrunt.stack.hcl files”

The include block is not supported in terragrunt.stack.hcl files. This isn’t functionality that is planned for future implementation, but may change based on community feedback, and proven use-cases.

The current design of explicit stacks is that, when necessary, stacks can be nested into other stacks making them better organized and reusable without relying on includes to share configuration between stacks.