2016-09-16 19:29:36 -05:00
|
|
|
package terraform
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2016-10-21 00:51:06 -05:00
|
|
|
|
|
|
|
"github.com/hashicorp/terraform/config"
|
2016-09-16 19:29:36 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
// NodeDestroyResource represents a resource that is to be destroyed.
|
2016-09-16 22:26:10 -05:00
|
|
|
type NodeDestroyResource struct {
|
2016-10-21 01:12:07 -05:00
|
|
|
*NodeAbstractResource
|
2016-09-16 19:29:36 -05:00
|
|
|
}
|
|
|
|
|
2016-09-16 22:26:10 -05:00
|
|
|
func (n *NodeDestroyResource) Name() string {
|
2016-09-21 20:48:21 -05:00
|
|
|
return n.NodeAbstractResource.Name() + " (destroy)"
|
2016-09-16 19:29:36 -05:00
|
|
|
}
|
|
|
|
|
2016-09-20 12:23:48 -05:00
|
|
|
// GraphNodeDestroyer
|
|
|
|
func (n *NodeDestroyResource) DestroyAddr() *ResourceAddress {
|
|
|
|
return n.Addr
|
|
|
|
}
|
|
|
|
|
2016-09-21 20:48:21 -05:00
|
|
|
// GraphNodeDestroyerCBD
|
|
|
|
func (n *NodeDestroyResource) CreateBeforeDestroy() bool {
|
|
|
|
// If we have no config, we just assume no
|
|
|
|
if n.Config == nil {
|
|
|
|
return false
|
|
|
|
}
|
2016-09-16 22:26:10 -05:00
|
|
|
|
2016-09-21 20:48:21 -05:00
|
|
|
return n.Config.Lifecycle.CreateBeforeDestroy
|
2016-09-16 22:26:10 -05:00
|
|
|
}
|
|
|
|
|
2016-09-22 12:10:57 -05:00
|
|
|
// GraphNodeReferenceable, overriding NodeAbstractResource
|
|
|
|
func (n *NodeDestroyResource) ReferenceableName() []string {
|
2016-10-12 20:43:26 -05:00
|
|
|
result := n.NodeAbstractResource.ReferenceableName()
|
|
|
|
for i, v := range result {
|
|
|
|
result[i] = v + ".destroy"
|
|
|
|
}
|
|
|
|
|
|
|
|
return result
|
2016-09-22 12:10:57 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// GraphNodeReferencer, overriding NodeAbstractResource
|
|
|
|
func (n *NodeDestroyResource) References() []string {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2016-09-22 13:03:03 -05:00
|
|
|
// GraphNodeDynamicExpandable
|
|
|
|
func (n *NodeDestroyResource) DynamicExpand(ctx EvalContext) (*Graph, error) {
|
|
|
|
// If we have no config we do nothing
|
|
|
|
if n.Config == nil {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
state, lock := ctx.State()
|
|
|
|
lock.RLock()
|
|
|
|
defer lock.RUnlock()
|
|
|
|
|
|
|
|
// Start creating the steps
|
|
|
|
steps := make([]GraphTransformer, 0, 5)
|
|
|
|
|
|
|
|
// We want deposed resources in the state to be destroyed
|
|
|
|
steps = append(steps, &DeposedTransformer{
|
|
|
|
State: state,
|
|
|
|
View: n.Config.Id(),
|
|
|
|
})
|
|
|
|
|
2016-10-20 20:59:02 -05:00
|
|
|
// Target
|
|
|
|
steps = append(steps, &TargetsTransformer{
|
|
|
|
ParsedTargets: n.Targets,
|
|
|
|
})
|
|
|
|
|
2016-09-22 13:03:03 -05:00
|
|
|
// Always end with the root being added
|
|
|
|
steps = append(steps, &RootTransformer{})
|
|
|
|
|
|
|
|
// Build the graph
|
|
|
|
b := &BasicGraphBuilder{Steps: steps}
|
|
|
|
return b.Build(ctx.Path())
|
|
|
|
}
|
|
|
|
|
2016-09-16 19:29:36 -05:00
|
|
|
// GraphNodeEvalable
|
2016-09-16 22:26:10 -05:00
|
|
|
func (n *NodeDestroyResource) EvalTree() EvalNode {
|
|
|
|
// stateId is the ID to put into the state
|
|
|
|
stateId := n.Addr.stateId()
|
|
|
|
if n.Addr.Index > -1 {
|
|
|
|
stateId = fmt.Sprintf("%s.%d", stateId, n.Addr.Index)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Build the instance info. More of this will be populated during eval
|
|
|
|
info := &InstanceInfo{
|
2016-10-19 15:51:11 -05:00
|
|
|
Id: stateId,
|
|
|
|
Type: n.Addr.Type,
|
|
|
|
uniqueExtra: "destroy",
|
2016-09-16 22:26:10 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// Get our state
|
|
|
|
rs := n.ResourceState
|
|
|
|
if rs == nil {
|
|
|
|
rs = &ResourceState{}
|
|
|
|
}
|
|
|
|
|
|
|
|
var diffApply *InstanceDiff
|
|
|
|
var provider ResourceProvider
|
|
|
|
var state *InstanceState
|
|
|
|
var err error
|
|
|
|
return &EvalOpFilter{
|
|
|
|
Ops: []walkOperation{walkApply, walkDestroy},
|
|
|
|
Node: &EvalSequence{
|
|
|
|
Nodes: []EvalNode{
|
|
|
|
// Get the saved diff for apply
|
|
|
|
&EvalReadDiff{
|
|
|
|
Name: stateId,
|
|
|
|
Diff: &diffApply,
|
|
|
|
},
|
|
|
|
|
|
|
|
// Filter the diff so we only get the destroy
|
|
|
|
&EvalFilterDiff{
|
|
|
|
Diff: &diffApply,
|
|
|
|
Output: &diffApply,
|
|
|
|
Destroy: true,
|
|
|
|
},
|
|
|
|
|
|
|
|
// If we're not destroying, then compare diffs
|
|
|
|
&EvalIf{
|
|
|
|
If: func(ctx EvalContext) (bool, error) {
|
|
|
|
if diffApply != nil && diffApply.GetDestroy() {
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return true, EvalEarlyExitError{}
|
|
|
|
},
|
|
|
|
Then: EvalNoop{},
|
|
|
|
},
|
|
|
|
|
|
|
|
// Load the instance info so we have the module path set
|
|
|
|
&EvalInstanceInfo{Info: info},
|
|
|
|
|
|
|
|
&EvalGetProvider{
|
|
|
|
Name: n.ProvidedBy()[0],
|
|
|
|
Output: &provider,
|
|
|
|
},
|
|
|
|
&EvalReadState{
|
|
|
|
Name: stateId,
|
|
|
|
Output: &state,
|
|
|
|
},
|
|
|
|
&EvalRequireState{
|
|
|
|
State: &state,
|
|
|
|
},
|
|
|
|
// Make sure we handle data sources properly.
|
|
|
|
&EvalIf{
|
|
|
|
If: func(ctx EvalContext) (bool, error) {
|
2016-10-21 01:12:07 -05:00
|
|
|
if n.Addr == nil {
|
|
|
|
return false, fmt.Errorf("nil address")
|
|
|
|
}
|
|
|
|
|
|
|
|
if n.Addr.Mode == config.DataResourceMode {
|
2016-09-16 22:26:10 -05:00
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return false, nil
|
|
|
|
},
|
|
|
|
|
|
|
|
Then: &EvalReadDataApply{
|
|
|
|
Info: info,
|
|
|
|
Diff: &diffApply,
|
|
|
|
Provider: &provider,
|
|
|
|
Output: &state,
|
|
|
|
},
|
|
|
|
Else: &EvalApply{
|
|
|
|
Info: info,
|
|
|
|
State: &state,
|
|
|
|
Diff: &diffApply,
|
|
|
|
Provider: &provider,
|
|
|
|
Output: &state,
|
|
|
|
Error: &err,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
&EvalWriteState{
|
|
|
|
Name: stateId,
|
|
|
|
ResourceType: n.Addr.Type,
|
|
|
|
Provider: rs.Provider,
|
|
|
|
Dependencies: rs.Dependencies,
|
|
|
|
State: &state,
|
|
|
|
},
|
|
|
|
&EvalApplyPost{
|
|
|
|
Info: info,
|
|
|
|
State: &state,
|
|
|
|
Error: &err,
|
|
|
|
},
|
2016-09-19 12:04:09 -05:00
|
|
|
&EvalUpdateStateHook{},
|
2016-09-16 22:26:10 -05:00
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
2016-09-16 19:29:36 -05:00
|
|
|
}
|