cosmos-sdk/collections/collections.go

179 lines
6.7 KiB
Go

package collections
import (
"context"
"errors"
"io"
"math"
"cosmossdk.io/collections/codec"
"cosmossdk.io/schema"
)
var (
// ErrNotFound is returned when the provided key is not present in the StorageProvider.
ErrNotFound = errors.New("collections: not found")
// ErrEncoding is returned when something fails during key or value encoding/decoding.
ErrEncoding = codec.ErrEncoding
// ErrConflict is returned when there are conflicts, for example in UniqueIndex.
ErrConflict = errors.New("collections: conflict")
)
// KEYS
var (
// Uint16Key can be used to encode uint16 keys. Encoding is big endian to retain ordering.
Uint16Key = codec.NewUint16Key[uint16]()
// Uint32Key can be used to encode uint32 keys. Encoding is big endian to retain ordering.
Uint32Key = codec.NewUint32Key[uint32]()
// Uint64Key can be used to encode uint64 keys. Encoding is big endian to retain ordering.
Uint64Key = codec.NewUint64Key[uint64]()
// Int32Key can be used to encode int32 keys. Encoding retains ordering by toggling the MSB.
Int32Key = codec.NewInt32Key[int32]()
// Int64Key can be used to encode int64 keys. Encoding retains ordering by toggling the MSB.
Int64Key = codec.NewInt64Key[int64]()
// StringKey can be used to encode string keys. The encoding just converts the string
// to bytes.
// Non-terminality in multipart keys is handled by appending the StringDelimiter,
// this means that a string key when used as the non final part of a multipart key cannot
// contain the StringDelimiter.
// Lexicographical ordering is retained both in non and multipart keys.
StringKey = codec.NewStringKeyCodec[string]()
// BytesKey can be used to encode bytes keys. The encoding will just use
// the provided bytes.
// When used as the non-terminal part of a multipart key, we prefix the bytes key
// with a single byte representing the length of the key. This means two things:
// 1. When used in multipart keys the length can be at maximum 255 (max number that
// can be represented with a single byte).
// 2. When used in multipart keys the lexicographical ordering is lost due to the
// length prefixing.
// JSON encoding represents a bytes key as a hex encoded string.
BytesKey = codec.NewBytesKey[[]byte]()
// BoolKey can be used to encode booleans. It uses a single byte to represent the boolean.
// 0x0 is used to represent false, and 0x1 is used to represent true.
BoolKey = codec.NewBoolKey[bool]()
)
// VALUES
var (
// BoolValue implements a ValueCodec for bool.
BoolValue = codec.KeyToValueCodec(BoolKey)
// Uint16Value implements a ValueCodec for uint16.
Uint16Value = codec.KeyToValueCodec(Uint16Key)
// Uint32Value implements a ValueCodec for uint32.
Uint32Value = codec.KeyToValueCodec(Uint32Key)
// Uint64Value implements a ValueCodec for uint64.
Uint64Value = codec.KeyToValueCodec(Uint64Key)
// Int32Value implements a ValueCodec for int32.
Int32Value = codec.KeyToValueCodec(Int32Key)
// Int64Value implements a ValueCodec for int64.
Int64Value = codec.KeyToValueCodec(Int64Key)
// StringValue implements a ValueCodec for string.
StringValue = codec.KeyToValueCodec(StringKey)
// BytesValue implements a ValueCodec for bytes.
BytesValue = codec.KeyToValueCodec(BytesKey)
)
// Collection is the interface that all collections implement. It will eventually
// include methods for importing/exporting genesis data and schema
// reflection for clients.
// NOTE: Unstable.
type Collection interface {
// GetName is the unique name of the collection within a schema. It must
// match format specified by NameRegex.
GetName() string
// GetPrefix is the unique prefix of the collection within a schema.
GetPrefix() []byte
// ValueCodec returns the codec used to encode/decode values of the collection.
ValueCodec() codec.UntypedValueCodec
genesisHandler
// collectionSchemaCodec returns the schema codec for this collection.
schemaCodec() (*collectionSchemaCodec, error)
// isSecondaryIndex indicates that this collection represents a secondary index
// in the schema and should be excluded from the module's user facing schema.
isSecondaryIndex() bool
}
// collectionSchemaCodec maps a collection to a schema object type and provides
// decoders and encoders to and from schema values and raw kv-store bytes.
type collectionSchemaCodec struct {
coll Collection
objectType schema.StateObjectType
keyDecoder func([]byte) (any, error)
valueDecoder func([]byte) (any, error)
}
// Prefix defines a segregation bytes namespace for specific collections objects.
type Prefix []byte
// Bytes returns the raw Prefix bytes.
func (n Prefix) Bytes() []byte { return n }
// NewPrefix returns a Prefix given the provided namespace identifier.
// In the same module, no prefixes should share the same starting bytes
// meaning that having two namespaces whose bytes representation is:
// p1 := []byte("prefix")
// p2 := []byte("prefix1")
// yields to iterations of p1 overlapping over p2.
// If a numeric prefix is provided, it must be between 0 and 255 (uint8).
// If out of bounds this function will panic.
// Reason for which this function is constrained to `int` instead of `uint8` is for
// API ergonomics, golang's type inference will infer int properly but not uint8
// meaning that developers would need to write NewPrefix(uint8(number)) for numeric
// prefixes.
func NewPrefix[T interface{ int | string | []byte }](identifier T) Prefix {
i := any(identifier)
var prefix []byte
switch c := i.(type) {
case int:
if c > math.MaxUint8 || c < 0 {
panic("invalid integer prefix value: must be between 0 and 255")
}
prefix = []byte{uint8(c)}
case string:
prefix = []byte(c)
case []byte:
identifierCopy := make([]byte, len(c))
copy(identifierCopy, c)
prefix = identifierCopy
}
return prefix
}
var _ Collection = (*collectionImpl[string, string])(nil)
// collectionImpl wraps a Map and implements Collection. This properly splits
// the generic and untyped Collection interface from the typed Map, which every
// collection builds on.
type collectionImpl[K, V any] struct {
m Map[K, V]
}
func (c collectionImpl[K, V]) ValueCodec() codec.UntypedValueCodec {
return codec.NewUntypedValueCodec(c.m.vc)
}
func (c collectionImpl[K, V]) GetName() string { return c.m.name }
func (c collectionImpl[K, V]) GetPrefix() []byte { return NewPrefix(c.m.prefix) }
func (c collectionImpl[K, V]) validateGenesis(r io.Reader) error { return c.m.validateGenesis(r) }
func (c collectionImpl[K, V]) importGenesis(ctx context.Context, r io.Reader) error {
return c.m.importGenesis(ctx, r)
}
func (c collectionImpl[K, V]) exportGenesis(ctx context.Context, w io.Writer) error {
return c.m.exportGenesis(ctx, w)
}
func (c collectionImpl[K, V]) defaultGenesis(w io.Writer) error { return c.m.defaultGenesis(w) }
func (c collectionImpl[K, V]) isSecondaryIndex() bool { return c.m.isSecondaryIndex }