126 lines
4 KiB
Go
126 lines
4 KiB
Go
package state
|
|
|
|
import (
|
|
"bytes"
|
|
"errors"
|
|
"fmt"
|
|
)
|
|
|
|
// indexerSingle implements both memdb.Indexer and memdb.SingleIndexer. It may
|
|
// be used in a memdb.IndexSchema to specify functions that generate the index
|
|
// value for memdb.Txn operations.
|
|
type indexerSingle struct {
|
|
// readIndex is used by memdb for Txn.Get, Txn.First, and other operations
|
|
// that read data.
|
|
readIndex
|
|
// writeIndex is used by memdb for Txn.Insert, Txn.Delete, for operations
|
|
// that write data to the index.
|
|
writeIndex
|
|
}
|
|
|
|
// indexerMulti implements both memdb.Indexer and memdb.MultiIndexer. It may
|
|
// be used in a memdb.IndexSchema to specify functions that generate the index
|
|
// value for memdb.Txn operations.
|
|
type indexerMulti struct {
|
|
// readIndex is used by memdb for Txn.Get, Txn.First, and other operations
|
|
// that read data.
|
|
readIndex
|
|
// writeIndexMulti is used by memdb for Txn.Insert, Txn.Delete, for operations
|
|
// that write data to the index.
|
|
writeIndexMulti
|
|
}
|
|
|
|
// indexerSingleWithPrefix is a indexerSingle which also supports prefix queries.
|
|
type indexerSingleWithPrefix struct {
|
|
readIndex
|
|
writeIndex
|
|
prefixIndex
|
|
}
|
|
|
|
// readIndex implements memdb.Indexer. It exists so that a function can be used
|
|
// to provide the interface.
|
|
//
|
|
// Unlike memdb.Indexer, a readIndex function accepts only a single argument. To
|
|
// generate an index from multiple values, use a struct type with multiple fields.
|
|
type readIndex func(arg interface{}) ([]byte, error)
|
|
|
|
func (f readIndex) FromArgs(args ...interface{}) ([]byte, error) {
|
|
if len(args) != 1 {
|
|
return nil, fmt.Errorf("index supports only a single arg")
|
|
}
|
|
return f(args[0])
|
|
}
|
|
|
|
var errMissingValueForIndex = fmt.Errorf("object is missing a value for this index")
|
|
|
|
// writeIndex implements memdb.SingleIndexer. It exists so that a function
|
|
// can be used to provide this interface.
|
|
//
|
|
// Instead of a bool return value, writeIndex expects errMissingValueForIndex to
|
|
// indicate that an index could not be build for the object. It will translate
|
|
// this error into a false value to satisfy the memdb.SingleIndexer interface.
|
|
type writeIndex func(raw interface{}) ([]byte, error)
|
|
|
|
func (f writeIndex) FromObject(raw interface{}) (bool, []byte, error) {
|
|
v, err := f(raw)
|
|
if errors.Is(err, errMissingValueForIndex) {
|
|
return false, nil, nil
|
|
}
|
|
return err == nil, v, err
|
|
}
|
|
|
|
// writeIndexMulti implements memdb.MultiIndexer. It exists so that a function
|
|
// can be used to provide this interface.
|
|
//
|
|
// Instead of a bool return value, writeIndexMulti expects errMissingValueForIndex to
|
|
// indicate that an index could not be build for the object. It will translate
|
|
// this error into a false value to satisfy the memdb.MultiIndexer interface.
|
|
type writeIndexMulti func(raw interface{}) ([][]byte, error)
|
|
|
|
func (f writeIndexMulti) FromObject(raw interface{}) (bool, [][]byte, error) {
|
|
v, err := f(raw)
|
|
if errors.Is(err, errMissingValueForIndex) {
|
|
return false, nil, nil
|
|
}
|
|
return err == nil, v, err
|
|
}
|
|
|
|
// prefixIndex implements memdb.PrefixIndexer. It exists so that a function
|
|
// can be used to provide this interface.
|
|
type prefixIndex func(args interface{}) ([]byte, error)
|
|
|
|
func (f prefixIndex) PrefixFromArgs(args ...interface{}) ([]byte, error) {
|
|
if len(args) != 1 {
|
|
return nil, fmt.Errorf("index supports only a single arg")
|
|
}
|
|
return f(args[0])
|
|
}
|
|
|
|
const null = "\x00"
|
|
|
|
// indexBuilder is a buffer used to construct memdb index values.
|
|
type indexBuilder bytes.Buffer
|
|
|
|
func newIndexBuilder(cap int) *indexBuilder {
|
|
buff := make([]byte, 0, cap)
|
|
b := bytes.NewBuffer(buff)
|
|
return (*indexBuilder)(b)
|
|
}
|
|
|
|
// String appends the string and a null terminator to the buffer.
|
|
func (b *indexBuilder) String(v string) {
|
|
(*bytes.Buffer)(b).WriteString(v)
|
|
(*bytes.Buffer)(b).WriteString(null)
|
|
}
|
|
|
|
// Raw appends the bytes without a null terminator to the buffer. Raw should
|
|
// only be used when v has a fixed length, or when building the last segment of
|
|
// a prefix index.
|
|
func (b *indexBuilder) Raw(v []byte) {
|
|
(*bytes.Buffer)(b).Write(v)
|
|
}
|
|
|
|
func (b *indexBuilder) Bytes() []byte {
|
|
return (*bytes.Buffer)(b).Bytes()
|
|
}
|