mirror of
https://github.com/opentofu/opentofu.git
synced 2025-01-15 19:22:46 -06:00
prevent errors from NoOp deposed changes
If a previously deposed object is deleted outside of Terraform, the next plan will result in a NoOp change for the deposed object. Fix the check to verify that the deposed object has an acceptable action rather than use the `update` flag.
This commit is contained in:
parent
8c98e1f4a4
commit
f78ecef5e7
@ -3682,3 +3682,69 @@ output "out" {
|
||||
|
||||
assertNoErrors(t, diags)
|
||||
}
|
||||
|
||||
// A deposed instances which no longer exists during ReadResource creates NoOp
|
||||
// change, which should not effect the plan.
|
||||
func TestContext2Plan_deposedNoLongerExists(t *testing.T) {
|
||||
m := testModuleInline(t, map[string]string{
|
||||
"main.tf": `
|
||||
resource "test_object" "b" {
|
||||
count = 1
|
||||
test_string = "updated"
|
||||
lifecycle {
|
||||
create_before_destroy = true
|
||||
}
|
||||
}
|
||||
`,
|
||||
})
|
||||
|
||||
p := simpleMockProvider()
|
||||
p.ReadResourceFn = func(req providers.ReadResourceRequest) (resp providers.ReadResourceResponse) {
|
||||
s := req.PriorState.GetAttr("test_string").AsString()
|
||||
if s == "current" {
|
||||
resp.NewState = req.PriorState
|
||||
return resp
|
||||
}
|
||||
// pretend the non-current instance has been deleted already
|
||||
resp.NewState = cty.NullVal(req.PriorState.Type())
|
||||
return resp
|
||||
}
|
||||
|
||||
// Here we introduce a cycle via state which only shows up in the apply
|
||||
// graph where the actual destroy instances are connected in the graph.
|
||||
// This could happen for example when a user has an existing state with
|
||||
// stored dependencies, and changes the config in such a way that
|
||||
// contradicts the stored dependencies.
|
||||
state := states.NewState()
|
||||
root := state.EnsureModule(addrs.RootModuleInstance)
|
||||
root.SetResourceInstanceDeposed(
|
||||
mustResourceInstanceAddr("test_object.a[0]").Resource,
|
||||
states.DeposedKey("deposed"),
|
||||
&states.ResourceInstanceObjectSrc{
|
||||
Status: states.ObjectTainted,
|
||||
AttrsJSON: []byte(`{"test_string":"old"}`),
|
||||
Dependencies: []addrs.ConfigResource{},
|
||||
},
|
||||
mustProviderConfig(`provider["registry.terraform.io/hashicorp/test"]`),
|
||||
)
|
||||
root.SetResourceInstanceCurrent(
|
||||
mustResourceInstanceAddr("test_object.a[0]").Resource,
|
||||
&states.ResourceInstanceObjectSrc{
|
||||
Status: states.ObjectTainted,
|
||||
AttrsJSON: []byte(`{"test_string":"current"}`),
|
||||
Dependencies: []addrs.ConfigResource{},
|
||||
},
|
||||
mustProviderConfig(`provider["registry.terraform.io/hashicorp/test"]`),
|
||||
)
|
||||
|
||||
ctx := testContext2(t, &ContextOpts{
|
||||
Providers: map[addrs.Provider]providers.Factory{
|
||||
addrs.NewDefaultProvider("test"): testProviderFuncFixed(p),
|
||||
},
|
||||
})
|
||||
|
||||
_, diags := ctx.Plan(m, state, &PlanOpts{
|
||||
Mode: plans.NormalMode,
|
||||
})
|
||||
assertNoErrors(t, diags)
|
||||
}
|
||||
|
@ -80,7 +80,10 @@ func (t *DiffTransformer) Transform(g *Graph) error {
|
||||
update = true
|
||||
}
|
||||
|
||||
if dk != states.NotDeposed && update {
|
||||
// A deposed instance may only have a change of Delete or NoOp. A NoOp
|
||||
// can happen if the provider shows it no longer exists during the most
|
||||
// recent ReadResource operation.
|
||||
if dk != states.NotDeposed && !(rc.Action == plans.Delete || rc.Action == plans.NoOp) {
|
||||
diags = diags.Append(tfdiags.Sourceless(
|
||||
tfdiags.Error,
|
||||
"Invalid planned change for deposed object",
|
||||
|
Loading…
Reference in New Issue
Block a user