From ab6a8b5a14694aa77bc40caa4d6918abd03ddc66 Mon Sep 17 00:00:00 2001 From: Liam Cervante Date: Fri, 19 Apr 2024 17:01:33 +0200 Subject: [PATCH] deferred actions: add common functionality for evaluating unexpanded and expanded modules --- internal/terraform/eval_context_builtin.go | 10 +- internal/terraform/evaluate.go | 897 --------------------- internal/terraform/evaluate_data.go | 190 +++++ internal/terraform/evaluate_placeholder.go | 55 +- internal/terraform/evaluate_state.go | 753 +++++++++++++++++ internal/terraform/evaluate_test.go | 28 +- 6 files changed, 974 insertions(+), 959 deletions(-) create mode 100644 internal/terraform/evaluate_data.go create mode 100644 internal/terraform/evaluate_state.go diff --git a/internal/terraform/eval_context_builtin.go b/internal/terraform/eval_context_builtin.go index a9ba7cbb8005..afe3e1ec9b75 100644 --- a/internal/terraform/eval_context_builtin.go +++ b/internal/terraform/eval_context_builtin.go @@ -440,7 +440,10 @@ func (ctx *BuiltinEvalContext) EvaluationScope(self addrs.Referenceable, source switch scope := ctx.scope.(type) { case evalContextModuleInstance: data := &evaluationStateData{ - Evaluator: ctx.Evaluator, + evaluationData: &evaluationData{ + Evaluator: ctx.Evaluator, + Module: scope.Addr.Module(), + }, ModulePath: scope.Addr, InstanceKeyData: keyData, Operation: ctx.Evaluator.Operation, @@ -460,7 +463,10 @@ func (ctx *BuiltinEvalContext) EvaluationScope(self addrs.Referenceable, source return evalScope case evalContextPartialExpandedModule: data := &evaluationPlaceholderData{ - Evaluator: ctx.Evaluator, + evaluationData: &evaluationData{ + Evaluator: ctx.Evaluator, + Module: scope.Addr.Module(), + }, ModulePath: scope.Addr, CountAvailable: keyData.CountIndex != cty.NilVal, EachAvailable: keyData.EachKey != cty.NilVal, diff --git a/internal/terraform/evaluate.go b/internal/terraform/evaluate.go index dc76b9988d0a..e57ca1be9d04 100644 --- a/internal/terraform/evaluate.go +++ b/internal/terraform/evaluate.go @@ -4,27 +4,16 @@ package terraform import ( - "fmt" - "log" - "os" - "path/filepath" "time" - "github.com/hashicorp/hcl/v2" - "github.com/zclconf/go-cty/cty" - "github.com/hashicorp/terraform/internal/addrs" "github.com/hashicorp/terraform/internal/configs" - "github.com/hashicorp/terraform/internal/configs/configschema" - "github.com/hashicorp/terraform/internal/didyoumean" "github.com/hashicorp/terraform/internal/instances" "github.com/hashicorp/terraform/internal/lang" - "github.com/hashicorp/terraform/internal/lang/marks" "github.com/hashicorp/terraform/internal/namedvals" "github.com/hashicorp/terraform/internal/plans" "github.com/hashicorp/terraform/internal/plans/deferring" "github.com/hashicorp/terraform/internal/states" - "github.com/hashicorp/terraform/internal/tfdiags" ) // Evaluator provides the necessary contextual data for evaluating expressions @@ -95,892 +84,6 @@ func (e *Evaluator) Scope(data lang.Data, self addrs.Referenceable, source addrs } } -// evaluationStateData is an implementation of lang.Data that resolves -// references primarily (but not exclusively) using information from a State. -type evaluationStateData struct { - Evaluator *Evaluator - - // ModulePath is the path through the dynamic module tree to the module - // that references will be resolved relative to. - ModulePath addrs.ModuleInstance - - // InstanceKeyData describes the values, if any, that are accessible due - // to repetition of a containing object using "count" or "for_each" - // arguments. (It is _not_ used for the for_each inside "dynamic" blocks, - // since the user specifies in that case which variable name to locally - // shadow.) - InstanceKeyData InstanceKeyEvalData - - // Operation records the type of walk the evaluationStateData is being used - // for. - Operation walkOperation -} - -// InstanceKeyEvalData is the old name for instances.RepetitionData, aliased -// here for compatibility. In new code, use instances.RepetitionData instead. -type InstanceKeyEvalData = instances.RepetitionData - -// EvalDataForInstanceKey constructs a suitable InstanceKeyEvalData for -// evaluating in a context that has the given instance key. -// -// The forEachMap argument can be nil when preparing for evaluation -// in a context where each.value is prohibited, such as a destroy-time -// provisioner. In that case, the returned EachValue will always be -// cty.NilVal. -func EvalDataForInstanceKey(key addrs.InstanceKey, forEachMap map[string]cty.Value) InstanceKeyEvalData { - var evalData InstanceKeyEvalData - if key == nil { - return evalData - } - - keyValue := key.Value() - switch keyValue.Type() { - case cty.String: - evalData.EachKey = keyValue - evalData.EachValue = forEachMap[keyValue.AsString()] - case cty.Number: - evalData.CountIndex = keyValue - } - return evalData -} - -// EvalDataForNoInstanceKey is a value of InstanceKeyData that sets no instance -// key values at all, suitable for use in contexts where no keyed instance -// is relevant. -var EvalDataForNoInstanceKey = InstanceKeyEvalData{} - -// evaluationStateData must implement lang.Data -var _ lang.Data = (*evaluationStateData)(nil) - -// StaticValidateReferences calls [Evaluator.StaticValidateReferences] on -// the evaluator embedded in this data object, using this data object's -// static module path. -func (d *evaluationStateData) StaticValidateReferences(refs []*addrs.Reference, self addrs.Referenceable, source addrs.Referenceable) tfdiags.Diagnostics { - return d.Evaluator.StaticValidateReferences(refs, d.ModulePath.Module(), self, source) -} - -func (d *evaluationStateData) GetCountAttr(addr addrs.CountAttr, rng tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { - var diags tfdiags.Diagnostics - switch addr.Name { - - case "index": - idxVal := d.InstanceKeyData.CountIndex - if idxVal == cty.NilVal { - diags = diags.Append(&hcl.Diagnostic{ - Severity: hcl.DiagError, - Summary: `Reference to "count" in non-counted context`, - Detail: `The "count" object can only be used in "module", "resource", and "data" blocks, and only when the "count" argument is set.`, - Subject: rng.ToHCL().Ptr(), - }) - return cty.UnknownVal(cty.Number), diags - } - return idxVal, diags - - default: - diags = diags.Append(&hcl.Diagnostic{ - Severity: hcl.DiagError, - Summary: `Invalid "count" attribute`, - Detail: fmt.Sprintf(`The "count" object does not have an attribute named %q. The only supported attribute is count.index, which is the index of each instance of a resource block that has the "count" argument set.`, addr.Name), - Subject: rng.ToHCL().Ptr(), - }) - return cty.DynamicVal, diags - } -} - -func (d *evaluationStateData) GetForEachAttr(addr addrs.ForEachAttr, rng tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { - var diags tfdiags.Diagnostics - var returnVal cty.Value - switch addr.Name { - - case "key": - returnVal = d.InstanceKeyData.EachKey - case "value": - returnVal = d.InstanceKeyData.EachValue - - if returnVal == cty.NilVal { - diags = diags.Append(&hcl.Diagnostic{ - Severity: hcl.DiagError, - Summary: `each.value cannot be used in this context`, - Detail: `A reference to "each.value" has been used in a context in which it is unavailable, such as when the configuration no longer contains the value in its "for_each" expression. Remove this reference to each.value in your configuration to work around this error.`, - Subject: rng.ToHCL().Ptr(), - }) - return cty.UnknownVal(cty.DynamicPseudoType), diags - } - default: - diags = diags.Append(&hcl.Diagnostic{ - Severity: hcl.DiagError, - Summary: `Invalid "each" attribute`, - Detail: fmt.Sprintf(`The "each" object does not have an attribute named %q. The supported attributes are each.key and each.value, the current key and value pair of the "for_each" attribute set.`, addr.Name), - Subject: rng.ToHCL().Ptr(), - }) - return cty.DynamicVal, diags - } - - if returnVal == cty.NilVal { - diags = diags.Append(&hcl.Diagnostic{ - Severity: hcl.DiagError, - Summary: `Reference to "each" in context without for_each`, - Detail: `The "each" object can be used only in "module" or "resource" blocks, and only when the "for_each" argument is set.`, - Subject: rng.ToHCL().Ptr(), - }) - return cty.UnknownVal(cty.DynamicPseudoType), diags - } - return returnVal, diags -} - -func (d *evaluationStateData) GetInputVariable(addr addrs.InputVariable, rng tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { - var diags tfdiags.Diagnostics - - // First we'll make sure the requested value is declared in configuration, - // so we can produce a nice message if not. - moduleConfig := d.Evaluator.Config.DescendentForInstance(d.ModulePath) - if moduleConfig == nil { - // should never happen, since we can't be evaluating in a module - // that wasn't mentioned in configuration. - panic(fmt.Sprintf("input variable read from %s, which has no configuration", d.ModulePath)) - } - - config := moduleConfig.Module.Variables[addr.Name] - if config == nil { - var suggestions []string - for k := range moduleConfig.Module.Variables { - suggestions = append(suggestions, k) - } - suggestion := didyoumean.NameSuggestion(addr.Name, suggestions) - if suggestion != "" { - suggestion = fmt.Sprintf(" Did you mean %q?", suggestion) - } else { - suggestion = fmt.Sprintf(" This variable can be declared with a variable %q {} block.", addr.Name) - } - - diags = diags.Append(&hcl.Diagnostic{ - Severity: hcl.DiagError, - Summary: `Reference to undeclared input variable`, - Detail: fmt.Sprintf(`An input variable with the name %q has not been declared.%s`, addr.Name, suggestion), - Subject: rng.ToHCL().Ptr(), - }) - return cty.DynamicVal, diags - } - - // During the validate walk, input variables are always unknown so - // that we are validating the configuration for all possible input values - // rather than for a specific set. Checking against a specific set of - // input values then happens during the plan walk. - // - // This is important because otherwise the validation walk will tend to be - // overly strict, requiring expressions throughout the configuration to - // be complicated to accommodate all possible inputs, whereas returning - // unknown here allows for simpler patterns like using input values as - // guards to broadly enable/disable resources, avoid processing things - // that are disabled, etc. Terraform's static validation leans towards - // being liberal in what it accepts because the subsequent plan walk has - // more information available and so can be more conservative. - if d.Operation == walkValidate { - // Ensure variable sensitivity is captured in the validate walk - if config.Sensitive { - return cty.UnknownVal(config.Type).Mark(marks.Sensitive), diags - } - return cty.UnknownVal(config.Type), diags - } - - val := d.Evaluator.NamedValues.GetInputVariableValue(d.ModulePath.InputVariable(addr.Name)) - - // Mark if sensitive - if config.Sensitive { - val = val.Mark(marks.Sensitive) - } - - return val, diags -} - -func (d *evaluationStateData) GetLocalValue(addr addrs.LocalValue, rng tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { - var diags tfdiags.Diagnostics - - // First we'll make sure the requested value is declared in configuration, - // so we can produce a nice message if not. - moduleConfig := d.Evaluator.Config.DescendentForInstance(d.ModulePath) - if moduleConfig == nil { - // should never happen, since we can't be evaluating in a module - // that wasn't mentioned in configuration. - panic(fmt.Sprintf("local value read from %s, which has no configuration", d.ModulePath)) - } - - config := moduleConfig.Module.Locals[addr.Name] - if config == nil { - var suggestions []string - for k := range moduleConfig.Module.Locals { - suggestions = append(suggestions, k) - } - suggestion := didyoumean.NameSuggestion(addr.Name, suggestions) - if suggestion != "" { - suggestion = fmt.Sprintf(" Did you mean %q?", suggestion) - } - - diags = diags.Append(&hcl.Diagnostic{ - Severity: hcl.DiagError, - Summary: `Reference to undeclared local value`, - Detail: fmt.Sprintf(`A local value with the name %q has not been declared.%s`, addr.Name, suggestion), - Subject: rng.ToHCL().Ptr(), - }) - return cty.DynamicVal, diags - } - - val := d.Evaluator.NamedValues.GetLocalValue(addr.Absolute(d.ModulePath)) - return val, diags -} - -func (d *evaluationStateData) GetModule(addr addrs.ModuleCall, rng tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { - var diags tfdiags.Diagnostics - - // Output results live in the module that declares them, which is one of - // the child module instances of our current module path. - moduleAddr := d.ModulePath.Module().Child(addr.Name) - absAddr := addr.Absolute(d.ModulePath) - - parentCfg := d.Evaluator.Config.DescendentForInstance(d.ModulePath) - callConfig, ok := parentCfg.Module.ModuleCalls[addr.Name] - if !ok { - diags = diags.Append(&hcl.Diagnostic{ - Severity: hcl.DiagError, - Summary: `Reference to undeclared module`, - Detail: fmt.Sprintf(`The configuration contains no %s.`, moduleAddr), - Subject: rng.ToHCL().Ptr(), - }) - return cty.DynamicVal, diags - } - - // We'll consult the configuration to see what output names we are - // expecting, so we can ensure the resulting object is of the expected - // type even if our data is incomplete for some reason. - moduleConfig := d.Evaluator.Config.Descendent(moduleAddr) - if moduleConfig == nil { - // should never happen, since we have a valid module call above, this - // should be caught during static validation. - panic(fmt.Sprintf("output value read from %s, which has no configuration", moduleAddr)) - } - outputConfigs := moduleConfig.Module.Outputs - - // We don't do instance expansion during validation, and so we need to - // return an unknown value. Technically we should always return - // cty.DynamicVal here because the final value during plan will always - // be an object or tuple type with unpredictable attributes/elements, - // but because we never actually carry values forward from validation to - // planning we lie a little here and return unknown list and map types, - // just to give us more opportunities to catch author mistakes during - // validation. - // - // This means that in practice any expression that refers to a module - // call must be written to be valid for either a collection type or - // structural type of similar kind, so that it can be considered as - // valid during both the validate and plan walks. - if d.Operation == walkValidate { - atys := make(map[string]cty.Type, len(outputConfigs)) - for name := range outputConfigs { - atys[name] = cty.DynamicPseudoType // output values are dynamically-typed - } - instTy := cty.Object(atys) - - switch { - case callConfig.Count != nil: - return cty.UnknownVal(cty.List(instTy)), diags - case callConfig.ForEach != nil: - return cty.UnknownVal(cty.Map(instTy)), diags - default: - return cty.UnknownVal(instTy), diags - } - } - - // For all other walk types, we proceed to dynamic evaluation of individual - // instances, using the global instance expander. An earlier graph node - // should always have registered the expansion of this module call before - // we get here, unless there's a bug in the graph builders. - allInstances := d.Evaluator.Instances - instKeyType, instKeys, known := allInstances.ExpandAbsModuleCall(absAddr) - if !known { - // If we don't know which instances exist then we can't really predict - // anything at all. We can't even predict the return type based on - // instKeyType because output values are dynamically-typed and so - // our final result will always be an object or tuple type whose - // attribute/element count we cannot predict. - return cty.DynamicVal, diags - } - - instanceObjVal := func(instKey addrs.InstanceKey) (cty.Value, tfdiags.Diagnostics) { - // This function must always return a valid value, even if it's - // just a cty.DynamicVal placeholder accompanying error diagnostics. - var diags tfdiags.Diagnostics - - namedVals := d.Evaluator.NamedValues - moduleInstAddr := absAddr.Instance(instKey) - attrs := make(map[string]cty.Value, len(outputConfigs)) - for name := range outputConfigs { - outputAddr := moduleInstAddr.OutputValue(name) - - // Although we do typically expect the graph dependencies to - // ensure that values get registered before they are needed, - // we track depedencies with specific output values where - // possible, instead of with entire module calls, and so - // in this specific case it's valid for some of this call's - // output values to not be known yet, with the graph builder - // being responsible for making sure that no expression - // in the configuration can actually observe that. - if !namedVals.HasOutputValue(outputAddr) { - attrs[name] = cty.DynamicVal - continue - } - outputVal := namedVals.GetOutputValue(outputAddr) - attrs[name] = outputVal - } - - return cty.ObjectVal(attrs), diags - } - - switch instKeyType { - - case addrs.NoKeyType: - // In this case we should always have exactly one instance that - // is addrs.NoKey. If not then there's a bug in the [instances.Expander] - // implementation. - if len(instKeys) != 1 { - panic(fmt.Sprintf("module call has no instance key type but has %d instances (should be 1)", len(instKeys))) - } - ret, moreDiags := instanceObjVal(instKeys[0]) - diags = diags.Append(moreDiags) - return ret, diags - - case addrs.IntKeyType: - // We can assume that the instance keys are in ascending numerical order - // and are consecutive, per the contract of allInstances.ExpandModuleCall. - elems := make([]cty.Value, 0, len(instKeys)) - for _, instKey := range instKeys { - instVal, moreDiags := instanceObjVal(instKey) - elems = append(elems, instVal) - diags = diags.Append(moreDiags) - } - return cty.TupleVal(elems), diags - - case addrs.StringKeyType: - attrs := make(map[string]cty.Value, len(instKeys)) - for _, instKey := range instKeys { - instVal, moreDiags := instanceObjVal(instKey) - attrs[string(instKey.(addrs.StringKey))] = instVal - diags = diags.Append(moreDiags) - } - return cty.ObjectVal(attrs), diags - - default: - diags = diags.Append(&hcl.Diagnostic{ - Severity: hcl.DiagError, - Summary: `Unsupported instance key type`, - Detail: fmt.Sprintf( - `Module call %s has instance key type %#v, which is not supported by the expression evaluator. This is a bug in Terraform.`, - absAddr, instKeyType, - ), - Subject: rng.ToHCL().Ptr(), - }) - return cty.DynamicVal, diags - } -} - -func (d *evaluationStateData) GetPathAttr(addr addrs.PathAttr, rng tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { - var diags tfdiags.Diagnostics - switch addr.Name { - - case "cwd": - var err error - var wd string - if d.Evaluator.Meta != nil { - // Meta is always non-nil in the normal case, but some test cases - // are not so realistic. - wd = d.Evaluator.Meta.OriginalWorkingDir - } - if wd == "" { - wd, err = os.Getwd() - if err != nil { - diags = diags.Append(&hcl.Diagnostic{ - Severity: hcl.DiagError, - Summary: `Failed to get working directory`, - Detail: fmt.Sprintf(`The value for path.cwd cannot be determined due to a system error: %s`, err), - Subject: rng.ToHCL().Ptr(), - }) - return cty.DynamicVal, diags - } - } - // The current working directory should always be absolute, whether we - // just looked it up or whether we were relying on ContextMeta's - // (possibly non-normalized) path. - wd, err = filepath.Abs(wd) - if err != nil { - diags = diags.Append(&hcl.Diagnostic{ - Severity: hcl.DiagError, - Summary: `Failed to get working directory`, - Detail: fmt.Sprintf(`The value for path.cwd cannot be determined due to a system error: %s`, err), - Subject: rng.ToHCL().Ptr(), - }) - return cty.DynamicVal, diags - } - - return cty.StringVal(filepath.ToSlash(wd)), diags - - case "module": - moduleConfig := d.Evaluator.Config.DescendentForInstance(d.ModulePath) - if moduleConfig == nil { - // should never happen, since we can't be evaluating in a module - // that wasn't mentioned in configuration. - panic(fmt.Sprintf("module.path read from module %s, which has no configuration", d.ModulePath)) - } - sourceDir := moduleConfig.Module.SourceDir - return cty.StringVal(filepath.ToSlash(sourceDir)), diags - - case "root": - sourceDir := d.Evaluator.Config.Module.SourceDir - return cty.StringVal(filepath.ToSlash(sourceDir)), diags - - default: - suggestion := didyoumean.NameSuggestion(addr.Name, []string{"cwd", "module", "root"}) - if suggestion != "" { - suggestion = fmt.Sprintf(" Did you mean %q?", suggestion) - } - diags = diags.Append(&hcl.Diagnostic{ - Severity: hcl.DiagError, - Summary: `Invalid "path" attribute`, - Detail: fmt.Sprintf(`The "path" object does not have an attribute named %q.%s`, addr.Name, suggestion), - Subject: rng.ToHCL().Ptr(), - }) - return cty.DynamicVal, diags - } -} - -func (d *evaluationStateData) GetResource(addr addrs.Resource, rng tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { - var diags tfdiags.Diagnostics - // First we'll consult the configuration to see if an resource of this - // name is declared at all. - moduleAddr := d.ModulePath - moduleConfig := d.Evaluator.Config.DescendentForInstance(moduleAddr) - if moduleConfig == nil { - // should never happen, since we can't be evaluating in a module - // that wasn't mentioned in configuration. - panic(fmt.Sprintf("resource value read from %s, which has no configuration", moduleAddr)) - } - - config := moduleConfig.Module.ResourceByAddr(addr) - if config == nil { - diags = diags.Append(&hcl.Diagnostic{ - Severity: hcl.DiagError, - Summary: `Reference to undeclared resource`, - Detail: fmt.Sprintf(`A resource %q %q has not been declared in %s`, addr.Type, addr.Name, moduleDisplayAddr(moduleAddr)), - Subject: rng.ToHCL().Ptr(), - }) - return cty.DynamicVal, diags - } - - // Much of this function was written before we had factored out the handling - // of instance keys into the separate instance expander model, and so it - // does a bunch of instance-related work itself below. - // - // Currently, unknown instance keys are only possible when planning with - // DeferralAllowed set to true in the PlanOpts, which should only be the - // case in the stacks runtime (not the "normal terraform" modules runtime). - // Thus, we have some amount of duplicated code remaining, to be more - // certain that stacks-specific behaviors won't leak out into the standard - // runtime. - // - // TODO: When deferred actions are more stable and robust in stacks, it - // would be nice to rework this function to rely on the ResourceInstanceKeys - // result for _all_ of its work, rather than continuing to duplicate a bunch - // of the logic we've tried to encapsulate over ther already. - if d.Operation == walkPlan { - if _, _, hasUnknownKeys := d.Evaluator.Instances.ResourceInstanceKeys(addr.Absolute(moduleAddr)); hasUnknownKeys { - // There really isn't anything interesting we can do in this situation, - // because it means we have an unknown for_each/count, in which case - // we can't even predict what the result type will be because it - // would be either an object or tuple type decided based on the instance - // keys. - // (We can't get in here for a single-instance resource because in that - // case we would know that there's only one key and it's addrs.NoKey, - // so we'll fall through to the other logic below.) - return cty.DynamicVal, diags - } - } - - // Build the provider address from configuration, since we may not have - // state available in all cases. - // We need to build an abs provider address, but we can use a default - // instance since we're only interested in the schema. - schema := d.getResourceSchema(addr, config.Provider) - if schema == nil { - // This shouldn't happen, since validation before we get here should've - // taken care of it, but we'll show a reasonable error message anyway. - diags = diags.Append(&hcl.Diagnostic{ - Severity: hcl.DiagError, - Summary: `Missing resource type schema`, - Detail: fmt.Sprintf("No schema is available for %s in %s. This is a bug in Terraform and should be reported.", addr, config.Provider), - Subject: rng.ToHCL().Ptr(), - }) - return cty.DynamicVal, diags - } - ty := schema.ImpliedType() - - rs := d.Evaluator.State.Resource(addr.Absolute(d.ModulePath)) - - if rs == nil { - switch d.Operation { - case walkPlan, walkApply: - // During plan and apply as we evaluate each removed instance they - // are removed from the working state. Since we know there are no - // instances, return an empty container of the expected type. - switch { - case config.Count != nil: - return cty.EmptyTupleVal, diags - case config.ForEach != nil: - return cty.EmptyObjectVal, diags - default: - // While we can reference an expanded resource with 0 - // instances, we cannot reference instances that do not exist. - // Due to the fact that we may have direct references to - // instances that may end up in a root output during destroy - // (since a planned destroy cannot yet remove root outputs), we - // need to return a dynamic value here to allow evaluation to - // continue. - log.Printf("[ERROR] unknown instance %q referenced during %s", addr.Absolute(d.ModulePath), d.Operation) - return cty.DynamicVal, diags - } - - case walkImport: - // Import does not yet plan resource changes, so new resources from - // config are not going to be found here. Once walkImport fully - // plans resources, this case should not longer be needed. - // In the single instance case, we can return a typed unknown value - // for the instance to better satisfy other expressions using the - // value. This of course will not help if statically known - // attributes are expected to be known elsewhere, but reduces the - // number of problematic configs for now. - // Unlike in plan and apply above we can't be sure the count or - // for_each instances are empty, so we return a DynamicVal. We - // don't really have a good value to return otherwise -- empty - // values will fail for direct index expressions, and unknown - // Lists and Maps could fail in some type unifications. - switch { - case config.Count != nil: - return cty.DynamicVal, diags - case config.ForEach != nil: - return cty.DynamicVal, diags - default: - return cty.UnknownVal(ty), diags - } - - default: - // We should only end up here during the validate walk, - // since later walks should have at least partial states populated - // for all resources in the configuration. - return cty.DynamicVal, diags - } - } - - // Decode all instances in the current state - instances := map[addrs.InstanceKey]cty.Value{} - pendingDestroy := d.Operation == walkDestroy - for key, is := range rs.Instances { - if is == nil || is.Current == nil { - // Assume we're dealing with an instance that hasn't been created yet. - instances[key] = cty.UnknownVal(ty) - continue - } - - instAddr := addr.Instance(key).Absolute(d.ModulePath) - - change := d.Evaluator.Changes.GetResourceInstanceChange(instAddr, addrs.NotDeposed) - if change != nil { - // Don't take any resources that are yet to be deleted into account. - // If the referenced resource is CreateBeforeDestroy, then orphaned - // instances will be in the state, as they are not destroyed until - // after their dependants are updated. - if change.Action == plans.Delete { - if !pendingDestroy { - continue - } - } - } - - // Planned resources are temporarily stored in state with empty values, - // and need to be replaced by the planned value here. - if is.Current.Status == states.ObjectPlanned { - if change == nil { - // If the object is in planned status then we should not get - // here, since we should have found a pending value in the plan - // above instead. - diags = diags.Append(&hcl.Diagnostic{ - Severity: hcl.DiagError, - Summary: "Missing pending object in plan", - Detail: fmt.Sprintf("Instance %s is marked as having a change pending but that change is not recorded in the plan. This is a bug in Terraform; please report it.", instAddr), - Subject: &config.DeclRange, - }) - continue - } - val, err := change.After.Decode(ty) - if err != nil { - diags = diags.Append(&hcl.Diagnostic{ - Severity: hcl.DiagError, - Summary: "Invalid resource instance data in plan", - Detail: fmt.Sprintf("Instance %s data could not be decoded from the plan: %s.", instAddr, err), - Subject: &config.DeclRange, - }) - continue - } - - // Unlike decoding state, decoding a change does not automatically - // mark values. - // FIXME: Correct that inconsistency by moving this logic into - // the decoder function in the plans package, so that we can - // test that behavior being implemented in only one place. - instances[key] = marks.MarkPaths(val, marks.Sensitive, change.AfterSensitivePaths) - continue - } - - ios, err := is.Current.Decode(ty) - if err != nil { - // This shouldn't happen, since by the time we get here we - // should have upgraded the state data already. - diags = diags.Append(&hcl.Diagnostic{ - Severity: hcl.DiagError, - Summary: "Invalid resource instance data in state", - Detail: fmt.Sprintf("Instance %s data could not be decoded from the state: %s.", instAddr, err), - Subject: &config.DeclRange, - }) - continue - } - - val := ios.Value - - instances[key] = val - } - - // ret should be populated with a valid value in all cases below - var ret cty.Value - - switch { - case config.Count != nil: - // figure out what the last index we have is - length := -1 - for key := range instances { - intKey, ok := key.(addrs.IntKey) - if !ok { - continue - } - if int(intKey) >= length { - length = int(intKey) + 1 - } - } - - if length > 0 { - vals := make([]cty.Value, length) - for key, instance := range instances { - intKey, ok := key.(addrs.IntKey) - if !ok { - // old key from state, which isn't valid for evaluation - continue - } - - vals[int(intKey)] = instance - } - - // Insert unknown values where there are any missing instances - for i, v := range vals { - if v == cty.NilVal { - vals[i] = cty.UnknownVal(ty) - } - } - ret = cty.TupleVal(vals) - } else { - ret = cty.EmptyTupleVal - } - - case config.ForEach != nil: - vals := make(map[string]cty.Value) - for key, instance := range instances { - strKey, ok := key.(addrs.StringKey) - if !ok { - // old key that is being dropped and not used for evaluation - continue - } - vals[string(strKey)] = instance - } - - if len(vals) > 0 { - // We use an object rather than a map here because resource schemas - // may include dynamically-typed attributes, which will then cause - // each instance to potentially have a different runtime type even - // though they all conform to the static schema. - ret = cty.ObjectVal(vals) - } else { - ret = cty.EmptyObjectVal - } - - default: - val, ok := instances[addrs.NoKey] - if !ok { - // if the instance is missing, insert an unknown value - val = cty.UnknownVal(ty) - } - - ret = val - } - - return ret, diags -} - -func (d *evaluationStateData) getResourceSchema(addr addrs.Resource, providerAddr addrs.Provider) *configschema.Block { - schema, _, err := d.Evaluator.Plugins.ResourceTypeSchema(providerAddr, addr.Mode, addr.Type) - if err != nil { - // We have plently other codepaths that will detect and report - // schema lookup errors before we'd reach this point, so we'll just - // treat a failure here the same as having no schema. - return nil - } - return schema -} - -func (d *evaluationStateData) GetTerraformAttr(addr addrs.TerraformAttr, rng tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { - var diags tfdiags.Diagnostics - - if d.Evaluator.Meta == nil || d.Evaluator.Meta.Env == "" { - // The absense of an "env" (really: workspace) name suggests that - // we're running in a non-workspace context, such as in a component - // of a stack. terraform.workspace -- and the terraform symbol in - // general -- is a legacy thing from workspaces mode that isn't - // carried forward to stacks, because stack configurations can instead - // vary their behavior based on input variables provided in the - // deployment configuration. - switch addr.Name { - case "workspace": - diags = diags.Append(&hcl.Diagnostic{ - Severity: hcl.DiagError, - Summary: `Invalid reference`, - Detail: `The terraform.workspace attribute is only available for modules used in Terraform workspaces. Use input variables instead to create variations between different instances of this module.`, - Subject: rng.ToHCL().Ptr(), - }) - default: - // A more generic error for any other attribute name, since no - // others are valid anyway but it would be confusing to mention - // terraform.workspace here. - diags = diags.Append(&hcl.Diagnostic{ - Severity: hcl.DiagError, - Summary: `Invalid reference`, - Detail: `The "terraform" object is only available for modules used in Terraform workspaces.`, - Subject: rng.ToHCL().Ptr(), - }) - } - return cty.DynamicVal, diags - } - - switch addr.Name { - - case "workspace": - workspaceName := d.Evaluator.Meta.Env - return cty.StringVal(workspaceName), diags - - case "env": - // Prior to Terraform 0.12 there was an attribute "env", which was - // an alias name for "workspace". This was deprecated and is now - // removed. - diags = diags.Append(&hcl.Diagnostic{ - Severity: hcl.DiagError, - Summary: `Invalid "terraform" attribute`, - Detail: `The terraform.env attribute was deprecated in v0.10 and removed in v0.12. The "state environment" concept was renamed to "workspace" in v0.12, and so the workspace name can now be accessed using the terraform.workspace attribute.`, - Subject: rng.ToHCL().Ptr(), - }) - return cty.DynamicVal, diags - - default: - diags = diags.Append(&hcl.Diagnostic{ - Severity: hcl.DiagError, - Summary: `Invalid "terraform" attribute`, - Detail: fmt.Sprintf(`The "terraform" object does not have an attribute named %q. The only supported attribute is terraform.workspace, the name of the currently-selected workspace.`, addr.Name), - Subject: rng.ToHCL().Ptr(), - }) - return cty.DynamicVal, diags - } -} - -func (d *evaluationStateData) GetOutput(addr addrs.OutputValue, rng tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { - var diags tfdiags.Diagnostics - - // First we'll make sure the requested value is declared in configuration, - // so we can produce a nice message if not. - moduleConfig := d.Evaluator.Config.DescendentForInstance(d.ModulePath) - if moduleConfig == nil { - // should never happen, since we can't be evaluating in a module - // that wasn't mentioned in configuration. - panic(fmt.Sprintf("output value read from %s, which has no configuration", d.ModulePath)) - } - - config := moduleConfig.Module.Outputs[addr.Name] - if config == nil { - var suggestions []string - for k := range moduleConfig.Module.Outputs { - suggestions = append(suggestions, k) - } - suggestion := didyoumean.NameSuggestion(addr.Name, suggestions) - if suggestion != "" { - suggestion = fmt.Sprintf(" Did you mean %q?", suggestion) - } - - diags = diags.Append(&hcl.Diagnostic{ - Severity: hcl.DiagError, - Summary: `Reference to undeclared output value`, - Detail: fmt.Sprintf(`An output value with the name %q has not been declared.%s`, addr.Name, suggestion), - Subject: rng.ToHCL().Ptr(), - }) - return cty.DynamicVal, diags - } - - output := d.Evaluator.State.OutputValue(addr.Absolute(d.ModulePath)) - if output == nil { - // Then the output itself returned null, so we'll package that up and - // pass it on. - output = &states.OutputValue{ - Addr: addr.Absolute(d.ModulePath), - Value: cty.NilVal, - Sensitive: config.Sensitive, - } - } else if output.Value == cty.NilVal || output.Value.IsNull() { - // Then we did get a value but Terraform itself thought it was NilVal - // so we treat this as if the value isn't yet known. - output.Value = cty.DynamicVal - } - - val := output.Value - if output.Sensitive { - val = val.Mark(marks.Sensitive) - } - - return val, diags -} - -func (d *evaluationStateData) GetCheckBlock(addr addrs.Check, rng tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { - // For now, check blocks don't contain any meaningful data and can only - // be referenced from the testing scope within an expect_failures attribute. - // - // We've added them into the scope explicitly since they are referencable, - // but we'll actually just return an error message saying they can't be - // referenced in this context. - var diags tfdiags.Diagnostics - diags = diags.Append(&hcl.Diagnostic{ - Severity: hcl.DiagError, - Summary: "Reference to \"check\" in invalid context", - Detail: "The \"check\" object can only be referenced from an \"expect_failures\" attribute within a Terraform testing \"run\" block.", - Subject: rng.ToHCL().Ptr(), - }) - return cty.NilVal, diags -} - -func (d *evaluationStateData) GetRunBlock(run addrs.Run, rng tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { - // We should not get here because any scope that has an [evaluationStateData] - // as its Data should have a reference parser that doesn't accept addrs.Run - // addresses. - panic("GetRunBlock called on non-test evaluation dataset") -} - // moduleDisplayAddr returns a string describing the given module instance // address that is appropriate for returning to users in situations where the // root module is possible. Specifically, it returns "the root module" if the diff --git a/internal/terraform/evaluate_data.go b/internal/terraform/evaluate_data.go new file mode 100644 index 000000000000..8fd58d8dd97e --- /dev/null +++ b/internal/terraform/evaluate_data.go @@ -0,0 +1,190 @@ +// Copyright (c) HashiCorp, Inc. +// SPDX-License-Identifier: BUSL-1.1 + +package terraform + +import ( + "fmt" + "os" + "path/filepath" + + "github.com/hashicorp/hcl/v2" + "github.com/zclconf/go-cty/cty" + + "github.com/hashicorp/terraform/internal/addrs" + "github.com/hashicorp/terraform/internal/didyoumean" + "github.com/hashicorp/terraform/internal/tfdiags" +) + +// evaluationData is the base struct for evaluating data from within Terraform +// Core. It contains some common data and functions shared by the various +// implemented evaluators. +type evaluationData struct { + Evaluator *Evaluator + + // Module is the unexpanded module that this data is being evaluated within. + Module addrs.Module +} + +// GetPathAttr implements lang.Data. +func (d *evaluationData) GetPathAttr(addr addrs.PathAttr, rng tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { + var diags tfdiags.Diagnostics + switch addr.Name { + + case "cwd": + var err error + var wd string + if d.Evaluator.Meta != nil { + // Meta is always non-nil in the normal case, but some test cases + // are not so realistic. + wd = d.Evaluator.Meta.OriginalWorkingDir + } + if wd == "" { + wd, err = os.Getwd() + if err != nil { + diags = diags.Append(&hcl.Diagnostic{ + Severity: hcl.DiagError, + Summary: `Failed to get working directory`, + Detail: fmt.Sprintf(`The value for path.cwd cannot be determined due to a system error: %s`, err), + Subject: rng.ToHCL().Ptr(), + }) + return cty.DynamicVal, diags + } + } + // The current working directory should always be absolute, whether we + // just looked it up or whether we were relying on ContextMeta's + // (possibly non-normalized) path. + wd, err = filepath.Abs(wd) + if err != nil { + diags = diags.Append(&hcl.Diagnostic{ + Severity: hcl.DiagError, + Summary: `Failed to get working directory`, + Detail: fmt.Sprintf(`The value for path.cwd cannot be determined due to a system error: %s`, err), + Subject: rng.ToHCL().Ptr(), + }) + return cty.DynamicVal, diags + } + + return cty.StringVal(filepath.ToSlash(wd)), diags + + case "module": + moduleConfig := d.Evaluator.Config.Descendent(d.Module) + if moduleConfig == nil { + // should never happen, since we can't be evaluating in a module + // that wasn't mentioned in configuration. + panic(fmt.Sprintf("module.path read from module %s, which has no configuration", d.Module)) + } + sourceDir := moduleConfig.Module.SourceDir + return cty.StringVal(filepath.ToSlash(sourceDir)), diags + + case "root": + sourceDir := d.Evaluator.Config.Module.SourceDir + return cty.StringVal(filepath.ToSlash(sourceDir)), diags + + default: + suggestion := didyoumean.NameSuggestion(addr.Name, []string{"cwd", "module", "root"}) + if suggestion != "" { + suggestion = fmt.Sprintf(" Did you mean %q?", suggestion) + } + diags = diags.Append(&hcl.Diagnostic{ + Severity: hcl.DiagError, + Summary: `Invalid "path" attribute`, + Detail: fmt.Sprintf(`The "path" object does not have an attribute named %q.%s`, addr.Name, suggestion), + Subject: rng.ToHCL().Ptr(), + }) + return cty.DynamicVal, diags + } +} + +// GetTerraformAttr implements lang.Data. +func (d *evaluationData) GetTerraformAttr(addr addrs.TerraformAttr, rng tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { + var diags tfdiags.Diagnostics + + if d.Evaluator.Meta == nil || d.Evaluator.Meta.Env == "" { + // The absense of an "env" (really: workspace) name suggests that + // we're running in a non-workspace context, such as in a component + // of a stack. terraform.workspace -- and the terraform symbol in + // general -- is a legacy thing from workspaces mode that isn't + // carried forward to stacks, because stack configurations can instead + // vary their behavior based on input variables provided in the + // deployment configuration. + switch addr.Name { + case "workspace": + diags = diags.Append(&hcl.Diagnostic{ + Severity: hcl.DiagError, + Summary: `Invalid reference`, + Detail: `The terraform.workspace attribute is only available for modules used in Terraform workspaces. Use input variables instead to create variations between different instances of this module.`, + Subject: rng.ToHCL().Ptr(), + }) + default: + // A more generic error for any other attribute name, since no + // others are valid anyway but it would be confusing to mention + // terraform.workspace here. + diags = diags.Append(&hcl.Diagnostic{ + Severity: hcl.DiagError, + Summary: `Invalid reference`, + Detail: `The "terraform" object is only available for modules used in Terraform workspaces.`, + Subject: rng.ToHCL().Ptr(), + }) + } + return cty.DynamicVal, diags + } + + switch addr.Name { + + case "workspace": + workspaceName := d.Evaluator.Meta.Env + return cty.StringVal(workspaceName), diags + + case "env": + // Prior to Terraform 0.12 there was an attribute "env", which was + // an alias name for "workspace". This was deprecated and is now + // removed. + diags = diags.Append(&hcl.Diagnostic{ + Severity: hcl.DiagError, + Summary: `Invalid "terraform" attribute`, + Detail: `The terraform.env attribute was deprecated in v0.10 and removed in v0.12. The "state environment" concept was renamed to "workspace" in v0.12, and so the workspace name can now be accessed using the terraform.workspace attribute.`, + Subject: rng.ToHCL().Ptr(), + }) + return cty.DynamicVal, diags + + default: + diags = diags.Append(&hcl.Diagnostic{ + Severity: hcl.DiagError, + Summary: `Invalid "terraform" attribute`, + Detail: fmt.Sprintf(`The "terraform" object does not have an attribute named %q. The only supported attribute is terraform.workspace, the name of the currently-selected workspace.`, addr.Name), + Subject: rng.ToHCL().Ptr(), + }) + return cty.DynamicVal, diags + } +} + +// StaticValidateReferences implements lang.Data. +func (d *evaluationData) StaticValidateReferences(refs []*addrs.Reference, self addrs.Referenceable, source addrs.Referenceable) tfdiags.Diagnostics { + return d.Evaluator.StaticValidateReferences(refs, d.Module, self, source) +} + +// GetRunBlock implements lang.Data. +func (d *evaluationData) GetRunBlock(addrs.Run, tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { + // We should not get here because any scope that has an [evaluationPlaceholderData] + // as its Data should have a reference parser that doesn't accept addrs.Run + // addresses. + panic("GetRunBlock called on non-test evaluation dataset") +} + +func (d *evaluationData) GetCheckBlock(addr addrs.Check, rng tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { + // For now, check blocks don't contain any meaningful data and can only + // be referenced from the testing scope within an expect_failures attribute. + // + // We've added them into the scope explicitly since they are referencable, + // but we'll actually just return an error message saying they can't be + // referenced in this context. + var diags tfdiags.Diagnostics + diags = diags.Append(&hcl.Diagnostic{ + Severity: hcl.DiagError, + Summary: "Reference to \"check\" in invalid context", + Detail: "The \"check\" object can only be referenced from an \"expect_failures\" attribute within a Terraform testing \"run\" block.", + Subject: rng.ToHCL().Ptr(), + }) + return cty.NilVal, diags +} diff --git a/internal/terraform/evaluate_placeholder.go b/internal/terraform/evaluate_placeholder.go index 70359bf6dccb..d087665b269b 100644 --- a/internal/terraform/evaluate_placeholder.go +++ b/internal/terraform/evaluate_placeholder.go @@ -9,6 +9,7 @@ import ( "github.com/zclconf/go-cty/cty" "github.com/hashicorp/hcl/v2" + "github.com/hashicorp/terraform/internal/addrs" "github.com/hashicorp/terraform/internal/lang" "github.com/hashicorp/terraform/internal/tfdiags" @@ -20,7 +21,7 @@ import ( // only what we know to be true for all possible final module instances // that could exist for the prefix. type evaluationPlaceholderData struct { - Evaluator *Evaluator + *evaluationData // ModulePath is the partially-expanded path through the dynamic module // tree to a set of possible module instances that share a common known @@ -55,21 +56,6 @@ type evaluationPlaceholderData struct { var _ lang.Data = (*evaluationPlaceholderData)(nil) -// GetCheckBlock implements lang.Data. -func (d *evaluationPlaceholderData) GetCheckBlock(addr addrs.Check, rng tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { - // check blocks don't produce any useful data and can only be referred - // to within an expect_failures attribute in the test language. - var diags tfdiags.Diagnostics - diags = diags.Append(&hcl.Diagnostic{ - Severity: hcl.DiagError, - Summary: "Reference to \"check\" in invalid context", - Detail: "The \"check\" object can only be used from an \"expect_failures\" attribute within a Terraform testing \"run\" block.", - Subject: rng.ToHCL().Ptr(), - }) - return cty.NilVal, diags - -} - // GetCountAttr implements lang.Data. func (d *evaluationPlaceholderData) GetCountAttr(addr addrs.CountAttr, rng tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { var diags tfdiags.Diagnostics @@ -224,21 +210,6 @@ func (d *evaluationPlaceholderData) GetOutput(addr addrs.OutputValue, rng tfdiag } -// GetPathAttr implements lang.Data. -func (d *evaluationPlaceholderData) GetPathAttr(addrs.PathAttr, tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { - // TODO: It would be helpful to perform the same logic here as we do - // in the full-evaluation case, since the paths we'd return here cannot - // vary based on dynamic data, but we'll need to factor out the logic - // into a common location we can call from both places first. For now, - // we'll just leave these all as unknown value placeholders. - // - // What we _do_ know is that all valid attributes of "path" are strings - // that are definitely not null, so we can at least catch situations - // where someone tries to use them in a place where a string is - // unacceptable. - return cty.UnknownVal(cty.String).RefineNotNull(), nil -} - // GetResource implements lang.Data. func (d *evaluationPlaceholderData) GetResource(addrs.Resource, tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { // TODO: Once we've implemented the evaluation of placeholders for @@ -251,25 +222,3 @@ func (d *evaluationPlaceholderData) GetResource(addrs.Resource, tfdiags.SourceRa // really help references to single-instance resources. return cty.DynamicVal, nil } - -// GetRunBlock implements lang.Data. -func (d *evaluationPlaceholderData) GetRunBlock(addrs.Run, tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { - // We should not get here because any scope that has an [evaluationPlaceholderData] - // as its Data should have a reference parser that doesn't accept addrs.Run - // addresses. - panic("GetRunBlock called on non-test evaluation dataset") -} - -// GetTerraformAttr implements lang.Data. -func (d *evaluationPlaceholderData) GetTerraformAttr(addrs.TerraformAttr, tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { - // TODO: It would be helpful to perform the same validation checks that - // occur in evaluationStateData.GetTerraformAttr, so authors can catch - // invalid usage of the "terraform" object even when under an unexpanded - // module prefix. - return cty.DynamicVal, nil -} - -// StaticValidateReferences implements lang.Data. -func (d *evaluationPlaceholderData) StaticValidateReferences(refs []*addrs.Reference, self addrs.Referenceable, source addrs.Referenceable) tfdiags.Diagnostics { - return d.Evaluator.StaticValidateReferences(refs, d.ModulePath.Module(), self, source) -} diff --git a/internal/terraform/evaluate_state.go b/internal/terraform/evaluate_state.go new file mode 100644 index 000000000000..7554791df9ef --- /dev/null +++ b/internal/terraform/evaluate_state.go @@ -0,0 +1,753 @@ +// Copyright (c) HashiCorp, Inc. +// SPDX-License-Identifier: BUSL-1.1 + +package terraform + +import ( + "fmt" + "log" + + "github.com/hashicorp/hcl/v2" + "github.com/zclconf/go-cty/cty" + + "github.com/hashicorp/terraform/internal/addrs" + "github.com/hashicorp/terraform/internal/configs/configschema" + "github.com/hashicorp/terraform/internal/didyoumean" + "github.com/hashicorp/terraform/internal/instances" + "github.com/hashicorp/terraform/internal/lang" + "github.com/hashicorp/terraform/internal/lang/marks" + "github.com/hashicorp/terraform/internal/plans" + "github.com/hashicorp/terraform/internal/states" + "github.com/hashicorp/terraform/internal/tfdiags" +) + +// evaluationStateData is an implementation of lang.Data that resolves +// references primarily (but not exclusively) using information from a State. +type evaluationStateData struct { + *evaluationData + + // ModulePath is the path through the dynamic module tree to the module + // that references will be resolved relative to. + ModulePath addrs.ModuleInstance + + // InstanceKeyData describes the values, if any, that are accessible due + // to repetition of a containing object using "count" or "for_each" + // arguments. (It is _not_ used for the for_each inside "dynamic" blocks, + // since the user specifies in that case which variable name to locally + // shadow.) + InstanceKeyData InstanceKeyEvalData + + // Operation records the type of walk the evaluationStateData is being used + // for. + Operation walkOperation +} + +// InstanceKeyEvalData is the old name for instances.RepetitionData, aliased +// here for compatibility. In new code, use instances.RepetitionData instead. +type InstanceKeyEvalData = instances.RepetitionData + +// EvalDataForInstanceKey constructs a suitable InstanceKeyEvalData for +// evaluating in a context that has the given instance key. +// +// The forEachMap argument can be nil when preparing for evaluation +// in a context where each.value is prohibited, such as a destroy-time +// provisioner. In that case, the returned EachValue will always be +// cty.NilVal. +func EvalDataForInstanceKey(key addrs.InstanceKey, forEachMap map[string]cty.Value) InstanceKeyEvalData { + var evalData InstanceKeyEvalData + if key == nil { + return evalData + } + + keyValue := key.Value() + switch keyValue.Type() { + case cty.String: + evalData.EachKey = keyValue + evalData.EachValue = forEachMap[keyValue.AsString()] + case cty.Number: + evalData.CountIndex = keyValue + } + return evalData +} + +// EvalDataForNoInstanceKey is a value of InstanceKeyData that sets no instance +// key values at all, suitable for use in contexts where no keyed instance +// is relevant. +var EvalDataForNoInstanceKey = InstanceKeyEvalData{} + +// evaluationStateData must implement lang.Data +var _ lang.Data = (*evaluationStateData)(nil) + +// StaticValidateReferences calls [Evaluator.StaticValidateReferences] on +// the evaluator embedded in this data object, using this data object's +// static module path. +func (d *evaluationStateData) StaticValidateReferences(refs []*addrs.Reference, self addrs.Referenceable, source addrs.Referenceable) tfdiags.Diagnostics { + return d.Evaluator.StaticValidateReferences(refs, d.ModulePath.Module(), self, source) +} + +func (d *evaluationStateData) GetCountAttr(addr addrs.CountAttr, rng tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { + var diags tfdiags.Diagnostics + switch addr.Name { + + case "index": + idxVal := d.InstanceKeyData.CountIndex + if idxVal == cty.NilVal { + diags = diags.Append(&hcl.Diagnostic{ + Severity: hcl.DiagError, + Summary: `Reference to "count" in non-counted context`, + Detail: `The "count" object can only be used in "module", "resource", and "data" blocks, and only when the "count" argument is set.`, + Subject: rng.ToHCL().Ptr(), + }) + return cty.UnknownVal(cty.Number), diags + } + return idxVal, diags + + default: + diags = diags.Append(&hcl.Diagnostic{ + Severity: hcl.DiagError, + Summary: `Invalid "count" attribute`, + Detail: fmt.Sprintf(`The "count" object does not have an attribute named %q. The only supported attribute is count.index, which is the index of each instance of a resource block that has the "count" argument set.`, addr.Name), + Subject: rng.ToHCL().Ptr(), + }) + return cty.DynamicVal, diags + } +} + +func (d *evaluationStateData) GetForEachAttr(addr addrs.ForEachAttr, rng tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { + var diags tfdiags.Diagnostics + var returnVal cty.Value + switch addr.Name { + + case "key": + returnVal = d.InstanceKeyData.EachKey + case "value": + returnVal = d.InstanceKeyData.EachValue + + if returnVal == cty.NilVal { + diags = diags.Append(&hcl.Diagnostic{ + Severity: hcl.DiagError, + Summary: `each.value cannot be used in this context`, + Detail: `A reference to "each.value" has been used in a context in which it is unavailable, such as when the configuration no longer contains the value in its "for_each" expression. Remove this reference to each.value in your configuration to work around this error.`, + Subject: rng.ToHCL().Ptr(), + }) + return cty.UnknownVal(cty.DynamicPseudoType), diags + } + default: + diags = diags.Append(&hcl.Diagnostic{ + Severity: hcl.DiagError, + Summary: `Invalid "each" attribute`, + Detail: fmt.Sprintf(`The "each" object does not have an attribute named %q. The supported attributes are each.key and each.value, the current key and value pair of the "for_each" attribute set.`, addr.Name), + Subject: rng.ToHCL().Ptr(), + }) + return cty.DynamicVal, diags + } + + if returnVal == cty.NilVal { + diags = diags.Append(&hcl.Diagnostic{ + Severity: hcl.DiagError, + Summary: `Reference to "each" in context without for_each`, + Detail: `The "each" object can be used only in "module" or "resource" blocks, and only when the "for_each" argument is set.`, + Subject: rng.ToHCL().Ptr(), + }) + return cty.UnknownVal(cty.DynamicPseudoType), diags + } + return returnVal, diags +} + +func (d *evaluationStateData) GetInputVariable(addr addrs.InputVariable, rng tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { + var diags tfdiags.Diagnostics + + // First we'll make sure the requested value is declared in configuration, + // so we can produce a nice message if not. + moduleConfig := d.Evaluator.Config.DescendentForInstance(d.ModulePath) + if moduleConfig == nil { + // should never happen, since we can't be evaluating in a module + // that wasn't mentioned in configuration. + panic(fmt.Sprintf("input variable read from %s, which has no configuration", d.ModulePath)) + } + + config := moduleConfig.Module.Variables[addr.Name] + if config == nil { + var suggestions []string + for k := range moduleConfig.Module.Variables { + suggestions = append(suggestions, k) + } + suggestion := didyoumean.NameSuggestion(addr.Name, suggestions) + if suggestion != "" { + suggestion = fmt.Sprintf(" Did you mean %q?", suggestion) + } else { + suggestion = fmt.Sprintf(" This variable can be declared with a variable %q {} block.", addr.Name) + } + + diags = diags.Append(&hcl.Diagnostic{ + Severity: hcl.DiagError, + Summary: `Reference to undeclared input variable`, + Detail: fmt.Sprintf(`An input variable with the name %q has not been declared.%s`, addr.Name, suggestion), + Subject: rng.ToHCL().Ptr(), + }) + return cty.DynamicVal, diags + } + + // During the validate walk, input variables are always unknown so + // that we are validating the configuration for all possible input values + // rather than for a specific set. Checking against a specific set of + // input values then happens during the plan walk. + // + // This is important because otherwise the validation walk will tend to be + // overly strict, requiring expressions throughout the configuration to + // be complicated to accommodate all possible inputs, whereas returning + // unknown here allows for simpler patterns like using input values as + // guards to broadly enable/disable resources, avoid processing things + // that are disabled, etc. Terraform's static validation leans towards + // being liberal in what it accepts because the subsequent plan walk has + // more information available and so can be more conservative. + if d.Operation == walkValidate { + // Ensure variable sensitivity is captured in the validate walk + if config.Sensitive { + return cty.UnknownVal(config.Type).Mark(marks.Sensitive), diags + } + return cty.UnknownVal(config.Type), diags + } + + val := d.Evaluator.NamedValues.GetInputVariableValue(d.ModulePath.InputVariable(addr.Name)) + + // Mark if sensitive + if config.Sensitive { + val = val.Mark(marks.Sensitive) + } + + return val, diags +} + +func (d *evaluationStateData) GetLocalValue(addr addrs.LocalValue, rng tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { + var diags tfdiags.Diagnostics + + // First we'll make sure the requested value is declared in configuration, + // so we can produce a nice message if not. + moduleConfig := d.Evaluator.Config.DescendentForInstance(d.ModulePath) + if moduleConfig == nil { + // should never happen, since we can't be evaluating in a module + // that wasn't mentioned in configuration. + panic(fmt.Sprintf("local value read from %s, which has no configuration", d.ModulePath)) + } + + config := moduleConfig.Module.Locals[addr.Name] + if config == nil { + var suggestions []string + for k := range moduleConfig.Module.Locals { + suggestions = append(suggestions, k) + } + suggestion := didyoumean.NameSuggestion(addr.Name, suggestions) + if suggestion != "" { + suggestion = fmt.Sprintf(" Did you mean %q?", suggestion) + } + + diags = diags.Append(&hcl.Diagnostic{ + Severity: hcl.DiagError, + Summary: `Reference to undeclared local value`, + Detail: fmt.Sprintf(`A local value with the name %q has not been declared.%s`, addr.Name, suggestion), + Subject: rng.ToHCL().Ptr(), + }) + return cty.DynamicVal, diags + } + + val := d.Evaluator.NamedValues.GetLocalValue(addr.Absolute(d.ModulePath)) + return val, diags +} + +func (d *evaluationStateData) GetModule(addr addrs.ModuleCall, rng tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { + var diags tfdiags.Diagnostics + + // Output results live in the module that declares them, which is one of + // the child module instances of our current module path. + moduleAddr := d.ModulePath.Module().Child(addr.Name) + absAddr := addr.Absolute(d.ModulePath) + + parentCfg := d.Evaluator.Config.DescendentForInstance(d.ModulePath) + callConfig, ok := parentCfg.Module.ModuleCalls[addr.Name] + if !ok { + diags = diags.Append(&hcl.Diagnostic{ + Severity: hcl.DiagError, + Summary: `Reference to undeclared module`, + Detail: fmt.Sprintf(`The configuration contains no %s.`, moduleAddr), + Subject: rng.ToHCL().Ptr(), + }) + return cty.DynamicVal, diags + } + + // We'll consult the configuration to see what output names we are + // expecting, so we can ensure the resulting object is of the expected + // type even if our data is incomplete for some reason. + moduleConfig := d.Evaluator.Config.Descendent(moduleAddr) + if moduleConfig == nil { + // should never happen, since we have a valid module call above, this + // should be caught during static validation. + panic(fmt.Sprintf("output value read from %s, which has no configuration", moduleAddr)) + } + outputConfigs := moduleConfig.Module.Outputs + + // We don't do instance expansion during validation, and so we need to + // return an unknown value. Technically we should always return + // cty.DynamicVal here because the final value during plan will always + // be an object or tuple type with unpredictable attributes/elements, + // but because we never actually carry values forward from validation to + // planning we lie a little here and return unknown list and map types, + // just to give us more opportunities to catch author mistakes during + // validation. + // + // This means that in practice any expression that refers to a module + // call must be written to be valid for either a collection type or + // structural type of similar kind, so that it can be considered as + // valid during both the validate and plan walks. + if d.Operation == walkValidate { + atys := make(map[string]cty.Type, len(outputConfigs)) + for name := range outputConfigs { + atys[name] = cty.DynamicPseudoType // output values are dynamically-typed + } + instTy := cty.Object(atys) + + switch { + case callConfig.Count != nil: + return cty.UnknownVal(cty.List(instTy)), diags + case callConfig.ForEach != nil: + return cty.UnknownVal(cty.Map(instTy)), diags + default: + return cty.UnknownVal(instTy), diags + } + } + + // For all other walk types, we proceed to dynamic evaluation of individual + // instances, using the global instance expander. An earlier graph node + // should always have registered the expansion of this module call before + // we get here, unless there's a bug in the graph builders. + allInstances := d.Evaluator.Instances + instKeyType, instKeys, known := allInstances.ExpandAbsModuleCall(absAddr) + if !known { + // If we don't know which instances exist then we can't really predict + // anything at all. We can't even predict the return type based on + // instKeyType because output values are dynamically-typed and so + // our final result will always be an object or tuple type whose + // attribute/element count we cannot predict. + return cty.DynamicVal, diags + } + + instanceObjVal := func(instKey addrs.InstanceKey) (cty.Value, tfdiags.Diagnostics) { + // This function must always return a valid value, even if it's + // just a cty.DynamicVal placeholder accompanying error diagnostics. + var diags tfdiags.Diagnostics + + namedVals := d.Evaluator.NamedValues + moduleInstAddr := absAddr.Instance(instKey) + attrs := make(map[string]cty.Value, len(outputConfigs)) + for name := range outputConfigs { + outputAddr := moduleInstAddr.OutputValue(name) + + // Although we do typically expect the graph dependencies to + // ensure that values get registered before they are needed, + // we track depedencies with specific output values where + // possible, instead of with entire module calls, and so + // in this specific case it's valid for some of this call's + // output values to not be known yet, with the graph builder + // being responsible for making sure that no expression + // in the configuration can actually observe that. + if !namedVals.HasOutputValue(outputAddr) { + attrs[name] = cty.DynamicVal + continue + } + outputVal := namedVals.GetOutputValue(outputAddr) + attrs[name] = outputVal + } + + return cty.ObjectVal(attrs), diags + } + + switch instKeyType { + + case addrs.NoKeyType: + // In this case we should always have exactly one instance that + // is addrs.NoKey. If not then there's a bug in the [instances.Expander] + // implementation. + if len(instKeys) != 1 { + panic(fmt.Sprintf("module call has no instance key type but has %d instances (should be 1)", len(instKeys))) + } + ret, moreDiags := instanceObjVal(instKeys[0]) + diags = diags.Append(moreDiags) + return ret, diags + + case addrs.IntKeyType: + // We can assume that the instance keys are in ascending numerical order + // and are consecutive, per the contract of allInstances.ExpandModuleCall. + elems := make([]cty.Value, 0, len(instKeys)) + for _, instKey := range instKeys { + instVal, moreDiags := instanceObjVal(instKey) + elems = append(elems, instVal) + diags = diags.Append(moreDiags) + } + return cty.TupleVal(elems), diags + + case addrs.StringKeyType: + attrs := make(map[string]cty.Value, len(instKeys)) + for _, instKey := range instKeys { + instVal, moreDiags := instanceObjVal(instKey) + attrs[string(instKey.(addrs.StringKey))] = instVal + diags = diags.Append(moreDiags) + } + return cty.ObjectVal(attrs), diags + + default: + diags = diags.Append(&hcl.Diagnostic{ + Severity: hcl.DiagError, + Summary: `Unsupported instance key type`, + Detail: fmt.Sprintf( + `Module call %s has instance key type %#v, which is not supported by the expression evaluator. This is a bug in Terraform.`, + absAddr, instKeyType, + ), + Subject: rng.ToHCL().Ptr(), + }) + return cty.DynamicVal, diags + } +} + +func (d *evaluationStateData) GetResource(addr addrs.Resource, rng tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { + var diags tfdiags.Diagnostics + // First we'll consult the configuration to see if an resource of this + // name is declared at all. + moduleAddr := d.ModulePath + moduleConfig := d.Evaluator.Config.DescendentForInstance(moduleAddr) + if moduleConfig == nil { + // should never happen, since we can't be evaluating in a module + // that wasn't mentioned in configuration. + panic(fmt.Sprintf("resource value read from %s, which has no configuration", moduleAddr)) + } + + config := moduleConfig.Module.ResourceByAddr(addr) + if config == nil { + diags = diags.Append(&hcl.Diagnostic{ + Severity: hcl.DiagError, + Summary: `Reference to undeclared resource`, + Detail: fmt.Sprintf(`A resource %q %q has not been declared in %s`, addr.Type, addr.Name, moduleDisplayAddr(moduleAddr)), + Subject: rng.ToHCL().Ptr(), + }) + return cty.DynamicVal, diags + } + + // Much of this function was written before we had factored out the handling + // of instance keys into the separate instance expander model, and so it + // does a bunch of instance-related work itself below. + // + // Currently, unknown instance keys are only possible when planning with + // DeferralAllowed set to true in the PlanOpts, which should only be the + // case in the stacks runtime (not the "normal terraform" modules runtime). + // Thus, we have some amount of duplicated code remaining, to be more + // certain that stacks-specific behaviors won't leak out into the standard + // runtime. + // + // TODO: When deferred actions are more stable and robust in stacks, it + // would be nice to rework this function to rely on the ResourceInstanceKeys + // result for _all_ of its work, rather than continuing to duplicate a bunch + // of the logic we've tried to encapsulate over ther already. + if d.Operation == walkPlan { + if _, _, hasUnknownKeys := d.Evaluator.Instances.ResourceInstanceKeys(addr.Absolute(moduleAddr)); hasUnknownKeys { + // There really isn't anything interesting we can do in this situation, + // because it means we have an unknown for_each/count, in which case + // we can't even predict what the result type will be because it + // would be either an object or tuple type decided based on the instance + // keys. + // (We can't get in here for a single-instance resource because in that + // case we would know that there's only one key and it's addrs.NoKey, + // so we'll fall through to the other logic below.) + return cty.DynamicVal, diags + } + } + + // Build the provider address from configuration, since we may not have + // state available in all cases. + // We need to build an abs provider address, but we can use a default + // instance since we're only interested in the schema. + schema := d.getResourceSchema(addr, config.Provider) + if schema == nil { + // This shouldn't happen, since validation before we get here should've + // taken care of it, but we'll show a reasonable error message anyway. + diags = diags.Append(&hcl.Diagnostic{ + Severity: hcl.DiagError, + Summary: `Missing resource type schema`, + Detail: fmt.Sprintf("No schema is available for %s in %s. This is a bug in Terraform and should be reported.", addr, config.Provider), + Subject: rng.ToHCL().Ptr(), + }) + return cty.DynamicVal, diags + } + ty := schema.ImpliedType() + + rs := d.Evaluator.State.Resource(addr.Absolute(d.ModulePath)) + + if rs == nil { + switch d.Operation { + case walkPlan, walkApply: + // During plan and apply as we evaluate each removed instance they + // are removed from the working state. Since we know there are no + // instances, return an empty container of the expected type. + switch { + case config.Count != nil: + return cty.EmptyTupleVal, diags + case config.ForEach != nil: + return cty.EmptyObjectVal, diags + default: + // While we can reference an expanded resource with 0 + // instances, we cannot reference instances that do not exist. + // Due to the fact that we may have direct references to + // instances that may end up in a root output during destroy + // (since a planned destroy cannot yet remove root outputs), we + // need to return a dynamic value here to allow evaluation to + // continue. + log.Printf("[ERROR] unknown instance %q referenced during %s", addr.Absolute(d.ModulePath), d.Operation) + return cty.DynamicVal, diags + } + + case walkImport: + // Import does not yet plan resource changes, so new resources from + // config are not going to be found here. Once walkImport fully + // plans resources, this case should not longer be needed. + // In the single instance case, we can return a typed unknown value + // for the instance to better satisfy other expressions using the + // value. This of course will not help if statically known + // attributes are expected to be known elsewhere, but reduces the + // number of problematic configs for now. + // Unlike in plan and apply above we can't be sure the count or + // for_each instances are empty, so we return a DynamicVal. We + // don't really have a good value to return otherwise -- empty + // values will fail for direct index expressions, and unknown + // Lists and Maps could fail in some type unifications. + switch { + case config.Count != nil: + return cty.DynamicVal, diags + case config.ForEach != nil: + return cty.DynamicVal, diags + default: + return cty.UnknownVal(ty), diags + } + + default: + // We should only end up here during the validate walk, + // since later walks should have at least partial states populated + // for all resources in the configuration. + return cty.DynamicVal, diags + } + } + + // Decode all instances in the current state + instances := map[addrs.InstanceKey]cty.Value{} + pendingDestroy := d.Operation == walkDestroy + for key, is := range rs.Instances { + if is == nil || is.Current == nil { + // Assume we're dealing with an instance that hasn't been created yet. + instances[key] = cty.UnknownVal(ty) + continue + } + + instAddr := addr.Instance(key).Absolute(d.ModulePath) + + change := d.Evaluator.Changes.GetResourceInstanceChange(instAddr, addrs.NotDeposed) + if change != nil { + // Don't take any resources that are yet to be deleted into account. + // If the referenced resource is CreateBeforeDestroy, then orphaned + // instances will be in the state, as they are not destroyed until + // after their dependants are updated. + if change.Action == plans.Delete { + if !pendingDestroy { + continue + } + } + } + + // Planned resources are temporarily stored in state with empty values, + // and need to be replaced by the planned value here. + if is.Current.Status == states.ObjectPlanned { + if change == nil { + // If the object is in planned status then we should not get + // here, since we should have found a pending value in the plan + // above instead. + diags = diags.Append(&hcl.Diagnostic{ + Severity: hcl.DiagError, + Summary: "Missing pending object in plan", + Detail: fmt.Sprintf("Instance %s is marked as having a change pending but that change is not recorded in the plan. This is a bug in Terraform; please report it.", instAddr), + Subject: &config.DeclRange, + }) + continue + } + val, err := change.After.Decode(ty) + if err != nil { + diags = diags.Append(&hcl.Diagnostic{ + Severity: hcl.DiagError, + Summary: "Invalid resource instance data in plan", + Detail: fmt.Sprintf("Instance %s data could not be decoded from the plan: %s.", instAddr, err), + Subject: &config.DeclRange, + }) + continue + } + + // Unlike decoding state, decoding a change does not automatically + // mark values. + // FIXME: Correct that inconsistency by moving this logic into + // the decoder function in the plans package, so that we can + // test that behavior being implemented in only one place. + instances[key] = marks.MarkPaths(val, marks.Sensitive, change.AfterSensitivePaths) + continue + } + + ios, err := is.Current.Decode(ty) + if err != nil { + // This shouldn't happen, since by the time we get here we + // should have upgraded the state data already. + diags = diags.Append(&hcl.Diagnostic{ + Severity: hcl.DiagError, + Summary: "Invalid resource instance data in state", + Detail: fmt.Sprintf("Instance %s data could not be decoded from the state: %s.", instAddr, err), + Subject: &config.DeclRange, + }) + continue + } + + val := ios.Value + + instances[key] = val + } + + // ret should be populated with a valid value in all cases below + var ret cty.Value + + switch { + case config.Count != nil: + // figure out what the last index we have is + length := -1 + for key := range instances { + intKey, ok := key.(addrs.IntKey) + if !ok { + continue + } + if int(intKey) >= length { + length = int(intKey) + 1 + } + } + + if length > 0 { + vals := make([]cty.Value, length) + for key, instance := range instances { + intKey, ok := key.(addrs.IntKey) + if !ok { + // old key from state, which isn't valid for evaluation + continue + } + + vals[int(intKey)] = instance + } + + // Insert unknown values where there are any missing instances + for i, v := range vals { + if v == cty.NilVal { + vals[i] = cty.UnknownVal(ty) + } + } + ret = cty.TupleVal(vals) + } else { + ret = cty.EmptyTupleVal + } + + case config.ForEach != nil: + vals := make(map[string]cty.Value) + for key, instance := range instances { + strKey, ok := key.(addrs.StringKey) + if !ok { + // old key that is being dropped and not used for evaluation + continue + } + vals[string(strKey)] = instance + } + + if len(vals) > 0 { + // We use an object rather than a map here because resource schemas + // may include dynamically-typed attributes, which will then cause + // each instance to potentially have a different runtime type even + // though they all conform to the static schema. + ret = cty.ObjectVal(vals) + } else { + ret = cty.EmptyObjectVal + } + + default: + val, ok := instances[addrs.NoKey] + if !ok { + // if the instance is missing, insert an unknown value + val = cty.UnknownVal(ty) + } + + ret = val + } + + return ret, diags +} + +func (d *evaluationStateData) getResourceSchema(addr addrs.Resource, providerAddr addrs.Provider) *configschema.Block { + schema, _, err := d.Evaluator.Plugins.ResourceTypeSchema(providerAddr, addr.Mode, addr.Type) + if err != nil { + // We have plently other codepaths that will detect and report + // schema lookup errors before we'd reach this point, so we'll just + // treat a failure here the same as having no schema. + return nil + } + return schema +} + +func (d *evaluationStateData) GetOutput(addr addrs.OutputValue, rng tfdiags.SourceRange) (cty.Value, tfdiags.Diagnostics) { + var diags tfdiags.Diagnostics + + // First we'll make sure the requested value is declared in configuration, + // so we can produce a nice message if not. + moduleConfig := d.Evaluator.Config.DescendentForInstance(d.ModulePath) + if moduleConfig == nil { + // should never happen, since we can't be evaluating in a module + // that wasn't mentioned in configuration. + panic(fmt.Sprintf("output value read from %s, which has no configuration", d.ModulePath)) + } + + config := moduleConfig.Module.Outputs[addr.Name] + if config == nil { + var suggestions []string + for k := range moduleConfig.Module.Outputs { + suggestions = append(suggestions, k) + } + suggestion := didyoumean.NameSuggestion(addr.Name, suggestions) + if suggestion != "" { + suggestion = fmt.Sprintf(" Did you mean %q?", suggestion) + } + + diags = diags.Append(&hcl.Diagnostic{ + Severity: hcl.DiagError, + Summary: `Reference to undeclared output value`, + Detail: fmt.Sprintf(`An output value with the name %q has not been declared.%s`, addr.Name, suggestion), + Subject: rng.ToHCL().Ptr(), + }) + return cty.DynamicVal, diags + } + + output := d.Evaluator.State.OutputValue(addr.Absolute(d.ModulePath)) + if output == nil { + // Then the output itself returned null, so we'll package that up and + // pass it on. + output = &states.OutputValue{ + Addr: addr.Absolute(d.ModulePath), + Value: cty.NilVal, + Sensitive: config.Sensitive, + } + } else if output.Value == cty.NilVal || output.Value.IsNull() { + // Then we did get a value but Terraform itself thought it was NilVal + // so we treat this as if the value isn't yet known. + output.Value = cty.DynamicVal + } + + val := output.Value + if output.Sensitive { + val = val.Mark(marks.Sensitive) + } + + return val, diags +} diff --git a/internal/terraform/evaluate_test.go b/internal/terraform/evaluate_test.go index c929e7c7e2fd..614aeccd0423 100644 --- a/internal/terraform/evaluate_test.go +++ b/internal/terraform/evaluate_test.go @@ -30,7 +30,9 @@ func TestEvaluatorGetTerraformAttr(t *testing.T) { NamedValues: namedvals.NewState(), } data := &evaluationStateData{ - Evaluator: evaluator, + evaluationData: &evaluationData{ + Evaluator: evaluator, + }, } scope := evaluator.Scope(data, nil, nil, lang.ExternalFuncs{}) @@ -61,7 +63,9 @@ func TestEvaluatorGetPathAttr(t *testing.T) { NamedValues: namedvals.NewState(), } data := &evaluationStateData{ - Evaluator: evaluator, + evaluationData: &evaluationData{ + Evaluator: evaluator, + }, } scope := evaluator.Scope(data, nil, nil, lang.ExternalFuncs{}) @@ -127,7 +131,9 @@ func TestEvaluatorGetOutputValue(t *testing.T) { } data := &evaluationStateData{ - Evaluator: evaluator, + evaluationData: &evaluationData{ + Evaluator: evaluator, + }, } scope := evaluator.Scope(data, nil, nil, lang.ExternalFuncs{}) @@ -196,7 +202,9 @@ func TestEvaluatorGetInputVariable(t *testing.T) { } data := &evaluationStateData{ - Evaluator: evaluator, + evaluationData: &evaluationData{ + Evaluator: evaluator, + }, } scope := evaluator.Scope(data, nil, nil, lang.ExternalFuncs{}) @@ -355,7 +363,9 @@ func TestEvaluatorGetResource(t *testing.T) { } data := &evaluationStateData{ - Evaluator: evaluator, + evaluationData: &evaluationData{ + Evaluator: evaluator, + }, } scope := evaluator.Scope(data, nil, nil, lang.ExternalFuncs{}) @@ -522,7 +532,9 @@ func TestEvaluatorGetResource_changes(t *testing.T) { } data := &evaluationStateData{ - Evaluator: evaluator, + evaluationData: &evaluationData{ + Evaluator: evaluator, + }, } scope := evaluator.Scope(data, nil, nil, lang.ExternalFuncs{}) @@ -554,7 +566,9 @@ func TestEvaluatorGetModule(t *testing.T) { cty.StringVal("bar").Mark(marks.Sensitive), ) data := &evaluationStateData{ - Evaluator: evaluator, + evaluationData: &evaluationData{ + Evaluator: evaluator, + }, } scope := evaluator.Scope(data, nil, nil, lang.ExternalFuncs{}) want := cty.ObjectVal(map[string]cty.Value{"out": cty.StringVal("bar").Mark(marks.Sensitive)})