databricks-cli/libs/dyn/mapping.go

150 lines
3.8 KiB
Go

package dyn
import (
"fmt"
"maps"
"slices"
)
// Pair represents a single key-value pair in a Mapping.
type Pair struct {
Key Value
Value Value
}
// Mapping represents a key-value map of dynamic values.
// It exists because plain Go maps cannot use dynamic values for keys.
// We need to use dynamic values for keys because it lets us associate metadata
// with keys (i.e. their definition location). Keys must be strings.
type Mapping struct {
pairs []Pair
index map[string]int
}
// NewMapping creates a new empty Mapping.
func NewMapping() Mapping {
return Mapping{
pairs: make([]Pair, 0),
index: make(map[string]int),
}
}
// newMappingWithSize creates a new Mapping preallocated to the specified size.
func newMappingWithSize(size int) Mapping {
return Mapping{
pairs: make([]Pair, 0, size),
index: make(map[string]int, size),
}
}
// newMappingFromGoMap creates a new Mapping from a Go map of string keys and dynamic values.
func newMappingFromGoMap(vin map[string]Value) Mapping {
m := newMappingWithSize(len(vin))
for k, v := range vin {
m.Set(V(k), v) //nolint:errcheck
}
return m
}
// Pairs returns all the key-value pairs in the Mapping. The pairs are sorted by
// their key in lexicographic order.
func (m Mapping) Pairs() []Pair {
return m.pairs
}
// Len returns the number of key-value pairs in the Mapping.
func (m Mapping) Len() int {
return len(m.pairs)
}
// GetPair returns the key-value pair with the specified key.
// It also returns a boolean indicating whether the pair was found.
func (m Mapping) GetPair(key Value) (Pair, bool) {
skey, ok := key.AsString()
if !ok {
return Pair{}, false
}
return m.GetPairByString(skey)
}
// GetPairByString returns the key-value pair with the specified string key.
// It also returns a boolean indicating whether the pair was found.
func (m Mapping) GetPairByString(skey string) (Pair, bool) {
if i, ok := m.index[skey]; ok {
return m.pairs[i], true
}
return Pair{}, false
}
// Get returns the value associated with the specified key.
// It also returns a boolean indicating whether the value was found.
func (m Mapping) Get(key Value) (Value, bool) {
p, ok := m.GetPair(key)
return p.Value, ok
}
// GetByString returns the value associated with the specified string key.
// It also returns a boolean indicating whether the value was found.
func (m *Mapping) GetByString(skey string) (Value, bool) {
p, ok := m.GetPairByString(skey)
return p.Value, ok
}
// Set sets the value for the given key in the mapping.
// If the key already exists, the value is updated.
// If the key does not exist, a new key-value pair is added.
// The key must be a string, otherwise an error is returned.
func (m *Mapping) Set(key, value Value) error {
skey, ok := key.AsString()
if !ok {
return fmt.Errorf("key must be a string, got %s", key.Kind())
}
// If the key already exists, update the value.
if i, ok := m.index[skey]; ok {
m.pairs[i].Value = value
return nil
}
// Otherwise, add a new pair.
m.pairs = append(m.pairs, Pair{key, value})
if m.index == nil {
m.index = make(map[string]int)
}
m.index[skey] = len(m.pairs) - 1
return nil
}
// Keys returns all the keys in the Mapping.
func (m Mapping) Keys() []Value {
keys := make([]Value, 0, len(m.pairs))
for _, p := range m.pairs {
keys = append(keys, p.Key)
}
return keys
}
// Values returns all the values in the Mapping.
func (m Mapping) Values() []Value {
values := make([]Value, 0, len(m.pairs))
for _, p := range m.pairs {
values = append(values, p.Value)
}
return values
}
// Clone creates a shallow copy of the Mapping.
func (m Mapping) Clone() Mapping {
return Mapping{
pairs: slices.Clone(m.pairs),
index: maps.Clone(m.index),
}
}
// Merge merges the key-value pairs from another Mapping into the current Mapping.
func (m *Mapping) Merge(n Mapping) {
for _, p := range n.pairs {
m.Set(p.Key, p.Value) //nolint:errcheck
}
}