2020-09-25 10:18:14 -05:00
|
|
|
package terraform
|
|
|
|
|
|
|
|
import (
|
|
|
|
"testing"
|
|
|
|
|
2021-05-17 14:00:50 -05:00
|
|
|
"github.com/hashicorp/terraform/internal/addrs"
|
2021-05-17 14:17:09 -05:00
|
|
|
"github.com/hashicorp/terraform/internal/configs/configschema"
|
2021-05-17 12:34:40 -05:00
|
|
|
"github.com/hashicorp/terraform/internal/instances"
|
2021-05-17 12:40:40 -05:00
|
|
|
"github.com/hashicorp/terraform/internal/providers"
|
2020-09-25 10:18:14 -05:00
|
|
|
"github.com/hashicorp/terraform/plans"
|
|
|
|
"github.com/hashicorp/terraform/states"
|
2021-05-06 13:35:05 -05:00
|
|
|
"github.com/zclconf/go-cty/cty"
|
2020-09-25 10:18:14 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestNodeResourcePlanOrphanExecute(t *testing.T) {
|
|
|
|
state := states.NewState()
|
|
|
|
state.Module(addrs.RootModuleInstance).SetResourceInstanceCurrent(
|
|
|
|
addrs.Resource{
|
|
|
|
Mode: addrs.ManagedResourceMode,
|
|
|
|
Type: "test_object",
|
|
|
|
Name: "foo",
|
|
|
|
}.Instance(addrs.NoKey),
|
|
|
|
&states.ResourceInstanceObjectSrc{
|
|
|
|
AttrsFlat: map[string]string{
|
|
|
|
"test_string": "foo",
|
|
|
|
},
|
|
|
|
Status: states.ObjectReady,
|
|
|
|
},
|
|
|
|
addrs.AbsProviderConfig{
|
|
|
|
Provider: addrs.NewDefaultProvider("test"),
|
|
|
|
Module: addrs.RootModule,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
p := simpleMockProvider()
|
|
|
|
ctx := &MockEvalContext{
|
|
|
|
StateState: state.SyncWrapper(),
|
2021-05-04 20:14:43 -05:00
|
|
|
RefreshStateState: state.DeepCopy().SyncWrapper(),
|
|
|
|
PrevRunStateState: state.DeepCopy().SyncWrapper(),
|
2020-09-25 10:18:14 -05:00
|
|
|
InstanceExpanderExpander: instances.NewExpander(),
|
|
|
|
ProviderProvider: p,
|
|
|
|
ProviderSchemaSchema: &ProviderSchema{
|
|
|
|
ResourceTypes: map[string]*configschema.Block{
|
|
|
|
"test_object": simpleTestSchema(),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
ChangesChanges: plans.NewChanges().SyncWrapper(),
|
|
|
|
}
|
|
|
|
|
|
|
|
node := NodePlannableResourceInstanceOrphan{
|
|
|
|
NodeAbstractResourceInstance: &NodeAbstractResourceInstance{
|
|
|
|
NodeAbstractResource: NodeAbstractResource{
|
|
|
|
ResolvedProvider: addrs.AbsProviderConfig{
|
|
|
|
Provider: addrs.NewDefaultProvider("test"),
|
|
|
|
Module: addrs.RootModule,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Addr: mustResourceInstanceAddr("test_object.foo"),
|
|
|
|
},
|
|
|
|
}
|
2020-10-28 12:47:04 -05:00
|
|
|
diags := node.Execute(ctx, walkApply)
|
|
|
|
if diags.HasErrors() {
|
|
|
|
t.Fatalf("unexpected error: %s", diags.Err())
|
2020-09-25 10:18:14 -05:00
|
|
|
}
|
|
|
|
if !state.Empty() {
|
|
|
|
t.Fatalf("expected empty state, got %s", state.String())
|
|
|
|
}
|
|
|
|
}
|
2021-05-06 13:35:05 -05:00
|
|
|
|
|
|
|
func TestNodeResourcePlanOrphanExecute_alreadyDeleted(t *testing.T) {
|
|
|
|
addr := addrs.Resource{
|
|
|
|
Mode: addrs.ManagedResourceMode,
|
|
|
|
Type: "test_object",
|
|
|
|
Name: "foo",
|
|
|
|
}.Instance(addrs.NoKey).Absolute(addrs.RootModuleInstance)
|
|
|
|
|
|
|
|
state := states.NewState()
|
|
|
|
state.Module(addrs.RootModuleInstance).SetResourceInstanceCurrent(
|
|
|
|
addr.Resource,
|
|
|
|
&states.ResourceInstanceObjectSrc{
|
|
|
|
AttrsFlat: map[string]string{
|
|
|
|
"test_string": "foo",
|
|
|
|
},
|
|
|
|
Status: states.ObjectReady,
|
|
|
|
},
|
|
|
|
addrs.AbsProviderConfig{
|
|
|
|
Provider: addrs.NewDefaultProvider("test"),
|
|
|
|
Module: addrs.RootModule,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
refreshState := state.DeepCopy()
|
|
|
|
prevRunState := state.DeepCopy()
|
|
|
|
changes := plans.NewChanges()
|
|
|
|
|
|
|
|
p := simpleMockProvider()
|
|
|
|
p.ReadResourceResponse = &providers.ReadResourceResponse{
|
|
|
|
NewState: cty.NullVal(p.GetProviderSchemaResponse.ResourceTypes["test_string"].Block.ImpliedType()),
|
|
|
|
}
|
|
|
|
ctx := &MockEvalContext{
|
|
|
|
StateState: state.SyncWrapper(),
|
|
|
|
RefreshStateState: refreshState.SyncWrapper(),
|
|
|
|
PrevRunStateState: prevRunState.SyncWrapper(),
|
|
|
|
InstanceExpanderExpander: instances.NewExpander(),
|
|
|
|
ProviderProvider: p,
|
|
|
|
ProviderSchemaSchema: &ProviderSchema{
|
|
|
|
ResourceTypes: map[string]*configschema.Block{
|
|
|
|
"test_object": simpleTestSchema(),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
ChangesChanges: changes.SyncWrapper(),
|
|
|
|
}
|
|
|
|
|
|
|
|
node := NodePlannableResourceInstanceOrphan{
|
|
|
|
NodeAbstractResourceInstance: &NodeAbstractResourceInstance{
|
|
|
|
NodeAbstractResource: NodeAbstractResource{
|
|
|
|
ResolvedProvider: addrs.AbsProviderConfig{
|
|
|
|
Provider: addrs.NewDefaultProvider("test"),
|
|
|
|
Module: addrs.RootModule,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Addr: mustResourceInstanceAddr("test_object.foo"),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
diags := node.Execute(ctx, walkPlan)
|
|
|
|
if diags.HasErrors() {
|
|
|
|
t.Fatalf("unexpected error: %s", diags.Err())
|
|
|
|
}
|
|
|
|
if !state.Empty() {
|
|
|
|
t.Fatalf("expected empty state, got %s", state.String())
|
|
|
|
}
|
|
|
|
|
|
|
|
if got := prevRunState.ResourceInstance(addr); got == nil {
|
|
|
|
t.Errorf("no entry for %s in the prev run state; should still be present", addr)
|
|
|
|
}
|
|
|
|
if got := refreshState.ResourceInstance(addr); got != nil {
|
|
|
|
t.Errorf("refresh state has entry for %s; should've been removed", addr)
|
|
|
|
}
|
|
|
|
if got := changes.ResourceInstance(addr); got == nil {
|
|
|
|
t.Errorf("no entry for %s in the planned changes; should have a NoOp change", addr)
|
|
|
|
} else {
|
|
|
|
if got, want := got.Action, plans.NoOp; got != want {
|
|
|
|
t.Errorf("planned change for %s has wrong action\ngot: %s\nwant: %s", addr, got, want)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|