diff --git a/cli/app.go b/cli/app.go
index 5c5ef060c..5fcb4e53b 100644
--- a/cli/app.go
+++ b/cli/app.go
@@ -343,6 +343,21 @@ func initialSetup(cliCtx *cli.Context, opts *options.TerragruntOptions) error {
opts.ExcludeByDefault = true
}
+ if !opts.ExcludeByDefault && len(opts.ModulesThatInclude) > 0 {
+ opts.Logger.Debugf("Modules that include set. Excluding by default.")
+ opts.ExcludeByDefault = true
+ }
+
+ if !opts.ExcludeByDefault && len(opts.UnitsReading) > 0 {
+ opts.Logger.Debugf("Units that read set. Excluding by default.")
+ opts.ExcludeByDefault = true
+ }
+
+ if !opts.ExcludeByDefault && opts.StrictInclude {
+ opts.Logger.Debugf("Strict include set. Excluding by default.")
+ opts.ExcludeByDefault = true
+ }
+
opts.IncludeDirs, err = util.GlobCanonicalPath(opts.WorkingDir, opts.IncludeDirs...)
if err != nil {
return err
diff --git a/cli/commands/flags.go b/cli/commands/flags.go
index 2b2fe48b5..56fc85567 100644
--- a/cli/commands/flags.go
+++ b/cli/commands/flags.go
@@ -67,30 +67,6 @@ const (
TerragruntIAMWebIdentityTokenFlagName = "terragrunt-iam-web-identity-token"
TerragruntIAMWebIdentityTokenEnvName = "TERRAGRUNT_IAM_ASSUME_ROLE_WEB_IDENTITY_TOKEN"
- TerragruntIgnoreDependencyErrorsFlagName = "terragrunt-ignore-dependency-errors"
- TerragruntIgnoreDependencyErrorsEnvName = "TERRAGRUNT_IGNORE_DEPENDENCY_ERRORS"
-
- TerragruntIgnoreDependencyOrderFlagName = "terragrunt-ignore-dependency-order"
- TerragruntIgnoreDependencyOrderEnvName = "TERRAGRUNT_IGNORE_DEPENDENCY_ORDER"
-
- TerragruntIgnoreExternalDependenciesFlagName = "terragrunt-ignore-external-dependencies"
- TerragruntIgnoreExternalDependenciesEnvName = "TERRAGRUNT_IGNORE_EXTERNAL_DEPENDENCIES"
-
- TerragruntIncludeExternalDependenciesFlagName = "terragrunt-include-external-dependencies"
- TerragruntIncludeExternalDependenciesEnvName = "TERRAGRUNT_INCLUDE_EXTERNAL_DEPENDENCIES"
-
- TerragruntExcludesFileFlagName = "terragrunt-excludes-file"
- TerragruntExcludesFileEnvName = "TERRAGRUNT_EXCLUDES_FILE"
-
- TerragruntExcludeDirFlagName = "terragrunt-exclude-dir"
- TerragruntExcludeDirEnvName = "TERRAGRUNT_EXCLUDE_DIR"
-
- TerragruntIncludeDirFlagName = "terragrunt-include-dir"
- TerragruntIncludeDirEnvName = "TERRAGRUNT_INCLUDE_DIR"
-
- TerragruntStrictIncludeFlagName = "terragrunt-strict-include"
- TerragruntStrictIncludeEnvName = "TERRAGRUNT_STRICT_INCLUDE"
-
TerragruntParallelismFlagName = "terragrunt-parallelism"
TerragruntParallelismEnvName = "TERRAGRUNT_PARALLELISM"
@@ -130,6 +106,35 @@ const (
TerragruntNoDestroyDependenciesCheckFlagEnvName = "TERRAGRUNT_NO_DESTROY_DEPENDENCIES_CHECK"
TerragruntNoDestroyDependenciesCheckFlagName = "terragrunt-no-destroy-dependencies-check"
+ // Queue related flags
+
+ TerragruntIgnoreDependencyErrorsFlagName = "terragrunt-ignore-dependency-errors"
+ TerragruntIgnoreDependencyErrorsEnvName = "TERRAGRUNT_IGNORE_DEPENDENCY_ERRORS"
+
+ TerragruntIgnoreDependencyOrderFlagName = "terragrunt-ignore-dependency-order"
+ TerragruntIgnoreDependencyOrderEnvName = "TERRAGRUNT_IGNORE_DEPENDENCY_ORDER"
+
+ TerragruntIgnoreExternalDependenciesFlagName = "terragrunt-ignore-external-dependencies"
+ TerragruntIgnoreExternalDependenciesEnvName = "TERRAGRUNT_IGNORE_EXTERNAL_DEPENDENCIES"
+
+ TerragruntIncludeExternalDependenciesFlagName = "terragrunt-include-external-dependencies"
+ TerragruntIncludeExternalDependenciesEnvName = "TERRAGRUNT_INCLUDE_EXTERNAL_DEPENDENCIES"
+
+ TerragruntExcludesFileFlagName = "terragrunt-excludes-file"
+ TerragruntExcludesFileEnvName = "TERRAGRUNT_EXCLUDES_FILE"
+
+ TerragruntExcludeDirFlagName = "terragrunt-exclude-dir"
+ TerragruntExcludeDirEnvName = "TERRAGRUNT_EXCLUDE_DIR"
+
+ TerragruntIncludeDirFlagName = "terragrunt-include-dir"
+ TerragruntIncludeDirEnvName = "TERRAGRUNT_INCLUDE_DIR"
+
+ TerragruntStrictIncludeFlagName = "terragrunt-strict-include"
+ TerragruntStrictIncludeEnvName = "TERRAGRUNT_STRICT_INCLUDE"
+
+ TerragruntUnitsReadingFlagName = "terragrunt-queue-include-units-reading"
+ TerragruntUnitsReadingEnvName = "TERRAGRUNT_QUEUE_INCLUDE_UNITS_READING"
+
// Logs related flags/envs
TerragruntLogLevelFlagName = "terragrunt-log-level"
@@ -453,6 +458,12 @@ func NewGlobalFlags(opts *options.TerragruntOptions) cli.Flags {
Destination: &opts.ModulesThatInclude,
Usage: "If flag is set, 'run-all' will only run the command against Terragrunt modules that include the specified file.",
},
+ &cli.SliceFlag[string]{
+ Name: TerragruntUnitsReadingFlagName,
+ EnvVar: TerragruntUnitsReadingEnvName,
+ Destination: &opts.UnitsReading,
+ Usage: "If flag is set, 'run-all' will only run the command against Terragrunt units that read the specified file via an HCL function.",
+ },
&cli.BoolFlag{
Name: TerragruntFailOnStateBucketCreationFlagName,
EnvVar: TerragruntFailOnStateBucketCreationEnvName,
diff --git a/config/config_helpers.go b/config/config_helpers.go
index 08f0bafc6..666217b26 100644
--- a/config/config_helpers.go
+++ b/config/config_helpers.go
@@ -68,6 +68,7 @@ const (
FuncNameEndsWith = "endswith"
FuncNameStrContains = "strcontains"
FuncNameTimeCmp = "timecmp"
+ FuncNameMarkAsRead = "mark_as_read"
sopsCacheName = "sopsCache"
)
@@ -167,6 +168,7 @@ func createTerragruntEvalContext(ctx *ParsingContext, configPath string) (*hcl.E
FuncNameGetDefaultRetryableErrors: wrapVoidToStringSliceAsFuncImpl(ctx, getDefaultRetryableErrors),
FuncNameReadTfvarsFile: wrapStringSliceToStringAsFuncImpl(ctx, readTFVarsFile),
FuncNameGetWorkingDir: wrapVoidToStringAsFuncImpl(ctx, getWorkingDir),
+ FuncNameMarkAsRead: wrapStringSliceToStringAsFuncImpl(ctx, markAsRead),
// Map with HCL functions introduced in Terraform after v0.15.3, since upgrade to a later version is not supported
// https://github.com/gruntwork-io/terragrunt/blob/master/go.mod#L22
@@ -627,12 +629,25 @@ func ParseTerragruntConfig(ctx *ParsingContext, configPath string, defaultVal *c
targetConfig := getCleanedTargetConfigPath(configPath, ctx.TerragruntOptions.TerragruntConfigPath)
targetConfigFileExists := util.FileExists(targetConfig)
+
if !targetConfigFileExists && defaultVal == nil {
return cty.NilVal, errors.New(TerragruntConfigNotFoundError{Path: targetConfig})
- } else if !targetConfigFileExists {
+ }
+
+ if !targetConfigFileExists {
return *defaultVal, nil
}
+ path, err := util.CanonicalPath(targetConfig, ctx.TerragruntOptions.WorkingDir)
+ if err != nil {
+ return cty.NilVal, errors.New(err)
+ }
+
+ ctx.TerragruntOptions.AppendReadFile(
+ path,
+ ctx.TerragruntOptions.WorkingDir,
+ )
+
// We update the ctx of terragruntOptions to the config being read in.
opts, err := ctx.TerragruntOptions.Clone(targetConfig)
if err != nil {
@@ -812,6 +827,11 @@ func sopsDecryptFile(ctx *ParsingContext, params []string) (string, error) {
return "", errors.New(err)
}
+ ctx.TerragruntOptions.AppendReadFile(
+ canonicalSourceFile,
+ ctx.TerragruntOptions.WorkingDir,
+ )
+
// Set environment variables from the TerragruntOptions.Env map.
// This is especially useful for integrations with things like the `terragrunt-auth-provider` flag,
// which can set environment variables that are used for decryption.
@@ -1008,6 +1028,11 @@ func readTFVarsFile(ctx *ParsingContext, args []string) (string, error) {
return "", errors.New(TFVarFileNotFoundError{File: varFile})
}
+ ctx.TerragruntOptions.AppendReadFile(
+ varFile,
+ ctx.TerragruntOptions.WorkingDir,
+ )
+
fileContents, err := os.ReadFile(varFile)
if err != nil {
return "", errors.New(fmt.Errorf("could not read file %q: %w", varFile, err))
@@ -1036,6 +1061,33 @@ func readTFVarsFile(ctx *ParsingContext, args []string) (string, error) {
return string(data), nil
}
+// markAsRead marks a file as explicitly read. This is useful for detection via TerragruntUnitsReading flag.
+func markAsRead(ctx *ParsingContext, args []string) (string, error) {
+ if len(args) != 1 {
+ return "", errors.New(WrongNumberOfParamsError{Func: "mark_as_read", Expected: "1", Actual: len(args)})
+ }
+
+ file := args[0]
+
+ path, err := util.CanonicalPath(file, ctx.TerragruntOptions.WorkingDir)
+ if err != nil {
+ return "", errors.New(err)
+ }
+
+ ctx.TerragruntOptions.AppendReadFile(
+ path,
+ ctx.TerragruntOptions.WorkingDir,
+ )
+
+ return file, nil
+}
+
+// warnWhenFileNotMarkedAsRead warns when a file is not being marked as read, even though a user might expect it to be.
+// Situations where this is the case include:
+// - A user specifies a file in the UnitsReading flag and that file is being read while parsing the inputs attribute.
+//
+// When the file is not marked as read, the function will return true, otherwise false.
+
// ParseAndDecodeVarFile uses the HCL2 file to parse the given varfile string into an HCL file body, and then decode it
// into the provided output.
func ParseAndDecodeVarFile(opts *options.TerragruntOptions, varFile string, fileContents []byte, out interface{}) error {
diff --git a/configstack/module.go b/configstack/module.go
index 2bf4f7176..ba845d5a6 100644
--- a/configstack/module.go
+++ b/configstack/module.go
@@ -96,11 +96,11 @@ func (module *TerraformModule) checkForCyclesUsingDepthFirstSearch(visitedPaths
}
// planFile - return plan file location, if output folder is set
-func (module *TerraformModule) planFile(terragruntOptions *options.TerragruntOptions) string {
+func (module *TerraformModule) planFile(opts *options.TerragruntOptions) string {
var planFile string
// set plan file location if output folder is set
- planFile = module.outputFile(terragruntOptions)
+ planFile = module.outputFile(opts)
planCommand := module.TerragruntOptions.TerraformCommand == terraform.CommandNamePlan || module.TerragruntOptions.TerraformCommand == terraform.CommandNameShow
@@ -158,26 +158,26 @@ func (module *TerraformModule) findModuleInPath(targetDirs []string) bool {
// Note that we skip the prompt for `run-all destroy` calls. Given the destructive and irreversible nature of destroy, we don't
// want to provide any risk to the user of accidentally destroying an external dependency unless explicitly included
// with the --terragrunt-include-external-dependencies or --terragrunt-include-dir flags.
-func (module *TerraformModule) confirmShouldApplyExternalDependency(ctx context.Context, dependency *TerraformModule, terragruntOptions *options.TerragruntOptions) (bool, error) {
- if terragruntOptions.IncludeExternalDependencies {
- terragruntOptions.Logger.Debugf("The --terragrunt-include-external-dependencies flag is set, so automatically including all external dependencies, and will run this command against module %s, which is a dependency of module %s.", dependency.Path, module.Path)
+func (module *TerraformModule) confirmShouldApplyExternalDependency(ctx context.Context, dependency *TerraformModule, opts *options.TerragruntOptions) (bool, error) {
+ if opts.IncludeExternalDependencies {
+ opts.Logger.Debugf("The --terragrunt-include-external-dependencies flag is set, so automatically including all external dependencies, and will run this command against module %s, which is a dependency of module %s.", dependency.Path, module.Path)
return true, nil
}
- if terragruntOptions.NonInteractive {
- terragruntOptions.Logger.Debugf("The --non-interactive flag is set. To avoid accidentally affecting external dependencies with a run-all command, will not run this command against module %s, which is a dependency of module %s.", dependency.Path, module.Path)
+ if opts.NonInteractive {
+ opts.Logger.Debugf("The --non-interactive flag is set. To avoid accidentally affecting external dependencies with a run-all command, will not run this command against module %s, which is a dependency of module %s.", dependency.Path, module.Path)
return false, nil
}
- stackCmd := terragruntOptions.TerraformCommand
+ stackCmd := opts.TerraformCommand
if stackCmd == "destroy" {
- terragruntOptions.Logger.Debugf("run-all command called with destroy. To avoid accidentally having destructive effects on external dependencies with run-all command, will not run this command against module %s, which is a dependency of module %s.", dependency.Path, module.Path)
+ opts.Logger.Debugf("run-all command called with destroy. To avoid accidentally having destructive effects on external dependencies with run-all command, will not run this command against module %s, which is a dependency of module %s.", dependency.Path, module.Path)
return false, nil
}
- terragruntOptions.Logger.Infof("Module %s has external dependency %s", module.Path, dependency.Path)
+ opts.Logger.Infof("Module %s has external dependency %s", module.Path, dependency.Path)
- return shell.PromptUserForYesNo(ctx, "Should Terragrunt apply the external dependency?", terragruntOptions)
+ return shell.PromptUserForYesNo(ctx, "Should Terragrunt apply the external dependency?", opts)
}
// Get the list of modules this module depends on
@@ -220,15 +220,15 @@ type TerraformModules []*TerraformModule
// FindWhereWorkingDirIsIncluded - find where working directory is included, flow:
// 1. Find root git top level directory and build list of modules
-// 2. Iterate over includes from terragruntOptions if git top level directory detection failed
+// 2. Iterate over includes from opts if git top level directory detection failed
// 3. Filter found module only items which has in dependencies working directory
-func FindWhereWorkingDirIsIncluded(ctx context.Context, terragruntOptions *options.TerragruntOptions, terragruntConfig *config.TerragruntConfig) TerraformModules {
+func FindWhereWorkingDirIsIncluded(ctx context.Context, opts *options.TerragruntOptions, terragruntConfig *config.TerragruntConfig) TerraformModules {
var (
pathsToCheck []string
matchedModulesMap = make(TerraformModulesMap)
)
- if gitTopLevelDir, err := shell.GitTopLevelDir(ctx, terragruntOptions, terragruntOptions.WorkingDir); err == nil {
+ if gitTopLevelDir, err := shell.GitTopLevelDir(ctx, opts, opts.WorkingDir); err == nil {
pathsToCheck = append(pathsToCheck, gitTopLevelDir)
} else {
// detection failed, trying to use include directories as source for stacks
@@ -247,14 +247,14 @@ func FindWhereWorkingDirIsIncluded(ctx context.Context, terragruntOptions *optio
cfgOptions, err := options.NewTerragruntOptionsWithConfigPath(dir)
if err != nil {
- terragruntOptions.Logger.Debugf("Failed to build terragrunt options from %s %v", dir, err)
+ opts.Logger.Debugf("Failed to build terragrunt options from %s %v", dir, err)
continue
}
- cfgOptions.Env = terragruntOptions.Env
- cfgOptions.LogLevel = terragruntOptions.LogLevel
- cfgOptions.OriginalTerragruntConfigPath = terragruntOptions.OriginalTerragruntConfigPath
- cfgOptions.TerraformCommand = terragruntOptions.TerraformCommand
+ cfgOptions.Env = opts.Env
+ cfgOptions.LogLevel = opts.LogLevel
+ cfgOptions.OriginalTerragruntConfigPath = opts.OriginalTerragruntConfigPath
+ cfgOptions.TerraformCommand = opts.TerraformCommand
cfgOptions.NonInteractive = true
cfgOptions.Logger.SetOptions(log.WithHooks(NewForceLogLevelHook(log.DebugLevel)))
@@ -263,13 +263,13 @@ func FindWhereWorkingDirIsIncluded(ctx context.Context, terragruntOptions *optio
if err != nil {
// log error as debug since in some cases stack building may fail because parent files can be designed
// to work with relative paths from downstream modules
- terragruntOptions.Logger.Debugf("Failed to build module stack %v", err)
+ opts.Logger.Debugf("Failed to build module stack %v", err)
continue
}
dependentModules := stack.ListStackDependentModules()
- deps, found := dependentModules[terragruntOptions.WorkingDir]
+ deps, found := dependentModules[opts.WorkingDir]
if found {
for _, module := range stack.Modules {
for _, dep := range deps {
@@ -295,19 +295,19 @@ func FindWhereWorkingDirIsIncluded(ctx context.Context, terragruntOptions *optio
// for a directed graph. It can be used to dump a .dot file.
// This is a similar implementation to terraform's digraph https://github.com/hashicorp/terraform/blob/master/digraph/graphviz.go
// adding some styling to modules that are excluded from the execution in *-all commands
-func (modules TerraformModules) WriteDot(w io.Writer, terragruntOptions *options.TerragruntOptions) error {
+func (modules TerraformModules) WriteDot(w io.Writer, opts *options.TerragruntOptions) error {
if _, err := w.Write([]byte("digraph {\n")); err != nil {
return errors.New(err)
}
defer func(w io.Writer, p []byte) {
_, err := w.Write(p)
if err != nil {
- terragruntOptions.Logger.Warnf("Failed to close graphviz output: %v", err)
+ opts.Logger.Warnf("Failed to close graphviz output: %v", err)
}
}(w, []byte("}\n"))
// all paths are relative to the TerragruntConfigPath
- prefix := filepath.Dir(terragruntOptions.TerragruntConfigPath) + "/"
+ prefix := filepath.Dir(opts.TerragruntConfigPath) + "/"
for _, source := range modules {
// apply a different coloring for excluded nodes
@@ -407,41 +407,17 @@ func (modules TerraformModules) CheckForCycles() error {
return nil
}
-// flagExcludedDirs iterates over a module slice and flags all entries as excluded, which should be ignored via the terragrunt-exclude-dir CLI flag.
-func (modules TerraformModules) flagExcludedDirs(terragruntOptions *options.TerragruntOptions) TerraformModules {
- for _, module := range modules {
- if module.findModuleInPath(terragruntOptions.ExcludeDirs) {
- // Mark module itself as excluded
- module.FlagExcluded = true
- }
-
- // Mark all affected dependencies as excluded
- for _, dependency := range module.Dependencies {
- if dependency.findModuleInPath(terragruntOptions.ExcludeDirs) {
- dependency.FlagExcluded = true
- }
- }
- }
-
- return modules
-}
-
-// flagIncludedDirs iterates over a module slice and flags all entries not in the list specified via the terragrunt-include-dir CLI flag as excluded.
-func (modules TerraformModules) flagIncludedDirs(terragruntOptions *options.TerragruntOptions) TerraformModules {
- // If we're not excluding by default, we should include everything by default.
- // This can happen when a user doesn't set include flags.
- if !terragruntOptions.ExcludeByDefault {
- // If we aren't given any include directories, but are given the strict include flag,
- // return no modules.
- if terragruntOptions.StrictInclude {
- return TerraformModules{}
- }
-
+// flagIncludedDirs includes all units by default.
+//
+// However, when anything that triggers ExcludeByDefault is set, the function will instead
+// selectively include only the units that are in the list specified via the IncludeDirs option.
+func (modules TerraformModules) flagIncludedDirs(opts *options.TerragruntOptions) TerraformModules {
+ if !opts.ExcludeByDefault {
return modules
}
for _, module := range modules {
- if module.findModuleInPath(terragruntOptions.IncludeDirs) {
+ if module.findModuleInPath(opts.IncludeDirs) {
module.FlagExcluded = false
} else {
module.FlagExcluded = true
@@ -449,7 +425,7 @@ func (modules TerraformModules) flagIncludedDirs(terragruntOptions *options.Terr
}
// Mark all affected dependencies as included before proceeding if not in strict include mode.
- if !terragruntOptions.StrictInclude {
+ if !opts.StrictInclude {
for _, module := range modules {
if !module.FlagExcluded {
for _, dependency := range module.Dependencies {
@@ -462,36 +438,26 @@ func (modules TerraformModules) flagIncludedDirs(terragruntOptions *options.Terr
return modules
}
-// flagModulesThatDontInclude iterates over a module slice and flags all modules that don't include at least one file in
-// the specified include list on the TerragruntOptions ModulesThatInclude attribute. Flagged modules will be filtered
-// out of the set.
-func (modules TerraformModules) flagModulesThatDontInclude(terragruntOptions *options.TerragruntOptions) (TerraformModules, error) {
+// flagUnitsThatAreIncluded iterates over a module slice and flags all modules that include at least one file in
+// the specified include list on the TerragruntOptions ModulesThatInclude attribute.
+func (modules TerraformModules) flagUnitsThatAreIncluded(opts *options.TerragruntOptions) (TerraformModules, error) {
// If no ModulesThatInclude is specified return the modules list instantly
- if len(terragruntOptions.ModulesThatInclude) == 0 {
+ if len(opts.ModulesThatInclude) == 0 {
return modules, nil
}
- modulesThatIncludeCanonicalPath := []string{}
+ modulesThatIncludeCanonicalPaths := []string{}
- for _, includePath := range terragruntOptions.ModulesThatInclude {
- canonicalPath, err := util.CanonicalPath(includePath, terragruntOptions.WorkingDir)
+ for _, includePath := range opts.ModulesThatInclude {
+ canonicalPath, err := util.CanonicalPath(includePath, opts.WorkingDir)
if err != nil {
return nil, err
}
- modulesThatIncludeCanonicalPath = append(modulesThatIncludeCanonicalPath, canonicalPath)
+ modulesThatIncludeCanonicalPaths = append(modulesThatIncludeCanonicalPaths, canonicalPath)
}
for _, module := range modules {
- // Ignore modules that are already excluded because this feature is a filter for excluding the subset, not
- // including modules that have already been excluded through other means.
- if module.FlagExcluded {
- continue
- }
-
- // Mark modules that don't include any of the specified paths as excluded. To do this, we first flag the module
- // as excluded, and if it includes any path in the set, we set the exclude flag back to false.
- module.FlagExcluded = true
for _, includeConfig := range module.Config.ProcessedIncludes {
// resolve include config to canonical path to compare with modulesThatIncludeCanonicalPath
// https://github.com/gruntwork-io/terragrunt/issues/1944
@@ -500,7 +466,7 @@ func (modules TerraformModules) flagModulesThatDontInclude(terragruntOptions *op
return nil, err
}
- if util.ListContainsElement(modulesThatIncludeCanonicalPath, canonicalPath) {
+ if util.ListContainsElement(modulesThatIncludeCanonicalPaths, canonicalPath) {
module.FlagExcluded = false
}
}
@@ -512,14 +478,13 @@ func (modules TerraformModules) flagModulesThatDontInclude(terragruntOptions *op
continue
}
- dependency.FlagExcluded = true
for _, includeConfig := range dependency.Config.ProcessedIncludes {
canonicalPath, err := util.CanonicalPath(includeConfig.Path, module.Path)
if err != nil {
return nil, err
}
- if util.ListContainsElement(modulesThatIncludeCanonicalPath, canonicalPath) {
+ if util.ListContainsElement(modulesThatIncludeCanonicalPaths, canonicalPath) {
dependency.FlagExcluded = false
}
}
@@ -529,6 +494,54 @@ func (modules TerraformModules) flagModulesThatDontInclude(terragruntOptions *op
return modules, nil
}
+// flagUnitsThatRead iterates over a module slice and flags all modules that read at least one file in the specified
+// file list in the TerragruntOptions UnitsReading attribute.
+func (modules TerraformModules) flagUnitsThatRead(opts *options.TerragruntOptions) (TerraformModules, error) {
+ // If no UnitsThatRead is specified return the modules list instantly
+ if len(opts.UnitsReading) == 0 {
+ return modules, nil
+ }
+
+ for _, readPath := range opts.UnitsReading {
+ path, err := util.CanonicalPath(readPath, opts.WorkingDir)
+ if err != nil {
+ return nil, err
+ }
+
+ for _, module := range modules {
+ if opts.DidReadFile(path, module.Path) {
+ module.FlagExcluded = false
+ }
+ }
+ }
+
+ return modules, nil
+}
+
+// flagExcludedDirs iterates over a module slice and flags all entries as excluded listed in the terragrunt-exclude-dir CLI flag.
+func (modules TerraformModules) flagExcludedDirs(opts *options.TerragruntOptions) TerraformModules {
+ // If we don't have any excludes, we don't need to do anything.
+ if len(opts.ExcludeDirs) == 0 {
+ return modules
+ }
+
+ for _, module := range modules {
+ if module.findModuleInPath(opts.ExcludeDirs) {
+ // Mark module itself as excluded
+ module.FlagExcluded = true
+ }
+
+ // Mark all affected dependencies as excluded
+ for _, dependency := range module.Dependencies {
+ if dependency.findModuleInPath(opts.ExcludeDirs) {
+ dependency.FlagExcluded = true
+ }
+ }
+ }
+
+ return modules
+}
+
var existingModules = cache.NewCache[*TerraformModulesMap](existingModulesCacheName)
type TerraformModulesMap map[string]*TerraformModule
diff --git a/configstack/stack.go b/configstack/stack.go
index b037e4ea9..0f99a72e2 100644
--- a/configstack/stack.go
+++ b/configstack/stack.go
@@ -399,12 +399,12 @@ func (stack *Stack) ResolveTerraformModules(ctx context.Context, terragruntConfi
return nil, err
}
- var includedModules TerraformModules
+ var withUnitsIncluded TerraformModules
err = telemetry.Telemetry(ctx, stack.terragruntOptions, "flag_included_dirs", map[string]interface{}{
"working_dir": stack.terragruntOptions.WorkingDir,
}, func(childCtx context.Context) error {
- includedModules = crossLinkedModules.flagIncludedDirs(stack.terragruntOptions)
+ withUnitsIncluded = crossLinkedModules.flagIncludedDirs(stack.terragruntOptions)
return nil
})
@@ -412,12 +412,18 @@ func (stack *Stack) ResolveTerraformModules(ctx context.Context, terragruntConfi
return nil, err
}
- var includedModulesWithExcluded TerraformModules
+ var withUnitsThatAreIncludedByOthers TerraformModules
- err = telemetry.Telemetry(ctx, stack.terragruntOptions, "flag_excluded_dirs", map[string]interface{}{
+ err = telemetry.Telemetry(ctx, stack.terragruntOptions, "flag_units_that_are_included", map[string]interface{}{
"working_dir": stack.terragruntOptions.WorkingDir,
}, func(childCtx context.Context) error {
- includedModulesWithExcluded = includedModules.flagExcludedDirs(stack.terragruntOptions)
+ result, err := withUnitsIncluded.flagUnitsThatAreIncluded(stack.terragruntOptions)
+ if err != nil {
+ return err
+ }
+
+ withUnitsThatAreIncludedByOthers = result
+
return nil
})
@@ -425,25 +431,39 @@ func (stack *Stack) ResolveTerraformModules(ctx context.Context, terragruntConfi
return nil, err
}
- var finalModules TerraformModules
+ var withUnitsRead TerraformModules
- err = telemetry.Telemetry(ctx, stack.terragruntOptions, "flag_modules_that_dont_include", map[string]interface{}{
+ err = telemetry.Telemetry(ctx, stack.terragruntOptions, "flag_units_that_read", map[string]interface{}{
"working_dir": stack.terragruntOptions.WorkingDir,
}, func(childCtx context.Context) error {
- result, err := includedModulesWithExcluded.flagModulesThatDontInclude(stack.terragruntOptions)
+ result, err := withUnitsThatAreIncludedByOthers.flagUnitsThatRead(stack.terragruntOptions)
if err != nil {
return err
}
- finalModules = result
+ withUnitsRead = result
return nil
})
+
if err != nil {
return nil, err
}
- return finalModules, nil
+ var withModulesExcluded TerraformModules
+
+ err = telemetry.Telemetry(ctx, stack.terragruntOptions, "flag_excluded_dirs", map[string]interface{}{
+ "working_dir": stack.terragruntOptions.WorkingDir,
+ }, func(childCtx context.Context) error {
+ withModulesExcluded = withUnitsRead.flagExcludedDirs(stack.terragruntOptions)
+ return nil
+ })
+
+ if err != nil {
+ return nil, err
+ }
+
+ return withModulesExcluded, nil
}
// Go through each of the given Terragrunt configuration files and resolve the module that configuration file represents
@@ -584,6 +604,9 @@ func (stack *Stack) resolveTerraformModule(ctx context.Context, terragruntConfig
})
}
+ // Hack to persist readFiles. Need to discuss with team to see if there is a better way to handle this.
+ stack.terragruntOptions.CloneReadFiles(opts.ReadFiles)
+
terragruntSource, err := config.GetTerragruntSourceForModule(stack.terragruntOptions.Source, modulePath, terragruntConfig)
if err != nil {
return nil, err
diff --git a/docs/_docs/04_reference/built-in-functions.md b/docs/_docs/04_reference/built-in-functions.md
index c829cbfe5..6d09017fb 100644
--- a/docs/_docs/04_reference/built-in-functions.md
+++ b/docs/_docs/04_reference/built-in-functions.md
@@ -40,6 +40,7 @@ Terragrunt allows you to use built-in functions anywhere in `terragrunt.hcl`, ju
- [sops\_decrypt\_file](#sops_decrypt_file)
- [get\_terragrunt\_source\_cli\_flag](#get_terragrunt_source_cli_flag)
- [read\_tfvars\_file](#read_tfvars_file)
+- [mark\_as\_read](#mark_as_read)
## OpenTofu/Terraform built-in functions
@@ -867,3 +868,30 @@ remote_state {
}
}
```
+
+## mark_as_read
+
+`mark_as_read(file_path)` marks a file as read, so that it can be picked up for inclusion by the [queue-include-units-reading](./cli-options.md#queue-include-units-reading) flag.
+
+This is useful for situations when you want to mark a file as read, but are not reading it using a native Terragrunt HCL function.
+
+For example:
+
+```hcl
+locals {
+ filename = mark_as_read("file-read-by-tofu.txt")
+}
+
+inputs = {
+ filename = local.filename
+}
+```
+
+By using `mark_as_read` on `file-read-by-tofu.txt`, you can ensure that the `terragrunt.hcl` file passing in the `file-read-by-tofu.txt` file as an input will be included in
+any `run-all` run where the flag `--queue-include-units-reading file-read-by-tofu.txt` is set.
+
+The same technique can be used to mark a file as read when reading a file using code in `run_cmd`, etc.
+
+**NOTE**: Due to the way that Terragrunt parses configurations during a `run-all`, functions will only properly mark files as read
+if they are used in the `locals` block. Reading a file directly in the `inputs` block will not mark the file as read, as the `inputs`
+block is not evaluated until *after* the queue has been populated with units to run.
diff --git a/docs/_docs/04_reference/cli-options.md b/docs/_docs/04_reference/cli-options.md
index f25460fd0..2b5a50a1a 100644
--- a/docs/_docs/04_reference/cli-options.md
+++ b/docs/_docs/04_reference/cli-options.md
@@ -1291,6 +1291,73 @@ passed in, the set will be the union of modules that includes at least one of th
NOTE: When using relative paths, the paths are relative to the working directory. This is either the current working
directory, or any path passed in to [terragrunt-working-dir](#terragrunt-working-dir).
+### terragrunt-queue-include-units-reading
+
+**CLI Arg**: `--terragrunt-queue-include-units-reading`
+**Environment Variable**: `TERRAGRUNT_QUEUE_INCLUDE_UNITS_READING`
+**Commands**:
+
+- [run-all](#run-all)
+- [plan-all (DEPRECATED: use run-all)](#plan-all-deprecated-use-run-all)
+- [apply-all (DEPRECATED: use run-all)](#apply-all-deprecated-use-run-all)
+- [output-all (DEPRECATED: use run-all)](#output-all-deprecated-use-run-all)
+- [destroy-all (DEPRECATED: use run-all)](#destroy-all-deprecated-use-run-all)
+- [validate-all (DEPRECATED: use run-all)](#validate-all-deprecated-use-run-all)
+
+This flag works very similarly to the `--terragrunt-modules-that-include` flag, but instead of looking for included configurations,
+it instead looks for configurations that read a given file.
+
+When passed in, the `*-all` commands will include all units (modules) that read a given file into the queue. This is useful
+when you want to trigger an update on all units that read a given file using HCL functions in their configurations.
+
+Consider the following folder structure:
+
+```tree
+.
+├── reading-shared-hcl
+│ └── terragrunt.hcl
+├── also-reading-shared-hcl
+│ └── terragrunt.hcl
+├── not-reading-shared-hcl
+│ └── terragrunt.hcl
+└── shared.hcl
+```
+
+Suppose that `reading-shared-hcl` and `also-reading-shared-hcl` both read `shared.hcl` in their configurations, like so:
+
+```hcl
+locals {
+ shared = read_terragrunt_config(find_in_parent_folders("shared.hcl"))
+}
+```
+
+If you run the command `run-all init --terragrunt-queue-include-units-reading shared.hcl` from the root folder, both
+`reading-shared-hcl` and `also-reading-shared-hcl` will be run; not `not-reading-shared-hcl`.
+
+This is because the `read_terragrunt_config` HCL function has a special hook that allows Terragrunt to track that it has
+read the file `shared.hcl`. This hook is used by all native HCL functions that Terragrunt supports which read files.
+
+Note, however, that there are certain scenarios where Terragrunt may not be able to track that a file has been read this way.
+
+For example, you may be using a bash script to read a file via `run_cmd`, or reading the file via OpenTofu code. To support these
+use-cases, the [mark_as_read](./built-in-functions.md#mark_as_read) function can be used to manually mark a file as read.
+
+That would look something like this:
+
+```hcl
+locals {
+ filename = mark_as_read("file-read-by-tofu.txt")
+}
+
+inputs = {
+ filename = local.filename
+}
+```
+
+**⚠️**: Due to the way that Terragrunt parses configurations during a `run-all`, functions will only properly mark files as read
+if they are used in the `locals` block. Reading a file directly in the `inputs` block will not mark the file as read, as the `inputs`
+block is not evaluated until _after_ the queue has been populated with units to run.
+
### terragrunt-fetch-dependency-output-from-state
**CLI Arg**: `--terragrunt-fetch-dependency-output-from-state`
diff --git a/options/options.go b/options/options.go
index 4fc5ca93c..2e98de2d6 100644
--- a/options/options.go
+++ b/options/options.go
@@ -248,6 +248,10 @@ type TerragruntOptions struct {
// in this list.
ModulesThatInclude []string
+ // When used with `run-all`, restrict the units in the stack to only those that read at least one of the files
+ // in this list.
+ UnitsReading []string
+
// A command that can be used to run Terragrunt with the given options. This is useful for running Terragrunt
// multiple times (e.g. when spinning up a stack of Terraform modules). The actual command is normally defined
// in the cli package, which depends on almost all other packages, so we declare it here so that other
@@ -356,6 +360,10 @@ type TerragruntOptions struct {
// FeatureFlags is a map of feature flags to enable.
FeatureFlags map[string]string
+
+ // ReadFiles is a map of files to the Units
+ // that read them using HCL functions in the unit.
+ ReadFiles map[string][]string
}
// TerragruntOptionsFunc is a functional option type used to pass options in certain integration tests
@@ -587,6 +595,8 @@ func (opts *TerragruntOptions) Clone(terragruntConfigPath string) (*TerragruntOp
IncludeDirs: opts.IncludeDirs,
ExcludeByDefault: opts.ExcludeByDefault,
ModulesThatInclude: opts.ModulesThatInclude,
+ UnitsReading: opts.UnitsReading,
+ ReadFiles: opts.ReadFiles,
Parallelism: opts.Parallelism,
StrictInclude: opts.StrictInclude,
RunTerragrunt: opts.RunTerragrunt,
@@ -721,6 +731,50 @@ func (opts *TerragruntOptions) DataDir() string {
return util.JoinPath(opts.WorkingDir, tfDataDir)
}
+// AppendReadFile appends to the list of files read by a given unit.
+func (opts *TerragruntOptions) AppendReadFile(file, unit string) {
+ if opts.ReadFiles == nil {
+ opts.ReadFiles = map[string][]string{}
+ }
+
+ for _, u := range opts.ReadFiles[file] {
+ if u == unit {
+ return
+ }
+ }
+
+ opts.Logger.Debugf("Tracking that file %s was read by %s.", file, unit)
+ opts.ReadFiles[file] = append(opts.ReadFiles[file], unit)
+}
+
+// DidReadFile checks if a given file was read by a given unit.
+func (opts *TerragruntOptions) DidReadFile(file, unit string) bool {
+ if opts.ReadFiles == nil {
+ return false
+ }
+
+ for _, u := range opts.ReadFiles[file] {
+ if u == unit {
+ return true
+ }
+ }
+
+ return false
+}
+
+// CloneReadFiles creates a copy of the ReadFiles map.
+func (opts *TerragruntOptions) CloneReadFiles(readFiles map[string][]string) {
+ if readFiles == nil {
+ return
+ }
+
+ for file, units := range readFiles {
+ for _, unit := range units {
+ opts.AppendReadFile(file, unit)
+ }
+ }
+}
+
// identifyDefaultWrappedExecutable returns default path used for wrapped executable.
func identifyDefaultWrappedExecutable() string {
if util.IsCommandExecutable(TofuDefaultPath, "-version") {
diff --git a/test/fixtures/units-reading/reading-from-tf/main.tf b/test/fixtures/units-reading/reading-from-tf/main.tf
new file mode 100644
index 000000000..7ee83502d
--- /dev/null
+++ b/test/fixtures/units-reading/reading-from-tf/main.tf
@@ -0,0 +1,4 @@
+variable "filename" {}
+output "shared" {
+ value = jsondecode(file(var.filename))
+}
diff --git a/test/fixtures/units-reading/reading-from-tf/terragrunt.hcl b/test/fixtures/units-reading/reading-from-tf/terragrunt.hcl
new file mode 100644
index 000000000..abd30a935
--- /dev/null
+++ b/test/fixtures/units-reading/reading-from-tf/terragrunt.hcl
@@ -0,0 +1,7 @@
+locals {
+ filename = mark_as_read(find_in_parent_folders("shared.json"))
+}
+
+inputs = {
+ filename = local.filename
+}
diff --git a/test/fixtures/units-reading/reading-hcl-and-tfvars/main.tf b/test/fixtures/units-reading/reading-hcl-and-tfvars/main.tf
new file mode 100644
index 000000000..674790dc7
--- /dev/null
+++ b/test/fixtures/units-reading/reading-hcl-and-tfvars/main.tf
@@ -0,0 +1,9 @@
+variable "shared_hcl" {}
+output "shared_hcl" {
+ value = var.shared_hcl
+}
+
+variable "shared_tfvars" {}
+output "shared_tfvars" {
+ value = var.shared_tfvars
+}
diff --git a/test/fixtures/units-reading/reading-hcl-and-tfvars/terragrunt.hcl b/test/fixtures/units-reading/reading-hcl-and-tfvars/terragrunt.hcl
new file mode 100644
index 000000000..fd8a3eced
--- /dev/null
+++ b/test/fixtures/units-reading/reading-hcl-and-tfvars/terragrunt.hcl
@@ -0,0 +1,9 @@
+locals {
+ shared_hcl = read_terragrunt_config(find_in_parent_folders("shared.hcl")).locals
+ shared_tfvars = read_tfvars_file(find_in_parent_folders("shared.tfvars"))
+}
+
+inputs = {
+ shared_hcl = local.shared_hcl
+ shared_tfvars = local.shared_tfvars
+}
diff --git a/test/fixtures/units-reading/reading-hcl/main.tf b/test/fixtures/units-reading/reading-hcl/main.tf
new file mode 100644
index 000000000..05d8c59b8
--- /dev/null
+++ b/test/fixtures/units-reading/reading-hcl/main.tf
@@ -0,0 +1,4 @@
+variable "shared" {}
+output "shared" {
+ value = var.shared
+}
diff --git a/test/fixtures/units-reading/reading-hcl/terragrunt.hcl b/test/fixtures/units-reading/reading-hcl/terragrunt.hcl
new file mode 100644
index 000000000..06fab2f1c
--- /dev/null
+++ b/test/fixtures/units-reading/reading-hcl/terragrunt.hcl
@@ -0,0 +1,7 @@
+locals {
+ shared = read_terragrunt_config(find_in_parent_folders("shared.hcl")).locals
+}
+
+inputs = {
+ shared = local.shared
+}
diff --git a/test/fixtures/units-reading/reading-json/main.tf b/test/fixtures/units-reading/reading-json/main.tf
new file mode 100644
index 000000000..05d8c59b8
--- /dev/null
+++ b/test/fixtures/units-reading/reading-json/main.tf
@@ -0,0 +1,4 @@
+variable "shared" {}
+output "shared" {
+ value = var.shared
+}
diff --git a/test/fixtures/units-reading/reading-json/terragrunt.hcl b/test/fixtures/units-reading/reading-json/terragrunt.hcl
new file mode 100644
index 000000000..11d8df178
--- /dev/null
+++ b/test/fixtures/units-reading/reading-json/terragrunt.hcl
@@ -0,0 +1,7 @@
+locals {
+ shared = jsondecode(file(mark_as_read(find_in_parent_folders("shared.json"))))
+}
+
+inputs = {
+ shared = local.shared
+}
diff --git a/test/fixtures/units-reading/reading-sops/main.tf b/test/fixtures/units-reading/reading-sops/main.tf
new file mode 100644
index 000000000..05d8c59b8
--- /dev/null
+++ b/test/fixtures/units-reading/reading-sops/main.tf
@@ -0,0 +1,4 @@
+variable "shared" {}
+output "shared" {
+ value = var.shared
+}
diff --git a/test/fixtures/units-reading/reading-sops/terragrunt.hcl b/test/fixtures/units-reading/reading-sops/terragrunt.hcl
new file mode 100644
index 000000000..3e8310fa1
--- /dev/null
+++ b/test/fixtures/units-reading/reading-sops/terragrunt.hcl
@@ -0,0 +1,7 @@
+locals {
+ shared = sops_decrypt_file(find_in_parent_folders("secrets.txt"))
+}
+
+inputs = {
+ shared = local.shared
+}
diff --git a/test/fixtures/units-reading/reading-tfvars/main.tf b/test/fixtures/units-reading/reading-tfvars/main.tf
new file mode 100644
index 000000000..05d8c59b8
--- /dev/null
+++ b/test/fixtures/units-reading/reading-tfvars/main.tf
@@ -0,0 +1,4 @@
+variable "shared" {}
+output "shared" {
+ value = var.shared
+}
diff --git a/test/fixtures/units-reading/reading-tfvars/terragrunt.hcl b/test/fixtures/units-reading/reading-tfvars/terragrunt.hcl
new file mode 100644
index 000000000..4c6441f83
--- /dev/null
+++ b/test/fixtures/units-reading/reading-tfvars/terragrunt.hcl
@@ -0,0 +1,7 @@
+locals {
+ shared = read_tfvars_file(find_in_parent_folders("shared.tfvars"))
+}
+
+inputs = {
+ shared = local.shared
+}
diff --git a/test/fixtures/units-reading/secrets.txt b/test/fixtures/units-reading/secrets.txt
new file mode 100644
index 000000000..37426edfc
--- /dev/null
+++ b/test/fixtures/units-reading/secrets.txt
@@ -0,0 +1,21 @@
+{
+ "data": "ENC[AES256_GCM,data:w2jDRJR9BeIMSKE4+qnKWhfM,iv:08ACLYrUGtWriOV/ua4X6NZt57VmiTmAcnxB5V+8AUc=,tag:cVdkIO4EXAmyV3y7n/zbiA==,type:str]",
+ "sops": {
+ "kms": null,
+ "gcp_kms": null,
+ "azure_kv": null,
+ "hc_vault": null,
+ "age": null,
+ "lastmodified": "2021-12-17T18:38:13Z",
+ "mac": "ENC[AES256_GCM,data:8lPZmY8YgA0DqPRxLC9hVoRUXmbzaXgUBv3MHTm4iK44/6URIgJBUnPFPUbwIN7xbIgXd+QPQEMvfsmifqXorynGEwt2WtMKCPANg+2Ctf2KMmj7fGpe3HIlRhQiixip7/xzrIMbSdIRMS098D42JTvOIFNbWVQhByfN64AnDJY=,iv:wtouC/mWjhFwiJKDS6+5LqnQMcAeejElXLaL3H15jbY=,tag:6Bmemr2BMgShaMO3v4uiXw==,type:str]",
+ "pgp": [
+ {
+ "created_at": "2021-12-17T18:38:12Z",
+ "enc": "-----BEGIN PGP MESSAGE-----\n\nhQEMA0sXzMgpEabgAQf+KHsPp4Pp8YNtG7ChRpZO2qB/bFncWtAF9evO+RjAEahb\nM+hzxkB5KDUSMYs0aeWeOrOqYPrjPPJxCspZtQhy8/qrC064kA7gq2PWhYAqGcKP\ntnPI8D0SYDZBgoyHRqFuuD5TZio8swE89SxphftL0W3KkHay7WKQHj/cFqNoISNl\nn0XeCgbacIwo5WxWz1qNFvaeo0rFFFhIhbfaegx/SWwUi1y6WK7sB0QobMRwXHj+\nORiUWVvx/fCIMCaerPN/SjIA/DgzbZ3DWaixYXpW85Ipz7myu/zUQcWnWcGXnMRQ\nERMYc6GyyLHwjZN1XuvXdPXvAt6vvaH4w5U9kW2l19JeAZXkcM14ivDoGwY1oLcX\n4d2/MAS7vM7SgmcPBGmpNsJJgkWTgoc8qeFtu9u3e4e9pR4+dcJCbGQLQ5RiyM2Z\nsyHjL6em/j4JLdtbM16orP6Q3oEPelphG7sxbDXBeA==\n=6u1S\n-----END PGP MESSAGE-----\n",
+ "fp": "3EF98802EEDCAF0C688B81F419546E0C123C664E"
+ }
+ ],
+ "unencrypted_suffix": "_unencrypted",
+ "version": "3.7.1"
+ }
+}
\ No newline at end of file
diff --git a/test/fixtures/units-reading/shared.hcl b/test/fixtures/units-reading/shared.hcl
new file mode 100644
index 000000000..d5f16bab7
--- /dev/null
+++ b/test/fixtures/units-reading/shared.hcl
@@ -0,0 +1,3 @@
+locals {
+ shared_hcl = "value"
+}
diff --git a/test/fixtures/units-reading/shared.json b/test/fixtures/units-reading/shared.json
new file mode 100644
index 000000000..ead884617
--- /dev/null
+++ b/test/fixtures/units-reading/shared.json
@@ -0,0 +1,3 @@
+{
+ "key": "value"
+}
diff --git a/test/fixtures/units-reading/shared.tfvars b/test/fixtures/units-reading/shared.tfvars
new file mode 100644
index 000000000..0ffcc1d99
--- /dev/null
+++ b/test/fixtures/units-reading/shared.tfvars
@@ -0,0 +1 @@
+shared_tfvars = "value"
diff --git a/test/fixtures/units-reading/test_pgp_key.asc b/test/fixtures/units-reading/test_pgp_key.asc
new file mode 100644
index 000000000..e42619f65
--- /dev/null
+++ b/test/fixtures/units-reading/test_pgp_key.asc
@@ -0,0 +1,87 @@
+-----BEGIN PGP PRIVATE KEY BLOCK-----
+
+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
+0S9IMjn4Mj7CYtAZarnIQw==
+=fO3n
+-----END PGP PRIVATE KEY BLOCK-----
+-----BEGIN PGP PUBLIC KEY BLOCK-----
+
+mQENBF69y60BCAC0Ton9jmk5/ZV3yKi2o0rY22G41dh9dLSoxTRJNKsKeSJePHK0
+qtrHriPjnHxFLXrXfYiv6FkdkPv9dhmKjMf1op/RRw0uTDZ6CsQd5C8nT9LBPiB6
+JsO89+GCwoVsPQGGVgqq7PWSbA8fJkAPBNvdQvxJnSbm4EJTcaU7XZmmY/eZnhPt
+pho66X3PIQSx8Xm9Dc7jHo/7bBCoEV15OdFYi44imDW9sCdCpYTVzMF2ZRtXGylQ
+U6P5Iu3SEMTnn4jwPpyKllT4qRFdqMQ4GvqE47uRb+k/FDXP8UC6h/5GNHfbQtEO
+CMngw5vJciLEUWW1e97baBPiccLXNaBXiecXABEBAAG0EFRlcnJhZ3J1bnQgVGVz
+dHOJAUwEEwEKADYWIQQ++YgC7tyvDGiLgfQZVG4MEjxmTgUCXr3LrQIbAwQLCQgH
+BBUKCQgFFgIDAQACHgECF4AACgkQGVRuDBI8Zk76SwgAkfEmyr9qOtNXpD4m8SnC
+rg8EJRX584hJyAu08/uB+QQV7XzjAzns4oVGGpZYEB7sUd22ejw8BGe6OI4puL43
+fnpDcFj4UA7FAIzQTKNztWFci2HoL0IAwcdzSlJb6qREMJacit8G4Aayn/bqZSNF
+4cu0XipRlOtK28FXP1PQThlNtSJLyPpYRyU6s2CLSeRRqbi1MJ4KgKcnfTvv/4+V
+YQ6y7/rsDnXDA0nLgzYKMqtNdBPYr4u7kZsGcWP8w9xQDWNJtOqj8NiRSAtnglqi
+YTHuEQsRjYGjH+vHqnGfw7dcK4+3Rlc7ZPGON2NCmJ6OajedIxZR6yNiDwj1AfuX
+xbkBDQRevcutAQgAtV1FnBgLKuqgBVs09UbyrFuVJIx6M4e0IFac4bVzyfGJKjmx
+Hx2n7D8slTZs+lcBNDy+CseAsAucixv/jVAsl6saip5nUIAMmHkPkPM+BVAuqBRW
+oYWGFw3TL83MaYAgBkSkurx+Koxvw2bTumxrb9xsyiy7Bn7kaIyZU/nN8Xb0xf6Q
+C+y7ckzaI6ZT4utrtJGXya4Cu1rpqUUYs/4ssWjZS7KuRwu8ijNwq/Zl0920IXH/
+sX/gUhL2AO5sgExC+52yjTTvhlIbee8tjMB4JT6Hq1RS9JGkNqcUOvJ99WlUoOtZ
+UJU9xAN4eyPUy+r7v9wxcwoguanCLWaKSNe9hQARAQABiQE2BBgBCgAgFiEEPvmI
+Au7crwxoi4H0GVRuDBI8Zk4FAl69y60CGwwACgkQGVRuDBI8Zk4gogf/SyXhch/E
+p9ESW+Zx7T+ImfQAVPBmka0NNCEvSF5i/ghTm700Iv/sLcRqhSJDiKLb1imeqVN6
+khmr6+d21uCqqgiv/6X4w6zmxH9h4uPMv3H6WnHi40kDUawR7hFctNLXtwArH+xC
+X5MJgzhqAP5Yzgsk5XLwzgjCuXz06RqtmZVW4ofOP4GWBT24Gg4TyUF9Ibl1/QLK
+eYOarD4a/PcEfHhJyvnAmD68uIFr1gABJJneW8vvO4OPjLcptZMIZ1NfxqUuLBAJ
+71OI3RSSowpI3qCpFhH2j5vICM7jX2gIL9PQEGGg2ljNJoorJIEjYZzS0S9IMjn4
+Mj7CYtAZarnIQw==
+=dO7W
+-----END PGP PUBLIC KEY BLOCK-----
diff --git a/test/integartion_units_reading_test.go b/test/integartion_units_reading_test.go
new file mode 100644
index 000000000..245100af4
--- /dev/null
+++ b/test/integartion_units_reading_test.go
@@ -0,0 +1,162 @@
+package test_test
+
+import (
+ "regexp"
+ "strings"
+ "testing"
+
+ "github.com/gruntwork-io/terragrunt/test/helpers"
+ "github.com/gruntwork-io/terragrunt/util"
+ "github.com/stretchr/testify/assert"
+ "github.com/stretchr/testify/require"
+)
+
+const (
+ testFixtureUnitsReading = "fixtures/units-reading/"
+)
+
+func TestUnitsReading(t *testing.T) {
+ t.Parallel()
+
+ cleanupTerraformFolder(t, testFixtureUnitsReading)
+
+ tc := []struct {
+ name string
+ unitsReading []string
+ unitsExcluding []string
+ unitsIncluding []string
+ expectedUnits []string
+ }{
+ {
+ name: "empty",
+ unitsReading: []string{},
+ expectedUnits: []string{
+ "reading-from-tf",
+ "reading-hcl",
+ "reading-hcl-and-tfvars",
+ "reading-json",
+ "reading-sops",
+ "reading-tfvars",
+ },
+ },
+ {
+ name: "reading_hcl",
+ unitsReading: []string{
+ "shared.hcl",
+ },
+ expectedUnits: []string{
+ "reading-hcl",
+ "reading-hcl-and-tfvars",
+ },
+ },
+ {
+ name: "reading_tfvars",
+ unitsReading: []string{
+ "shared.tfvars",
+ },
+ expectedUnits: []string{
+ "reading-tfvars",
+ "reading-hcl-and-tfvars",
+ },
+ },
+ {
+ name: "reading_json",
+ unitsReading: []string{
+ "shared.json",
+ },
+ expectedUnits: []string{
+ "reading-from-tf",
+ "reading-json",
+ },
+ },
+ {
+ name: "reading_sops",
+ unitsReading: []string{
+ "secrets.txt",
+ },
+ expectedUnits: []string{
+ "reading-sops",
+ },
+ },
+ {
+ name: "reading_from_hcl_with_exclude",
+ unitsReading: []string{
+ "shared.hcl",
+ },
+ unitsExcluding: []string{
+ "reading-hcl-and-tfvars",
+ },
+ expectedUnits: []string{
+ "reading-hcl",
+ },
+ },
+ {
+ name: "reading_from_hcl_with_include",
+ unitsReading: []string{
+ "shared.hcl",
+ },
+ unitsIncluding: []string{
+ "reading-tfvars",
+ },
+ expectedUnits: []string{
+ "reading-hcl",
+ "reading-hcl-and-tfvars",
+ "reading-tfvars",
+ },
+ },
+ {
+ name: "reading_from_hcl_with_include_and_exclude",
+ unitsReading: []string{
+ "shared.hcl",
+ "shared.tfvars",
+ },
+ unitsIncluding: []string{
+ "reading-tfvars",
+ },
+ unitsExcluding: []string{
+ "reading-hcl-and-tfvars",
+ },
+ expectedUnits: []string{
+ "reading-hcl",
+ "reading-tfvars",
+ },
+ },
+ }
+
+ includedLogEntryRegex := regexp.MustCompile(`=> Module ./([^ ]+) \(excluded: false`)
+
+ for _, tt := range tc {
+ t.Run(tt.name, func(t *testing.T) {
+ t.Parallel()
+
+ tmpEnvPath := helpers.CopyEnvironment(t, testFixtureUnitsReading)
+ rootPath := util.JoinPath(tmpEnvPath, testFixtureUnitsReading)
+
+ cmd := "terragrunt run-all plan --terragrunt-non-interactive --terragrunt-log-level debug --terragrunt-working-dir " + rootPath
+
+ for _, unit := range tt.unitsReading {
+ cmd = cmd + " --terragrunt-queue-include-units-reading " + unit
+ }
+
+ for _, unit := range tt.unitsIncluding {
+ cmd = cmd + " --terragrunt-include-dir " + unit
+ }
+
+ for _, unit := range tt.unitsExcluding {
+ cmd = cmd + " --terragrunt-exclude-dir " + unit
+ }
+
+ _, stderr, err := helpers.RunTerragruntCommandWithOutput(t, cmd)
+ require.NoError(t, err)
+
+ includedUnits := []string{}
+ for _, line := range strings.Split(stderr, "\n") {
+ if includedLogEntryRegex.MatchString(line) {
+ includedUnits = append(includedUnits, includedLogEntryRegex.FindStringSubmatch(line)[1])
+ }
+ }
+
+ assert.ElementsMatch(t, tt.expectedUnits, includedUnits)
+ })
+ }
+}
diff --git a/test/integration_test.go b/test/integration_test.go
index 43eb5c822..715992094 100644
--- a/test/integration_test.go
+++ b/test/integration_test.go
@@ -3020,12 +3020,12 @@ func TestHclFmtStdin(t *testing.T) {
t.Parallel()
cleanupTerraformFolder(t, testFixtureHclfmtStdin)
- tmpEnvPath := copyEnvironment(t, testFixtureHclfmtStdin)
+ tmpEnvPath := helpers.CopyEnvironment(t, testFixtureHclfmtStdin)
rootPath := util.JoinPath(tmpEnvPath, testFixtureHclfmtStdin)
os.Stdin, _ = os.Open(util.JoinPath(rootPath, "terragrunt.hcl"))
- stdout, _, err := runTerragruntCommandWithOutput(t, "terragrunt hclfmt --terragrunt-hclfmt-stdin")
+ stdout, _, err := helpers.RunTerragruntCommandWithOutput(t, "terragrunt hclfmt --terragrunt-hclfmt-stdin")
require.NoError(t, err)
expectedDiff, err := os.ReadFile(util.JoinPath(rootPath, "expected.hcl"))