2016-05-08 01:41:27 -05:00
|
|
|
package terraform
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
)
|
|
|
|
|
|
|
|
// EvalReadDataDiff is an EvalNode implementation that executes a data
|
|
|
|
// resource's ReadDataDiff method to discover what attributes it exports.
|
|
|
|
type EvalReadDataDiff struct {
|
|
|
|
Provider *ResourceProvider
|
|
|
|
Output **InstanceDiff
|
|
|
|
OutputState **InstanceState
|
|
|
|
Config **ResourceConfig
|
|
|
|
Info *InstanceInfo
|
2016-05-17 09:58:11 -05:00
|
|
|
|
|
|
|
// Set Previous when re-evaluating diff during apply, to ensure that
|
|
|
|
// the "Destroy" flag is preserved.
|
|
|
|
Previous **InstanceDiff
|
2016-05-08 01:41:27 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
func (n *EvalReadDataDiff) Eval(ctx EvalContext) (interface{}, error) {
|
|
|
|
// TODO: test
|
|
|
|
|
|
|
|
err := ctx.Hook(func(h Hook) (HookAction, error) {
|
|
|
|
return h.PreDiff(n.Info, nil)
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2016-05-17 09:58:11 -05:00
|
|
|
var diff *InstanceDiff
|
2016-05-08 01:41:27 -05:00
|
|
|
|
2016-07-29 12:17:48 -05:00
|
|
|
if n.Previous != nil && *n.Previous != nil && (*n.Previous).GetDestroy() {
|
2016-05-17 09:58:11 -05:00
|
|
|
// If we're re-diffing for a diff that was already planning to
|
|
|
|
// destroy, then we'll just continue with that plan.
|
|
|
|
diff = &InstanceDiff{Destroy: true}
|
|
|
|
} else {
|
|
|
|
provider := *n.Provider
|
|
|
|
config := *n.Config
|
|
|
|
|
2016-05-21 15:23:28 -05:00
|
|
|
var err error
|
|
|
|
diff, err = provider.ReadDataDiff(n.Info, config)
|
2016-05-17 09:58:11 -05:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if diff == nil {
|
|
|
|
diff = new(InstanceDiff)
|
|
|
|
}
|
|
|
|
|
2016-05-22 09:55:39 -05:00
|
|
|
// if id isn't explicitly set then it's always computed, because we're
|
|
|
|
// always "creating a new resource".
|
2016-05-17 09:58:11 -05:00
|
|
|
diff.init()
|
2016-05-22 09:55:39 -05:00
|
|
|
if _, ok := diff.Attributes["id"]; !ok {
|
|
|
|
diff.SetAttribute("id", &ResourceAttrDiff{
|
|
|
|
Old: "",
|
|
|
|
NewComputed: true,
|
|
|
|
RequiresNew: true,
|
|
|
|
Type: DiffAttrOutput,
|
|
|
|
})
|
|
|
|
}
|
2016-05-08 01:41:27 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
err = ctx.Hook(func(h Hook) (HookAction, error) {
|
|
|
|
return h.PostDiff(n.Info, diff)
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
*n.Output = diff
|
|
|
|
|
|
|
|
if n.OutputState != nil {
|
|
|
|
state := &InstanceState{}
|
|
|
|
*n.OutputState = state
|
|
|
|
|
|
|
|
// Apply the diff to the returned state, so the state includes
|
|
|
|
// any attribute values that are not computed.
|
|
|
|
if !diff.Empty() && n.OutputState != nil {
|
|
|
|
*n.OutputState = state.MergeDiff(diff)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// EvalReadDataApply is an EvalNode implementation that executes a data
|
|
|
|
// resource's ReadDataApply method to read data from the data source.
|
|
|
|
type EvalReadDataApply struct {
|
|
|
|
Provider *ResourceProvider
|
|
|
|
Output **InstanceState
|
|
|
|
Diff **InstanceDiff
|
|
|
|
Info *InstanceInfo
|
|
|
|
}
|
|
|
|
|
|
|
|
func (n *EvalReadDataApply) Eval(ctx EvalContext) (interface{}, error) {
|
|
|
|
// TODO: test
|
|
|
|
provider := *n.Provider
|
|
|
|
diff := *n.Diff
|
|
|
|
|
2016-05-08 03:27:46 -05:00
|
|
|
// If the diff is for *destroying* this resource then we'll
|
|
|
|
// just drop its state and move on, since data resources don't
|
|
|
|
// support an actual "destroy" action.
|
2016-07-29 12:17:48 -05:00
|
|
|
if diff != nil && diff.GetDestroy() {
|
2016-05-08 03:27:46 -05:00
|
|
|
if n.Output != nil {
|
|
|
|
*n.Output = nil
|
|
|
|
}
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
2016-05-08 01:41:27 -05:00
|
|
|
// For the purpose of external hooks we present a data apply as a
|
|
|
|
// "Refresh" rather than an "Apply" because creating a data source
|
|
|
|
// is presented to users/callers as a "read" operation.
|
|
|
|
err := ctx.Hook(func(h Hook) (HookAction, error) {
|
|
|
|
// We don't have a state yet, so we'll just give the hook an
|
|
|
|
// empty one to work with.
|
|
|
|
return h.PreRefresh(n.Info, &InstanceState{})
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
state, err := provider.ReadDataApply(n.Info, diff)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("%s: %s", n.Info.Id, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
err = ctx.Hook(func(h Hook) (HookAction, error) {
|
|
|
|
return h.PostRefresh(n.Info, state)
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if n.Output != nil {
|
|
|
|
*n.Output = state
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, nil
|
|
|
|
}
|