mirror of
https://github.com/opentofu/opentofu.git
synced 2025-01-07 14:44:11 -06:00
terraform: remove old resource count transformer
This commit is contained in:
parent
4e4d51e6ba
commit
d820d186e1
@ -156,13 +156,6 @@ func (n *GraphNodeConfigResource) DynamicExpand(ctx EvalContext) (*Graph, error)
|
|||||||
// Start creating the steps
|
// Start creating the steps
|
||||||
steps := make([]GraphTransformer, 0, 5)
|
steps := make([]GraphTransformer, 0, 5)
|
||||||
|
|
||||||
// Expand counts.
|
|
||||||
steps = append(steps, &ResourceCountTransformerOld{
|
|
||||||
Resource: n.Resource,
|
|
||||||
Destroy: n.Destroy,
|
|
||||||
Targets: n.Targets,
|
|
||||||
})
|
|
||||||
|
|
||||||
// Additional destroy modifications.
|
// Additional destroy modifications.
|
||||||
if n.Destroy {
|
if n.Destroy {
|
||||||
steps = append(steps, &DeposedTransformer{
|
steps = append(steps, &DeposedTransformer{
|
||||||
|
@ -5,92 +5,8 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/hashicorp/terraform/config"
|
"github.com/hashicorp/terraform/config"
|
||||||
"github.com/hashicorp/terraform/dag"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// ResourceCountTransformerOld is a GraphTransformer that expands the count
|
|
||||||
// out for a specific resource.
|
|
||||||
type ResourceCountTransformerOld struct {
|
|
||||||
Resource *config.Resource
|
|
||||||
Destroy bool
|
|
||||||
Targets []ResourceAddress
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *ResourceCountTransformerOld) Transform(g *Graph) error {
|
|
||||||
// Expand the resource count
|
|
||||||
count, err := t.Resource.Count()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Don't allow the count to be negative
|
|
||||||
if count < 0 {
|
|
||||||
return fmt.Errorf("negative count: %d", count)
|
|
||||||
}
|
|
||||||
|
|
||||||
// For each count, build and add the node
|
|
||||||
nodes := make([]dag.Vertex, 0, count)
|
|
||||||
for i := 0; i < count; i++ {
|
|
||||||
// Set the index. If our count is 1 we special case it so that
|
|
||||||
// we handle the "resource.0" and "resource" boundary properly.
|
|
||||||
index := i
|
|
||||||
if count == 1 {
|
|
||||||
index = -1
|
|
||||||
}
|
|
||||||
|
|
||||||
// Save the node for later so we can do connections. Make the
|
|
||||||
// proper node depending on if we're just a destroy node or if
|
|
||||||
// were a regular node.
|
|
||||||
var node dag.Vertex = &graphNodeExpandedResource{
|
|
||||||
Index: index,
|
|
||||||
Resource: t.Resource,
|
|
||||||
Path: g.Path,
|
|
||||||
}
|
|
||||||
if t.Destroy {
|
|
||||||
node = &graphNodeExpandedResourceDestroy{
|
|
||||||
graphNodeExpandedResource: node.(*graphNodeExpandedResource),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Skip nodes if targeting excludes them
|
|
||||||
if !t.nodeIsTargeted(node) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add the node now
|
|
||||||
nodes = append(nodes, node)
|
|
||||||
g.Add(node)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Make the dependency connections
|
|
||||||
for _, n := range nodes {
|
|
||||||
// Connect the dependents. We ignore the return value for missing
|
|
||||||
// dependents since that should've been caught at a higher level.
|
|
||||||
g.ConnectDependent(n)
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *ResourceCountTransformerOld) nodeIsTargeted(node dag.Vertex) bool {
|
|
||||||
// no targets specified, everything stays in the graph
|
|
||||||
if len(t.Targets) == 0 {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
addressable, ok := node.(GraphNodeAddressable)
|
|
||||||
if !ok {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
addr := addressable.ResourceAddress()
|
|
||||||
for _, targetAddr := range t.Targets {
|
|
||||||
if targetAddr.Equals(addr) {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
type graphNodeExpandedResource struct {
|
type graphNodeExpandedResource struct {
|
||||||
Index int
|
Index int
|
||||||
Resource *config.Resource
|
Resource *config.Resource
|
||||||
@ -194,650 +110,6 @@ func (n *graphNodeExpandedResource) StateDependencies() []string {
|
|||||||
return deps
|
return deps
|
||||||
}
|
}
|
||||||
|
|
||||||
// GraphNodeEvalable impl.
|
|
||||||
func (n *graphNodeExpandedResource) EvalTree() EvalNode {
|
|
||||||
var provider ResourceProvider
|
|
||||||
var resourceConfig *ResourceConfig
|
|
||||||
|
|
||||||
// Build the resource. If we aren't part of a multi-resource, then
|
|
||||||
// we still consider ourselves as count index zero.
|
|
||||||
index := n.Index
|
|
||||||
if index < 0 {
|
|
||||||
index = 0
|
|
||||||
}
|
|
||||||
resource := &Resource{
|
|
||||||
Name: n.Resource.Name,
|
|
||||||
Type: n.Resource.Type,
|
|
||||||
CountIndex: index,
|
|
||||||
}
|
|
||||||
|
|
||||||
seq := &EvalSequence{Nodes: make([]EvalNode, 0, 5)}
|
|
||||||
|
|
||||||
// Validate the resource
|
|
||||||
vseq := &EvalSequence{Nodes: make([]EvalNode, 0, 5)}
|
|
||||||
vseq.Nodes = append(vseq.Nodes, &EvalGetProvider{
|
|
||||||
Name: n.ProvidedBy()[0],
|
|
||||||
Output: &provider,
|
|
||||||
})
|
|
||||||
vseq.Nodes = append(vseq.Nodes, &EvalInterpolate{
|
|
||||||
Config: n.Resource.RawConfig.Copy(),
|
|
||||||
Resource: resource,
|
|
||||||
Output: &resourceConfig,
|
|
||||||
})
|
|
||||||
vseq.Nodes = append(vseq.Nodes, &EvalValidateResource{
|
|
||||||
Provider: &provider,
|
|
||||||
Config: &resourceConfig,
|
|
||||||
ResourceName: n.Resource.Name,
|
|
||||||
ResourceType: n.Resource.Type,
|
|
||||||
ResourceMode: n.Resource.Mode,
|
|
||||||
})
|
|
||||||
|
|
||||||
// Validate all the provisioners
|
|
||||||
for _, p := range n.Resource.Provisioners {
|
|
||||||
var provisioner ResourceProvisioner
|
|
||||||
vseq.Nodes = append(vseq.Nodes, &EvalGetProvisioner{
|
|
||||||
Name: p.Type,
|
|
||||||
Output: &provisioner,
|
|
||||||
}, &EvalInterpolate{
|
|
||||||
Config: p.RawConfig.Copy(),
|
|
||||||
Resource: resource,
|
|
||||||
Output: &resourceConfig,
|
|
||||||
}, &EvalValidateProvisioner{
|
|
||||||
Provisioner: &provisioner,
|
|
||||||
Config: &resourceConfig,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add the validation operations
|
|
||||||
seq.Nodes = append(seq.Nodes, &EvalOpFilter{
|
|
||||||
Ops: []walkOperation{walkValidate},
|
|
||||||
Node: vseq,
|
|
||||||
})
|
|
||||||
|
|
||||||
// Build instance info
|
|
||||||
info := n.instanceInfo()
|
|
||||||
seq.Nodes = append(seq.Nodes, &EvalInstanceInfo{Info: info})
|
|
||||||
|
|
||||||
// Each resource mode has its own lifecycle
|
|
||||||
switch n.Resource.Mode {
|
|
||||||
case config.ManagedResourceMode:
|
|
||||||
seq.Nodes = append(
|
|
||||||
seq.Nodes,
|
|
||||||
n.managedResourceEvalNodes(resource, info, resourceConfig)...,
|
|
||||||
)
|
|
||||||
case config.DataResourceMode:
|
|
||||||
seq.Nodes = append(
|
|
||||||
seq.Nodes,
|
|
||||||
n.dataResourceEvalNodes(resource, info, resourceConfig)...,
|
|
||||||
)
|
|
||||||
default:
|
|
||||||
panic(fmt.Errorf("unsupported resource mode %s", n.Resource.Mode))
|
|
||||||
}
|
|
||||||
|
|
||||||
return seq
|
|
||||||
}
|
|
||||||
|
|
||||||
func (n *graphNodeExpandedResource) managedResourceEvalNodes(resource *Resource, info *InstanceInfo, resourceConfig *ResourceConfig) []EvalNode {
|
|
||||||
var diff *InstanceDiff
|
|
||||||
var provider ResourceProvider
|
|
||||||
var state *InstanceState
|
|
||||||
|
|
||||||
nodes := make([]EvalNode, 0, 5)
|
|
||||||
|
|
||||||
// Refresh the resource
|
|
||||||
nodes = append(nodes, &EvalOpFilter{
|
|
||||||
Ops: []walkOperation{walkRefresh},
|
|
||||||
Node: &EvalSequence{
|
|
||||||
Nodes: []EvalNode{
|
|
||||||
&EvalGetProvider{
|
|
||||||
Name: n.ProvidedBy()[0],
|
|
||||||
Output: &provider,
|
|
||||||
},
|
|
||||||
&EvalReadState{
|
|
||||||
Name: n.stateId(),
|
|
||||||
Output: &state,
|
|
||||||
},
|
|
||||||
&EvalRefresh{
|
|
||||||
Info: info,
|
|
||||||
Provider: &provider,
|
|
||||||
State: &state,
|
|
||||||
Output: &state,
|
|
||||||
},
|
|
||||||
&EvalWriteState{
|
|
||||||
Name: n.stateId(),
|
|
||||||
ResourceType: n.Resource.Type,
|
|
||||||
Provider: n.Resource.Provider,
|
|
||||||
Dependencies: n.StateDependencies(),
|
|
||||||
State: &state,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
})
|
|
||||||
|
|
||||||
// Diff the resource
|
|
||||||
nodes = append(nodes, &EvalOpFilter{
|
|
||||||
Ops: []walkOperation{walkPlan},
|
|
||||||
Node: &EvalSequence{
|
|
||||||
Nodes: []EvalNode{
|
|
||||||
&EvalInterpolate{
|
|
||||||
Config: n.Resource.RawConfig.Copy(),
|
|
||||||
Resource: resource,
|
|
||||||
Output: &resourceConfig,
|
|
||||||
},
|
|
||||||
&EvalGetProvider{
|
|
||||||
Name: n.ProvidedBy()[0],
|
|
||||||
Output: &provider,
|
|
||||||
},
|
|
||||||
// Re-run validation to catch any errors we missed, e.g. type
|
|
||||||
// mismatches on computed values.
|
|
||||||
&EvalValidateResource{
|
|
||||||
Provider: &provider,
|
|
||||||
Config: &resourceConfig,
|
|
||||||
ResourceName: n.Resource.Name,
|
|
||||||
ResourceType: n.Resource.Type,
|
|
||||||
ResourceMode: n.Resource.Mode,
|
|
||||||
IgnoreWarnings: true,
|
|
||||||
},
|
|
||||||
&EvalReadState{
|
|
||||||
Name: n.stateId(),
|
|
||||||
Output: &state,
|
|
||||||
},
|
|
||||||
&EvalDiff{
|
|
||||||
Info: info,
|
|
||||||
Config: &resourceConfig,
|
|
||||||
Resource: n.Resource,
|
|
||||||
Provider: &provider,
|
|
||||||
State: &state,
|
|
||||||
OutputDiff: &diff,
|
|
||||||
OutputState: &state,
|
|
||||||
},
|
|
||||||
&EvalCheckPreventDestroy{
|
|
||||||
Resource: n.Resource,
|
|
||||||
Diff: &diff,
|
|
||||||
},
|
|
||||||
&EvalWriteState{
|
|
||||||
Name: n.stateId(),
|
|
||||||
ResourceType: n.Resource.Type,
|
|
||||||
Provider: n.Resource.Provider,
|
|
||||||
Dependencies: n.StateDependencies(),
|
|
||||||
State: &state,
|
|
||||||
},
|
|
||||||
&EvalWriteDiff{
|
|
||||||
Name: n.stateId(),
|
|
||||||
Diff: &diff,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
})
|
|
||||||
|
|
||||||
// Diff the resource for destruction
|
|
||||||
nodes = append(nodes, &EvalOpFilter{
|
|
||||||
Ops: []walkOperation{walkPlanDestroy},
|
|
||||||
Node: &EvalSequence{
|
|
||||||
Nodes: []EvalNode{
|
|
||||||
&EvalReadState{
|
|
||||||
Name: n.stateId(),
|
|
||||||
Output: &state,
|
|
||||||
},
|
|
||||||
&EvalDiffDestroy{
|
|
||||||
Info: info,
|
|
||||||
State: &state,
|
|
||||||
Output: &diff,
|
|
||||||
},
|
|
||||||
&EvalCheckPreventDestroy{
|
|
||||||
Resource: n.Resource,
|
|
||||||
Diff: &diff,
|
|
||||||
},
|
|
||||||
&EvalWriteDiff{
|
|
||||||
Name: n.stateId(),
|
|
||||||
Diff: &diff,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
})
|
|
||||||
|
|
||||||
// Apply
|
|
||||||
var diffApply *InstanceDiff
|
|
||||||
var err error
|
|
||||||
var createNew bool
|
|
||||||
var createBeforeDestroyEnabled bool
|
|
||||||
nodes = append(nodes, &EvalOpFilter{
|
|
||||||
Ops: []walkOperation{walkApply, walkDestroy},
|
|
||||||
Node: &EvalSequence{
|
|
||||||
Nodes: []EvalNode{
|
|
||||||
// Get the saved diff for apply
|
|
||||||
&EvalReadDiff{
|
|
||||||
Name: n.stateId(),
|
|
||||||
Diff: &diffApply,
|
|
||||||
},
|
|
||||||
|
|
||||||
// We don't want to do any destroys
|
|
||||||
&EvalIf{
|
|
||||||
If: func(ctx EvalContext) (bool, error) {
|
|
||||||
if diffApply == nil {
|
|
||||||
return true, EvalEarlyExitError{}
|
|
||||||
}
|
|
||||||
|
|
||||||
if diffApply.GetDestroy() && diffApply.GetAttributesLen() == 0 {
|
|
||||||
return true, EvalEarlyExitError{}
|
|
||||||
}
|
|
||||||
|
|
||||||
diffApply.SetDestroy(false)
|
|
||||||
return true, nil
|
|
||||||
},
|
|
||||||
Then: EvalNoop{},
|
|
||||||
},
|
|
||||||
|
|
||||||
&EvalIf{
|
|
||||||
If: func(ctx EvalContext) (bool, error) {
|
|
||||||
destroy := false
|
|
||||||
if diffApply != nil {
|
|
||||||
destroy = diffApply.GetDestroy() || diffApply.RequiresNew()
|
|
||||||
}
|
|
||||||
|
|
||||||
createBeforeDestroyEnabled =
|
|
||||||
n.Resource.Lifecycle.CreateBeforeDestroy &&
|
|
||||||
destroy
|
|
||||||
|
|
||||||
return createBeforeDestroyEnabled, nil
|
|
||||||
},
|
|
||||||
Then: &EvalDeposeState{
|
|
||||||
Name: n.stateId(),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
&EvalInterpolate{
|
|
||||||
Config: n.Resource.RawConfig.Copy(),
|
|
||||||
Resource: resource,
|
|
||||||
Output: &resourceConfig,
|
|
||||||
},
|
|
||||||
&EvalGetProvider{
|
|
||||||
Name: n.ProvidedBy()[0],
|
|
||||||
Output: &provider,
|
|
||||||
},
|
|
||||||
&EvalReadState{
|
|
||||||
Name: n.stateId(),
|
|
||||||
Output: &state,
|
|
||||||
},
|
|
||||||
// Re-run validation to catch any errors we missed, e.g. type
|
|
||||||
// mismatches on computed values.
|
|
||||||
&EvalValidateResource{
|
|
||||||
Provider: &provider,
|
|
||||||
Config: &resourceConfig,
|
|
||||||
ResourceName: n.Resource.Name,
|
|
||||||
ResourceType: n.Resource.Type,
|
|
||||||
ResourceMode: n.Resource.Mode,
|
|
||||||
IgnoreWarnings: true,
|
|
||||||
},
|
|
||||||
&EvalDiff{
|
|
||||||
Info: info,
|
|
||||||
Config: &resourceConfig,
|
|
||||||
Resource: n.Resource,
|
|
||||||
Provider: &provider,
|
|
||||||
Diff: &diffApply,
|
|
||||||
State: &state,
|
|
||||||
OutputDiff: &diffApply,
|
|
||||||
},
|
|
||||||
|
|
||||||
// Get the saved diff
|
|
||||||
&EvalReadDiff{
|
|
||||||
Name: n.stateId(),
|
|
||||||
Diff: &diff,
|
|
||||||
},
|
|
||||||
|
|
||||||
// Compare the diffs
|
|
||||||
&EvalCompareDiff{
|
|
||||||
Info: info,
|
|
||||||
One: &diff,
|
|
||||||
Two: &diffApply,
|
|
||||||
},
|
|
||||||
|
|
||||||
&EvalGetProvider{
|
|
||||||
Name: n.ProvidedBy()[0],
|
|
||||||
Output: &provider,
|
|
||||||
},
|
|
||||||
&EvalReadState{
|
|
||||||
Name: n.stateId(),
|
|
||||||
Output: &state,
|
|
||||||
},
|
|
||||||
&EvalApplyPre{
|
|
||||||
Info: info,
|
|
||||||
State: &state,
|
|
||||||
Diff: &diffApply,
|
|
||||||
},
|
|
||||||
&EvalApply{
|
|
||||||
Info: info,
|
|
||||||
State: &state,
|
|
||||||
Diff: &diffApply,
|
|
||||||
Provider: &provider,
|
|
||||||
Output: &state,
|
|
||||||
Error: &err,
|
|
||||||
CreateNew: &createNew,
|
|
||||||
},
|
|
||||||
&EvalWriteState{
|
|
||||||
Name: n.stateId(),
|
|
||||||
ResourceType: n.Resource.Type,
|
|
||||||
Provider: n.Resource.Provider,
|
|
||||||
Dependencies: n.StateDependencies(),
|
|
||||||
State: &state,
|
|
||||||
},
|
|
||||||
&EvalApplyProvisioners{
|
|
||||||
Info: info,
|
|
||||||
State: &state,
|
|
||||||
Resource: n.Resource,
|
|
||||||
InterpResource: resource,
|
|
||||||
CreateNew: &createNew,
|
|
||||||
Error: &err,
|
|
||||||
},
|
|
||||||
&EvalIf{
|
|
||||||
If: func(ctx EvalContext) (bool, error) {
|
|
||||||
return createBeforeDestroyEnabled && err != nil, nil
|
|
||||||
},
|
|
||||||
Then: &EvalUndeposeState{
|
|
||||||
Name: n.stateId(),
|
|
||||||
State: &state,
|
|
||||||
},
|
|
||||||
Else: &EvalWriteState{
|
|
||||||
Name: n.stateId(),
|
|
||||||
ResourceType: n.Resource.Type,
|
|
||||||
Provider: n.Resource.Provider,
|
|
||||||
Dependencies: n.StateDependencies(),
|
|
||||||
State: &state,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
// We clear the diff out here so that future nodes
|
|
||||||
// don't see a diff that is already complete. There
|
|
||||||
// is no longer a diff!
|
|
||||||
&EvalWriteDiff{
|
|
||||||
Name: n.stateId(),
|
|
||||||
Diff: nil,
|
|
||||||
},
|
|
||||||
|
|
||||||
&EvalApplyPost{
|
|
||||||
Info: info,
|
|
||||||
State: &state,
|
|
||||||
Error: &err,
|
|
||||||
},
|
|
||||||
&EvalUpdateStateHook{},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
})
|
|
||||||
|
|
||||||
return nodes
|
|
||||||
}
|
|
||||||
|
|
||||||
func (n *graphNodeExpandedResource) dataResourceEvalNodes(resource *Resource, info *InstanceInfo, resourceConfig *ResourceConfig) []EvalNode {
|
|
||||||
//var diff *InstanceDiff
|
|
||||||
var provider ResourceProvider
|
|
||||||
var config *ResourceConfig
|
|
||||||
var diff *InstanceDiff
|
|
||||||
var state *InstanceState
|
|
||||||
|
|
||||||
nodes := make([]EvalNode, 0, 5)
|
|
||||||
|
|
||||||
// Refresh the resource
|
|
||||||
nodes = append(nodes, &EvalOpFilter{
|
|
||||||
Ops: []walkOperation{walkRefresh},
|
|
||||||
Node: &EvalSequence{
|
|
||||||
Nodes: []EvalNode{
|
|
||||||
|
|
||||||
// Always destroy the existing state first, since we must
|
|
||||||
// make sure that values from a previous read will not
|
|
||||||
// get interpolated if we end up needing to defer our
|
|
||||||
// loading until apply time.
|
|
||||||
&EvalWriteState{
|
|
||||||
Name: n.stateId(),
|
|
||||||
ResourceType: n.Resource.Type,
|
|
||||||
Provider: n.Resource.Provider,
|
|
||||||
Dependencies: n.StateDependencies(),
|
|
||||||
State: &state, // state is nil here
|
|
||||||
},
|
|
||||||
|
|
||||||
&EvalInterpolate{
|
|
||||||
Config: n.Resource.RawConfig.Copy(),
|
|
||||||
Resource: resource,
|
|
||||||
Output: &config,
|
|
||||||
},
|
|
||||||
|
|
||||||
// The rest of this pass can proceed only if there are no
|
|
||||||
// computed values in our config.
|
|
||||||
// (If there are, we'll deal with this during the plan and
|
|
||||||
// apply phases.)
|
|
||||||
&EvalIf{
|
|
||||||
If: func(ctx EvalContext) (bool, error) {
|
|
||||||
|
|
||||||
if config.ComputedKeys != nil && len(config.ComputedKeys) > 0 {
|
|
||||||
return true, EvalEarlyExitError{}
|
|
||||||
}
|
|
||||||
|
|
||||||
// If the config explicitly has a depends_on for this
|
|
||||||
// data source, assume the intention is to prevent
|
|
||||||
// refreshing ahead of that dependency.
|
|
||||||
if len(n.Resource.DependsOn) > 0 {
|
|
||||||
return true, EvalEarlyExitError{}
|
|
||||||
}
|
|
||||||
|
|
||||||
return true, nil
|
|
||||||
},
|
|
||||||
Then: EvalNoop{},
|
|
||||||
},
|
|
||||||
|
|
||||||
// The remainder of this pass is the same as running
|
|
||||||
// a "plan" pass immediately followed by an "apply" pass,
|
|
||||||
// populating the state early so it'll be available to
|
|
||||||
// provider configurations that need this data during
|
|
||||||
// refresh/plan.
|
|
||||||
|
|
||||||
&EvalGetProvider{
|
|
||||||
Name: n.ProvidedBy()[0],
|
|
||||||
Output: &provider,
|
|
||||||
},
|
|
||||||
|
|
||||||
&EvalReadDataDiff{
|
|
||||||
Info: info,
|
|
||||||
Config: &config,
|
|
||||||
Provider: &provider,
|
|
||||||
Output: &diff,
|
|
||||||
OutputState: &state,
|
|
||||||
},
|
|
||||||
|
|
||||||
&EvalReadDataApply{
|
|
||||||
Info: info,
|
|
||||||
Diff: &diff,
|
|
||||||
Provider: &provider,
|
|
||||||
Output: &state,
|
|
||||||
},
|
|
||||||
|
|
||||||
&EvalWriteState{
|
|
||||||
Name: n.stateId(),
|
|
||||||
ResourceType: n.Resource.Type,
|
|
||||||
Provider: n.Resource.Provider,
|
|
||||||
Dependencies: n.StateDependencies(),
|
|
||||||
State: &state,
|
|
||||||
},
|
|
||||||
|
|
||||||
&EvalUpdateStateHook{},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
})
|
|
||||||
|
|
||||||
// Diff the resource
|
|
||||||
nodes = append(nodes, &EvalOpFilter{
|
|
||||||
Ops: []walkOperation{walkPlan},
|
|
||||||
Node: &EvalSequence{
|
|
||||||
Nodes: []EvalNode{
|
|
||||||
|
|
||||||
&EvalReadState{
|
|
||||||
Name: n.stateId(),
|
|
||||||
Output: &state,
|
|
||||||
},
|
|
||||||
|
|
||||||
// We need to re-interpolate the config here because some
|
|
||||||
// of the attributes may have become computed during
|
|
||||||
// earlier planning, due to other resources having
|
|
||||||
// "requires new resource" diffs.
|
|
||||||
&EvalInterpolate{
|
|
||||||
Config: n.Resource.RawConfig.Copy(),
|
|
||||||
Resource: resource,
|
|
||||||
Output: &config,
|
|
||||||
},
|
|
||||||
|
|
||||||
&EvalIf{
|
|
||||||
If: func(ctx EvalContext) (bool, error) {
|
|
||||||
computed := config.ComputedKeys != nil && len(config.ComputedKeys) > 0
|
|
||||||
|
|
||||||
// If the configuration is complete and we
|
|
||||||
// already have a state then we don't need to
|
|
||||||
// do any further work during apply, because we
|
|
||||||
// already populated the state during refresh.
|
|
||||||
if !computed && state != nil {
|
|
||||||
return true, EvalEarlyExitError{}
|
|
||||||
}
|
|
||||||
|
|
||||||
return true, nil
|
|
||||||
},
|
|
||||||
Then: EvalNoop{},
|
|
||||||
},
|
|
||||||
|
|
||||||
&EvalGetProvider{
|
|
||||||
Name: n.ProvidedBy()[0],
|
|
||||||
Output: &provider,
|
|
||||||
},
|
|
||||||
|
|
||||||
&EvalReadDataDiff{
|
|
||||||
Info: info,
|
|
||||||
Config: &config,
|
|
||||||
Provider: &provider,
|
|
||||||
Output: &diff,
|
|
||||||
OutputState: &state,
|
|
||||||
},
|
|
||||||
|
|
||||||
&EvalWriteState{
|
|
||||||
Name: n.stateId(),
|
|
||||||
ResourceType: n.Resource.Type,
|
|
||||||
Provider: n.Resource.Provider,
|
|
||||||
Dependencies: n.StateDependencies(),
|
|
||||||
State: &state,
|
|
||||||
},
|
|
||||||
|
|
||||||
&EvalWriteDiff{
|
|
||||||
Name: n.stateId(),
|
|
||||||
Diff: &diff,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
})
|
|
||||||
|
|
||||||
// Diff the resource for destruction
|
|
||||||
nodes = append(nodes, &EvalOpFilter{
|
|
||||||
Ops: []walkOperation{walkPlanDestroy},
|
|
||||||
Node: &EvalSequence{
|
|
||||||
Nodes: []EvalNode{
|
|
||||||
|
|
||||||
&EvalReadState{
|
|
||||||
Name: n.stateId(),
|
|
||||||
Output: &state,
|
|
||||||
},
|
|
||||||
|
|
||||||
// Since EvalDiffDestroy doesn't interact with the
|
|
||||||
// provider at all, we can safely share the same
|
|
||||||
// implementation for data vs. managed resources.
|
|
||||||
&EvalDiffDestroy{
|
|
||||||
Info: info,
|
|
||||||
State: &state,
|
|
||||||
Output: &diff,
|
|
||||||
},
|
|
||||||
|
|
||||||
&EvalWriteDiff{
|
|
||||||
Name: n.stateId(),
|
|
||||||
Diff: &diff,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
})
|
|
||||||
|
|
||||||
// Apply
|
|
||||||
nodes = append(nodes, &EvalOpFilter{
|
|
||||||
Ops: []walkOperation{walkApply, walkDestroy},
|
|
||||||
Node: &EvalSequence{
|
|
||||||
Nodes: []EvalNode{
|
|
||||||
// Get the saved diff for apply
|
|
||||||
&EvalReadDiff{
|
|
||||||
Name: n.stateId(),
|
|
||||||
Diff: &diff,
|
|
||||||
},
|
|
||||||
|
|
||||||
// Stop here if we don't actually have a diff
|
|
||||||
&EvalIf{
|
|
||||||
If: func(ctx EvalContext) (bool, error) {
|
|
||||||
if diff == nil {
|
|
||||||
return true, EvalEarlyExitError{}
|
|
||||||
}
|
|
||||||
|
|
||||||
if diff.GetAttributesLen() == 0 {
|
|
||||||
return true, EvalEarlyExitError{}
|
|
||||||
}
|
|
||||||
|
|
||||||
return true, nil
|
|
||||||
},
|
|
||||||
Then: EvalNoop{},
|
|
||||||
},
|
|
||||||
|
|
||||||
// We need to re-interpolate the config here, rather than
|
|
||||||
// just using the diff's values directly, because we've
|
|
||||||
// potentially learned more variable values during the
|
|
||||||
// apply pass that weren't known when the diff was produced.
|
|
||||||
&EvalInterpolate{
|
|
||||||
Config: n.Resource.RawConfig.Copy(),
|
|
||||||
Resource: resource,
|
|
||||||
Output: &config,
|
|
||||||
},
|
|
||||||
|
|
||||||
&EvalGetProvider{
|
|
||||||
Name: n.ProvidedBy()[0],
|
|
||||||
Output: &provider,
|
|
||||||
},
|
|
||||||
|
|
||||||
// Make a new diff with our newly-interpolated config.
|
|
||||||
&EvalReadDataDiff{
|
|
||||||
Info: info,
|
|
||||||
Config: &config,
|
|
||||||
Previous: &diff,
|
|
||||||
Provider: &provider,
|
|
||||||
Output: &diff,
|
|
||||||
},
|
|
||||||
|
|
||||||
&EvalReadDataApply{
|
|
||||||
Info: info,
|
|
||||||
Diff: &diff,
|
|
||||||
Provider: &provider,
|
|
||||||
Output: &state,
|
|
||||||
},
|
|
||||||
|
|
||||||
&EvalWriteState{
|
|
||||||
Name: n.stateId(),
|
|
||||||
ResourceType: n.Resource.Type,
|
|
||||||
Provider: n.Resource.Provider,
|
|
||||||
Dependencies: n.StateDependencies(),
|
|
||||||
State: &state,
|
|
||||||
},
|
|
||||||
|
|
||||||
// Clear the diff now that we've applied it, so
|
|
||||||
// later nodes won't see a diff that's now a no-op.
|
|
||||||
&EvalWriteDiff{
|
|
||||||
Name: n.stateId(),
|
|
||||||
Diff: nil,
|
|
||||||
},
|
|
||||||
|
|
||||||
&EvalUpdateStateHook{},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
})
|
|
||||||
|
|
||||||
return nodes
|
|
||||||
}
|
|
||||||
|
|
||||||
// instanceInfo is used for EvalTree.
|
// instanceInfo is used for EvalTree.
|
||||||
func (n *graphNodeExpandedResource) instanceInfo() *InstanceInfo {
|
func (n *graphNodeExpandedResource) instanceInfo() *InstanceInfo {
|
||||||
return &InstanceInfo{Id: n.stateId(), Type: n.Resource.Type}
|
return &InstanceInfo{Id: n.stateId(), Type: n.Resource.Type}
|
||||||
|
@ -1,69 +0,0 @@
|
|||||||
package terraform
|
|
||||||
|
|
||||||
import (
|
|
||||||
"strings"
|
|
||||||
"testing"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestResourceCountTransformerOld(t *testing.T) {
|
|
||||||
cfg := testModule(t, "transform-resource-count-basic").Config()
|
|
||||||
resource := cfg.Resources[0]
|
|
||||||
|
|
||||||
g := Graph{Path: RootModulePath}
|
|
||||||
{
|
|
||||||
tf := &ResourceCountTransformerOld{Resource: resource}
|
|
||||||
if err := tf.Transform(&g); err != nil {
|
|
||||||
t.Fatalf("err: %s", err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
actual := strings.TrimSpace(g.String())
|
|
||||||
expected := strings.TrimSpace(testResourceCountTransformOldStr)
|
|
||||||
if actual != expected {
|
|
||||||
t.Fatalf("bad:\n\n%s", actual)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestResourceCountTransformerOld_countNegative(t *testing.T) {
|
|
||||||
cfg := testModule(t, "transform-resource-count-negative").Config()
|
|
||||||
resource := cfg.Resources[0]
|
|
||||||
|
|
||||||
g := Graph{Path: RootModulePath}
|
|
||||||
{
|
|
||||||
tf := &ResourceCountTransformerOld{Resource: resource}
|
|
||||||
if err := tf.Transform(&g); err == nil {
|
|
||||||
t.Fatal("should error")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestResourceCountTransformerOld_deps(t *testing.T) {
|
|
||||||
cfg := testModule(t, "transform-resource-count-deps").Config()
|
|
||||||
resource := cfg.Resources[0]
|
|
||||||
|
|
||||||
g := Graph{Path: RootModulePath}
|
|
||||||
{
|
|
||||||
tf := &ResourceCountTransformerOld{Resource: resource}
|
|
||||||
if err := tf.Transform(&g); err != nil {
|
|
||||||
t.Fatalf("err: %s", err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
actual := strings.TrimSpace(g.String())
|
|
||||||
expected := strings.TrimSpace(testResourceCountTransformOldDepsStr)
|
|
||||||
if actual != expected {
|
|
||||||
t.Fatalf("bad:\n\n%s", actual)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const testResourceCountTransformOldStr = `
|
|
||||||
aws_instance.foo #0
|
|
||||||
aws_instance.foo #1
|
|
||||||
aws_instance.foo #2
|
|
||||||
`
|
|
||||||
|
|
||||||
const testResourceCountTransformOldDepsStr = `
|
|
||||||
aws_instance.foo #0
|
|
||||||
aws_instance.foo #1
|
|
||||||
aws_instance.foo #0
|
|
||||||
`
|
|
Loading…
Reference in New Issue
Block a user