mirror of
https://github.com/opentofu/opentofu.git
synced 2025-01-09 15:43:53 -06:00
cc8e8a55de
A common issue with new resource implementations is not considering parts of a complex structure that's used inside a set, which causes quirky behavior. The schema helper has enough information to provide a default reasonable implementation of a set function that includes all non-computed attributes in a deterministic way. Here we implement such a function and use it when no explicit hashing function is provided. In order to achieve this we encapsulate the construction of the zero value for a schema in a new method schema.ZeroValue, which allows us to put the fallback logic to the new default function in a single spot. It is no longer valid to use &Set{F: schema.Set} and all uses of that construct should be replaced with schema.ZeroValue().(*Set) .
106 lines
2.5 KiB
Go
106 lines
2.5 KiB
Go
package schema
|
|
|
|
import (
|
|
"bytes"
|
|
"sort"
|
|
"strconv"
|
|
)
|
|
|
|
func SerializeValueForHash(buf *bytes.Buffer, val interface{}, schema *Schema) {
|
|
if val == nil {
|
|
buf.WriteRune(';')
|
|
return
|
|
}
|
|
|
|
switch schema.Type {
|
|
case TypeBool:
|
|
if val.(bool) {
|
|
buf.WriteRune('1')
|
|
} else {
|
|
buf.WriteRune('0')
|
|
}
|
|
case TypeInt:
|
|
buf.WriteString(strconv.Itoa(val.(int)))
|
|
case TypeFloat:
|
|
buf.WriteString(strconv.FormatFloat(val.(float64), 'g', -1, 64))
|
|
case TypeString:
|
|
buf.WriteString(val.(string))
|
|
case TypeList:
|
|
buf.WriteRune('(')
|
|
l := val.([]interface{})
|
|
for _, innerVal := range l {
|
|
serializeCollectionMemberForHash(buf, innerVal, schema.Elem)
|
|
}
|
|
buf.WriteRune(')')
|
|
case TypeMap:
|
|
m := val.(map[string]interface{})
|
|
var keys []string
|
|
for k := range m {
|
|
keys = append(keys, k)
|
|
}
|
|
sort.Strings(keys)
|
|
buf.WriteRune('[')
|
|
for _, k := range keys {
|
|
innerVal := m[k]
|
|
buf.WriteString(k)
|
|
buf.WriteRune(':')
|
|
serializeCollectionMemberForHash(buf, innerVal, schema.Elem)
|
|
}
|
|
buf.WriteRune(']')
|
|
case TypeSet:
|
|
buf.WriteRune('{')
|
|
s := val.(*Set)
|
|
for _, innerVal := range s.List() {
|
|
serializeCollectionMemberForHash(buf, innerVal, schema.Elem)
|
|
}
|
|
buf.WriteRune('}')
|
|
default:
|
|
panic("unknown schema type to serialize")
|
|
}
|
|
buf.WriteRune(';')
|
|
}
|
|
|
|
// SerializeValueForHash appends a serialization of the given resource config
|
|
// to the given buffer, guaranteeing deterministic results given the same value
|
|
// and schema.
|
|
//
|
|
// Its primary purpose is as input into a hashing function in order
|
|
// to hash complex substructures when used in sets, and so the serialization
|
|
// is not reversible.
|
|
func SerializeResourceForHash(buf *bytes.Buffer, val interface{}, resource *Resource) {
|
|
sm := resource.Schema
|
|
m := val.(map[string]interface{})
|
|
var keys []string
|
|
for k := range sm {
|
|
keys = append(keys, k)
|
|
}
|
|
sort.Strings(keys)
|
|
for _, k := range keys {
|
|
innerSchema := sm[k]
|
|
// Skip attributes that are not user-provided. Computed attributes
|
|
// do not contribute to the hash since their ultimate value cannot
|
|
// be known at plan/diff time.
|
|
if !(innerSchema.Required || innerSchema.Optional) {
|
|
continue
|
|
}
|
|
|
|
buf.WriteString(k)
|
|
buf.WriteRune(':')
|
|
innerVal := m[k]
|
|
SerializeValueForHash(buf, innerVal, innerSchema)
|
|
}
|
|
}
|
|
|
|
func serializeCollectionMemberForHash(buf *bytes.Buffer, val interface{}, elem interface{}) {
|
|
switch tElem := elem.(type) {
|
|
case *Schema:
|
|
SerializeValueForHash(buf, val, tElem)
|
|
case *Resource:
|
|
buf.WriteRune('<')
|
|
SerializeResourceForHash(buf, val, tElem)
|
|
buf.WriteString(">;")
|
|
default:
|
|
panic("invalid element type")
|
|
}
|
|
}
|