You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
tea/env.go

201 lines
4.5 KiB
Go

package tea
import (
"fmt"
"reflect"
)
type env struct {
data map[string]interface{}
parent *env
}
func mkenv(test Test) *env {
var e *env
return e.save(test)
}
// save looks at the Test t and saves the values of its fields marked with a
// save tag. All of the fields for that tests are stored together as a data
// layer.
func (e *env) save(test Test) *env {
V := reflect.ValueOf(test)
if V.Type().Kind() == reflect.Ptr {
V = V.Elem()
}
T := V.Type()
if T.Kind() != reflect.Struct {
return e
}
saved := make(map[string]interface{})
for i := 0; i < T.NumField(); i++ {
f := T.Field(i)
if !isSaveField(f) {
continue
}
fv := V.Field(i)
saved[f.Name] = fv.Interface()
}
if len(saved) > 0 {
return &env{
data: saved,
parent: e,
}
}
return e
}
func (e *env) load(dest Test) error {
destV := reflect.ValueOf(dest).Elem()
destT := destV.Type()
e, err := e.match(dest)
if err != nil {
return fmt.Errorf("match failed: %w", err)
}
for i := 0; i < destT.NumField(); i++ {
f := destT.Field(i)
if !isLoadField(f) {
continue
}
fv := destV.Field(i)
if !fv.IsZero() {
// the value is already populated, so we don't want to overwrite
// it.
continue
}
set := false
for e := e; e != nil; e = e.parent {
v, ok := e.data[f.Name]
if !ok {
continue
}
ev := reflect.ValueOf(v)
if ev.Type().AssignableTo(fv.Type()) {
set = true
fv.Set(ev)
break
}
}
if !set {
return fmt.Errorf("%w: failed to set required field: %q", PlanError, f.Name)
}
}
return nil
}
func (e *env) match(dest Test) (*env, error) {
destV := reflect.ValueOf(dest).Elem()
destT := destV.Type()
required := getMatchFields(destT)
if len(required) == 0 {
return e, nil
}
var (
last *env
leaf *env
foundWithWrongType = make(map[string]bool)
foundWithMatchingType = make(map[string]bool)
foundWithWrongValue = make(map[string]bool)
foundWithCorrectValue = make(map[string]bool)
)
for e := e; e != nil; e = e.parent {
present := make([]reflect.StructField, 0, len(required))
for _, f := range required {
ev, ok := e.data[f.Name]
if !ok {
break
}
if reflect.TypeOf(ev).AssignableTo(f.Type) {
foundWithMatchingType[f.Name] = true
present = append(present, f)
} else {
foundWithWrongType[f.Name] = true
}
}
// all required fields are present in this layer
if len(present) == len(required) {
// check that the values in the env match the values that were
// asked for.
matched := make(map[string]interface{})
for _, f := range required {
fv := destV.FieldByName(f.Name)
if fv.Interface() == e.data[f.Name] {
foundWithCorrectValue[f.Name] = true
matched[f.Name] = e.data[f.Name]
} else {
foundWithWrongValue[f.Name] = true
}
}
// all required match conditions are met
if len(matched) == len(required) {
if leaf == nil {
// if this is the first matched layer, it is the leaf of the
// resultant env.
leaf = e
last = leaf
} else {
// otherwise we keep this layer, since it matched our match
// requirements. Another layer already did, but there may
// be other things in the layer we want to keep.
last.parent = e
last = e
}
}
} else {
// the required fields do not exist in the layer, so this layer
// does not conflict with the match requirement.
if leaf != nil {
// since we have a leaf node, we have found a matching layer,
// and since this layer does not conflict, we keep it.
last.parent = e
last = e
}
}
}
if leaf == nil {
var notFound []string
for _, f := range required {
if !foundWithMatchingType[f.Name] && !foundWithWrongType[f.Name] {
notFound = append(notFound, f.Name)
}
}
switch len(notFound) {
case 0:
break
case 1:
return nil, fmt.Errorf("%w: missing required field: %q", PlanError, notFound[0])
default:
return nil, fmt.Errorf("%w: missing %d required fields: %s", PlanError, len(notFound), notFound)
}
for f, _ := range foundWithWrongType {
if !foundWithMatchingType[f] {
return nil, fmt.Errorf("%w: field %s was only found with unmatching types", PlanError, f)
}
}
for f, _ := range foundWithWrongValue {
if !foundWithCorrectValue[f] {
return nil, fmt.Errorf("%w: field %s was only found with unmatching values", RunError, f)
}
}
return nil, fmt.Errorf("%w: required match fields not encountered on the same layer", PlanError)
}
return leaf, nil
}