core: Always update resource metadata in state during apply
Previously we had a bug where we would fail to populate resource-level metadata in the state during apply when count = 0, because the apply graph would contain only instance nodes, not whole-resource nodes. To address this, we add to the apply graph a node for each resource in the configuration alongside the separate resource instance nodes. This node's job is just to populate the state metadata for the resource, which ensures it gets updated correctly even when count = 0. When count is not zero this ends up doing some redundant work that would've happened as a side-effect of applying individual resource instances anyway, but it's harmless and makes the updating of our resource-level metadata more explicit.
This commit is contained in:
parent
5390fb1eed
commit
0a97daf3de
|
@ -216,13 +216,13 @@ func TestContext2Apply_resourceCountZeroList(t *testing.T) {
|
|||
t.Fatalf("diags: %s", diags.Err())
|
||||
}
|
||||
|
||||
actual := strings.TrimSpace(state.String())
|
||||
expected := strings.TrimSpace(`<no state>
|
||||
got := strings.TrimSpace(state.String())
|
||||
want := strings.TrimSpace(`<no state>
|
||||
Outputs:
|
||||
|
||||
test = []`)
|
||||
if actual != expected {
|
||||
t.Fatalf("expected: \n%s\n\ngot: \n%s\n", expected, actual)
|
||||
if got != want {
|
||||
t.Fatalf("wrong state\n\ngot:\n%s\n\nwant:\n%s\n", got, want)
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -4,6 +4,10 @@ import (
|
|||
"fmt"
|
||||
"log"
|
||||
|
||||
"github.com/hashicorp/terraform/tfdiags"
|
||||
|
||||
"github.com/hashicorp/terraform/configs"
|
||||
|
||||
"github.com/hashicorp/terraform/addrs"
|
||||
"github.com/hashicorp/terraform/providers"
|
||||
"github.com/hashicorp/terraform/states"
|
||||
|
@ -345,3 +349,45 @@ func (n *EvalUndeposeState) Eval(ctx EvalContext) (interface{}, error) {
|
|||
|
||||
return nil, nil
|
||||
}
|
||||
|
||||
// EvalWriteResourceState is an EvalNode implementation that ensures that
|
||||
// a suitable resource-level state record is present in the state, if that's
|
||||
// required for the "each mode" of that resource.
|
||||
//
|
||||
// This is important primarily for the situation where count = 0, since this
|
||||
// eval is the only change we get to set the resource "each mode" to list
|
||||
// in that case, allowing expression evaluation to see it as a zero-element
|
||||
// list rather than as not set at all.
|
||||
type EvalWriteResourceState struct {
|
||||
Addr addrs.Resource
|
||||
Config *configs.Resource
|
||||
ProviderAddr addrs.AbsProviderConfig
|
||||
}
|
||||
|
||||
// TODO: test
|
||||
func (n *EvalWriteResourceState) Eval(ctx EvalContext) (interface{}, error) {
|
||||
var diags tfdiags.Diagnostics
|
||||
absAddr := n.Addr.Absolute(ctx.Path())
|
||||
state := ctx.State()
|
||||
|
||||
count, countDiags := evaluateResourceCountExpression(n.Config.Count, ctx)
|
||||
diags = diags.Append(countDiags)
|
||||
if countDiags.HasErrors() {
|
||||
return nil, diags.Err()
|
||||
}
|
||||
|
||||
// Currently we ony support NoEach and EachList, because for_each support
|
||||
// is not fully wired up across Terraform. Once for_each support is added,
|
||||
// we'll need to handle that here too, setting states.EachMap if the
|
||||
// assigned expression is a map.
|
||||
eachMode := states.NoEach
|
||||
if count >= 0 { // -1 signals "count not set"
|
||||
eachMode = states.EachList
|
||||
}
|
||||
|
||||
// This method takes care of all of the business logic of updating this
|
||||
// while ensuring that any existing instances are preserved, etc.
|
||||
state.SetResourceMeta(absAddr, eachMode, n.ProviderAddr)
|
||||
|
||||
return nil, nil
|
||||
}
|
||||
|
|
|
@ -68,16 +68,33 @@ func (b *ApplyGraphBuilder) Steps() []GraphTransformer {
|
|||
}
|
||||
}
|
||||
|
||||
concreteResource := func(a *NodeAbstractResourceInstance) dag.Vertex {
|
||||
concreteResource := func(a *NodeAbstractResource) dag.Vertex {
|
||||
return &NodeApplyableResource{
|
||||
NodeAbstractResource: a,
|
||||
}
|
||||
}
|
||||
|
||||
concreteResourceInstance := func(a *NodeAbstractResourceInstance) dag.Vertex {
|
||||
return &NodeApplyableResourceInstance{
|
||||
NodeAbstractResourceInstance: a,
|
||||
}
|
||||
}
|
||||
|
||||
steps := []GraphTransformer{
|
||||
// Creates all the nodes represented in the diff.
|
||||
&DiffTransformer{
|
||||
// Creates all the resources represented in the config. During apply,
|
||||
// we use this just to ensure that the whole-resource metadata is
|
||||
// updated to reflect things such as whether the count argument is
|
||||
// set in config, or which provider configuration manages each resource.
|
||||
&ConfigTransformer{
|
||||
Concrete: concreteResource,
|
||||
Config: b.Config,
|
||||
},
|
||||
|
||||
// Creates all the resource instances represented in the diff, along
|
||||
// with dependency edges against the whole-resource nodes added by
|
||||
// ConfigTransformer above.
|
||||
&DiffTransformer{
|
||||
Concrete: concreteResourceInstance,
|
||||
Changes: b.Changes,
|
||||
},
|
||||
|
||||
|
|
|
@ -1,387 +1,47 @@
|
|||
package terraform
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
|
||||
"github.com/hashicorp/terraform/providers"
|
||||
"github.com/hashicorp/terraform/states"
|
||||
|
||||
"github.com/hashicorp/terraform/plans"
|
||||
|
||||
"github.com/hashicorp/terraform/addrs"
|
||||
"github.com/hashicorp/terraform/configs"
|
||||
"github.com/zclconf/go-cty/cty"
|
||||
"log"
|
||||
)
|
||||
|
||||
// NodeApplyableResourceInstance represents a resource that is "applyable":
|
||||
// it is ready to be applied and is represented by a diff.
|
||||
type NodeApplyableResourceInstance struct {
|
||||
*NodeAbstractResourceInstance
|
||||
|
||||
destroyNode GraphNodeDestroyerCBD
|
||||
// NodeApplyableResource represents a resource that is "applyable":
|
||||
// it may need to have its record in the state adjusted to match configuration.
|
||||
//
|
||||
// Unlike in the plan walk, this resource node does not DynamicExpand. Instead,
|
||||
// it should be inserted into the same graph as any instances of the nodes
|
||||
// with dependency edges ensuring that the resource is evaluated before any
|
||||
// of its instances, which will turn ensure that the whole-resource record
|
||||
// in the state is suitably prepared to receive any updates to instances.
|
||||
type NodeApplyableResource struct {
|
||||
*NodeAbstractResource
|
||||
}
|
||||
|
||||
var (
|
||||
_ GraphNodeResource = (*NodeApplyableResourceInstance)(nil)
|
||||
_ GraphNodeResourceInstance = (*NodeApplyableResourceInstance)(nil)
|
||||
_ GraphNodeCreator = (*NodeApplyableResourceInstance)(nil)
|
||||
_ GraphNodeReferencer = (*NodeApplyableResourceInstance)(nil)
|
||||
_ GraphNodeEvalable = (*NodeApplyableResourceInstance)(nil)
|
||||
_ GraphNodeResource = (*NodeApplyableResource)(nil)
|
||||
_ GraphNodeEvalable = (*NodeApplyableResource)(nil)
|
||||
_ GraphNodeProviderConsumer = (*NodeApplyableResource)(nil)
|
||||
_ GraphNodeAttachResourceConfig = (*NodeApplyableResource)(nil)
|
||||
)
|
||||
|
||||
// GraphNodeAttachDestroyer
|
||||
func (n *NodeApplyableResourceInstance) AttachDestroyNode(d GraphNodeDestroyerCBD) {
|
||||
n.destroyNode = d
|
||||
}
|
||||
|
||||
// createBeforeDestroy checks this nodes config status and the status af any
|
||||
// companion destroy node for CreateBeforeDestroy.
|
||||
func (n *NodeApplyableResourceInstance) createBeforeDestroy() bool {
|
||||
cbd := false
|
||||
|
||||
if n.Config != nil && n.Config.Managed != nil {
|
||||
cbd = n.Config.Managed.CreateBeforeDestroy
|
||||
}
|
||||
|
||||
if n.destroyNode != nil {
|
||||
cbd = cbd || n.destroyNode.CreateBeforeDestroy()
|
||||
}
|
||||
|
||||
return cbd
|
||||
}
|
||||
|
||||
// GraphNodeCreator
|
||||
func (n *NodeApplyableResourceInstance) CreateAddr() *addrs.AbsResourceInstance {
|
||||
addr := n.ResourceInstanceAddr()
|
||||
return &addr
|
||||
}
|
||||
|
||||
// GraphNodeReferencer, overriding NodeAbstractResourceInstance
|
||||
func (n *NodeApplyableResourceInstance) References() []*addrs.Reference {
|
||||
// Start with the usual resource instance implementation
|
||||
ret := n.NodeAbstractResourceInstance.References()
|
||||
|
||||
// Applying a resource must also depend on the destruction of any of its
|
||||
// dependencies, since this may for example affect the outcome of
|
||||
// evaluating an entire list of resources with "count" set (by reducing
|
||||
// the count).
|
||||
//
|
||||
// However, we can't do this in create_before_destroy mode because that
|
||||
// would create a dependency cycle. We make a compromise here of requiring
|
||||
// changes to be updated across two applies in this case, since the first
|
||||
// plan will use the old values.
|
||||
if !n.createBeforeDestroy() {
|
||||
for _, ref := range ret {
|
||||
switch tr := ref.Subject.(type) {
|
||||
case addrs.ResourceInstance:
|
||||
newRef := *ref // shallow copy so we can mutate
|
||||
newRef.Subject = tr.Phase(addrs.ResourceInstancePhaseDestroy)
|
||||
newRef.Remaining = nil // can't access attributes of something being destroyed
|
||||
ret = append(ret, &newRef)
|
||||
case addrs.Resource:
|
||||
newRef := *ref // shallow copy so we can mutate
|
||||
newRef.Subject = tr.Phase(addrs.ResourceInstancePhaseDestroy)
|
||||
newRef.Remaining = nil // can't access attributes of something being destroyed
|
||||
ret = append(ret, &newRef)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return ret
|
||||
func (n *NodeApplyableResource) Name() string {
|
||||
return n.NodeAbstractResource.Name() + " (prepare state)"
|
||||
}
|
||||
|
||||
// GraphNodeEvalable
|
||||
func (n *NodeApplyableResourceInstance) EvalTree() EvalNode {
|
||||
addr := n.ResourceInstanceAddr()
|
||||
func (n *NodeApplyableResource) EvalTree() EvalNode {
|
||||
addr := n.ResourceAddr()
|
||||
config := n.Config
|
||||
providerAddr := n.ResolvedProvider
|
||||
|
||||
// State still uses legacy-style internal ids, so we need to shim to get
|
||||
// a suitable key to use.
|
||||
stateId := NewLegacyResourceInstanceAddress(addr).stateId()
|
||||
if config == nil {
|
||||
// Nothing to do, then.
|
||||
log.Printf("[TRACE] NodeApplyableResource: no configuration present for %s", addr)
|
||||
return &EvalNoop{}
|
||||
}
|
||||
|
||||
// Determine the dependencies for the state.
|
||||
stateDeps := n.StateReferences()
|
||||
|
||||
// Eval info is different depending on what kind of resource this is
|
||||
switch n.Config.Mode {
|
||||
case addrs.ManagedResourceMode:
|
||||
return n.evalTreeManagedResource(addr, stateId, stateDeps)
|
||||
case addrs.DataResourceMode:
|
||||
return n.evalTreeDataResource(addr, stateId, stateDeps)
|
||||
default:
|
||||
panic(fmt.Errorf("unsupported resource mode %s", n.Config.Mode))
|
||||
}
|
||||
}
|
||||
|
||||
func (n *NodeApplyableResourceInstance) evalTreeDataResource(addr addrs.AbsResourceInstance, stateId string, stateDeps []addrs.Referenceable) EvalNode {
|
||||
var provider providers.Interface
|
||||
var providerSchema *ProviderSchema
|
||||
var change *plans.ResourceInstanceChange
|
||||
var state *states.ResourceInstanceObject
|
||||
var configVal cty.Value
|
||||
|
||||
return &EvalSequence{
|
||||
Nodes: []EvalNode{
|
||||
&EvalGetProvider{
|
||||
Addr: n.ResolvedProvider,
|
||||
Output: &provider,
|
||||
Schema: &providerSchema,
|
||||
},
|
||||
|
||||
// Get the saved diff for apply
|
||||
&EvalReadDiff{
|
||||
Addr: addr.Resource,
|
||||
ProviderSchema: &providerSchema,
|
||||
Change: &change,
|
||||
},
|
||||
|
||||
// Stop early if we don't actually have a diff
|
||||
&EvalIf{
|
||||
If: func(ctx EvalContext) (bool, error) {
|
||||
if change == nil {
|
||||
return true, EvalEarlyExitError{}
|
||||
}
|
||||
return true, nil
|
||||
},
|
||||
Then: EvalNoop{},
|
||||
},
|
||||
|
||||
// Make a new diff, in case we've learned new values in the state
|
||||
// during apply which we can now incorporate.
|
||||
&EvalReadDataDiff{
|
||||
Addr: addr.Resource,
|
||||
Config: n.Config,
|
||||
ProviderAddr: n.ResolvedProvider,
|
||||
ProviderSchema: &providerSchema,
|
||||
Output: &change,
|
||||
OutputValue: &configVal,
|
||||
OutputState: &state,
|
||||
},
|
||||
|
||||
&EvalReadDataApply{
|
||||
Addr: addr.Resource,
|
||||
Config: n.Config,
|
||||
Change: &change,
|
||||
Provider: &provider,
|
||||
ProviderAddr: n.ResolvedProvider,
|
||||
ProviderSchema: &providerSchema,
|
||||
Output: &state,
|
||||
StateReferences: n.StateReferences(),
|
||||
},
|
||||
|
||||
&EvalWriteState{
|
||||
Addr: addr.Resource,
|
||||
ProviderAddr: n.ResolvedProvider,
|
||||
ProviderSchema: &providerSchema,
|
||||
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{
|
||||
Addr: addr.Resource,
|
||||
ProviderSchema: &providerSchema,
|
||||
Change: nil,
|
||||
},
|
||||
|
||||
&EvalUpdateStateHook{},
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
func (n *NodeApplyableResourceInstance) evalTreeManagedResource(addr addrs.AbsResourceInstance, stateId string, stateDeps []addrs.Referenceable) EvalNode {
|
||||
// Declare a bunch of variables that are used for state during
|
||||
// evaluation. Most of this are written to by-address below.
|
||||
var provider providers.Interface
|
||||
var providerSchema *ProviderSchema
|
||||
var diff, diffApply *plans.ResourceInstanceChange
|
||||
var state *states.ResourceInstanceObject
|
||||
var err error
|
||||
var createNew bool
|
||||
var createBeforeDestroyEnabled bool
|
||||
var configVal cty.Value
|
||||
var deposedKey states.DeposedKey
|
||||
|
||||
return &EvalSequence{
|
||||
Nodes: []EvalNode{
|
||||
&EvalGetProvider{
|
||||
Addr: n.ResolvedProvider,
|
||||
Output: &provider,
|
||||
Schema: &providerSchema,
|
||||
},
|
||||
|
||||
// Get the saved diff for apply
|
||||
&EvalReadDiff{
|
||||
Addr: addr.Resource,
|
||||
ProviderSchema: &providerSchema,
|
||||
Change: &diffApply,
|
||||
},
|
||||
|
||||
// We don't want to do any destroys
|
||||
// (these are handled by NodeDestroyResourceInstance instead)
|
||||
&EvalIf{
|
||||
If: func(ctx EvalContext) (bool, error) {
|
||||
if diffApply == nil {
|
||||
return true, EvalEarlyExitError{}
|
||||
}
|
||||
if diffApply.Action == plans.Delete {
|
||||
return true, EvalEarlyExitError{}
|
||||
}
|
||||
return true, nil
|
||||
},
|
||||
Then: EvalNoop{},
|
||||
},
|
||||
|
||||
&EvalIf{
|
||||
If: func(ctx EvalContext) (bool, error) {
|
||||
destroy := false
|
||||
if diffApply != nil {
|
||||
destroy = (diffApply.Action == plans.Delete || diffApply.Action == plans.Replace)
|
||||
}
|
||||
if destroy && n.createBeforeDestroy() {
|
||||
createBeforeDestroyEnabled = true
|
||||
}
|
||||
return createBeforeDestroyEnabled, nil
|
||||
},
|
||||
Then: &EvalDeposeState{
|
||||
Addr: addr.Resource,
|
||||
OutputKey: &deposedKey,
|
||||
},
|
||||
},
|
||||
|
||||
&EvalReadState{
|
||||
Addr: addr.Resource,
|
||||
Provider: &provider,
|
||||
ProviderSchema: &providerSchema,
|
||||
|
||||
Output: &state,
|
||||
},
|
||||
|
||||
// Make a new diff, in case we've learned new values in the state
|
||||
// during apply which we can now incorporate.
|
||||
&EvalDiff{
|
||||
Addr: addr.Resource,
|
||||
Config: n.Config,
|
||||
Provider: &provider,
|
||||
ProviderAddr: n.ResolvedProvider,
|
||||
ProviderSchema: &providerSchema,
|
||||
State: &state,
|
||||
OutputChange: &diffApply,
|
||||
OutputValue: &configVal,
|
||||
OutputState: &state,
|
||||
},
|
||||
|
||||
// Get the saved diff
|
||||
&EvalReadDiff{
|
||||
Addr: addr.Resource,
|
||||
ProviderSchema: &providerSchema,
|
||||
Change: &diff,
|
||||
},
|
||||
|
||||
// Compare the diffs
|
||||
&EvalCheckPlannedChange{
|
||||
Addr: addr.Resource,
|
||||
ProviderAddr: n.ResolvedProvider,
|
||||
ProviderSchema: &providerSchema,
|
||||
Planned: &diff,
|
||||
Actual: &diffApply,
|
||||
},
|
||||
|
||||
&EvalGetProvider{
|
||||
Addr: n.ResolvedProvider,
|
||||
Output: &provider,
|
||||
Schema: &providerSchema,
|
||||
},
|
||||
&EvalReadState{
|
||||
Addr: addr.Resource,
|
||||
Provider: &provider,
|
||||
ProviderSchema: &providerSchema,
|
||||
|
||||
Output: &state,
|
||||
},
|
||||
|
||||
&EvalReduceDiff{
|
||||
Addr: addr.Resource,
|
||||
InChange: &diffApply,
|
||||
Destroy: false,
|
||||
OutChange: &diffApply,
|
||||
},
|
||||
|
||||
// EvalReduceDiff may have simplified our planned change
|
||||
// into a NoOp if it only requires destroying, since destroying
|
||||
// is handled by NodeDestroyResourceInstance.
|
||||
&EvalIf{
|
||||
If: func(ctx EvalContext) (bool, error) {
|
||||
if diffApply == nil || diffApply.Action == plans.NoOp {
|
||||
return true, EvalEarlyExitError{}
|
||||
}
|
||||
return true, nil
|
||||
},
|
||||
Then: EvalNoop{},
|
||||
},
|
||||
|
||||
// Call pre-apply hook
|
||||
&EvalApplyPre{
|
||||
Addr: addr.Resource,
|
||||
State: &state,
|
||||
Change: &diffApply,
|
||||
},
|
||||
&EvalApply{
|
||||
Addr: addr.Resource,
|
||||
Config: n.Config,
|
||||
State: &state,
|
||||
Change: &diffApply,
|
||||
Provider: &provider,
|
||||
ProviderAddr: n.ResolvedProvider,
|
||||
ProviderSchema: &providerSchema,
|
||||
Output: &state,
|
||||
Error: &err,
|
||||
CreateNew: &createNew,
|
||||
},
|
||||
&EvalWriteState{
|
||||
Addr: addr.Resource,
|
||||
ProviderAddr: n.ResolvedProvider,
|
||||
ProviderSchema: &providerSchema,
|
||||
State: &state,
|
||||
},
|
||||
&EvalApplyProvisioners{
|
||||
Addr: addr.Resource,
|
||||
State: &state,
|
||||
ResourceConfig: n.Config,
|
||||
CreateNew: &createNew,
|
||||
Error: &err,
|
||||
When: configs.ProvisionerWhenCreate,
|
||||
},
|
||||
&EvalWriteState{
|
||||
Addr: addr.Resource,
|
||||
ProviderAddr: n.ResolvedProvider,
|
||||
ProviderSchema: &providerSchema,
|
||||
State: &state,
|
||||
},
|
||||
&EvalIf{
|
||||
If: func(ctx EvalContext) (bool, error) {
|
||||
return createBeforeDestroyEnabled && err != nil, nil
|
||||
},
|
||||
Then: &EvalUndeposeState{
|
||||
Addr: addr.Resource,
|
||||
Key: &deposedKey,
|
||||
},
|
||||
},
|
||||
|
||||
// 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{
|
||||
Addr: addr.Resource,
|
||||
ProviderSchema: &providerSchema,
|
||||
Change: nil,
|
||||
},
|
||||
|
||||
&EvalApplyPost{
|
||||
Addr: addr.Resource,
|
||||
State: &state,
|
||||
Error: &err,
|
||||
},
|
||||
&EvalUpdateStateHook{},
|
||||
},
|
||||
return &EvalWriteResourceState{
|
||||
Addr: addr.Resource,
|
||||
Config: config,
|
||||
ProviderAddr: providerAddr,
|
||||
}
|
||||
}
|
||||
|
|
|
@ -0,0 +1,391 @@
|
|||
package terraform
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
|
||||
"github.com/zclconf/go-cty/cty"
|
||||
|
||||
"github.com/hashicorp/terraform/addrs"
|
||||
"github.com/hashicorp/terraform/configs"
|
||||
"github.com/hashicorp/terraform/plans"
|
||||
"github.com/hashicorp/terraform/providers"
|
||||
"github.com/hashicorp/terraform/states"
|
||||
)
|
||||
|
||||
// NodeApplyableResourceInstance represents a resource instance that is
|
||||
// "applyable": it is ready to be applied and is represented by a diff.
|
||||
//
|
||||
// This node is for a specific instance of a resource. It will usually be
|
||||
// accompanied in the graph by a NodeApplyableResource representing its
|
||||
// containing resource, and should depend on that node to ensure that the
|
||||
// state is properly prepared to receive changes to instances.
|
||||
type NodeApplyableResourceInstance struct {
|
||||
*NodeAbstractResourceInstance
|
||||
|
||||
destroyNode GraphNodeDestroyerCBD
|
||||
}
|
||||
|
||||
var (
|
||||
_ GraphNodeResource = (*NodeApplyableResourceInstance)(nil)
|
||||
_ GraphNodeResourceInstance = (*NodeApplyableResourceInstance)(nil)
|
||||
_ GraphNodeCreator = (*NodeApplyableResourceInstance)(nil)
|
||||
_ GraphNodeReferencer = (*NodeApplyableResourceInstance)(nil)
|
||||
_ GraphNodeEvalable = (*NodeApplyableResourceInstance)(nil)
|
||||
)
|
||||
|
||||
// GraphNodeAttachDestroyer
|
||||
func (n *NodeApplyableResourceInstance) AttachDestroyNode(d GraphNodeDestroyerCBD) {
|
||||
n.destroyNode = d
|
||||
}
|
||||
|
||||
// createBeforeDestroy checks this nodes config status and the status af any
|
||||
// companion destroy node for CreateBeforeDestroy.
|
||||
func (n *NodeApplyableResourceInstance) createBeforeDestroy() bool {
|
||||
cbd := false
|
||||
|
||||
if n.Config != nil && n.Config.Managed != nil {
|
||||
cbd = n.Config.Managed.CreateBeforeDestroy
|
||||
}
|
||||
|
||||
if n.destroyNode != nil {
|
||||
cbd = cbd || n.destroyNode.CreateBeforeDestroy()
|
||||
}
|
||||
|
||||
return cbd
|
||||
}
|
||||
|
||||
// GraphNodeCreator
|
||||
func (n *NodeApplyableResourceInstance) CreateAddr() *addrs.AbsResourceInstance {
|
||||
addr := n.ResourceInstanceAddr()
|
||||
return &addr
|
||||
}
|
||||
|
||||
// GraphNodeReferencer, overriding NodeAbstractResourceInstance
|
||||
func (n *NodeApplyableResourceInstance) References() []*addrs.Reference {
|
||||
// Start with the usual resource instance implementation
|
||||
ret := n.NodeAbstractResourceInstance.References()
|
||||
|
||||
// Applying a resource must also depend on the destruction of any of its
|
||||
// dependencies, since this may for example affect the outcome of
|
||||
// evaluating an entire list of resources with "count" set (by reducing
|
||||
// the count).
|
||||
//
|
||||
// However, we can't do this in create_before_destroy mode because that
|
||||
// would create a dependency cycle. We make a compromise here of requiring
|
||||
// changes to be updated across two applies in this case, since the first
|
||||
// plan will use the old values.
|
||||
if !n.createBeforeDestroy() {
|
||||
for _, ref := range ret {
|
||||
switch tr := ref.Subject.(type) {
|
||||
case addrs.ResourceInstance:
|
||||
newRef := *ref // shallow copy so we can mutate
|
||||
newRef.Subject = tr.Phase(addrs.ResourceInstancePhaseDestroy)
|
||||
newRef.Remaining = nil // can't access attributes of something being destroyed
|
||||
ret = append(ret, &newRef)
|
||||
case addrs.Resource:
|
||||
newRef := *ref // shallow copy so we can mutate
|
||||
newRef.Subject = tr.Phase(addrs.ResourceInstancePhaseDestroy)
|
||||
newRef.Remaining = nil // can't access attributes of something being destroyed
|
||||
ret = append(ret, &newRef)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return ret
|
||||
}
|
||||
|
||||
// GraphNodeEvalable
|
||||
func (n *NodeApplyableResourceInstance) EvalTree() EvalNode {
|
||||
addr := n.ResourceInstanceAddr()
|
||||
|
||||
// State still uses legacy-style internal ids, so we need to shim to get
|
||||
// a suitable key to use.
|
||||
stateId := NewLegacyResourceInstanceAddress(addr).stateId()
|
||||
|
||||
// Determine the dependencies for the state.
|
||||
stateDeps := n.StateReferences()
|
||||
|
||||
// Eval info is different depending on what kind of resource this is
|
||||
switch n.Config.Mode {
|
||||
case addrs.ManagedResourceMode:
|
||||
return n.evalTreeManagedResource(addr, stateId, stateDeps)
|
||||
case addrs.DataResourceMode:
|
||||
return n.evalTreeDataResource(addr, stateId, stateDeps)
|
||||
default:
|
||||
panic(fmt.Errorf("unsupported resource mode %s", n.Config.Mode))
|
||||
}
|
||||
}
|
||||
|
||||
func (n *NodeApplyableResourceInstance) evalTreeDataResource(addr addrs.AbsResourceInstance, stateId string, stateDeps []addrs.Referenceable) EvalNode {
|
||||
var provider providers.Interface
|
||||
var providerSchema *ProviderSchema
|
||||
var change *plans.ResourceInstanceChange
|
||||
var state *states.ResourceInstanceObject
|
||||
var configVal cty.Value
|
||||
|
||||
return &EvalSequence{
|
||||
Nodes: []EvalNode{
|
||||
&EvalGetProvider{
|
||||
Addr: n.ResolvedProvider,
|
||||
Output: &provider,
|
||||
Schema: &providerSchema,
|
||||
},
|
||||
|
||||
// Get the saved diff for apply
|
||||
&EvalReadDiff{
|
||||
Addr: addr.Resource,
|
||||
ProviderSchema: &providerSchema,
|
||||
Change: &change,
|
||||
},
|
||||
|
||||
// Stop early if we don't actually have a diff
|
||||
&EvalIf{
|
||||
If: func(ctx EvalContext) (bool, error) {
|
||||
if change == nil {
|
||||
return true, EvalEarlyExitError{}
|
||||
}
|
||||
return true, nil
|
||||
},
|
||||
Then: EvalNoop{},
|
||||
},
|
||||
|
||||
// Make a new diff, in case we've learned new values in the state
|
||||
// during apply which we can now incorporate.
|
||||
&EvalReadDataDiff{
|
||||
Addr: addr.Resource,
|
||||
Config: n.Config,
|
||||
ProviderAddr: n.ResolvedProvider,
|
||||
ProviderSchema: &providerSchema,
|
||||
Output: &change,
|
||||
OutputValue: &configVal,
|
||||
OutputState: &state,
|
||||
},
|
||||
|
||||
&EvalReadDataApply{
|
||||
Addr: addr.Resource,
|
||||
Config: n.Config,
|
||||
Change: &change,
|
||||
Provider: &provider,
|
||||
ProviderAddr: n.ResolvedProvider,
|
||||
ProviderSchema: &providerSchema,
|
||||
Output: &state,
|
||||
StateReferences: n.StateReferences(),
|
||||
},
|
||||
|
||||
&EvalWriteState{
|
||||
Addr: addr.Resource,
|
||||
ProviderAddr: n.ResolvedProvider,
|
||||
ProviderSchema: &providerSchema,
|
||||
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{
|
||||
Addr: addr.Resource,
|
||||
ProviderSchema: &providerSchema,
|
||||
Change: nil,
|
||||
},
|
||||
|
||||
&EvalUpdateStateHook{},
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
func (n *NodeApplyableResourceInstance) evalTreeManagedResource(addr addrs.AbsResourceInstance, stateId string, stateDeps []addrs.Referenceable) EvalNode {
|
||||
// Declare a bunch of variables that are used for state during
|
||||
// evaluation. Most of this are written to by-address below.
|
||||
var provider providers.Interface
|
||||
var providerSchema *ProviderSchema
|
||||
var diff, diffApply *plans.ResourceInstanceChange
|
||||
var state *states.ResourceInstanceObject
|
||||
var err error
|
||||
var createNew bool
|
||||
var createBeforeDestroyEnabled bool
|
||||
var configVal cty.Value
|
||||
var deposedKey states.DeposedKey
|
||||
|
||||
return &EvalSequence{
|
||||
Nodes: []EvalNode{
|
||||
&EvalGetProvider{
|
||||
Addr: n.ResolvedProvider,
|
||||
Output: &provider,
|
||||
Schema: &providerSchema,
|
||||
},
|
||||
|
||||
// Get the saved diff for apply
|
||||
&EvalReadDiff{
|
||||
Addr: addr.Resource,
|
||||
ProviderSchema: &providerSchema,
|
||||
Change: &diffApply,
|
||||
},
|
||||
|
||||
// We don't want to do any destroys
|
||||
// (these are handled by NodeDestroyResourceInstance instead)
|
||||
&EvalIf{
|
||||
If: func(ctx EvalContext) (bool, error) {
|
||||
if diffApply == nil {
|
||||
return true, EvalEarlyExitError{}
|
||||
}
|
||||
if diffApply.Action == plans.Delete {
|
||||
return true, EvalEarlyExitError{}
|
||||
}
|
||||
return true, nil
|
||||
},
|
||||
Then: EvalNoop{},
|
||||
},
|
||||
|
||||
&EvalIf{
|
||||
If: func(ctx EvalContext) (bool, error) {
|
||||
destroy := false
|
||||
if diffApply != nil {
|
||||
destroy = (diffApply.Action == plans.Delete || diffApply.Action == plans.Replace)
|
||||
}
|
||||
if destroy && n.createBeforeDestroy() {
|
||||
createBeforeDestroyEnabled = true
|
||||
}
|
||||
return createBeforeDestroyEnabled, nil
|
||||
},
|
||||
Then: &EvalDeposeState{
|
||||
Addr: addr.Resource,
|
||||
OutputKey: &deposedKey,
|
||||
},
|
||||
},
|
||||
|
||||
&EvalReadState{
|
||||
Addr: addr.Resource,
|
||||
Provider: &provider,
|
||||
ProviderSchema: &providerSchema,
|
||||
|
||||
Output: &state,
|
||||
},
|
||||
|
||||
// Make a new diff, in case we've learned new values in the state
|
||||
// during apply which we can now incorporate.
|
||||
&EvalDiff{
|
||||
Addr: addr.Resource,
|
||||
Config: n.Config,
|
||||
Provider: &provider,
|
||||
ProviderAddr: n.ResolvedProvider,
|
||||
ProviderSchema: &providerSchema,
|
||||
State: &state,
|
||||
OutputChange: &diffApply,
|
||||
OutputValue: &configVal,
|
||||
OutputState: &state,
|
||||
},
|
||||
|
||||
// Get the saved diff
|
||||
&EvalReadDiff{
|
||||
Addr: addr.Resource,
|
||||
ProviderSchema: &providerSchema,
|
||||
Change: &diff,
|
||||
},
|
||||
|
||||
// Compare the diffs
|
||||
&EvalCheckPlannedChange{
|
||||
Addr: addr.Resource,
|
||||
ProviderAddr: n.ResolvedProvider,
|
||||
ProviderSchema: &providerSchema,
|
||||
Planned: &diff,
|
||||
Actual: &diffApply,
|
||||
},
|
||||
|
||||
&EvalGetProvider{
|
||||
Addr: n.ResolvedProvider,
|
||||
Output: &provider,
|
||||
Schema: &providerSchema,
|
||||
},
|
||||
&EvalReadState{
|
||||
Addr: addr.Resource,
|
||||
Provider: &provider,
|
||||
ProviderSchema: &providerSchema,
|
||||
|
||||
Output: &state,
|
||||
},
|
||||
|
||||
&EvalReduceDiff{
|
||||
Addr: addr.Resource,
|
||||
InChange: &diffApply,
|
||||
Destroy: false,
|
||||
OutChange: &diffApply,
|
||||
},
|
||||
|
||||
// EvalReduceDiff may have simplified our planned change
|
||||
// into a NoOp if it only requires destroying, since destroying
|
||||
// is handled by NodeDestroyResourceInstance.
|
||||
&EvalIf{
|
||||
If: func(ctx EvalContext) (bool, error) {
|
||||
if diffApply == nil || diffApply.Action == plans.NoOp {
|
||||
return true, EvalEarlyExitError{}
|
||||
}
|
||||
return true, nil
|
||||
},
|
||||
Then: EvalNoop{},
|
||||
},
|
||||
|
||||
// Call pre-apply hook
|
||||
&EvalApplyPre{
|
||||
Addr: addr.Resource,
|
||||
State: &state,
|
||||
Change: &diffApply,
|
||||
},
|
||||
&EvalApply{
|
||||
Addr: addr.Resource,
|
||||
Config: n.Config,
|
||||
State: &state,
|
||||
Change: &diffApply,
|
||||
Provider: &provider,
|
||||
ProviderAddr: n.ResolvedProvider,
|
||||
ProviderSchema: &providerSchema,
|
||||
Output: &state,
|
||||
Error: &err,
|
||||
CreateNew: &createNew,
|
||||
},
|
||||
&EvalWriteState{
|
||||
Addr: addr.Resource,
|
||||
ProviderAddr: n.ResolvedProvider,
|
||||
ProviderSchema: &providerSchema,
|
||||
State: &state,
|
||||
},
|
||||
&EvalApplyProvisioners{
|
||||
Addr: addr.Resource,
|
||||
State: &state,
|
||||
ResourceConfig: n.Config,
|
||||
CreateNew: &createNew,
|
||||
Error: &err,
|
||||
When: configs.ProvisionerWhenCreate,
|
||||
},
|
||||
&EvalWriteState{
|
||||
Addr: addr.Resource,
|
||||
ProviderAddr: n.ResolvedProvider,
|
||||
ProviderSchema: &providerSchema,
|
||||
State: &state,
|
||||
},
|
||||
&EvalIf{
|
||||
If: func(ctx EvalContext) (bool, error) {
|
||||
return createBeforeDestroyEnabled && err != nil, nil
|
||||
},
|
||||
Then: &EvalUndeposeState{
|
||||
Addr: addr.Resource,
|
||||
Key: &deposedKey,
|
||||
},
|
||||
},
|
||||
|
||||
// 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{
|
||||
Addr: addr.Resource,
|
||||
ProviderSchema: &providerSchema,
|
||||
Change: nil,
|
||||
},
|
||||
|
||||
&EvalApplyPost{
|
||||
Addr: addr.Resource,
|
||||
State: &state,
|
||||
Error: &err,
|
||||
},
|
||||
&EvalUpdateStateHook{},
|
||||
},
|
||||
}
|
||||
}
|
|
@ -25,6 +25,28 @@ func (t *DiffTransformer) Transform(g *Graph) error {
|
|||
// Go through all the modules in the diff.
|
||||
log.Printf("[TRACE] DiffTransformer starting")
|
||||
|
||||
// DiffTransformer creates resource _instance_ nodes. If there are any
|
||||
// whole-resource nodes already in the graph, we must ensure that they
|
||||
// get evaluated before any of the corresponding instances by creating
|
||||
// dependency edges, so we'll do some prep work here to ensure we'll only
|
||||
// create connections to nodes that existed before we started here.
|
||||
resourceNodes := map[string][]GraphNodeResource{}
|
||||
for _, node := range g.Vertices() {
|
||||
rn, ok := node.(GraphNodeResource)
|
||||
if !ok {
|
||||
continue
|
||||
}
|
||||
// We ignore any instances that _also_ implement
|
||||
// GraphNodeResourceInstance, since in the unlikely event that they
|
||||
// do exist we'd probably end up creating cycles by connecting them.
|
||||
if _, ok := node.(GraphNodeResourceInstance); ok {
|
||||
continue
|
||||
}
|
||||
|
||||
addr := rn.ResourceAddr().String()
|
||||
resourceNodes[addr] = append(resourceNodes[addr], rn)
|
||||
}
|
||||
|
||||
for _, rc := range t.Changes.Resources {
|
||||
addr := rc.Addr
|
||||
dk := rc.DeposedKey
|
||||
|
@ -62,6 +84,10 @@ func (t *DiffTransformer) Transform(g *Graph) error {
|
|||
|
||||
log.Printf("[TRACE] DiffTransformer: %s will be represented by %s", addr, dag.VertexName(node))
|
||||
g.Add(node)
|
||||
rsrcAddr := addr.ContainingResource().String()
|
||||
for _, rsrcNode := range resourceNodes[rsrcAddr] {
|
||||
g.Connect(dag.BasicEdge(node, rsrcNode))
|
||||
}
|
||||
}
|
||||
|
||||
if delete {
|
||||
|
@ -77,6 +103,14 @@ func (t *DiffTransformer) Transform(g *Graph) error {
|
|||
log.Printf("[TRACE] DiffTransformer: %s deposed object %s will be represented for destruction by %s", addr, dk, dag.VertexName(node))
|
||||
}
|
||||
g.Add(node)
|
||||
rsrcAddr := addr.ContainingResource().String()
|
||||
for _, rsrcNode := range resourceNodes[rsrcAddr] {
|
||||
// We connect this edge "forwards" (even though destroy dependencies
|
||||
// are often inverted) because evaluating the resource node
|
||||
// after the destroy node could cause an unnecessary husk of
|
||||
// a resource state to be re-added.
|
||||
g.Connect(dag.BasicEdge(node, rsrcNode))
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue