terraform/helper/schema/resource_data.go

865 lines
18 KiB
Go
Raw Normal View History

2014-08-13 23:23:22 +02:00
package schema
import (
"fmt"
2014-08-18 19:00:41 +02:00
"reflect"
"strconv"
"strings"
2014-08-19 00:41:12 +02:00
"sync"
"github.com/hashicorp/terraform/terraform"
2014-08-17 00:02:51 +02:00
"github.com/mitchellh/mapstructure"
)
2014-08-18 18:58:44 +02:00
// getSource represents the level we want to get for a value (internally).
// Any source less than or equal to the level will be loaded (whichever
// has a value first).
type getSource byte
const (
getSourceState getSource = iota
getSourceConfig
2014-08-18 18:58:44 +02:00
getSourceDiff
getSourceSet
)
2014-08-13 23:23:22 +02:00
// ResourceData is used to query and set the attributes of a resource.
type ResourceData struct {
schema map[string]*Schema
config *terraform.ResourceConfig
state *terraform.ResourceState
diff *terraform.ResourceDiff
diffing bool
2014-08-18 04:45:26 +02:00
setMap map[string]string
newState *terraform.ResourceState
2014-08-19 00:41:12 +02:00
once sync.Once
}
2014-08-15 04:55:47 +02:00
// Get returns the data for the given key, or nil if the key doesn't exist.
2014-08-16 02:46:05 +02:00
//
// The type of the data returned will be according to the schema specified.
// Primitives will be their respective types in Go, lists will always be
// []interface{}, and sub-resources will be map[string]interface{}.
2014-08-15 04:55:47 +02:00
func (d *ResourceData) Get(key string) interface{} {
var parts []string
if key != "" {
parts = strings.Split(key, ".")
}
2014-08-18 18:58:44 +02:00
return d.getObject("", parts, d.schema, getSourceSet)
}
// GetChange returns the old and new value for a given key.
//
// If there is no change, then old and new will simply be the same.
func (d *ResourceData) GetChange(key string) (interface{}, interface{}) {
return d.getChange(key, getSourceConfig, getSourceDiff)
}
2014-08-18 19:00:41 +02:00
// HasChange returns whether or not the given key has been changed.
func (d *ResourceData) HasChange(key string) bool {
o, n := d.GetChange(key)
return !reflect.DeepEqual(o, n)
}
2014-08-17 00:02:51 +02:00
// Set sets the value for the given key.
//
// If the key is invalid or the value is not a correct type, an error
// will be returned.
func (d *ResourceData) Set(key string, value interface{}) error {
2014-08-17 20:38:16 +02:00
if d.setMap == nil {
d.setMap = make(map[string]string)
2014-08-17 00:02:51 +02:00
}
parts := strings.Split(key, ".")
return d.setObject("", parts, d.schema, value)
}
// Id returns the ID of the resource.
func (d *ResourceData) Id() string {
var result string
if d.state != nil {
result = d.state.ID
}
if d.newState != nil {
result = d.newState.ID
}
return result
}
2014-08-22 07:15:47 +02:00
// ConnInfo returns the connection info for this resource.
func (d *ResourceData) ConnInfo() map[string]string {
if d.newState != nil {
return d.newState.ConnInfo
}
if d.state != nil {
return d.state.ConnInfo
}
return nil
}
2014-08-19 00:41:12 +02:00
// Dependencies returns the dependencies in this state.
func (d *ResourceData) Dependencies() []terraform.ResourceDependency {
if d.newState != nil {
return d.newState.Dependencies
}
if d.state != nil {
return d.state.Dependencies
}
return nil
}
// SetId sets the ID of the resource. If the value is blank, then the
// resource is destroyed.
func (d *ResourceData) SetId(v string) {
2014-08-19 00:41:12 +02:00
d.once.Do(d.init)
d.newState.ID = v
}
2014-08-22 07:15:47 +02:00
// SetConnInfo sets the connection info for a resource.
func (d *ResourceData) SetConnInfo(v map[string]string) {
d.once.Do(d.init)
d.newState.ConnInfo = v
}
2014-08-19 00:41:12 +02:00
// SetDependencies sets the dependencies of a resource.
func (d *ResourceData) SetDependencies(ds []terraform.ResourceDependency) {
d.once.Do(d.init)
d.newState.Dependencies = ds
}
// State returns the new ResourceState after the diff and any Set
// calls.
func (d *ResourceData) State() *terraform.ResourceState {
var result terraform.ResourceState
result.ID = d.Id()
result.Attributes = d.stateObject("", d.schema)
2014-08-22 07:15:47 +02:00
result.ConnInfo = d.ConnInfo()
2014-08-19 00:41:12 +02:00
result.Dependencies = d.Dependencies()
2014-08-18 04:45:26 +02:00
if v := d.Id(); v != "" {
result.Attributes["id"] = d.Id()
}
return &result
}
2014-08-19 00:41:12 +02:00
func (d *ResourceData) init() {
var copyState terraform.ResourceState
if d.state != nil {
copyState = *d.state
}
d.newState = &copyState
}
func (d *ResourceData) diffChange(k string) (interface{}, interface{}, bool) {
// Get the change between the state and the config.
o, n := d.getChange(k, getSourceState, getSourceConfig)
// Return the old, new, and whether there is a change
return o, n, !reflect.DeepEqual(o, n)
}
func (d *ResourceData) getChange(
key string,
oldLevel getSource,
newLevel getSource) (interface{}, interface{}) {
2014-08-21 06:02:42 +02:00
var parts, parts2 []string
if key != "" {
parts = strings.Split(key, ".")
2014-08-21 06:02:42 +02:00
parts2 = strings.Split(key, ".")
}
o := d.getObject("", parts, d.schema, oldLevel)
2014-08-21 06:02:42 +02:00
n := d.getObject("", parts2, d.schema, newLevel)
return o, n
}
func (d *ResourceData) get(
k string,
parts []string,
2014-08-18 18:58:44 +02:00
schema *Schema,
source getSource) interface{} {
switch schema.Type {
case TypeList:
2014-08-18 18:58:44 +02:00
return d.getList(k, parts, schema, source)
2014-08-18 23:00:03 +02:00
case TypeMap:
return d.getMap(k, parts, schema, source)
case TypeSet:
return d.getSet(k, parts, schema, source)
case TypeBool:
fallthrough
case TypeInt:
fallthrough
case TypeString:
2014-08-18 18:58:44 +02:00
return d.getPrimitive(k, parts, schema, source)
default:
panic(fmt.Sprintf("%s: unknown type %s", k, schema.Type))
}
}
func (d *ResourceData) getSet(
k string,
parts []string,
schema *Schema,
source getSource) interface{} {
2014-08-21 06:02:42 +02:00
s := &Set{F: schema.Set}
raw := d.getList(k, nil, schema, source)
if raw == nil {
2014-08-21 06:02:42 +02:00
if len(parts) > 0 {
return d.getList(k, parts, schema, source)
}
return s
}
list := raw.([]interface{})
if len(list) == 0 {
2014-08-21 06:02:42 +02:00
if len(parts) > 0 {
return d.getList(k, parts, schema, source)
}
return s
}
// This is a reverse map of hash code => index in config used to
// resolve direct set item lookup for turning into state. Confused?
// Read on...
//
// To create the state (the state* functions), a Get call is done
// with a full key such as "ports.0". The index of a set ("0") doesn't
// make a lot of sense, but we need to deterministically list out
// elements of the set like this. Luckily, same sets have a deterministic
// List() output, so we can use that to look things up.
//
// This mapping makes it so that we can look up the hash code of an
// object back to its index in the REAL config.
var indexMap map[int]int
if len(parts) > 0 {
indexMap = make(map[int]int)
}
// Build the set from all the items using the given hash code
for i, v := range list {
code := s.add(v)
if indexMap != nil {
indexMap[code] = i
}
}
// If we're trying to get a specific element, then rewrite the
// index to be just that, then jump direct to getList.
if len(parts) > 0 {
index := parts[0]
indexInt, err := strconv.ParseInt(index, 0, 0)
if err != nil {
return nil
}
codes := s.listCode()
if int(indexInt) >= len(codes) {
return nil
}
code := codes[indexInt]
realIndex := indexMap[code]
parts[0] = strconv.FormatInt(int64(realIndex), 10)
return d.getList(k, parts, schema, source)
}
return s
}
2014-08-18 23:00:03 +02:00
func (d *ResourceData) getMap(
k string,
parts []string,
schema *Schema,
source getSource) interface{} {
elemSchema := &Schema{Type: TypeString}
2014-08-19 00:07:09 +02:00
result := make(map[string]interface{})
2014-08-18 23:00:03 +02:00
prefix := k + "."
2014-08-19 00:07:09 +02:00
if d.state != nil && source >= getSourceState {
for k, _ := range d.state.Attributes {
2014-08-18 23:00:03 +02:00
if !strings.HasPrefix(k, prefix) {
continue
}
single := k[len(prefix):]
result[single] = d.getPrimitive(k, nil, elemSchema, source)
}
}
if d.config != nil && source == getSourceConfig {
// For config, we always set the result to exactly what was requested
if m, ok := d.config.Get(k); ok {
result = m.(map[string]interface{})
} else {
result = nil
}
}
2014-08-19 00:07:09 +02:00
if d.diff != nil && source >= getSourceDiff {
for k, v := range d.diff.Attributes {
2014-08-18 23:00:03 +02:00
if !strings.HasPrefix(k, prefix) {
continue
}
single := k[len(prefix):]
2014-08-19 00:07:09 +02:00
if v.NewRemoved {
delete(result, single)
} else {
result[single] = d.getPrimitive(k, nil, elemSchema, source)
}
2014-08-18 23:00:03 +02:00
}
}
2014-08-19 00:07:09 +02:00
if d.setMap != nil && source >= getSourceSet {
cleared := false
for k, _ := range d.setMap {
2014-08-18 23:00:03 +02:00
if !strings.HasPrefix(k, prefix) {
continue
}
2014-08-19 00:07:09 +02:00
if !cleared {
// We clear the results if they are in the set map
2014-08-18 23:00:03 +02:00
result = make(map[string]interface{})
2014-08-19 00:07:09 +02:00
cleared = true
2014-08-18 23:00:03 +02:00
}
2014-08-19 00:07:09 +02:00
single := k[len(prefix):]
2014-08-18 23:00:03 +02:00
result[single] = d.getPrimitive(k, nil, elemSchema, source)
}
}
// If we're requesting a specific element, return that
if len(parts) > 0 {
return result[parts[0]]
}
2014-08-18 23:00:03 +02:00
return result
}
func (d *ResourceData) getObject(
k string,
parts []string,
2014-08-18 18:58:44 +02:00
schema map[string]*Schema,
source getSource) interface{} {
if len(parts) > 0 {
// We're requesting a specific key in an object
key := parts[0]
parts = parts[1:]
s, ok := schema[key]
if !ok {
return nil
}
if k != "" {
// If we're not at the root, then we need to append
// the key to get the full key path.
key = fmt.Sprintf("%s.%s", k, key)
}
2014-08-18 18:58:44 +02:00
return d.get(key, parts, s, source)
}
// Get the entire object
result := make(map[string]interface{})
for field, _ := range schema {
2014-08-18 18:58:44 +02:00
result[field] = d.getObject(k, []string{field}, schema, source)
}
return result
}
func (d *ResourceData) getList(
k string,
parts []string,
2014-08-18 18:58:44 +02:00
schema *Schema,
source getSource) interface{} {
if len(parts) > 0 {
// We still have parts left over meaning we're accessing an
// element of this list.
idx := parts[0]
parts = parts[1:]
// Special case if we're accessing the count of the list
if idx == "#" {
schema := &Schema{Type: TypeInt}
2014-08-18 18:58:44 +02:00
result := d.get(k+".#", parts, schema, source)
2014-08-17 00:02:51 +02:00
if result == nil {
result = 0
}
return result
}
key := fmt.Sprintf("%s.%s", k, idx)
switch t := schema.Elem.(type) {
case *Resource:
2014-08-18 18:58:44 +02:00
return d.getObject(key, parts, t.Schema, source)
case *Schema:
2014-08-18 18:58:44 +02:00
return d.get(key, parts, t, source)
}
}
// Get the entire list.
2014-08-18 18:58:44 +02:00
result := make(
[]interface{},
d.getList(k, []string{"#"}, schema, source).(int))
for i, _ := range result {
is := strconv.FormatInt(int64(i), 10)
2014-08-18 18:58:44 +02:00
result[i] = d.getList(k, []string{is}, schema, source)
}
return result
}
func (d *ResourceData) getPrimitive(
k string,
parts []string,
2014-08-18 18:58:44 +02:00
schema *Schema,
source getSource) interface{} {
var result string
2014-08-17 00:02:51 +02:00
var resultSet bool
2014-08-18 18:58:44 +02:00
if d.state != nil && source >= getSourceState {
2014-08-17 00:02:51 +02:00
result, resultSet = d.state.Attributes[k]
}
if d.config != nil && source == getSourceConfig {
// For config, we always return the exact value
if v, ok := d.config.Get(k); ok {
if err := mapstructure.WeakDecode(v, &result); err != nil {
panic(err)
}
2014-08-21 06:02:42 +02:00
resultSet = true
} else {
result = ""
2014-08-21 06:02:42 +02:00
resultSet = false
}
}
2014-08-18 18:58:44 +02:00
if d.diff != nil && source >= getSourceDiff {
attrD, ok := d.diff.Attributes[k]
if ok && !attrD.NewComputed {
result = attrD.New
2014-08-17 00:02:51 +02:00
resultSet = true
}
}
2014-08-18 18:58:44 +02:00
if d.setMap != nil && source >= getSourceSet {
2014-08-17 20:38:16 +02:00
if v, ok := d.setMap[k]; ok {
2014-08-17 00:02:51 +02:00
result = v
resultSet = true
}
}
if !resultSet {
return nil
}
switch schema.Type {
2014-08-20 01:46:36 +02:00
case TypeBool:
if result == "" {
return false
}
v, err := strconv.ParseBool(result)
if err != nil {
panic(err)
}
return v
case TypeString:
// Use the value as-is. We just put this case here to be explicit.
return result
case TypeInt:
if result == "" {
return 0
}
v, err := strconv.ParseInt(result, 0, 0)
if err != nil {
panic(err)
}
return int(v)
default:
panic(fmt.Sprintf("Unknown type: %s", schema.Type))
}
2014-08-15 04:55:47 +02:00
}
2014-08-17 00:02:51 +02:00
2014-08-17 20:38:16 +02:00
func (d *ResourceData) set(
k string,
parts []string,
schema *Schema,
value interface{}) error {
switch schema.Type {
case TypeList:
return d.setList(k, parts, schema, value)
2014-08-18 23:00:03 +02:00
case TypeMap:
return d.setMapValue(k, parts, schema, value)
2014-08-21 03:11:14 +02:00
case TypeSet:
return d.setSet(k, parts, schema, value)
case TypeBool:
fallthrough
case TypeInt:
fallthrough
case TypeString:
2014-08-17 20:38:16 +02:00
return d.setPrimitive(k, schema, value)
2014-08-21 03:11:14 +02:00
default:
panic(fmt.Sprintf("%s: unknown type %s", k, schema.Type))
2014-08-17 20:38:16 +02:00
}
}
func (d *ResourceData) setList(
k string,
parts []string,
schema *Schema,
value interface{}) error {
if len(parts) > 0 {
// We're setting a specific element
idx := parts[0]
parts = parts[1:]
// Special case if we're accessing the count of the list
if idx == "#" {
return fmt.Errorf("%s: can't set count of list", k)
}
key := fmt.Sprintf("%s.%s", k, idx)
switch t := schema.Elem.(type) {
case *Resource:
return d.setObject(key, parts, t.Schema, value)
case *Schema:
return d.set(key, parts, t, value)
}
}
var vs []interface{}
if err := mapstructure.Decode(value, &vs); err != nil {
return fmt.Errorf("%s: %s", k, err)
}
// Set the entire list.
var err error
for i, elem := range vs {
is := strconv.FormatInt(int64(i), 10)
err = d.setList(k, []string{is}, schema, elem)
if err != nil {
break
}
}
if err != nil {
for i, _ := range vs {
is := strconv.FormatInt(int64(i), 10)
d.setList(k, []string{is}, schema, nil)
}
return err
}
d.setMap[k+".#"] = strconv.FormatInt(int64(len(vs)), 10)
return nil
}
2014-08-18 23:00:03 +02:00
func (d *ResourceData) setMapValue(
k string,
parts []string,
schema *Schema,
value interface{}) error {
elemSchema := &Schema{Type: TypeString}
if len(parts) > 0 {
return fmt.Errorf("%s: full map must be set, no a single element", k)
}
2014-08-19 00:07:09 +02:00
// Delete any prior map set
/*
v := d.getMap(k, nil, schema, getSourceSet)
for subKey, _ := range v.(map[string]interface{}) {
delete(d.setMap, fmt.Sprintf("%s.%s", k, subKey))
}
*/
v := reflect.ValueOf(value)
if v.Kind() != reflect.Map {
return fmt.Errorf("%s: must be a map", k)
}
if v.Type().Key().Kind() != reflect.String {
return fmt.Errorf("%s: keys must strings", k)
}
vs := make(map[string]interface{})
for _, mk := range v.MapKeys() {
mv := v.MapIndex(mk)
vs[mk.String()] = mv.Interface()
}
2014-08-18 23:00:03 +02:00
for subKey, v := range vs {
err := d.set(fmt.Sprintf("%s.%s", k, subKey), nil, elemSchema, v)
if err != nil {
return err
}
}
return nil
}
2014-08-17 00:02:51 +02:00
func (d *ResourceData) setObject(
k string,
parts []string,
schema map[string]*Schema,
value interface{}) error {
if len(parts) > 0 {
// We're setting a specific key in an object
key := parts[0]
parts = parts[1:]
s, ok := schema[key]
if !ok {
return fmt.Errorf("%s (internal): unknown key to set: %s", k, key)
}
if k != "" {
// If we're not at the root, then we need to append
// the key to get the full key path.
key = fmt.Sprintf("%s.%s", k, key)
}
2014-08-17 20:38:16 +02:00
return d.set(key, parts, s, value)
}
// Set the entire object. First decode into a proper structure
var v map[string]interface{}
if err := mapstructure.Decode(value, &v); err != nil {
return fmt.Errorf("%s: %s", k, err)
2014-08-17 00:02:51 +02:00
}
2014-08-17 20:38:16 +02:00
// Set each element in turn
var err error
for k1, v1 := range v {
err = d.setObject(k, []string{k1}, schema, v1)
if err != nil {
break
}
}
if err != nil {
for k1, _ := range v {
d.setObject(k, []string{k1}, schema, nil)
}
}
return err
2014-08-17 00:02:51 +02:00
}
func (d *ResourceData) setPrimitive(
k string,
schema *Schema,
v interface{}) error {
2014-08-17 20:38:16 +02:00
if v == nil {
delete(d.setMap, k)
return nil
}
2014-08-17 00:02:51 +02:00
var set string
switch schema.Type {
2014-08-20 01:46:36 +02:00
case TypeBool:
var b bool
if err := mapstructure.Decode(v, &b); err != nil {
return fmt.Errorf("%s: %s", k, err)
}
set = strconv.FormatBool(b)
2014-08-17 00:02:51 +02:00
case TypeString:
if err := mapstructure.Decode(v, &set); err != nil {
return fmt.Errorf("%s: %s", k, err)
}
case TypeInt:
var n int
if err := mapstructure.Decode(v, &n); err != nil {
return fmt.Errorf("%s: %s", k, err)
}
set = strconv.FormatInt(int64(n), 10)
default:
return fmt.Errorf("Unknown type: %s", schema.Type)
}
2014-08-17 20:38:16 +02:00
d.setMap[k] = set
2014-08-17 00:02:51 +02:00
return nil
}
2014-08-21 03:11:14 +02:00
func (d *ResourceData) setSet(
k string,
parts []string,
schema *Schema,
value interface{}) error {
if len(parts) > 0 {
return fmt.Errorf("%s: can only set the full set, not elements", k)
}
if s, ok := value.(*Set); ok {
value = s.List()
}
return d.setList(k, nil, schema, value)
}
func (d *ResourceData) stateList(
prefix string,
schema *Schema) map[string]string {
2014-08-18 18:58:44 +02:00
countRaw := d.get(prefix, []string{"#"}, schema, getSourceSet)
if countRaw == nil {
return nil
}
count := countRaw.(int)
result := make(map[string]string)
if count > 0 {
result[prefix+".#"] = strconv.FormatInt(int64(count), 10)
}
for i := 0; i < count; i++ {
key := fmt.Sprintf("%s.%d", prefix, i)
var m map[string]string
switch t := schema.Elem.(type) {
case *Resource:
m = d.stateObject(key, t.Schema)
case *Schema:
m = d.stateSingle(key, t)
}
for k, v := range m {
result[k] = v
}
}
return result
}
2014-08-18 23:00:03 +02:00
func (d *ResourceData) stateMap(
prefix string,
schema *Schema) map[string]string {
v := d.getMap(prefix, nil, schema, getSourceSet)
if v == nil {
return nil
}
elemSchema := &Schema{Type: TypeString}
result := make(map[string]string)
for mk, _ := range v.(map[string]interface{}) {
mp := fmt.Sprintf("%s.%s", prefix, mk)
for k, v := range d.stateSingle(mp, elemSchema) {
result[k] = v
}
}
return result
}
func (d *ResourceData) stateObject(
prefix string,
schema map[string]*Schema) map[string]string {
result := make(map[string]string)
for k, v := range schema {
key := k
if prefix != "" {
key = prefix + "." + key
}
for k1, v1 := range d.stateSingle(key, v) {
result[k1] = v1
}
}
return result
}
func (d *ResourceData) statePrimitive(
prefix string,
schema *Schema) map[string]string {
v := d.Get(prefix)
if v == nil {
return nil
}
var vs string
switch schema.Type {
2014-08-20 01:46:36 +02:00
case TypeBool:
vs = strconv.FormatBool(v.(bool))
case TypeString:
vs = v.(string)
case TypeInt:
vs = strconv.FormatInt(int64(v.(int)), 10)
default:
panic(fmt.Sprintf("Unknown type: %s", schema.Type))
}
return map[string]string{
prefix: vs,
}
}
func (d *ResourceData) stateSet(
prefix string,
schema *Schema) map[string]string {
raw := d.get(prefix, nil, schema, getSourceSet)
if raw == nil {
return nil
}
set := raw.(*Set)
list := set.List()
result := make(map[string]string)
result[prefix+".#"] = strconv.FormatInt(int64(len(list)), 10)
for i := 0; i < len(list); i++ {
key := fmt.Sprintf("%s.%d", prefix, i)
var m map[string]string
switch t := schema.Elem.(type) {
case *Resource:
m = d.stateObject(key, t.Schema)
case *Schema:
m = d.stateSingle(key, t)
}
for k, v := range m {
result[k] = v
}
}
return result
}
func (d *ResourceData) stateSingle(
prefix string,
schema *Schema) map[string]string {
switch schema.Type {
case TypeList:
return d.stateList(prefix, schema)
2014-08-18 23:00:03 +02:00
case TypeMap:
return d.stateMap(prefix, schema)
case TypeSet:
return d.stateSet(prefix, schema)
case TypeBool:
fallthrough
case TypeInt:
fallthrough
case TypeString:
return d.statePrimitive(prefix, schema)
default:
panic(fmt.Sprintf("%s: unknown type %s", prefix, schema.Type))
}
}