Skip to content

Commit

Permalink
stacks: handle deferred actions in refresh
Browse files Browse the repository at this point in the history
Co-authored-by: Matej Risek <[email protected]>
  • Loading branch information
DanielMSchmidt and matejrisek committed Apr 5, 2024
1 parent 15058db commit b2fe26d
Show file tree
Hide file tree
Showing 5 changed files with 161 additions and 9 deletions.
2 changes: 1 addition & 1 deletion internal/plugin/convert/deferred_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -42,7 +42,7 @@ func TestProtoDeferred(t *testing.T) {

deferred := ProtoToDeferred(d)
if deferred.Reason != tc.expected {
t.Fatalf("expected %d, got %d", tc.expected, deferred.Reason)
t.Fatalf("expected %q, got %q", tc.expected, deferred.Reason)
}
})
}
Expand Down
2 changes: 1 addition & 1 deletion internal/plugin6/convert/deferred_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -42,7 +42,7 @@ func TestProtoDeferred(t *testing.T) {

deferred := ProtoToDeferred(d)
if deferred.Reason != providers.DeferredReason(tc.expected) {
t.Fatalf("expected %d, got %d", tc.expected, deferred.Reason)
t.Fatalf("expected %q, got %q", tc.expected, deferred.Reason)
}
})
}
Expand Down
76 changes: 74 additions & 2 deletions internal/terraform/context_apply_deferred_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -57,6 +57,10 @@ type deferredActionsTestStage struct {

// Whether the plan should be completed during this stage.
complete bool

// buildOpts is an optional field, that lets the test specify additional
// options to be used when building the plan.
buildOpts func(opts *PlanOpts)
}

var (
Expand Down Expand Up @@ -343,11 +347,54 @@ output "c" {
},
},
}

// resourceReadTest is a test that covers the behavior of reading resources
// in a refresh when the refresh is responding with a deferral.
resourceReadTest = deferredActionsTest{
configs: map[string]string{
"main.tf": `
resource "test" "a" {
name = "deferred_read"
}
output "a" {
value = test.a
}
`,
},
stages: []deferredActionsTestStage{
{
buildOpts: func(opts *PlanOpts) {
opts.Mode = plans.RefreshOnlyMode
},
inputs: map[string]cty.Value{},
wantPlanned: map[string]cty.Value{
// The all resources will be deferred, so shouldn't
// have any action at this stage.
},

wantActions: map[string]plans.Action{},
wantApplied: map[string]cty.Value{
// The all resources will be deferred, so shouldn't
// have any action at this stage.
},
wantOutputs: map[string]cty.Value{
"a": cty.ObjectVal(map[string]cty.Value{
"name": cty.StringVal("a"),
"upstream_names": cty.NullVal(cty.Set(cty.String)),
}),
},
wantDeferred: make(map[string]providers.DeferredReason),
complete: true,
},
},
}
)

func TestContextApply_deferredActions(t *testing.T) {
tests := map[string]deferredActionsTest{
"resource_for_each": resourceForEachTest,
"resource_read": resourceReadTest,
}
for name, test := range tests {
t.Run(name, func(t *testing.T) {
Expand Down Expand Up @@ -381,7 +428,7 @@ func TestContextApply_deferredActions(t *testing.T) {
},
})

plan, diags := ctx.Plan(cfg, state, &PlanOpts{
opts := &PlanOpts{
Mode: plans.NormalMode,
DeferralAllowed: true,
SetVariables: func() InputValues {
Expand All @@ -394,7 +441,13 @@ func TestContextApply_deferredActions(t *testing.T) {
}
return values
}(),
})
}

if stage.buildOpts != nil {
stage.buildOpts(opts)
}

plan, diags := ctx.Plan(cfg, state, opts)
if plan.Complete != stage.complete {
t.Errorf("wrong completion status in plan: got %v, want %v", plan.Complete, stage.complete)
}
Expand Down Expand Up @@ -425,6 +478,11 @@ func TestContextApply_deferredActions(t *testing.T) {
return
}

if opts.Mode == plans.RefreshOnlyMode {
// Don't execute the apply stage if wantApplied is nil.
return
}

updatedState, diags := ctx.Apply(plan, cfg, nil)

// We expect the correct applied changes and no diagnostics.
Expand Down Expand Up @@ -506,6 +564,20 @@ func (provider *deferredActionsProvider) Provider() providers.Interface {
},
},
},
ReadResourceFn: func(req providers.ReadResourceRequest) providers.ReadResourceResponse {
if key := req.PriorState.GetAttr("name"); key.IsKnown() && key.AsString() == "deferred_read" {
return providers.ReadResourceResponse{
NewState: req.PriorState,
Deferred: &providers.Deferred{
Reason: providers.DeferredReasonProviderConfigUnknown,
},
}
}

return providers.ReadResourceResponse{
NewState: req.PriorState,
}
},
PlanResourceChangeFn: func(req providers.PlanResourceChangeRequest) providers.PlanResourceChangeResponse {
key := "<unknown>"
if v := req.Config.GetAttr("name"); v.IsKnown() {
Expand Down
24 changes: 19 additions & 5 deletions internal/terraform/node_resource_abstract_instance.go
Original file line number Diff line number Diff line change
Expand Up @@ -641,11 +641,23 @@ func (n *NodeAbstractResourceInstance) refresh(ctx EvalContext, deposedKey state
}
} else {
resp = provider.ReadResource(providers.ReadResourceRequest{
TypeName: n.Addr.Resource.Resource.Type,
PriorState: priorVal,
Private: state.Private,
ProviderMeta: metaConfigVal,
TypeName: n.Addr.Resource.Resource.Type,
PriorState: priorVal,
Private: state.Private,
ProviderMeta: metaConfigVal,
DeferralAllowed: ctx.Deferrals().DeferralAllowed(),
})

if resp.Deferred != nil {
deferrals := ctx.Deferrals()
deferrals.ReportResourceInstanceDeferred(absAddr, resp.Deferred.Reason, &plans.ResourceInstanceChange{
Addr: absAddr,
Change: plans.Change{
Action: plans.Create,
After: cty.DynamicVal,
},
})
}
}
if n.Config != nil {
resp.Diagnostics = resp.Diagnostics.InConfigBody(n.Config.Config, n.Addr.String())
Expand All @@ -663,7 +675,9 @@ func (n *NodeAbstractResourceInstance) refresh(ctx EvalContext, deposedKey state
panic("new state is cty.NilVal")
}

if !resp.NewState.IsWhollyKnown() {
// If we have deferred the refresh, we expect the new state not to be wholly known
// and callers should be prepared to handle this.
if !resp.NewState.IsWhollyKnown() && deferred == nil {
diags = diags.Append(tfdiags.Sourceless(
tfdiags.Error,
"Provider produced invalid object",
Expand Down
66 changes: 66 additions & 0 deletions internal/terraform/node_resource_abstract_instance_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,8 @@ import (
"github.com/hashicorp/terraform/internal/addrs"
"github.com/hashicorp/terraform/internal/configs"
"github.com/hashicorp/terraform/internal/configs/configschema"
"github.com/hashicorp/terraform/internal/plans/deferring"
"github.com/hashicorp/terraform/internal/providers"
"github.com/hashicorp/terraform/internal/states"
"github.com/zclconf/go-cty/cty"
)
Expand Down Expand Up @@ -184,3 +186,67 @@ aws_instance.foo:
provider = provider["registry.terraform.io/hashicorp/aws"]
`)
}

func TestNodeAbstractResourceInstance_refresh_with_deferred_read(t *testing.T) {
state := states.NewState()
evalCtx := &MockEvalContext{}
evalCtx.StateState = state.SyncWrapper()
evalCtx.Scope = evalContextModuleInstance{Addr: addrs.RootModuleInstance}

mockProvider := mockProviderWithResourceTypeSchema("aws_instance", &configschema.Block{
Attributes: map[string]*configschema.Attribute{
"id": {
Type: cty.String,
Optional: true,
},
},
})
mockProvider.ConfigureProviderCalled = true

mockProvider.ReadResourceFn = func(providers.ReadResourceRequest) providers.ReadResourceResponse {
return providers.ReadResourceResponse{
NewState: cty.ObjectVal(map[string]cty.Value{
"id": cty.UnknownVal(cty.String),
}),
Deferred: &providers.Deferred{
Reason: providers.DeferredReasonAbsentPrereq,
},
}
}

obj := &states.ResourceInstanceObject{
Value: cty.ObjectVal(map[string]cty.Value{
"id": cty.StringVal("i-abc123"),
}),
Status: states.ObjectReady,
}

node := &NodeAbstractResourceInstance{
Addr: mustResourceInstanceAddr("aws_instance.foo"),
NodeAbstractResource: NodeAbstractResource{
ResolvedProvider: mustProviderConfig(`provider["registry.terraform.io/hashicorp/aws"]`),
},
}
evalCtx.ProviderProvider = mockProvider
evalCtx.ProviderSchemaSchema = mockProvider.GetProviderSchema()
resourceGraph := addrs.NewDirectedGraph[addrs.ConfigResource]()
evalCtx.DeferralsState = deferring.NewDeferred(resourceGraph, true)

rio, diags := node.refresh(evalCtx, states.NotDeposed, obj)
if diags.HasErrors() {
t.Fatal(diags.Err())
}

value := rio.Value
if value.IsWhollyKnown() {
t.Fatalf("value was known: %v", value)
}

if !evalCtx.DeferralsCalled {
t.Fatalf("expected deferral to be called")
}

if !evalCtx.DeferralsState.HaveAnyDeferrals() {
t.Fatalf("expected deferral to be present")
}
}

0 comments on commit b2fe26d

Please sign in to comment.