4 "github.com/hashicorp/terraform/dag"
5 "github.com/hashicorp/terraform/plans"
6 "github.com/hashicorp/terraform/providers"
7 "github.com/hashicorp/terraform/states"
8 "github.com/hashicorp/terraform/tfdiags"
9 "github.com/zclconf/go-cty/cty"
12 // NodeRefreshableDataResource represents a resource that is "refreshable".
13 type NodeRefreshableDataResource struct {
18 _ GraphNodeSubPath = (*NodeRefreshableDataResource)(nil)
19 _ GraphNodeDynamicExpandable = (*NodeRefreshableDataResource)(nil)
20 _ GraphNodeReferenceable = (*NodeRefreshableDataResource)(nil)
21 _ GraphNodeReferencer = (*NodeRefreshableDataResource)(nil)
22 _ GraphNodeResource = (*NodeRefreshableDataResource)(nil)
23 _ GraphNodeAttachResourceConfig = (*NodeRefreshableDataResource)(nil)
26 // GraphNodeDynamicExpandable
27 func (n *NodeRefreshableDataResource) DynamicExpand(ctx EvalContext) (*Graph, error) {
28 var diags tfdiags.Diagnostics
30 count, countKnown, countDiags := evaluateResourceCountExpressionKnown(n.Config.Count, ctx)
31 diags = diags.Append(countDiags)
32 if countDiags.HasErrors() {
33 return nil, diags.Err()
36 // If the count isn't known yet, we'll skip refreshing and try expansion
37 // again during the plan walk.
41 // Next we need to potentially rename an instance address in the state
42 // if we're transitioning whether "count" is set at all.
43 fixResourceCountSetTransition(ctx, n.ResourceAddr(), count != -1)
45 // Our graph transformers require access to the full state, so we'll
46 // temporarily lock it while we work on this.
47 state := ctx.State().Lock()
48 defer ctx.State().Unlock()
50 // The concrete resource factory we'll use
51 concreteResource := func(a *NodeAbstractResourceInstance) dag.Vertex {
52 // Add the config and state since we don't do that via transforms
54 a.ResolvedProvider = n.ResolvedProvider
56 return &NodeRefreshableDataResourceInstance{
57 NodeAbstractResourceInstance: a,
61 // We also need a destroyable resource for orphans that are a result of a
63 concreteResourceDestroyable := func(a *NodeAbstractResourceInstance) dag.Vertex {
64 // Add the config and provider since we don't do that via transforms
66 a.ResolvedProvider = n.ResolvedProvider
68 return &NodeDestroyableDataResourceInstance{
69 NodeAbstractResourceInstance: a,
73 // Start creating the steps
74 steps := []GraphTransformer{
76 &ResourceCountTransformer{
77 Concrete: concreteResource,
80 Addr: n.ResourceAddr(),
83 // Add the count orphans. As these are orphaned refresh nodes, we add them
84 // directly as NodeDestroyableDataResource.
85 &OrphanResourceCountTransformer{
86 Concrete: concreteResourceDestroyable,
88 Addr: n.ResourceAddr(),
93 &AttachStateTransformer{State: state},
96 &TargetsTransformer{Targets: n.Targets},
98 // Connect references so ordering is correct
99 &ReferenceTransformer{},
101 // Make sure there is a single root
106 b := &BasicGraphBuilder{
109 Name: "NodeRefreshableDataResource",
112 graph, diags := b.Build(ctx.Path())
113 return graph, diags.ErrWithWarnings()
116 // NodeRefreshableDataResourceInstance represents a single resource instance
117 // that is refreshable.
118 type NodeRefreshableDataResourceInstance struct {
119 *NodeAbstractResourceInstance
123 func (n *NodeRefreshableDataResourceInstance) EvalTree() EvalNode {
124 addr := n.ResourceInstanceAddr()
126 // These variables are the state for the eval sequence below, and are
127 // updated through pointers.
128 var provider providers.Interface
129 var providerSchema *ProviderSchema
130 var change *plans.ResourceInstanceChange
131 var state *states.ResourceInstanceObject
132 var configVal cty.Value
134 return &EvalSequence{
137 Addr: n.ResolvedProvider,
139 Schema: &providerSchema,
142 // Always destroy the existing state first, since we must
143 // make sure that values from a previous read will not
144 // get interpolated if we end up needing to defer our
145 // loading until apply time.
148 ProviderAddr: n.ResolvedProvider,
149 State: &state, // a pointer to nil, here
150 ProviderSchema: &providerSchema,
154 If: func(ctx EvalContext) (bool, error) {
155 // If the config explicitly has a depends_on for this
156 // data source, assume the intention is to prevent
157 // refreshing ahead of that dependency, and therefore
158 // we need to deal with this resource during the apply
160 if len(n.Config.DependsOn) > 0 {
161 return true, EvalEarlyExitError{}
169 // EvalReadData will _attempt_ to read the data source, but may
170 // generate an incomplete planned object if the configuration
171 // includes values that won't be known until apply.
175 Dependencies: n.StateReferences(),
177 ProviderAddr: n.ResolvedProvider,
178 ProviderSchema: &providerSchema,
179 OutputChange: &change,
180 OutputConfigValue: &configVal,
185 If: func(ctx EvalContext) (bool, error) {
186 return (*state).Status != states.ObjectPlanned, nil
192 ProviderAddr: n.ResolvedProvider,
194 ProviderSchema: &providerSchema,
196 &EvalUpdateStateHook{},
200 // We can't deal with this yet, so we'll repeat this step
201 // during the plan walk to produce a planned change to read
202 // this during the apply walk. However, we do still need to
203 // save the generated change and partial state so that
204 // results from it can be included in other data resources
205 // or provider configurations during the refresh walk.
206 // (The planned object we save in the state here will be
207 // pruned out at the end of the refresh walk, returning
208 // it back to being unset again for subsequent walks.)
213 ProviderSchema: &providerSchema,
217 ProviderAddr: n.ResolvedProvider,
219 ProviderSchema: &providerSchema,