opentofu/terraform/transform_resource.go

551 lines
12 KiB
Go
Raw Normal View History

package terraform
import (
"fmt"
"github.com/hashicorp/terraform/config"
"github.com/hashicorp/terraform/dag"
)
// ResourceCountTransformer is a GraphTransformer that expands the count
// out for a specific resource.
type ResourceCountTransformer struct {
Resource *config.Resource
2015-02-12 13:40:48 -06:00
Destroy bool
}
func (t *ResourceCountTransformer) 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, count)
for i := 0; i < count; i++ {
2015-02-11 19:18:16 -06:00
// 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
}
2015-02-12 13:40:48 -06:00
// 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{
2015-02-11 19:18:16 -06:00
Index: index,
Resource: t.Resource,
}
2015-02-12 13:40:48 -06:00
if t.Destroy {
node = &graphNodeExpandedResourceDestroy{
graphNodeExpandedResource: node.(*graphNodeExpandedResource),
}
}
// Add the node now
2015-02-12 13:40:48 -06:00
nodes[i] = node
g.Add(nodes[i])
}
// 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
}
type graphNodeExpandedResource struct {
Index int
Resource *config.Resource
}
func (n *graphNodeExpandedResource) Name() string {
if n.Index == -1 {
return n.Resource.Id()
}
return fmt.Sprintf("%s #%d", n.Resource.Id(), n.Index)
}
// GraphNodeDependable impl.
func (n *graphNodeExpandedResource) DependableName() []string {
return []string{
n.Resource.Id(),
2015-02-10 16:12:49 -06:00
n.stateId(),
}
}
// GraphNodeDependent impl.
func (n *graphNodeExpandedResource) DependentOn() []string {
config := &GraphNodeConfigResource{Resource: n.Resource}
return config.DependentOn()
}
// GraphNodeProviderConsumer
2015-02-11 16:14:05 -06:00
func (n *graphNodeExpandedResource) ProvidedBy() []string {
return []string{resourceProvider(n.Resource.Type)}
}
// GraphNodeEvalable impl.
func (n *graphNodeExpandedResource) EvalTree() EvalNode {
var diff *InstanceDiff
2015-02-14 00:58:41 -06:00
var provider ResourceProvider
var resourceConfig *ResourceConfig
2015-02-12 16:46:22 -06:00
var state *InstanceState
// 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
}
2015-02-23 16:56:02 -06:00
resource := &Resource{
Name: n.Resource.Name,
Type: n.Resource.Type,
CountIndex: index,
}
2015-02-09 13:15:54 -06:00
seq := &EvalSequence{Nodes: make([]EvalNode, 0, 5)}
// Validate the resource
vseq := &EvalSequence{Nodes: make([]EvalNode, 0, 5)}
2015-02-14 00:58:41 -06:00
vseq.Nodes = append(vseq.Nodes, &EvalGetProvider{
Name: n.ProvidedBy()[0],
Output: &provider,
})
vseq.Nodes = append(vseq.Nodes, &EvalInterpolate{
Config: n.Resource.RawConfig,
Resource: resource,
Output: &resourceConfig,
})
vseq.Nodes = append(vseq.Nodes, &EvalValidateResource{
2015-02-14 00:58:41 -06:00
Provider: &provider,
Config: &resourceConfig,
2015-02-09 13:15:54 -06:00
ResourceName: n.Resource.Name,
ResourceType: n.Resource.Type,
})
// Validate all the provisioners
for _, p := range n.Resource.Provisioners {
2015-02-14 00:58:41 -06:00
var provisioner ResourceProvisioner
vseq.Nodes = append(vseq.Nodes, &EvalGetProvisioner{
Name: p.Type,
Output: &provisioner,
}, &EvalInterpolate{
Config: p.RawConfig,
Resource: resource,
Output: &resourceConfig,
2015-02-14 00:58:41 -06:00
}, &EvalValidateProvisioner{
Provisioner: &provisioner,
Config: &resourceConfig,
2015-02-09 13:15:54 -06:00
})
}
2015-02-09 13:15:54 -06:00
// Add the validation operations
seq.Nodes = append(seq.Nodes, &EvalOpFilter{
Ops: []walkOperation{walkValidate},
Node: vseq,
})
2015-02-11 15:43:07 -06:00
// Build instance info
2015-02-13 14:05:34 -06:00
info := n.instanceInfo()
2015-02-11 15:43:07 -06:00
seq.Nodes = append(seq.Nodes, &EvalInstanceInfo{Info: info})
2015-02-11 10:48:45 -06:00
// Refresh the resource
seq.Nodes = append(seq.Nodes, &EvalOpFilter{
Ops: []walkOperation{walkRefresh},
2015-02-12 16:46:22 -06:00
Node: &EvalSequence{
Nodes: []EvalNode{
2015-02-14 00:58:41 -06:00
&EvalGetProvider{
Name: n.ProvidedBy()[0],
Output: &provider,
},
2015-02-12 16:46:22 -06:00
&EvalReadState{
Name: n.stateId(),
Output: &state,
},
&EvalRefresh{
Info: info,
2015-02-14 00:58:41 -06:00
Provider: &provider,
2015-02-12 16:46:22 -06:00
State: &state,
Output: &state,
},
&EvalWriteState{
Name: n.stateId(),
ResourceType: n.Resource.Type,
Dependencies: n.DependentOn(),
State: &state,
},
2015-02-11 10:48:45 -06:00
},
},
})
2015-02-11 17:22:03 -06:00
// Diff the resource
seq.Nodes = append(seq.Nodes, &EvalOpFilter{
Ops: []walkOperation{walkPlan},
Node: &EvalSequence{
Nodes: []EvalNode{
2015-02-14 00:58:41 -06:00
&EvalInterpolate{
Config: n.Resource.RawConfig,
Resource: resource,
Output: &resourceConfig,
},
&EvalGetProvider{
Name: n.ProvidedBy()[0],
Output: &provider,
},
&EvalReadState{
Name: n.stateId(),
Output: &state,
},
2015-02-12 16:46:22 -06:00
&EvalDiff{
Info: info,
2015-02-14 00:58:41 -06:00
Config: &resourceConfig,
Provider: &provider,
State: &state,
2015-02-12 16:46:22 -06:00
Output: &diff,
OutputState: &state,
},
2015-02-11 17:22:03 -06:00
&EvalWriteState{
Name: n.stateId(),
ResourceType: n.Resource.Type,
Dependencies: n.DependentOn(),
2015-02-12 16:46:22 -06:00
State: &state,
2015-02-11 17:22:03 -06:00
},
&EvalDiffTainted{
Diff: &diff,
Name: n.stateId(),
},
2015-02-11 17:22:03 -06:00
&EvalWriteDiff{
Name: n.stateId(),
Diff: &diff,
},
},
},
})
2015-02-12 14:42:33 -06:00
// Diff the resource for destruction
seq.Nodes = append(seq.Nodes, &EvalOpFilter{
Ops: []walkOperation{walkPlanDestroy},
Node: &EvalSequence{
Nodes: []EvalNode{
2015-02-14 00:58:41 -06:00
&EvalReadState{
Name: n.stateId(),
Output: &state,
},
2015-02-12 14:42:33 -06:00
&EvalDiffDestroy{
Info: info,
2015-02-14 00:58:41 -06:00
State: &state,
2015-02-12 14:42:33 -06:00
Output: &diff,
},
&EvalWriteDiff{
Name: n.stateId(),
Diff: &diff,
},
},
},
})
2015-02-16 21:15:58 -06:00
// Apply
var diffApply *InstanceDiff
2015-02-13 11:49:29 -06:00
var err error
var createNew, tainted bool
var createBeforeDestroyEnabled bool
2015-02-12 16:46:22 -06:00
seq.Nodes = append(seq.Nodes, &EvalOpFilter{
Ops: []walkOperation{walkApply},
Node: &EvalSequence{
Nodes: []EvalNode{
2015-02-13 14:05:34 -06:00
// 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 {
2015-02-13 14:13:39 -06:00
return true, EvalEarlyExitError{}
2015-02-13 14:05:34 -06:00
}
2015-02-13 16:03:17 -06:00
if diffApply.Destroy && len(diffApply.Attributes) == 0 {
2015-02-13 14:05:34 -06:00
return true, EvalEarlyExitError{}
}
2015-02-13 16:03:17 -06:00
diffApply.Destroy = false
2015-02-13 14:05:34 -06:00
return true, nil
},
Then: EvalNoop{},
2015-02-13 14:05:34 -06:00
},
2015-02-13 17:57:37 -06:00
&EvalIf{
If: func(ctx EvalContext) (bool, error) {
destroy := false
if diffApply != nil {
destroy = diffApply.Destroy || diffApply.RequiresNew()
}
createBeforeDestroyEnabled =
n.Resource.Lifecycle.CreateBeforeDestroy &&
destroy
return createBeforeDestroyEnabled, nil
2015-02-13 17:57:37 -06:00
},
Then: &EvalDeposeState{
2015-02-13 17:57:37 -06:00
Name: n.stateId(),
},
},
2015-02-14 00:58:41 -06:00
&EvalInterpolate{
Config: n.Resource.RawConfig,
Resource: resource,
Output: &resourceConfig,
},
&EvalGetProvider{
Name: n.ProvidedBy()[0],
Output: &provider,
},
&EvalReadState{
Name: n.stateId(),
Output: &state,
},
2015-02-12 21:48:57 -06:00
&EvalDiff{
Info: info,
2015-02-14 00:58:41 -06:00
Config: &resourceConfig,
Provider: &provider,
State: &state,
Output: &diffApply,
2015-02-12 16:46:22 -06:00
},
2015-02-12 21:48:57 -06:00
// Get the saved diff
2015-02-12 16:46:22 -06:00
&EvalReadDiff{
Name: n.stateId(),
Diff: &diff,
},
2015-02-12 21:48:57 -06:00
// Compare the diffs
&EvalCompareDiff{
Info: info,
One: &diff,
Two: &diffApply,
2015-02-12 21:48:57 -06:00
},
&EvalGetProvider{
Name: n.ProvidedBy()[0],
Output: &provider,
},
2015-02-12 16:46:22 -06:00
&EvalReadState{
Name: n.stateId(),
Output: &state,
},
&EvalApply{
Info: info,
State: &state,
Diff: &diffApply,
Provider: &provider,
Output: &state,
Error: &err,
CreateNew: &createNew,
2015-02-12 16:46:22 -06:00
},
&EvalWriteState{
Name: n.stateId(),
ResourceType: n.Resource.Type,
Dependencies: n.DependentOn(),
State: &state,
},
2015-02-13 11:49:29 -06:00
&EvalApplyProvisioners{
Info: info,
State: &state,
Resource: n.Resource,
InterpResource: resource,
CreateNew: &createNew,
2015-02-13 11:49:29 -06:00
Tainted: &tainted,
Error: &err,
},
2015-02-13 17:57:37 -06:00
&EvalIf{
If: func(ctx EvalContext) (bool, error) {
if createBeforeDestroyEnabled {
tainted = err != nil
}
failure := tainted || err != nil
return createBeforeDestroyEnabled && failure, nil
2015-02-13 17:57:37 -06:00
},
Then: &EvalUndeposeState{
2015-02-13 17:57:37 -06:00
Name: n.stateId(),
},
},
// 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,
},
&EvalIf{
If: func(ctx EvalContext) (bool, error) {
return tainted, nil
},
Then: &EvalSequence{
Nodes: []EvalNode{
&EvalWriteStateTainted{
Name: n.stateId(),
ResourceType: n.Resource.Type,
Dependencies: n.DependentOn(),
State: &state,
TaintedIndex: -1,
},
&EvalIf{
If: func(ctx EvalContext) (bool, error) {
return !n.Resource.Lifecycle.CreateBeforeDestroy, nil
},
Then: &EvalClearPrimaryState{
Name: n.stateId(),
},
},
},
},
Else: &EvalWriteState{
Name: n.stateId(),
ResourceType: n.Resource.Type,
Dependencies: n.DependentOn(),
State: &state,
},
2015-02-13 11:49:29 -06:00
},
2015-02-13 11:52:11 -06:00
&EvalApplyPost{
Info: info,
State: &state,
Error: &err,
},
&EvalUpdateStateHook{},
2015-02-12 16:46:22 -06:00
},
},
})
2015-02-09 13:15:54 -06:00
return seq
}
2015-02-10 16:12:49 -06:00
2015-02-13 14:05:34 -06:00
// instanceInfo is used for EvalTree.
func (n *graphNodeExpandedResource) instanceInfo() *InstanceInfo {
return &InstanceInfo{Id: n.stateId(), Type: n.Resource.Type}
}
2015-02-10 16:12:49 -06:00
// stateId is the name used for the state key
func (n *graphNodeExpandedResource) stateId() string {
2015-02-11 19:18:16 -06:00
if n.Index == -1 {
2015-02-11 10:48:45 -06:00
return n.Resource.Id()
}
2015-02-10 16:12:49 -06:00
return fmt.Sprintf("%s.%d", n.Resource.Id(), n.Index)
}
2015-02-12 13:40:48 -06:00
// GraphNodeStateRepresentative impl.
func (n *graphNodeExpandedResource) StateId() []string {
return []string{n.stateId()}
}
2015-02-12 13:40:48 -06:00
// graphNodeExpandedResourceDestroy represents an expanded resource that
// is to be destroyed.
type graphNodeExpandedResourceDestroy struct {
*graphNodeExpandedResource
}
func (n *graphNodeExpandedResourceDestroy) Name() string {
return fmt.Sprintf("%s (destroy)", n.graphNodeExpandedResource.Name())
}
// GraphNodeEvalable impl.
func (n *graphNodeExpandedResourceDestroy) EvalTree() EvalNode {
2015-02-13 14:05:34 -06:00
info := n.instanceInfo()
var diffApply *InstanceDiff
var provider ResourceProvider
var state *InstanceState
var err error
return &EvalOpFilter{
Ops: []walkOperation{walkApply},
Node: &EvalSequence{
Nodes: []EvalNode{
// Get the saved diff for apply
&EvalReadDiff{
Name: n.stateId(),
Diff: &diffApply,
},
// Filter the diff so we only get the destroy
&EvalFilterDiff{
Diff: &diffApply,
Output: &diffApply,
Destroy: true,
},
2015-02-13 14:05:34 -06:00
// If we're not destroying, then compare diffs
&EvalIf{
If: func(ctx EvalContext) (bool, error) {
if diffApply != nil && diffApply.Destroy {
return true, nil
}
return true, EvalEarlyExitError{}
},
Then: EvalNoop{},
2015-02-13 14:05:34 -06:00
},
&EvalGetProvider{
Name: n.ProvidedBy()[0],
Output: &provider,
},
&EvalIf{
If: func(ctx EvalContext) (bool, error) {
return n.Resource.Lifecycle.CreateBeforeDestroy, nil
},
Then: &EvalReadStateTainted{
Name: n.stateId(),
Output: &state,
TaintedIndex: -1,
},
Else: &EvalReadState{
Name: n.stateId(),
Output: &state,
},
2015-02-13 14:05:34 -06:00
},
&EvalRequireState{
State: &state,
},
2015-02-13 14:05:34 -06:00
&EvalApply{
Info: info,
State: &state,
Diff: &diffApply,
Provider: &provider,
Output: &state,
Error: &err,
},
&EvalWriteState{
Name: n.stateId(),
ResourceType: n.Resource.Type,
Dependencies: n.DependentOn(),
State: &state,
},
&EvalApplyPost{
Info: info,
State: &state,
Error: &err,
},
},
},
}
2015-02-12 13:40:48 -06:00
}