mirror of https://github.com/databricks/cli.git
92 lines
2.5 KiB
Go
92 lines
2.5 KiB
Go
package merge
|
|
|
|
import "github.com/databricks/cli/libs/dyn"
|
|
|
|
type elementsByKey struct {
|
|
key string
|
|
keyFunc func(dyn.Value) string
|
|
}
|
|
|
|
func (e elementsByKey) doMap(_ dyn.Path, v dyn.Value, mergeFunc func(a, b dyn.Value) (dyn.Value, error)) (dyn.Value, error) {
|
|
// We know the type of this value is a sequence.
|
|
// For additional defence, return self if it is not.
|
|
elements, ok := v.AsSequence()
|
|
if !ok {
|
|
return v, nil
|
|
}
|
|
|
|
seen := make(map[string]dyn.Value, len(elements))
|
|
keys := make([]string, 0, len(elements))
|
|
|
|
// Iterate in natural order. For a given key, we first see the
|
|
// base definition and merge instances that come after it.
|
|
for i := range elements {
|
|
kv := elements[i].Get(e.key)
|
|
key := e.keyFunc(kv)
|
|
|
|
// Register element with key if not yet seen before.
|
|
ref, ok := seen[key]
|
|
if !ok {
|
|
keys = append(keys, key)
|
|
seen[key] = elements[i]
|
|
continue
|
|
}
|
|
|
|
// Merge this instance into the reference.
|
|
nv, err := mergeFunc(ref, elements[i])
|
|
if err != nil {
|
|
return v, err
|
|
}
|
|
|
|
// Overwrite reference.
|
|
seen[key] = nv
|
|
}
|
|
|
|
// Gather resulting elements in natural order.
|
|
out := make([]dyn.Value, 0, len(keys))
|
|
for _, key := range keys {
|
|
nv, err := dyn.Set(seen[key], e.key, dyn.V(key))
|
|
if err != nil {
|
|
return dyn.InvalidValue, err
|
|
}
|
|
out = append(out, nv)
|
|
}
|
|
|
|
return dyn.NewValue(out, v.Locations()), nil
|
|
}
|
|
|
|
func (e elementsByKey) Map(_ dyn.Path, v dyn.Value) (dyn.Value, error) {
|
|
return e.doMap(nil, v, Merge)
|
|
}
|
|
|
|
func (e elementsByKey) MapWithOverride(p dyn.Path, v dyn.Value) (dyn.Value, error) {
|
|
return e.doMap(nil, v, func(a, b dyn.Value) (dyn.Value, error) {
|
|
return Override(a, b, OverrideVisitor{
|
|
VisitInsert: func(_ dyn.Path, v dyn.Value) (dyn.Value, error) {
|
|
return v, nil
|
|
},
|
|
VisitDelete: func(valuePath dyn.Path, left dyn.Value) error {
|
|
return nil
|
|
},
|
|
VisitUpdate: func(_ dyn.Path, a, b dyn.Value) (dyn.Value, error) {
|
|
return b, nil
|
|
},
|
|
})
|
|
})
|
|
}
|
|
|
|
// ElementsByKey returns a [dyn.MapFunc] that operates on a sequence
|
|
// where each element is a map. It groups elements by a key and merges
|
|
// elements with the same key.
|
|
//
|
|
// The function that extracts the key from an element is provided as
|
|
// a parameter. The resulting elements get their key field overwritten
|
|
// with the value as returned by the key function.
|
|
func ElementsByKey(key string, keyFunc func(dyn.Value) string) dyn.MapFunc {
|
|
return elementsByKey{key, keyFunc}.Map
|
|
}
|
|
|
|
func ElementsByKeyWithOverride(key string, keyFunc func(dyn.Value) string) dyn.MapFunc {
|
|
return elementsByKey{key, keyFunc}.MapWithOverride
|
|
}
|