2015-02-04 17:44:23 -06:00
|
|
|
package terraform
|
|
|
|
|
|
|
|
import (
|
|
|
|
"sync"
|
|
|
|
|
2015-02-07 11:53:46 -06:00
|
|
|
"github.com/hashicorp/go-multierror"
|
2015-02-04 17:44:23 -06:00
|
|
|
"github.com/hashicorp/terraform/config/module"
|
|
|
|
)
|
|
|
|
|
|
|
|
// ContextOpts are the user-configurable options to create a context with
|
|
|
|
// NewContext.
|
|
|
|
type ContextOpts struct {
|
|
|
|
Diff *Diff
|
|
|
|
Hooks []Hook
|
|
|
|
Module *module.Tree
|
|
|
|
Parallelism int
|
|
|
|
State *State
|
|
|
|
Providers map[string]ResourceProviderFactory
|
|
|
|
Provisioners map[string]ResourceProvisionerFactory
|
|
|
|
Variables map[string]string
|
|
|
|
|
|
|
|
UIInput UIInput
|
|
|
|
}
|
|
|
|
|
|
|
|
// Context represents all the context that Terraform needs in order to
|
|
|
|
// perform operations on infrastructure. This structure is built using
|
|
|
|
// NewContext. See the documentation for that.
|
|
|
|
type Context2 struct {
|
2015-02-11 17:22:03 -06:00
|
|
|
diff *Diff
|
2015-02-11 15:43:07 -06:00
|
|
|
hooks []Hook
|
2015-02-09 13:15:54 -06:00
|
|
|
module *module.Tree
|
|
|
|
providers map[string]ResourceProviderFactory
|
|
|
|
provisioners map[string]ResourceProvisionerFactory
|
|
|
|
state *State
|
|
|
|
stateLock sync.RWMutex
|
|
|
|
variables map[string]string
|
2015-02-04 17:44:23 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewContext creates a new Context structure.
|
|
|
|
//
|
|
|
|
// Once a Context is creator, the pointer values within ContextOpts
|
|
|
|
// should not be mutated in any way, since the pointers are copied, not
|
|
|
|
// the values themselves.
|
|
|
|
func NewContext2(opts *ContextOpts) *Context2 {
|
2015-02-11 17:22:03 -06:00
|
|
|
state := opts.State
|
|
|
|
if state == nil {
|
|
|
|
state = new(State)
|
|
|
|
state.init()
|
|
|
|
}
|
|
|
|
|
2015-02-04 17:44:23 -06:00
|
|
|
return &Context2{
|
2015-02-11 17:22:03 -06:00
|
|
|
diff: opts.Diff,
|
2015-02-11 15:43:07 -06:00
|
|
|
hooks: opts.Hooks,
|
2015-02-09 13:15:54 -06:00
|
|
|
module: opts.Module,
|
|
|
|
providers: opts.Providers,
|
|
|
|
provisioners: opts.Provisioners,
|
2015-02-11 17:22:03 -06:00
|
|
|
state: state,
|
2015-02-09 13:15:54 -06:00
|
|
|
variables: opts.Variables,
|
2015-02-04 17:44:23 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// GraphBuilder returns the GraphBuilder that will be used to create
|
|
|
|
// the graphs for this context.
|
|
|
|
func (c *Context2) GraphBuilder() GraphBuilder {
|
|
|
|
// TODO test
|
|
|
|
providers := make([]string, 0, len(c.providers))
|
|
|
|
for k, _ := range c.providers {
|
|
|
|
providers = append(providers, k)
|
|
|
|
}
|
|
|
|
|
2015-02-09 13:15:54 -06:00
|
|
|
provisioners := make([]string, 0, len(c.provisioners))
|
|
|
|
for k, _ := range c.provisioners {
|
|
|
|
provisioners = append(provisioners, k)
|
|
|
|
}
|
|
|
|
|
2015-02-04 17:44:23 -06:00
|
|
|
return &BuiltinGraphBuilder{
|
2015-02-09 13:15:54 -06:00
|
|
|
Root: c.module,
|
|
|
|
Providers: providers,
|
|
|
|
Provisioners: provisioners,
|
|
|
|
State: c.state,
|
2015-02-04 17:44:23 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-11 17:22:03 -06:00
|
|
|
// Plan generates an execution plan for the given context.
|
|
|
|
//
|
|
|
|
// The execution plan encapsulates the context and can be stored
|
|
|
|
// in order to reinstantiate a context later for Apply.
|
|
|
|
//
|
|
|
|
// Plan also updates the diff of this context to be the diff generated
|
|
|
|
// by the plan, so Apply can be called after.
|
|
|
|
func (c *Context2) Plan(opts *PlanOpts) (*Plan, error) {
|
|
|
|
p := &Plan{
|
|
|
|
Module: c.module,
|
|
|
|
Vars: c.variables,
|
|
|
|
State: c.state,
|
|
|
|
}
|
|
|
|
|
|
|
|
var operation walkOperation
|
|
|
|
if opts != nil && opts.Destroy {
|
|
|
|
operation = walkPlanDestroy
|
|
|
|
} else {
|
|
|
|
// Set our state to be something temporary. We do this so that
|
|
|
|
// the plan can update a fake state so that variables work, then
|
|
|
|
// we replace it back with our old state.
|
|
|
|
old := c.state
|
|
|
|
if old == nil {
|
|
|
|
c.state = &State{}
|
|
|
|
c.state.init()
|
|
|
|
} else {
|
|
|
|
c.state = old.deepcopy()
|
|
|
|
}
|
|
|
|
defer func() {
|
|
|
|
c.state = old
|
|
|
|
}()
|
|
|
|
|
|
|
|
operation = walkPlan
|
|
|
|
}
|
|
|
|
|
|
|
|
// Do the walk
|
|
|
|
walker, err := c.walk(operation)
|
|
|
|
p.Diff = walker.Diff
|
|
|
|
|
|
|
|
// Update the diff so that our context is up-to-date
|
|
|
|
c.diff = p.Diff
|
|
|
|
|
|
|
|
return p, err
|
|
|
|
}
|
|
|
|
|
2015-02-10 10:58:14 -06:00
|
|
|
// Refresh goes through all the resources in the state and refreshes them
|
|
|
|
// to their latest state. This will update the state that this context
|
|
|
|
// works with, along with returning it.
|
|
|
|
//
|
|
|
|
// Even in the case an error is returned, the state will be returned and
|
|
|
|
// will potentially be partially updated.
|
|
|
|
func (c *Context2) Refresh() (*State, []error) {
|
2015-02-11 15:52:13 -06:00
|
|
|
// Copy our own state
|
|
|
|
c.state = c.state.deepcopy()
|
|
|
|
|
|
|
|
// Do the walk
|
2015-02-10 10:58:14 -06:00
|
|
|
if _, err := c.walk(walkRefresh); err != nil {
|
|
|
|
var errs error
|
|
|
|
return nil, multierror.Append(errs, err).Errors
|
|
|
|
}
|
|
|
|
|
2015-02-11 10:51:17 -06:00
|
|
|
// Clean out any unused things
|
|
|
|
c.state.prune()
|
|
|
|
|
2015-02-11 10:48:45 -06:00
|
|
|
return c.state, nil
|
2015-02-10 10:58:14 -06:00
|
|
|
}
|
|
|
|
|
2015-02-04 17:44:23 -06:00
|
|
|
// Validate validates the configuration and returns any warnings or errors.
|
|
|
|
func (c *Context2) Validate() ([]string, []error) {
|
2015-02-07 11:53:46 -06:00
|
|
|
var errs error
|
2015-02-04 19:02:18 -06:00
|
|
|
|
|
|
|
// Validate the configuration itself
|
|
|
|
if err := c.module.Validate(); err != nil {
|
2015-02-07 11:53:46 -06:00
|
|
|
errs = multierror.Append(errs, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// This only needs to be done for the root module, since inter-module
|
|
|
|
// variables are validated in the module tree.
|
|
|
|
if config := c.module.Config(); config != nil {
|
|
|
|
// Validate the user variables
|
|
|
|
if err := smcUserVariables(config, c.variables); len(err) > 0 {
|
|
|
|
errs = multierror.Append(errs, err...)
|
|
|
|
}
|
2015-02-04 19:02:18 -06:00
|
|
|
}
|
|
|
|
|
2015-02-10 10:58:14 -06:00
|
|
|
// Walk
|
|
|
|
walker, err := c.walk(walkValidate)
|
2015-02-04 17:44:23 -06:00
|
|
|
if err != nil {
|
2015-02-07 11:55:11 -06:00
|
|
|
return nil, multierror.Append(errs, err).Errors
|
2015-02-04 17:44:23 -06:00
|
|
|
}
|
|
|
|
|
2015-02-07 15:29:55 -06:00
|
|
|
// Return the result
|
|
|
|
rerrs := multierror.Append(errs, walker.ValidationErrors...)
|
|
|
|
return walker.ValidationWarnings, rerrs.Errors
|
2015-02-04 17:44:23 -06:00
|
|
|
}
|
2015-02-10 10:58:14 -06:00
|
|
|
|
|
|
|
func (c *Context2) walk(operation walkOperation) (*ContextGraphWalker, error) {
|
|
|
|
// Build the graph
|
|
|
|
graph, err := c.GraphBuilder().Build(RootModulePath)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Walk the graph
|
|
|
|
walker := &ContextGraphWalker{Context: c, Operation: operation}
|
|
|
|
return walker, graph.Walk(walker)
|
|
|
|
}
|