opentofu/helper/schema/resource_data.go

230 lines
4.7 KiB
Go
Raw Normal View History

2014-08-13 16:23:22 -05:00
package schema
import (
"fmt"
"strconv"
"strings"
"github.com/hashicorp/terraform/terraform"
2014-08-16 17:02:51 -05:00
"github.com/mitchellh/mapstructure"
)
2014-08-13 16:23:22 -05:00
// ResourceData is used to query and set the attributes of a resource.
type ResourceData struct {
schema map[string]*Schema
state *terraform.ResourceState
diff *terraform.ResourceDiff
2014-08-16 17:02:51 -05:00
set map[string]string
}
2014-08-14 21:55:47 -05:00
// Get returns the data for the given key, or nil if the key doesn't exist.
2014-08-15 19:46:05 -05: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-14 21:55:47 -05:00
func (d *ResourceData) Get(key string) interface{} {
var parts []string
if key != "" {
parts = strings.Split(key, ".")
}
return d.getObject("", parts, d.schema)
}
2014-08-16 17:02:51 -05: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 {
if d.set == nil {
d.set = make(map[string]string)
}
parts := strings.Split(key, ".")
return d.setObject("", parts, d.schema, value)
}
func (d *ResourceData) get(
k string,
parts []string,
schema *Schema) interface{} {
switch schema.Type {
case TypeList:
return d.getList(k, parts, schema)
default:
return d.getPrimitive(k, parts, schema)
}
}
func (d *ResourceData) getObject(
k string,
parts []string,
schema map[string]*Schema) 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)
}
return d.get(key, parts, s)
}
// Get the entire object
result := make(map[string]interface{})
for field, _ := range schema {
result[field] = d.getObject(k, []string{field}, schema)
}
return result
}
func (d *ResourceData) getList(
k string,
parts []string,
schema *Schema) 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-16 17:02:51 -05:00
result := d.get(k+".#", parts, schema)
if result == nil {
result = 0
}
return result
}
key := fmt.Sprintf("%s.%s", k, idx)
switch t := schema.Elem.(type) {
case *Resource:
return d.getObject(key, parts, t.Schema)
case *Schema:
return d.get(key, parts, t)
}
}
// Get the entire list.
result := make([]interface{}, d.getList(k, []string{"#"}, schema).(int))
for i, _ := range result {
is := strconv.FormatInt(int64(i), 10)
result[i] = d.getList(k, []string{is}, schema)
}
return result
}
func (d *ResourceData) getPrimitive(
k string,
parts []string,
schema *Schema) interface{} {
var result string
2014-08-16 17:02:51 -05:00
var resultSet bool
if d.state != nil {
2014-08-16 17:02:51 -05:00
result, resultSet = d.state.Attributes[k]
}
if d.diff != nil {
attrD, ok := d.diff.Attributes[k]
if ok {
result = attrD.New
2014-08-16 17:02:51 -05:00
resultSet = true
}
}
2014-08-16 17:02:51 -05:00
if d.set != nil {
if v, ok := d.set[k]; ok {
result = v
resultSet = true
}
}
if !resultSet {
return nil
}
switch schema.Type {
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-14 21:55:47 -05:00
}
2014-08-16 17:02:51 -05: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)
}
return d.setPrimitive(key, s, value)
}
panic("can't set full object yet")
}
func (d *ResourceData) setPrimitive(
k string,
schema *Schema,
v interface{}) error {
var set string
switch schema.Type {
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)
}
d.set[k] = set
return nil
}