mirror of
https://github.com/opentofu/opentofu.git
synced 2025-01-18 12:42:58 -06:00
a8e3787afc
References: * https://github.com/hashicorp/terraform/issues/14418 * v0.9.5 (original bug report):a59ee0b30e/helper/schema/field_writer_map.go (L311)
* v0.11.12 (Terraform AWS Provider discovery):057286e522/helper/schema/field_writer_map.go (L343)
When creating flatten functions in Terraform Providers that return *schema.Set, its possible to return a typed `nil`, e.g. ```go func flattenHeaders(h *cloudfront.Headers) *schema.Set { if h.Items != nil { return schema.NewSet(schema.HashString, flattenStringList(h.Items)) } return nil } ``` This previously could cause a panic, e.g. ``` panic: runtime error: invalid memory address or nil pointer dereference [signal SIGSEGV: segmentation violation code=0x1 addr=0x8 pc=0x1881911] goroutine 1325 [running]: github.com/hashicorp/terraform/helper/schema.(*MapFieldWriter).setSet(0xc00054bf00, 0xc00073efa0, 0x5, 0x5, 0x5828140, 0x0, 0xc0002cea50, 0xc000e996a8, 0xc001026e40) /Users/bflad/go/pkg/mod/github.com/hashicorp/terraform@v0.11.12/helper/schema/field_writer_map.go:343 +0x211 ``` Here we catch the typed `nil` and return an empty list flatmap result instead. Unit testing result prior to code update: ``` --- FAIL: TestMapFieldWriter (0.00s) panic: runtime error: invalid memory address or nil pointer dereference [recovered] panic: runtime error: invalid memory address or nil pointer dereference [signal SIGSEGV: segmentation violation code=0x1 addr=0x8 pc=0x1777cdc] goroutine 913 [running]: testing.tRunner.func1(0xc00045b800) /usr/local/Cellar/go/1.12.1/libexec/src/testing/testing.go:830 +0x392 panic(0x192cf20, 0x2267ca0) /usr/local/Cellar/go/1.12.1/libexec/src/runtime/panic.go:522 +0x1b5 github.com/hashicorp/terraform/helper/schema.(*MapFieldWriter).setSet(0xc0004648a0, 0xc0004408d0, 0x1, 0x1, 0x19e3de0, 0x0, 0xc00045c600, 0x30, 0x19e0080) /Users/bflad/src/github.com/hashicorp/terraform/helper/schema/field_writer_map.go:344 +0x68c github.com/hashicorp/terraform/helper/schema.(*MapFieldWriter).set(0xc0004648a0, 0xc0004408d0, 0x1, 0x1, 0x19e3de0, 0x0, 0x1, 0x18) /Users/bflad/src/github.com/hashicorp/terraform/helper/schema/field_writer_map.go:107 +0x28b github.com/hashicorp/terraform/helper/schema.(*MapFieldWriter).WriteField(0xc0004648a0, 0xc0004408d0, 0x1, 0x1, 0x19e3de0, 0x0, 0x0, 0x0) /Users/bflad/src/github.com/hashicorp/terraform/helper/schema/field_writer_map.go:89 +0x504 github.com/hashicorp/terraform/helper/schema.TestMapFieldWriter(0xc00045b800) /Users/bflad/src/github.com/hashicorp/terraform/helper/schema/field_writer_map_test.go:337 +0x2ddd testing.tRunner(0xc00045b800, 0x1a44f90) /usr/local/Cellar/go/1.12.1/libexec/src/testing/testing.go:865 +0xc0 created by testing.(*T).Run /usr/local/Cellar/go/1.12.1/libexec/src/testing/testing.go:916 +0x35a ```
358 lines
8.7 KiB
Go
358 lines
8.7 KiB
Go
package schema
|
|
|
|
import (
|
|
"fmt"
|
|
"reflect"
|
|
"strconv"
|
|
"strings"
|
|
"sync"
|
|
|
|
"github.com/mitchellh/mapstructure"
|
|
)
|
|
|
|
// MapFieldWriter writes data into a single map[string]string structure.
|
|
type MapFieldWriter struct {
|
|
Schema map[string]*Schema
|
|
|
|
lock sync.Mutex
|
|
result map[string]string
|
|
}
|
|
|
|
// Map returns the underlying map that is being written to.
|
|
func (w *MapFieldWriter) Map() map[string]string {
|
|
w.lock.Lock()
|
|
defer w.lock.Unlock()
|
|
if w.result == nil {
|
|
w.result = make(map[string]string)
|
|
}
|
|
|
|
return w.result
|
|
}
|
|
|
|
func (w *MapFieldWriter) unsafeWriteField(addr string, value string) {
|
|
w.lock.Lock()
|
|
defer w.lock.Unlock()
|
|
if w.result == nil {
|
|
w.result = make(map[string]string)
|
|
}
|
|
|
|
w.result[addr] = value
|
|
}
|
|
|
|
// clearTree clears a field and any sub-fields of the given address out of the
|
|
// map. This should be used to reset some kind of complex structures (namely
|
|
// sets) before writing to make sure that any conflicting data is removed (for
|
|
// example, if the set was previously written to the writer's layer).
|
|
func (w *MapFieldWriter) clearTree(addr []string) {
|
|
prefix := strings.Join(addr, ".") + "."
|
|
for k := range w.result {
|
|
if strings.HasPrefix(k, prefix) {
|
|
delete(w.result, k)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (w *MapFieldWriter) WriteField(addr []string, value interface{}) error {
|
|
w.lock.Lock()
|
|
defer w.lock.Unlock()
|
|
if w.result == nil {
|
|
w.result = make(map[string]string)
|
|
}
|
|
|
|
schemaList := addrToSchema(addr, w.Schema)
|
|
if len(schemaList) == 0 {
|
|
return fmt.Errorf("Invalid address to set: %#v", addr)
|
|
}
|
|
|
|
// If we're setting anything other than a list root or set root,
|
|
// then disallow it.
|
|
for _, schema := range schemaList[:len(schemaList)-1] {
|
|
if schema.Type == TypeList {
|
|
return fmt.Errorf(
|
|
"%s: can only set full list",
|
|
strings.Join(addr, "."))
|
|
}
|
|
|
|
if schema.Type == TypeMap {
|
|
return fmt.Errorf(
|
|
"%s: can only set full map",
|
|
strings.Join(addr, "."))
|
|
}
|
|
|
|
if schema.Type == TypeSet {
|
|
return fmt.Errorf(
|
|
"%s: can only set full set",
|
|
strings.Join(addr, "."))
|
|
}
|
|
}
|
|
|
|
return w.set(addr, value)
|
|
}
|
|
|
|
func (w *MapFieldWriter) set(addr []string, value interface{}) error {
|
|
schemaList := addrToSchema(addr, w.Schema)
|
|
if len(schemaList) == 0 {
|
|
return fmt.Errorf("Invalid address to set: %#v", addr)
|
|
}
|
|
|
|
schema := schemaList[len(schemaList)-1]
|
|
switch schema.Type {
|
|
case TypeBool, TypeInt, TypeFloat, TypeString:
|
|
return w.setPrimitive(addr, value, schema)
|
|
case TypeList:
|
|
return w.setList(addr, value, schema)
|
|
case TypeMap:
|
|
return w.setMap(addr, value, schema)
|
|
case TypeSet:
|
|
return w.setSet(addr, value, schema)
|
|
case typeObject:
|
|
return w.setObject(addr, value, schema)
|
|
default:
|
|
panic(fmt.Sprintf("Unknown type: %#v", schema.Type))
|
|
}
|
|
}
|
|
|
|
func (w *MapFieldWriter) setList(
|
|
addr []string,
|
|
v interface{},
|
|
schema *Schema) error {
|
|
k := strings.Join(addr, ".")
|
|
setElement := func(idx string, value interface{}) error {
|
|
addrCopy := make([]string, len(addr), len(addr)+1)
|
|
copy(addrCopy, addr)
|
|
return w.set(append(addrCopy, idx), value)
|
|
}
|
|
|
|
var vs []interface{}
|
|
if err := mapstructure.Decode(v, &vs); err != nil {
|
|
return fmt.Errorf("%s: %s", k, err)
|
|
}
|
|
|
|
// Wipe the set from the current writer prior to writing if it exists.
|
|
// Multiple writes to the same layer is a lot safer for lists than sets due
|
|
// to the fact that indexes are always deterministic and the length will
|
|
// always be updated with the current length on the last write, but making
|
|
// sure we have a clean namespace removes any chance for edge cases to pop up
|
|
// and ensures that the last write to the set is the correct value.
|
|
w.clearTree(addr)
|
|
|
|
// Set the entire list.
|
|
var err error
|
|
for i, elem := range vs {
|
|
is := strconv.FormatInt(int64(i), 10)
|
|
err = setElement(is, elem)
|
|
if err != nil {
|
|
break
|
|
}
|
|
}
|
|
if err != nil {
|
|
for i, _ := range vs {
|
|
is := strconv.FormatInt(int64(i), 10)
|
|
setElement(is, nil)
|
|
}
|
|
|
|
return err
|
|
}
|
|
|
|
w.result[k+".#"] = strconv.FormatInt(int64(len(vs)), 10)
|
|
return nil
|
|
}
|
|
|
|
func (w *MapFieldWriter) setMap(
|
|
addr []string,
|
|
value interface{},
|
|
schema *Schema) error {
|
|
k := strings.Join(addr, ".")
|
|
v := reflect.ValueOf(value)
|
|
vs := make(map[string]interface{})
|
|
|
|
if value == nil {
|
|
// The empty string here means the map is removed.
|
|
w.result[k] = ""
|
|
return nil
|
|
}
|
|
|
|
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)
|
|
}
|
|
for _, mk := range v.MapKeys() {
|
|
mv := v.MapIndex(mk)
|
|
vs[mk.String()] = mv.Interface()
|
|
}
|
|
|
|
// Wipe this address tree. The contents of the map should always reflect the
|
|
// last write made to it.
|
|
w.clearTree(addr)
|
|
|
|
// Remove the pure key since we're setting the full map value
|
|
delete(w.result, k)
|
|
|
|
// Set each subkey
|
|
addrCopy := make([]string, len(addr), len(addr)+1)
|
|
copy(addrCopy, addr)
|
|
for subKey, v := range vs {
|
|
if err := w.set(append(addrCopy, subKey), v); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
// Set the count
|
|
w.result[k+".%"] = strconv.Itoa(len(vs))
|
|
|
|
return nil
|
|
}
|
|
|
|
func (w *MapFieldWriter) setObject(
|
|
addr []string,
|
|
value interface{},
|
|
schema *Schema) error {
|
|
// 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", strings.Join(addr, "."), err)
|
|
}
|
|
|
|
// Make space for additional elements in the address
|
|
addrCopy := make([]string, len(addr), len(addr)+1)
|
|
copy(addrCopy, addr)
|
|
|
|
// Set each element in turn
|
|
var err error
|
|
for k1, v1 := range v {
|
|
if err = w.set(append(addrCopy, k1), v1); err != nil {
|
|
break
|
|
}
|
|
}
|
|
if err != nil {
|
|
for k1, _ := range v {
|
|
w.set(append(addrCopy, k1), nil)
|
|
}
|
|
}
|
|
|
|
return err
|
|
}
|
|
|
|
func (w *MapFieldWriter) setPrimitive(
|
|
addr []string,
|
|
v interface{},
|
|
schema *Schema) error {
|
|
k := strings.Join(addr, ".")
|
|
|
|
if v == nil {
|
|
// The empty string here means the value is removed.
|
|
w.result[k] = ""
|
|
return nil
|
|
}
|
|
|
|
var set string
|
|
switch schema.Type {
|
|
case TypeBool:
|
|
var b bool
|
|
if err := mapstructure.Decode(v, &b); err != nil {
|
|
return fmt.Errorf("%s: %s", k, err)
|
|
}
|
|
|
|
set = strconv.FormatBool(b)
|
|
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)
|
|
case TypeFloat:
|
|
var n float64
|
|
if err := mapstructure.Decode(v, &n); err != nil {
|
|
return fmt.Errorf("%s: %s", k, err)
|
|
}
|
|
set = strconv.FormatFloat(float64(n), 'G', -1, 64)
|
|
default:
|
|
return fmt.Errorf("Unknown type: %#v", schema.Type)
|
|
}
|
|
|
|
w.result[k] = set
|
|
return nil
|
|
}
|
|
|
|
func (w *MapFieldWriter) setSet(
|
|
addr []string,
|
|
value interface{},
|
|
schema *Schema) error {
|
|
addrCopy := make([]string, len(addr), len(addr)+1)
|
|
copy(addrCopy, addr)
|
|
k := strings.Join(addr, ".")
|
|
|
|
if value == nil {
|
|
w.result[k+".#"] = "0"
|
|
return nil
|
|
}
|
|
|
|
// If it is a slice, then we have to turn it into a *Set so that
|
|
// we get the proper order back based on the hash code.
|
|
if v := reflect.ValueOf(value); v.Kind() == reflect.Slice {
|
|
// Build a temp *ResourceData to use for the conversion
|
|
tempAddr := addr[len(addr)-1:]
|
|
tempSchema := *schema
|
|
tempSchema.Type = TypeList
|
|
tempSchemaMap := map[string]*Schema{tempAddr[0]: &tempSchema}
|
|
tempW := &MapFieldWriter{Schema: tempSchemaMap}
|
|
|
|
// Set the entire list, this lets us get sane values out of it
|
|
if err := tempW.WriteField(tempAddr, value); err != nil {
|
|
return err
|
|
}
|
|
|
|
// Build the set by going over the list items in order and
|
|
// hashing them into the set. The reason we go over the list and
|
|
// not the `value` directly is because this forces all types
|
|
// to become []interface{} (generic) instead of []string, which
|
|
// most hash functions are expecting.
|
|
s := schema.ZeroValue().(*Set)
|
|
tempR := &MapFieldReader{
|
|
Map: BasicMapReader(tempW.Map()),
|
|
Schema: tempSchemaMap,
|
|
}
|
|
for i := 0; i < v.Len(); i++ {
|
|
is := strconv.FormatInt(int64(i), 10)
|
|
result, err := tempR.ReadField(append(tempAddr, is))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if !result.Exists {
|
|
panic("set item just set doesn't exist")
|
|
}
|
|
|
|
s.Add(result.Value)
|
|
}
|
|
|
|
value = s
|
|
}
|
|
|
|
// Clear any keys that match the set address first. This is necessary because
|
|
// it's always possible and sometimes may be necessary to write to a certain
|
|
// writer layer more than once with different set data each time, which will
|
|
// lead to different keys being inserted, which can lead to determinism
|
|
// problems when the old data isn't wiped first.
|
|
w.clearTree(addr)
|
|
|
|
if value.(*Set) == nil {
|
|
w.result[k+".#"] = "0"
|
|
return nil
|
|
}
|
|
|
|
for code, elem := range value.(*Set).m {
|
|
if err := w.set(append(addrCopy, code), elem); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
w.result[k+".#"] = strconv.Itoa(value.(*Set).Len())
|
|
return nil
|
|
}
|