79 lines
2.4 KiB
Go
79 lines
2.4 KiB
Go
|
package indexer
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"encoding/binary"
|
||
|
"errors"
|
||
|
"fmt"
|
||
|
"time"
|
||
|
|
||
|
"github.com/hashicorp/go-memdb"
|
||
|
)
|
||
|
|
||
|
var (
|
||
|
// Ensure the required memdb interfaces are met at compile time.
|
||
|
_ memdb.Indexer = SingleIndexer{}
|
||
|
_ memdb.SingleIndexer = SingleIndexer{}
|
||
|
)
|
||
|
|
||
|
// SingleIndexer 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 SingleIndexer 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, and other
|
||
|
// operations that write data to the index.
|
||
|
WriteIndex
|
||
|
}
|
||
|
|
||
|
// 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 any) ([]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 any) ([]byte, error)
|
||
|
|
||
|
func (f WriteIndex) FromObject(raw any) (bool, []byte, error) {
|
||
|
v, err := f(raw)
|
||
|
if errors.Is(err, ErrMissingValueForIndex) {
|
||
|
return false, nil, nil
|
||
|
}
|
||
|
return err == nil, v, err
|
||
|
}
|
||
|
|
||
|
// IndexBuilder is a buffer used to construct memdb index values.
|
||
|
type IndexBuilder bytes.Buffer
|
||
|
|
||
|
// Bytes returns the stored IndexBuilder value as a byte array.
|
||
|
func (b *IndexBuilder) Bytes() []byte { return (*bytes.Buffer)(b).Bytes() }
|
||
|
|
||
|
// Time is used to write the passed time into the IndexBuilder for use as a
|
||
|
// memdb index value.
|
||
|
func (b *IndexBuilder) Time(t time.Time) {
|
||
|
val := t.Unix()
|
||
|
buf := make([]byte, 8)
|
||
|
binary.BigEndian.PutUint64(buf, uint64(val))
|
||
|
(*bytes.Buffer)(b).Write(buf)
|
||
|
}
|