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 { module *module.Tree providers map[string]ResourceProviderFactory 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 { return &Context2{ module: opts.Module, providers: opts.Providers, state: opts.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) } return &BuiltinGraphBuilder{ Root: c.module, Providers: providers, State: c.state, } } // 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...) } } // Build the graph graph, err := c.GraphBuilder().Build(RootModulePath) if err != nil { return nil, multierror.Append(errs, err).Errors } // Walk the graph walker := &ContextGraphWalker{Context: c, Operation: walkValidate} graph.Walk(walker) // Return the result rerrs := multierror.Append(errs, walker.ValidationErrors...) return walker.ValidationWarnings, rerrs.Errors }