terraform: refactor var loading out to a helper
Small style changes too
This commit is contained in:
parent
181b4b188b
commit
af7085f671
|
@ -3,7 +3,6 @@ package terraform
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"log"
|
"log"
|
||||||
"os"
|
|
||||||
"sort"
|
"sort"
|
||||||
"strings"
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
|
@ -129,109 +128,10 @@ func NewContext(opts *ContextOpts) (*Context, error) {
|
||||||
variables := make(map[string]interface{})
|
variables := make(map[string]interface{})
|
||||||
|
|
||||||
if opts.Module != nil {
|
if opts.Module != nil {
|
||||||
for _, v := range opts.Module.Config().Variables {
|
var err error
|
||||||
if v.Default != nil {
|
variables, err = Variables(opts.Module, opts.Variables)
|
||||||
if v.Type() == config.VariableTypeString {
|
if err != nil {
|
||||||
// v.Default has already been parsed as HCL so there may be
|
return nil, err
|
||||||
// some stray ints in there
|
|
||||||
switch typedDefault := v.Default.(type) {
|
|
||||||
case string:
|
|
||||||
if typedDefault == "" {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
variables[v.Name] = typedDefault
|
|
||||||
case int, int64:
|
|
||||||
variables[v.Name] = fmt.Sprintf("%d", typedDefault)
|
|
||||||
case float32, float64:
|
|
||||||
variables[v.Name] = fmt.Sprintf("%f", typedDefault)
|
|
||||||
case bool:
|
|
||||||
variables[v.Name] = fmt.Sprintf("%t", typedDefault)
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
variables[v.Name] = v.Default
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, v := range os.Environ() {
|
|
||||||
if !strings.HasPrefix(v, VarEnvPrefix) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
// Strip off the prefix and get the value after the first "="
|
|
||||||
idx := strings.Index(v, "=")
|
|
||||||
k := v[len(VarEnvPrefix):idx]
|
|
||||||
v = v[idx+1:]
|
|
||||||
|
|
||||||
// Override the configuration-default values. Note that *not* finding the variable
|
|
||||||
// in configuration is OK, as we don't want to preclude people from having multiple
|
|
||||||
// sets of TF_VAR_whatever in their environment even if it is a little weird.
|
|
||||||
for _, schema := range opts.Module.Config().Variables {
|
|
||||||
if schema.Name == k {
|
|
||||||
varType := schema.Type()
|
|
||||||
varVal, err := parseVariableAsHCL(k, v, varType)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
switch varType {
|
|
||||||
case config.VariableTypeMap:
|
|
||||||
if existing, hasMap := variables[k]; !hasMap {
|
|
||||||
variables[k] = varVal
|
|
||||||
} else {
|
|
||||||
if existingMap, ok := existing.(map[string]interface{}); !ok {
|
|
||||||
panic(fmt.Sprintf("%s is not a map, this is a bug in Terraform.", k))
|
|
||||||
} else {
|
|
||||||
switch typedV := varVal.(type) {
|
|
||||||
case []map[string]interface{}:
|
|
||||||
for newKey, newVal := range typedV[0] {
|
|
||||||
existingMap[newKey] = newVal
|
|
||||||
}
|
|
||||||
case map[string]interface{}:
|
|
||||||
for newKey, newVal := range typedV {
|
|
||||||
existingMap[newKey] = newVal
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
panic(fmt.Sprintf("%s is not a map, this is a bug in Terraform.", k))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
variables[k] = varVal
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
for k, v := range opts.Variables {
|
|
||||||
for _, schema := range opts.Module.Config().Variables {
|
|
||||||
if schema.Name == k {
|
|
||||||
switch schema.Type() {
|
|
||||||
case config.VariableTypeMap:
|
|
||||||
if existing, hasMap := variables[k]; !hasMap {
|
|
||||||
variables[k] = v
|
|
||||||
} else {
|
|
||||||
if existingMap, ok := existing.(map[string]interface{}); !ok {
|
|
||||||
panic(fmt.Sprintf("%s is not a map, this is a bug in Terraform.", k))
|
|
||||||
} else {
|
|
||||||
switch typedV := v.(type) {
|
|
||||||
case []map[string]interface{}:
|
|
||||||
for newKey, newVal := range typedV[0] {
|
|
||||||
existingMap[newKey] = newVal
|
|
||||||
}
|
|
||||||
case map[string]interface{}:
|
|
||||||
for newKey, newVal := range typedV {
|
|
||||||
existingMap[newKey] = newVal
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
panic(fmt.Sprintf("%s is not a map, this is a bug in Terraform.", k))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
variables[k] = v
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,145 @@
|
||||||
|
package terraform
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/hashicorp/terraform/config"
|
||||||
|
"github.com/hashicorp/terraform/config/module"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Variables returns the fully loaded set of variables to use with
|
||||||
|
// ContextOpts and NewContext, loading any additional variables from
|
||||||
|
// the environment or any other sources.
|
||||||
|
//
|
||||||
|
// The given module tree doesn't need to be loaded.
|
||||||
|
func Variables(
|
||||||
|
m *module.Tree,
|
||||||
|
override map[string]interface{}) (map[string]interface{}, error) {
|
||||||
|
result := make(map[string]interface{})
|
||||||
|
|
||||||
|
// Variables are loaded in the following sequence. Each additional step
|
||||||
|
// will override conflicting variable keys from prior steps:
|
||||||
|
//
|
||||||
|
// * Take default values from config
|
||||||
|
// * Take values from TF_VAR_x env vars
|
||||||
|
// * Take values specified in the "override" param which is usually
|
||||||
|
// from -var, -var-file, etc.
|
||||||
|
//
|
||||||
|
|
||||||
|
// First load from the config
|
||||||
|
for _, v := range m.Config().Variables {
|
||||||
|
// If the var has no default, ignore
|
||||||
|
if v.Default == nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// If the type isn't a string, we use it as-is since it is a rich type
|
||||||
|
if v.Type() != config.VariableTypeString {
|
||||||
|
result[v.Name] = v.Default
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// v.Default has already been parsed as HCL but it may be an int type
|
||||||
|
switch typedDefault := v.Default.(type) {
|
||||||
|
case string:
|
||||||
|
if typedDefault == "" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
result[v.Name] = typedDefault
|
||||||
|
case int, int64:
|
||||||
|
result[v.Name] = fmt.Sprintf("%d", typedDefault)
|
||||||
|
case float32, float64:
|
||||||
|
result[v.Name] = fmt.Sprintf("%f", typedDefault)
|
||||||
|
case bool:
|
||||||
|
result[v.Name] = fmt.Sprintf("%t", typedDefault)
|
||||||
|
default:
|
||||||
|
panic(fmt.Sprintf(
|
||||||
|
"Unknown default var type: %T\n\n"+
|
||||||
|
"THIS IS A BUG. Please report it.",
|
||||||
|
v.Default))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Load from env vars
|
||||||
|
for _, v := range os.Environ() {
|
||||||
|
if !strings.HasPrefix(v, VarEnvPrefix) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// Strip off the prefix and get the value after the first "="
|
||||||
|
idx := strings.Index(v, "=")
|
||||||
|
k := v[len(VarEnvPrefix):idx]
|
||||||
|
v = v[idx+1:]
|
||||||
|
|
||||||
|
// Override the configuration-default values. Note that *not* finding the variable
|
||||||
|
// in configuration is OK, as we don't want to preclude people from having multiple
|
||||||
|
// sets of TF_VAR_whatever in their environment even if it is a little weird.
|
||||||
|
for _, schema := range m.Config().Variables {
|
||||||
|
if schema.Name != k {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
varType := schema.Type()
|
||||||
|
varVal, err := parseVariableAsHCL(k, v, varType)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
switch varType {
|
||||||
|
case config.VariableTypeMap:
|
||||||
|
varSetMap(result, k, varVal)
|
||||||
|
default:
|
||||||
|
result[k] = varVal
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Load from overrides
|
||||||
|
for k, v := range override {
|
||||||
|
for _, schema := range m.Config().Variables {
|
||||||
|
if schema.Name != k {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
switch schema.Type() {
|
||||||
|
case config.VariableTypeMap:
|
||||||
|
varSetMap(result, k, v)
|
||||||
|
default:
|
||||||
|
result[k] = v
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// varSetMap sets or merges the map in "v" with the key "k" in the
|
||||||
|
// "current" set of variables. This is just a private function to remove
|
||||||
|
// duplicate logic in Variables
|
||||||
|
func varSetMap(current map[string]interface{}, k string, v interface{}) {
|
||||||
|
existing, ok := current[k]
|
||||||
|
if !ok {
|
||||||
|
current[k] = v
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
existingMap, ok := existing.(map[string]interface{})
|
||||||
|
if !ok {
|
||||||
|
panic(fmt.Sprintf("%s is not a map, this is a bug in Terraform.", k))
|
||||||
|
}
|
||||||
|
|
||||||
|
switch typedV := v.(type) {
|
||||||
|
case []map[string]interface{}:
|
||||||
|
for newKey, newVal := range typedV[0] {
|
||||||
|
existingMap[newKey] = newVal
|
||||||
|
}
|
||||||
|
case map[string]interface{}:
|
||||||
|
for newKey, newVal := range typedV {
|
||||||
|
existingMap[newKey] = newVal
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
panic(fmt.Sprintf("%s is not a map, this is a bug in Terraform.", k))
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue