2024-03-25 11:01:09 +00:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2024-07-29 13:04:02 +00:00
|
|
|
// Pairs returns all the key-value pairs in the Mapping. The pairs are sorted by
|
|
|
|
// their key in lexicographic order.
|
2024-03-25 11:01:09 +00:00
|
|
|
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
|
|
|
|
}
|
|
|
|
}
|