databricks-cli/libs/dyn/visit.go

175 lines
4.0 KiB
Go

package dyn
import (
"errors"
"fmt"
"slices"
)
// This error is returned if the path indicates that a map or sequence is expected, but the value is nil.
type cannotTraverseNilError struct {
p Path
}
func (e cannotTraverseNilError) Error() string {
component := e.p[len(e.p)-1]
switch {
case component.isKey():
return fmt.Sprintf("expected a map to index %q, found nil", e.p)
case component.isIndex():
return fmt.Sprintf("expected a sequence to index %q, found nil", e.p)
default:
panic("invalid component")
}
}
func IsCannotTraverseNilError(err error) bool {
var target cannotTraverseNilError
return errors.As(err, &target)
}
type noSuchKeyError struct {
p Path
}
func (e noSuchKeyError) Error() string {
return fmt.Sprintf("key not found at %q", e.p)
}
func IsNoSuchKeyError(err error) bool {
var target noSuchKeyError
return errors.As(err, &target)
}
type indexOutOfBoundsError struct {
p Path
}
func (e indexOutOfBoundsError) Error() string {
return fmt.Sprintf("index out of bounds at %q", e.p)
}
func IsIndexOutOfBoundsError(err error) bool {
var target indexOutOfBoundsError
return errors.As(err, &target)
}
type visitOptions struct {
// The function to apply to the value once found.
//
// If this function returns the same value as it receives as argument,
// the original visit function call returns the original value unmodified.
//
// If this function returns a new value, the original visit function call
// returns a value with all the intermediate values updated.
//
// If this function returns an error, the original visit function call
// returns this error and the value is left unmodified.
fn func(Path, Value) (Value, error)
}
func visit(v Value, prefix Path, suffix Pattern, opts visitOptions) (Value, error) {
if len(suffix) == 0 {
return opts.fn(slices.Clone(prefix), v)
}
// Initialize prefix if it is empty.
// It is pre-allocated to its maximum size to avoid additional allocations.
if len(prefix) == 0 {
prefix = make(Path, 0, len(suffix))
}
component := suffix[0]
suffix = suffix[1:]
// Visit the value with the current component.
return component.visit(v, prefix, suffix, opts)
}
func (component pathComponent) visit(v Value, prefix Path, suffix Pattern, opts visitOptions) (Value, error) {
path := append(prefix, component)
switch {
case component.isKey():
// Expect a map to be set if this is a key.
switch v.Kind() {
case KindMap:
// OK
case KindNil:
return InvalidValue, cannotTraverseNilError{path}
default:
return InvalidValue, fmt.Errorf("expected a map to index %q, found %s", path, v.Kind())
}
m := v.MustMap()
// Lookup current value in the map.
ev, ok := m.GetByString(component.key)
if !ok {
return InvalidValue, noSuchKeyError{path}
}
// Recursively transform the value.
nv, err := visit(ev, path, suffix, opts)
if err != nil {
return InvalidValue, err
}
// Return the original value if the value hasn't changed.
if nv.eq(ev) {
return v, nil
}
// Return an updated map value.
m = m.Clone()
m.Set(V(component.key), nv) //nolint:errcheck
return Value{
v: m,
k: KindMap,
l: v.l,
}, nil
case component.isIndex():
// Expect a sequence to be set if this is an index.
switch v.Kind() {
case KindSequence:
// OK
case KindNil:
return InvalidValue, cannotTraverseNilError{path}
default:
return InvalidValue, fmt.Errorf("expected a sequence to index %q, found %s", path, v.Kind())
}
s := v.MustSequence()
// Lookup current value in the sequence.
if component.index < 0 || component.index >= len(s) {
return InvalidValue, indexOutOfBoundsError{path}
}
// Recursively transform the value.
ev := s[component.index]
nv, err := visit(ev, path, suffix, opts)
if err != nil {
return InvalidValue, err
}
// Return the original value if the value hasn't changed.
if nv.eq(ev) {
return v, nil
}
// Return an updated sequence value.
s = slices.Clone(s)
s[component.index] = nv
return Value{
v: s,
k: KindSequence,
l: v.l,
}, nil
default:
panic("invalid component")
}
}