mirror of
https://github.com/opentofu/opentofu.git
synced 2025-01-06 14:13:16 -06:00
343279110a
Previously our graph walker expected to recieve a data structure containing schemas for all of the provider and provisioner plugins used in the configuration and state. That made sense back when terraform.NewContext was responsible for loading all of the schemas before taking any other action, but it no longer has that responsiblity. Instead, we'll now make sure that the "contextPlugins" object reaches all of the locations where we need schema -- many of which already had access to that object anyway -- and then load the needed schemas just in time. The contextPlugins object memoizes schema lookups, so we can safely call it many times with the same provider address or provisioner type name and know that it'll still only load each distinct plugin once per Context object. As of this commit, the Context.Schemas method is now a public interface only and not used by logic in the "terraform" package at all. However, that does leave us in a rather tenuous situation of relying on the fact that all practical users of terraform.Context end up calling "Schemas" at some point in order to verify that we have all of the expected versions of plugins. That's a non-obvious implicit dependency, and so in subsequent commits we'll gradually move all responsibility for verifying plugin versions into the caller of terraform.NewContext, which'll heal a long-standing architectural wart whereby the caller is responsible for installing and locating the plugin executables but not for verifying that what's installed is conforming to the current configuration and dependency lock file.
129 lines
4.6 KiB
Go
129 lines
4.6 KiB
Go
package terraform
|
|
|
|
import (
|
|
"fmt"
|
|
"log"
|
|
|
|
"github.com/hashicorp/terraform/internal/addrs"
|
|
"github.com/hashicorp/terraform/internal/configs"
|
|
"github.com/hashicorp/terraform/internal/plans"
|
|
"github.com/hashicorp/terraform/internal/states"
|
|
"github.com/hashicorp/terraform/internal/tfdiags"
|
|
"github.com/zclconf/go-cty/cty"
|
|
)
|
|
|
|
// Apply performs the actions described by the given Plan object and returns
|
|
// the resulting updated state.
|
|
//
|
|
// The given configuration *must* be the same configuration that was passed
|
|
// earlier to Context.Plan in order to create this plan.
|
|
//
|
|
// Even if the returned diagnostics contains errors, Apply always returns the
|
|
// resulting state which is likely to have been partially-updated.
|
|
func (c *Context) Apply(plan *plans.Plan, config *configs.Config) (*states.State, tfdiags.Diagnostics) {
|
|
defer c.acquireRun("apply")()
|
|
var diags tfdiags.Diagnostics
|
|
|
|
log.Printf("[DEBUG] Building and walking apply graph for %s plan", plan.UIMode)
|
|
|
|
graph, operation, moreDiags := c.applyGraph(plan, config, true)
|
|
if moreDiags.HasErrors() {
|
|
return nil, diags
|
|
}
|
|
|
|
variables := InputValues{}
|
|
for name, dyVal := range plan.VariableValues {
|
|
val, err := dyVal.Decode(cty.DynamicPseudoType)
|
|
if err != nil {
|
|
diags = diags.Append(tfdiags.Sourceless(
|
|
tfdiags.Error,
|
|
"Invalid variable value in plan",
|
|
fmt.Sprintf("Invalid value for variable %q recorded in plan file: %s.", name, err),
|
|
))
|
|
continue
|
|
}
|
|
|
|
variables[name] = &InputValue{
|
|
Value: val,
|
|
SourceType: ValueFromPlan,
|
|
}
|
|
}
|
|
|
|
workingState := plan.PriorState.DeepCopy()
|
|
walker, walkDiags := c.walk(graph, operation, &graphWalkOpts{
|
|
Config: config,
|
|
InputState: workingState,
|
|
Changes: plan.Changes,
|
|
RootVariableValues: variables,
|
|
})
|
|
diags = diags.Append(walker.NonFatalDiagnostics)
|
|
diags = diags.Append(walkDiags)
|
|
|
|
newState := walker.State.Close()
|
|
if plan.UIMode == plans.DestroyMode && !diags.HasErrors() {
|
|
// NOTE: This is a vestigial violation of the rule that we mustn't
|
|
// use plan.UIMode to affect apply-time behavior.
|
|
// We ideally ought to just call newState.PruneResourceHusks
|
|
// unconditionally here, but we historically didn't and haven't yet
|
|
// verified that it'd be safe to do so.
|
|
newState.PruneResourceHusks()
|
|
}
|
|
|
|
if len(plan.TargetAddrs) > 0 {
|
|
diags = diags.Append(tfdiags.Sourceless(
|
|
tfdiags.Warning,
|
|
"Applied changes may be incomplete",
|
|
`The plan was created with the -target option in effect, so some changes requested in the configuration may have been ignored and the output values may not be fully updated. Run the following command to verify that no other changes are pending:
|
|
terraform plan
|
|
|
|
Note that the -target option is not suitable for routine use, and is provided only for exceptional situations such as recovering from errors or mistakes, or when Terraform specifically suggests to use it as part of an error message.`,
|
|
))
|
|
}
|
|
|
|
return newState, diags
|
|
}
|
|
|
|
func (c *Context) applyGraph(plan *plans.Plan, config *configs.Config, validate bool) (*Graph, walkOperation, tfdiags.Diagnostics) {
|
|
graph, diags := (&ApplyGraphBuilder{
|
|
Config: config,
|
|
Changes: plan.Changes,
|
|
State: plan.PriorState,
|
|
Plugins: c.plugins,
|
|
Targets: plan.TargetAddrs,
|
|
ForceReplace: plan.ForceReplaceAddrs,
|
|
Validate: validate,
|
|
}).Build(addrs.RootModuleInstance)
|
|
|
|
operation := walkApply
|
|
if plan.UIMode == plans.DestroyMode {
|
|
// NOTE: This is a vestigial violation of the rule that we mustn't
|
|
// use plan.UIMode to affect apply-time behavior. It's a design error
|
|
// if anything downstream switches behavior when operation is set
|
|
// to walkDestroy, but we've not yet fully audited that.
|
|
// TODO: Audit that and remove walkDestroy as an operation mode.
|
|
operation = walkDestroy
|
|
}
|
|
|
|
return graph, operation, diags
|
|
}
|
|
|
|
// ApplyGraphForUI is a last vestage of graphs in the public interface of
|
|
// Context (as opposed to graphs as an implementation detail) intended only for
|
|
// use by the "terraform graph" command when asked to render an apply-time
|
|
// graph.
|
|
//
|
|
// The result of this is intended only for rendering ot the user as a dot
|
|
// graph, and so may change in future in order to make the result more useful
|
|
// in that context, even if drifts away from the physical graph that Terraform
|
|
// Core currently uses as an implementation detail of planning.
|
|
func (c *Context) ApplyGraphForUI(plan *plans.Plan, config *configs.Config) (*Graph, tfdiags.Diagnostics) {
|
|
// For now though, this really is just the internal graph, confusing
|
|
// implementation details and all.
|
|
|
|
var diags tfdiags.Diagnostics
|
|
|
|
graph, _, moreDiags := c.applyGraph(plan, config, false)
|
|
diags = diags.Append(moreDiags)
|
|
return graph, diags
|
|
}
|