2015-02-03 19:46:11 -06:00
|
|
|
package terraform
|
|
|
|
|
|
|
|
import (
|
2016-12-22 13:33:26 -06:00
|
|
|
"context"
|
2015-02-03 19:46:11 -06:00
|
|
|
"fmt"
|
2015-02-13 11:05:09 -06:00
|
|
|
"log"
|
2015-02-03 19:46:11 -06:00
|
|
|
"sync"
|
|
|
|
|
|
|
|
"github.com/hashicorp/terraform/config"
|
|
|
|
)
|
|
|
|
|
|
|
|
// BuiltinEvalContext is an EvalContext implementation that is used by
|
|
|
|
// Terraform by default.
|
|
|
|
type BuiltinEvalContext struct {
|
2016-12-22 13:33:26 -06:00
|
|
|
// StopContext is the context used to track whether we're complete
|
|
|
|
StopContext context.Context
|
|
|
|
|
2015-05-04 12:51:34 -05:00
|
|
|
// PathValue is the Path that this context is operating within.
|
|
|
|
PathValue []string
|
|
|
|
|
|
|
|
// Interpolater setting below affect the interpolation of variables.
|
|
|
|
//
|
|
|
|
// The InterpolaterVars are the exact value for ${var.foo} values.
|
|
|
|
// The map is shared between all contexts and is a mapping of
|
|
|
|
// PATH to KEY to VALUE. Because it is shared by all contexts as well
|
|
|
|
// as the Interpolater itself, it is protected by InterpolaterVarLock
|
|
|
|
// which must be locked during any access to the map.
|
2015-02-10 01:32:28 -06:00
|
|
|
Interpolater *Interpolater
|
2016-04-11 12:40:06 -05:00
|
|
|
InterpolaterVars map[string]map[string]interface{}
|
2015-05-01 18:29:19 -05:00
|
|
|
InterpolaterVarLock *sync.Mutex
|
2015-05-04 12:51:34 -05:00
|
|
|
|
2017-10-13 09:34:03 -05:00
|
|
|
Components contextComponentFactory
|
|
|
|
Hooks []Hook
|
|
|
|
InputValue UIInput
|
|
|
|
ProviderCache map[string]ResourceProvider
|
2018-04-19 16:47:36 -05:00
|
|
|
ProviderSchemas map[string]*ProviderSchema
|
2015-02-13 19:59:54 -06:00
|
|
|
ProviderInputConfig map[string]map[string]interface{}
|
2015-02-10 01:32:28 -06:00
|
|
|
ProviderLock *sync.Mutex
|
|
|
|
ProvisionerCache map[string]ResourceProvisioner
|
|
|
|
ProvisionerLock *sync.Mutex
|
2015-02-11 17:22:03 -06:00
|
|
|
DiffValue *Diff
|
|
|
|
DiffLock *sync.RWMutex
|
2015-02-11 10:48:45 -06:00
|
|
|
StateValue *State
|
|
|
|
StateLock *sync.RWMutex
|
2015-02-03 19:46:11 -06:00
|
|
|
|
2015-05-01 18:29:19 -05:00
|
|
|
once sync.Once
|
2015-02-03 19:46:11 -06:00
|
|
|
}
|
|
|
|
|
2016-12-22 13:33:26 -06:00
|
|
|
func (ctx *BuiltinEvalContext) Stopped() <-chan struct{} {
|
|
|
|
// This can happen during tests. During tests, we just block forever.
|
|
|
|
if ctx.StopContext == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return ctx.StopContext.Done()
|
|
|
|
}
|
|
|
|
|
2015-02-11 15:43:07 -06:00
|
|
|
func (ctx *BuiltinEvalContext) Hook(fn func(Hook) (HookAction, error)) error {
|
|
|
|
for _, h := range ctx.Hooks {
|
|
|
|
action, err := fn(h)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
switch action {
|
|
|
|
case HookActionContinue:
|
|
|
|
continue
|
|
|
|
case HookActionHalt:
|
|
|
|
// Return an early exit error to trigger an early exit
|
2015-02-13 11:05:09 -06:00
|
|
|
log.Printf("[WARN] Early exit triggered by hook: %T", h)
|
2015-02-11 15:43:07 -06:00
|
|
|
return EvalEarlyExitError{}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-02-13 19:59:54 -06:00
|
|
|
func (ctx *BuiltinEvalContext) Input() UIInput {
|
|
|
|
return ctx.InputValue
|
|
|
|
}
|
|
|
|
|
2017-11-01 17:34:18 -05:00
|
|
|
func (ctx *BuiltinEvalContext) InitProvider(typeName, name string) (ResourceProvider, error) {
|
2015-02-03 19:46:11 -06:00
|
|
|
ctx.once.Do(ctx.init)
|
|
|
|
|
2015-02-08 19:58:02 -06:00
|
|
|
// If we already initialized, it is an error
|
2017-11-01 17:34:18 -05:00
|
|
|
if p := ctx.Provider(name); p != nil {
|
|
|
|
return nil, fmt.Errorf("Provider '%s' already initialized", name)
|
2015-02-03 19:46:11 -06:00
|
|
|
}
|
|
|
|
|
2015-02-08 19:58:02 -06:00
|
|
|
// Warning: make sure to acquire these locks AFTER the call to Provider
|
|
|
|
// above, since it also acquires locks.
|
|
|
|
ctx.ProviderLock.Lock()
|
|
|
|
defer ctx.ProviderLock.Unlock()
|
|
|
|
|
2017-11-01 17:34:18 -05:00
|
|
|
p, err := ctx.Components.ResourceProvider(typeName, name)
|
2015-02-04 19:02:18 -06:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2017-11-01 17:34:18 -05:00
|
|
|
ctx.ProviderCache[name] = p
|
2018-04-19 16:47:36 -05:00
|
|
|
|
|
|
|
// Also fetch and cache the provider's schema.
|
|
|
|
// FIXME: This is using a non-ideal provider API that requires us to
|
|
|
|
// request specific resource types, but we actually just want _all_ the
|
|
|
|
// resource types, so we'll list these first. Once the provider API is
|
|
|
|
// updated we'll get enough data to populate this whole structure in
|
|
|
|
// a single call.
|
|
|
|
resourceTypes := p.Resources()
|
|
|
|
dataSources := p.DataSources()
|
|
|
|
resourceTypeNames := make([]string, len(resourceTypes))
|
|
|
|
for i, t := range resourceTypes {
|
|
|
|
resourceTypeNames[i] = t.Name
|
|
|
|
}
|
|
|
|
dataSourceNames := make([]string, len(dataSources))
|
|
|
|
for i, t := range dataSources {
|
|
|
|
dataSourceNames[i] = t.Name
|
|
|
|
}
|
|
|
|
schema, err := p.GetSchema(&ProviderSchemaRequest{
|
|
|
|
DataSources: dataSourceNames,
|
|
|
|
ResourceTypes: resourceTypeNames,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("error fetching schema for %s: %s", name, err)
|
|
|
|
}
|
|
|
|
if ctx.ProviderSchemas == nil {
|
|
|
|
ctx.ProviderSchemas = make(map[string]*ProviderSchema)
|
|
|
|
}
|
|
|
|
ctx.ProviderSchemas[name] = schema
|
|
|
|
|
2015-02-04 19:02:18 -06:00
|
|
|
return p, nil
|
2015-02-03 19:46:11 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
func (ctx *BuiltinEvalContext) Provider(n string) ResourceProvider {
|
|
|
|
ctx.once.Do(ctx.init)
|
2015-02-08 19:58:02 -06:00
|
|
|
|
|
|
|
ctx.ProviderLock.Lock()
|
|
|
|
defer ctx.ProviderLock.Unlock()
|
|
|
|
|
2017-10-31 16:41:44 -05:00
|
|
|
return ctx.ProviderCache[n]
|
2015-02-10 01:32:28 -06:00
|
|
|
}
|
|
|
|
|
2018-04-19 16:47:36 -05:00
|
|
|
func (ctx *BuiltinEvalContext) ProviderSchema(n string) *ProviderSchema {
|
|
|
|
ctx.once.Do(ctx.init)
|
|
|
|
|
|
|
|
ctx.ProviderLock.Lock()
|
|
|
|
defer ctx.ProviderLock.Unlock()
|
|
|
|
|
|
|
|
return ctx.ProviderSchemas[n]
|
|
|
|
}
|
|
|
|
|
2015-06-19 14:52:50 -05:00
|
|
|
func (ctx *BuiltinEvalContext) CloseProvider(n string) error {
|
|
|
|
ctx.once.Do(ctx.init)
|
|
|
|
|
|
|
|
ctx.ProviderLock.Lock()
|
|
|
|
defer ctx.ProviderLock.Unlock()
|
|
|
|
|
|
|
|
var provider interface{}
|
2017-10-31 16:41:44 -05:00
|
|
|
provider = ctx.ProviderCache[n]
|
2015-06-19 14:52:50 -05:00
|
|
|
if provider != nil {
|
|
|
|
if p, ok := provider.(ResourceProviderCloser); ok {
|
2017-10-31 16:41:44 -05:00
|
|
|
delete(ctx.ProviderCache, n)
|
2015-06-19 14:52:50 -05:00
|
|
|
return p.Close()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-02-10 01:32:28 -06:00
|
|
|
func (ctx *BuiltinEvalContext) ConfigureProvider(
|
|
|
|
n string, cfg *ResourceConfig) error {
|
|
|
|
p := ctx.Provider(n)
|
|
|
|
if p == nil {
|
|
|
|
return fmt.Errorf("Provider '%s' not initialized", n)
|
|
|
|
}
|
2015-04-09 10:48:08 -05:00
|
|
|
return p.Configure(cfg)
|
|
|
|
}
|
|
|
|
|
2015-02-13 19:59:54 -06:00
|
|
|
func (ctx *BuiltinEvalContext) ProviderInput(n string) map[string]interface{} {
|
|
|
|
ctx.ProviderLock.Lock()
|
|
|
|
defer ctx.ProviderLock.Unlock()
|
|
|
|
|
2015-06-24 11:34:21 -05:00
|
|
|
// Make a copy of the path so we can safely edit it
|
|
|
|
path := ctx.Path()
|
|
|
|
pathCopy := make([]string, len(path)+1)
|
|
|
|
copy(pathCopy, path)
|
|
|
|
|
|
|
|
// Go up the tree.
|
|
|
|
for i := len(path) - 1; i >= 0; i-- {
|
|
|
|
pathCopy[i+1] = n
|
|
|
|
k := PathCacheKey(pathCopy[:i+2])
|
|
|
|
if v, ok := ctx.ProviderInputConfig[k]; ok {
|
|
|
|
return v
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
2015-02-13 19:59:54 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
func (ctx *BuiltinEvalContext) SetProviderInput(n string, c map[string]interface{}) {
|
2015-06-24 11:34:21 -05:00
|
|
|
providerPath := make([]string, len(ctx.Path())+1)
|
|
|
|
copy(providerPath, ctx.Path())
|
|
|
|
providerPath[len(providerPath)-1] = n
|
2015-02-13 19:59:54 -06:00
|
|
|
|
2015-06-24 11:34:21 -05:00
|
|
|
// Save the configuration
|
|
|
|
ctx.ProviderLock.Lock()
|
|
|
|
ctx.ProviderInputConfig[PathCacheKey(providerPath)] = c
|
|
|
|
ctx.ProviderLock.Unlock()
|
2015-02-13 19:59:54 -06:00
|
|
|
}
|
|
|
|
|
2015-02-09 13:15:54 -06:00
|
|
|
func (ctx *BuiltinEvalContext) InitProvisioner(
|
|
|
|
n string) (ResourceProvisioner, error) {
|
|
|
|
ctx.once.Do(ctx.init)
|
|
|
|
|
|
|
|
// If we already initialized, it is an error
|
|
|
|
if p := ctx.Provisioner(n); p != nil {
|
|
|
|
return nil, fmt.Errorf("Provisioner '%s' already initialized", n)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Warning: make sure to acquire these locks AFTER the call to Provisioner
|
|
|
|
// above, since it also acquires locks.
|
|
|
|
ctx.ProvisionerLock.Lock()
|
|
|
|
defer ctx.ProvisionerLock.Unlock()
|
|
|
|
|
2015-02-20 12:50:36 -06:00
|
|
|
provPath := make([]string, len(ctx.Path())+1)
|
|
|
|
copy(provPath, ctx.Path())
|
|
|
|
provPath[len(provPath)-1] = n
|
2016-10-03 21:34:07 -05:00
|
|
|
key := PathCacheKey(provPath)
|
|
|
|
|
|
|
|
p, err := ctx.Components.ResourceProvisioner(n, key)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2015-02-20 12:50:36 -06:00
|
|
|
|
2016-10-03 21:34:07 -05:00
|
|
|
ctx.ProvisionerCache[key] = p
|
2015-02-09 13:15:54 -06:00
|
|
|
return p, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ctx *BuiltinEvalContext) Provisioner(n string) ResourceProvisioner {
|
|
|
|
ctx.once.Do(ctx.init)
|
|
|
|
|
|
|
|
ctx.ProvisionerLock.Lock()
|
|
|
|
defer ctx.ProvisionerLock.Unlock()
|
|
|
|
|
2015-02-20 12:50:36 -06:00
|
|
|
provPath := make([]string, len(ctx.Path())+1)
|
|
|
|
copy(provPath, ctx.Path())
|
|
|
|
provPath[len(provPath)-1] = n
|
|
|
|
|
|
|
|
return ctx.ProvisionerCache[PathCacheKey(provPath)]
|
2015-02-09 13:15:54 -06:00
|
|
|
}
|
|
|
|
|
2015-06-19 14:52:50 -05:00
|
|
|
func (ctx *BuiltinEvalContext) CloseProvisioner(n string) error {
|
|
|
|
ctx.once.Do(ctx.init)
|
|
|
|
|
|
|
|
ctx.ProvisionerLock.Lock()
|
|
|
|
defer ctx.ProvisionerLock.Unlock()
|
|
|
|
|
|
|
|
provPath := make([]string, len(ctx.Path())+1)
|
|
|
|
copy(provPath, ctx.Path())
|
|
|
|
provPath[len(provPath)-1] = n
|
|
|
|
|
|
|
|
var prov interface{}
|
|
|
|
prov = ctx.ProvisionerCache[PathCacheKey(provPath)]
|
|
|
|
if prov != nil {
|
|
|
|
if p, ok := prov.(ResourceProvisionerCloser); ok {
|
|
|
|
delete(ctx.ProvisionerCache, PathCacheKey(provPath))
|
|
|
|
return p.Close()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-02-03 19:46:11 -06:00
|
|
|
func (ctx *BuiltinEvalContext) Interpolate(
|
2015-02-05 19:09:57 -06:00
|
|
|
cfg *config.RawConfig, r *Resource) (*ResourceConfig, error) {
|
2017-10-13 09:11:10 -05:00
|
|
|
|
2015-02-05 19:09:57 -06:00
|
|
|
if cfg != nil {
|
|
|
|
scope := &InterpolationScope{
|
2015-02-08 16:00:13 -06:00
|
|
|
Path: ctx.Path(),
|
2015-02-05 19:09:57 -06:00
|
|
|
Resource: r,
|
|
|
|
}
|
2016-05-19 12:46:51 -05:00
|
|
|
|
2015-02-05 19:09:57 -06:00
|
|
|
vs, err := ctx.Interpolater.Values(scope, cfg.Variables)
|
2017-10-13 09:11:10 -05:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Do the interpolation
|
|
|
|
if err := cfg.Interpolate(vs); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
result := NewResourceConfig(cfg)
|
|
|
|
result.interpolateForce()
|
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (ctx *BuiltinEvalContext) InterpolateProvider(
|
|
|
|
pc *config.ProviderConfig, r *Resource) (*ResourceConfig, error) {
|
|
|
|
|
|
|
|
var cfg *config.RawConfig
|
|
|
|
|
|
|
|
if pc != nil && pc.RawConfig != nil {
|
|
|
|
scope := &InterpolationScope{
|
2017-11-06 20:57:06 -06:00
|
|
|
Path: ctx.Path(),
|
2017-10-13 09:11:10 -05:00
|
|
|
Resource: r,
|
|
|
|
}
|
|
|
|
|
|
|
|
cfg = pc.RawConfig
|
|
|
|
|
|
|
|
vs, err := ctx.Interpolater.Values(scope, cfg.Variables)
|
2015-02-05 19:09:57 -06:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2015-02-04 10:30:53 -06:00
|
|
|
}
|
|
|
|
|
2015-02-05 19:09:57 -06:00
|
|
|
// Do the interpolation
|
|
|
|
if err := cfg.Interpolate(vs); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2015-02-04 10:30:53 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
result := NewResourceConfig(cfg)
|
|
|
|
result.interpolateForce()
|
|
|
|
return result, nil
|
2015-02-03 19:46:11 -06:00
|
|
|
}
|
|
|
|
|
2015-02-08 16:00:13 -06:00
|
|
|
func (ctx *BuiltinEvalContext) Path() []string {
|
|
|
|
return ctx.PathValue
|
|
|
|
}
|
|
|
|
|
2016-04-11 12:40:06 -05:00
|
|
|
func (ctx *BuiltinEvalContext) SetVariables(n string, vs map[string]interface{}) {
|
2015-05-01 18:29:19 -05:00
|
|
|
ctx.InterpolaterVarLock.Lock()
|
|
|
|
defer ctx.InterpolaterVarLock.Unlock()
|
|
|
|
|
|
|
|
path := make([]string, len(ctx.Path())+1)
|
|
|
|
copy(path, ctx.Path())
|
|
|
|
path[len(path)-1] = n
|
|
|
|
key := PathCacheKey(path)
|
|
|
|
|
|
|
|
vars := ctx.InterpolaterVars[key]
|
|
|
|
if vars == nil {
|
2016-04-11 12:40:06 -05:00
|
|
|
vars = make(map[string]interface{})
|
2015-05-01 18:29:19 -05:00
|
|
|
ctx.InterpolaterVars[key] = vars
|
|
|
|
}
|
2015-05-01 16:10:41 -05:00
|
|
|
|
2015-02-11 19:01:08 -06:00
|
|
|
for k, v := range vs {
|
2015-05-01 18:29:19 -05:00
|
|
|
vars[k] = v
|
2015-02-11 19:01:08 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-11 17:22:03 -06:00
|
|
|
func (ctx *BuiltinEvalContext) Diff() (*Diff, *sync.RWMutex) {
|
|
|
|
return ctx.DiffValue, ctx.DiffLock
|
|
|
|
}
|
|
|
|
|
2015-02-11 10:48:45 -06:00
|
|
|
func (ctx *BuiltinEvalContext) State() (*State, *sync.RWMutex) {
|
|
|
|
return ctx.StateValue, ctx.StateLock
|
|
|
|
}
|
|
|
|
|
2015-02-03 19:46:11 -06:00
|
|
|
func (ctx *BuiltinEvalContext) init() {
|
2015-02-08 19:58:02 -06:00
|
|
|
}
|