2023-12-22 13:20:45 +00:00
|
|
|
package dyn
|
2023-12-22 10:38:09 +00:00
|
|
|
|
|
|
|
import "errors"
|
|
|
|
|
|
|
|
// WalkValueFunc is the type of the function called by Walk to traverse the configuration tree.
|
|
|
|
type WalkValueFunc func(p Path, v Value) (Value, error)
|
|
|
|
|
|
|
|
// ErrDrop may be returned by WalkValueFunc to remove a value from the subtree.
|
|
|
|
var ErrDrop = errors.New("drop value from subtree")
|
|
|
|
|
|
|
|
// ErrSkip may be returned by WalkValueFunc to skip traversal of a subtree.
|
|
|
|
var ErrSkip = errors.New("skip traversal of subtree")
|
|
|
|
|
|
|
|
// Walk walks the configuration tree and calls the given function on each node.
|
|
|
|
// The callback may return ErrDrop to remove a value from the subtree.
|
|
|
|
// The callback may return ErrSkip to skip traversal of a subtree.
|
|
|
|
// If the callback returns another error, the walk is aborted, and the error is returned.
|
|
|
|
func Walk(v Value, fn func(p Path, v Value) (Value, error)) (Value, error) {
|
|
|
|
return walk(v, EmptyPath, fn)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Unexported counterpart to Walk.
|
|
|
|
// It carries the path leading up to the current node,
|
|
|
|
// such that it can be passed to the WalkValueFunc.
|
|
|
|
func walk(v Value, p Path, fn func(p Path, v Value) (Value, error)) (Value, error) {
|
|
|
|
v, err := fn(p, v)
|
|
|
|
if err != nil {
|
|
|
|
if err == ErrSkip {
|
|
|
|
return v, nil
|
|
|
|
}
|
2024-06-19 15:24:57 +00:00
|
|
|
return InvalidValue, err
|
2023-12-22 10:38:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
switch v.Kind() {
|
|
|
|
case KindMap:
|
|
|
|
m := v.MustMap()
|
2024-03-25 11:01:09 +00:00
|
|
|
out := newMappingWithSize(m.Len())
|
|
|
|
for _, pair := range m.Pairs() {
|
|
|
|
pk := pair.Key
|
|
|
|
pv := pair.Value
|
|
|
|
nv, err := walk(pv, append(p, Key(pk.MustString())), fn)
|
2023-12-22 10:38:09 +00:00
|
|
|
if err == ErrDrop {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if err != nil {
|
2024-06-19 15:24:57 +00:00
|
|
|
return InvalidValue, err
|
2023-12-22 10:38:09 +00:00
|
|
|
}
|
2024-12-11 12:26:00 +00:00
|
|
|
out.Set(pk, nv) //nolint:errcheck
|
2023-12-22 10:38:09 +00:00
|
|
|
}
|
|
|
|
v.v = out
|
|
|
|
case KindSequence:
|
|
|
|
s := v.MustSequence()
|
|
|
|
out := make([]Value, 0, len(s))
|
|
|
|
for i := range s {
|
2024-03-19 09:49:26 +00:00
|
|
|
nv, err := walk(s[i], append(p, Index(i)), fn)
|
2023-12-22 10:38:09 +00:00
|
|
|
if err == ErrDrop {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if err != nil {
|
2024-06-19 15:24:57 +00:00
|
|
|
return InvalidValue, err
|
2023-12-22 10:38:09 +00:00
|
|
|
}
|
|
|
|
out = append(out, nv)
|
|
|
|
}
|
|
|
|
v.v = out
|
|
|
|
}
|
|
|
|
|
|
|
|
return v, nil
|
|
|
|
}
|