package terraform import ( "sync" "github.com/hashicorp/go-multierror" "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 { diff *Diff hooks []Hook module *module.Tree providers map[string]ResourceProviderFactory provisioners map[string]ResourceProvisionerFactory state *State stateLock sync.RWMutex variables map[string]string } // 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 { state := opts.State if state == nil { state = new(State) state.init() } return &Context2{ diff: opts.Diff, hooks: opts.Hooks, module: opts.Module, providers: opts.Providers, provisioners: opts.Provisioners, state: state, variables: opts.Variables, } } // 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) } provisioners := make([]string, 0, len(c.provisioners)) for k, _ := range c.provisioners { provisioners = append(provisioners, k) } return &BuiltinGraphBuilder{ Root: c.module, Providers: providers, Provisioners: provisioners, State: c.state, } } // Apply applies the changes represented by this context and returns // the resulting state. // // In addition to returning the resulting state, this context is updated // with the latest state. func (c *Context2) Apply() (*State, error) { // Copy our own state c.state = c.state.deepcopy() // Do the walk _, err := c.walk(walkApply) // Clean out any unused things c.state.prune() return c.state, err } // 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) if err != nil { return nil, err } p.Diff = walker.Diff // Update the diff so that our context is up-to-date c.diff = p.Diff return p, err } // 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) { // Copy our own state c.state = c.state.deepcopy() // Do the walk if _, err := c.walk(walkRefresh); err != nil { var errs error return nil, multierror.Append(errs, err).Errors } // Clean out any unused things c.state.prune() return c.state, nil } // Validate validates the configuration and returns any warnings or errors. func (c *Context2) Validate() ([]string, []error) { var errs error // Validate the configuration itself if err := c.module.Validate(); err != nil { 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...) } } // Walk walker, err := c.walk(walkValidate) if err != nil { return nil, multierror.Append(errs, err).Errors } // Return the result rerrs := multierror.Append(errs, walker.ValidationErrors...) return walker.ValidationWarnings, rerrs.Errors } 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) }