578 lines
14 KiB
Go
578 lines
14 KiB
Go
package kv
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
|
|
"github.com/influxdata/influxdb/v2"
|
|
"github.com/influxdata/influxdb/v2/kit/tracing"
|
|
)
|
|
|
|
type Entity struct {
|
|
PK EncodeFn
|
|
UniqueKey EncodeFn
|
|
|
|
Body interface{}
|
|
}
|
|
|
|
// EncodeEntFn encodes the entity. This is used both for the key and vals in the store base.
|
|
type EncodeEntFn func(ent Entity) ([]byte, string, error)
|
|
|
|
// EncIDKey encodes an entity into a key that represents the encoded ID provided.
|
|
func EncIDKey(ent Entity) ([]byte, string, error) {
|
|
if ent.PK == nil {
|
|
return nil, "ID", errors.New("no ID provided")
|
|
}
|
|
key, err := ent.PK()
|
|
return key, "ID", err
|
|
}
|
|
|
|
// EncUniqKey encodes the unique key.
|
|
func EncUniqKey(ent Entity) ([]byte, string, error) {
|
|
if ent.UniqueKey == nil {
|
|
return nil, "Unique Key", errors.New("no unique key provided")
|
|
}
|
|
key, err := ent.UniqueKey()
|
|
return key, "Unique Key", err
|
|
}
|
|
|
|
// EncBodyJSON JSON encodes the entity body and returns the raw bytes and indicates
|
|
// that it uses the entity body.
|
|
func EncBodyJSON(ent Entity) ([]byte, string, error) {
|
|
v, err := json.Marshal(ent.Body)
|
|
return v, "entity body", err
|
|
}
|
|
|
|
// DecodeBucketValFn decodes the raw []byte.
|
|
type DecodeBucketValFn func(key, val []byte) (keyRepeat []byte, decodedVal interface{}, err error)
|
|
|
|
// DecIndexID decodes the bucket val into an influxdb.ID.
|
|
func DecIndexID(key, val []byte) ([]byte, interface{}, error) {
|
|
var i influxdb.ID
|
|
return key, i, i.Decode(val)
|
|
}
|
|
|
|
// ConvertValToEntFn converts a key and decoded bucket value to an entity.
|
|
type ConvertValToEntFn func(k []byte, v interface{}) (Entity, error)
|
|
|
|
// DecodeOrgNameKey decodes a raw bucket key into the organization id and name
|
|
// used to create it.
|
|
func DecodeOrgNameKey(k []byte) (influxdb.ID, string, error) {
|
|
var orgID influxdb.ID
|
|
if err := orgID.Decode(k[:influxdb.IDLength]); err != nil {
|
|
return 0, "", err
|
|
}
|
|
return orgID, string(k[influxdb.IDLength:]), nil
|
|
}
|
|
|
|
// NewOrgNameKeyStore creates a store for an entity's unique index on organization id and name.
|
|
// This is used throughout the kv pkg here to provide an entity uniquness by name within an org.
|
|
func NewOrgNameKeyStore(resource string, bktName []byte, caseSensitive bool) *StoreBase {
|
|
var decValToEntFn ConvertValToEntFn = func(k []byte, v interface{}) (Entity, error) {
|
|
id, ok := v.(influxdb.ID)
|
|
if err := IsErrUnexpectedDecodeVal(ok); err != nil {
|
|
return Entity{}, err
|
|
}
|
|
|
|
ent := Entity{PK: EncID(id)}
|
|
if len(k) == 0 {
|
|
return ent, nil
|
|
}
|
|
|
|
orgID, name, err := DecodeOrgNameKey(k)
|
|
if err != nil {
|
|
return Entity{}, err
|
|
}
|
|
nameEnc := EncString(name)
|
|
if !caseSensitive {
|
|
nameEnc = EncStringCaseInsensitive(name)
|
|
}
|
|
ent.UniqueKey = Encode(EncID(orgID), nameEnc)
|
|
return ent, nil
|
|
}
|
|
|
|
return NewStoreBase(resource, bktName, EncUniqKey, EncIDKey, DecIndexID, decValToEntFn)
|
|
}
|
|
|
|
// StoreBase is the base behavior for accessing buckets in kv. It provides mechanisms that can
|
|
// be used in composing stores together (i.e. IndexStore).
|
|
type StoreBase struct {
|
|
Resource string
|
|
BktName []byte
|
|
|
|
EncodeEntKeyFn EncodeEntFn
|
|
EncodeEntBodyFn EncodeEntFn
|
|
DecodeEntFn DecodeBucketValFn
|
|
ConvertValToEntFn ConvertValToEntFn
|
|
}
|
|
|
|
// NewStoreBase creates a new store base.
|
|
func NewStoreBase(resource string, bktName []byte, encKeyFn, encBodyFn EncodeEntFn, decFn DecodeBucketValFn, decToEntFn ConvertValToEntFn) *StoreBase {
|
|
return &StoreBase{
|
|
Resource: resource,
|
|
BktName: bktName,
|
|
EncodeEntKeyFn: encKeyFn,
|
|
EncodeEntBodyFn: encBodyFn,
|
|
DecodeEntFn: decFn,
|
|
ConvertValToEntFn: decToEntFn,
|
|
}
|
|
}
|
|
|
|
// EntKey returns the key for the entity provided. This is a shortcut for grabbing the EntKey without
|
|
// having to juggle the encoding funcs.
|
|
func (s *StoreBase) EntKey(ctx context.Context, ent Entity) ([]byte, error) {
|
|
return s.encodeEnt(ctx, ent, s.EncodeEntKeyFn)
|
|
}
|
|
|
|
// Init creates the buckets.
|
|
func (s *StoreBase) Init(ctx context.Context, tx Tx) error {
|
|
span, ctx := tracing.StartSpanFromContextWithOperationName(ctx, "bucket_"+string(s.BktName))
|
|
defer span.Finish()
|
|
|
|
if _, err := s.bucket(ctx, tx); err != nil {
|
|
return &influxdb.Error{
|
|
Code: influxdb.EInternal,
|
|
Msg: fmt.Sprintf("failed to create bucket: %s", string(s.BktName)),
|
|
Err: err,
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
type (
|
|
// DeleteOpts provides indicators to the store.Delete call for deleting a given
|
|
// entity. The FilterFn indicates the current value should be deleted when returning
|
|
// true.
|
|
DeleteOpts struct {
|
|
DeleteRelationFns []DeleteRelationsFn
|
|
FilterFn FilterFn
|
|
}
|
|
|
|
// DeleteRelationsFn is a hook that a store that composes other stores can use to
|
|
// delete an entity and any relations it may share. An example would be deleting an
|
|
// an entity and its associated index.
|
|
DeleteRelationsFn func(key []byte, decodedVal interface{}) error
|
|
)
|
|
|
|
// Delete deletes entities by the provided options.
|
|
func (s *StoreBase) Delete(ctx context.Context, tx Tx, opts DeleteOpts) error {
|
|
span, ctx := tracing.StartSpanFromContext(ctx)
|
|
defer span.Finish()
|
|
|
|
if opts.FilterFn == nil {
|
|
return nil
|
|
}
|
|
|
|
findOpts := FindOpts{
|
|
CaptureFn: func(k []byte, v interface{}) error {
|
|
for _, deleteFn := range opts.DeleteRelationFns {
|
|
if err := deleteFn(k, v); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return s.bucketDelete(ctx, tx, k)
|
|
},
|
|
FilterEntFn: opts.FilterFn,
|
|
}
|
|
return s.Find(ctx, tx, findOpts)
|
|
}
|
|
|
|
// DeleteEnt deletes an entity.
|
|
func (s *StoreBase) DeleteEnt(ctx context.Context, tx Tx, ent Entity) error {
|
|
span, ctx := tracing.StartSpanFromContext(ctx)
|
|
defer span.Finish()
|
|
|
|
encodedID, err := s.EntKey(ctx, ent)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return s.bucketDelete(ctx, tx, encodedID)
|
|
}
|
|
|
|
type (
|
|
// FindOpts provided a means to search through the bucket. When a filter func
|
|
// is provided, that will run against the entity and if the filter responds true,
|
|
// will count it towards the number of entries seen and the capture func will be
|
|
// run with it provided to it.
|
|
FindOpts struct {
|
|
Descending bool
|
|
Offset int
|
|
Limit int
|
|
Prefix []byte
|
|
CaptureFn FindCaptureFn
|
|
FilterEntFn FilterFn
|
|
}
|
|
|
|
// FindCaptureFn is the mechanism for closing over the key and decoded value pair
|
|
// for adding results to the call sites collection. This generic implementation allows
|
|
// it to be reused. The returned decodedVal should always satisfy whatever decoding
|
|
// of the bucket value was set on the storeo that calls Find.
|
|
FindCaptureFn func(key []byte, decodedVal interface{}) error
|
|
|
|
// FilterFn will provide an indicator to the Find or Delete calls that the entity that
|
|
// was seen is one that is valid and should be either captured or deleted (depending on
|
|
// the caller of the filter func).
|
|
FilterFn func(key []byte, decodedVal interface{}) bool
|
|
)
|
|
|
|
// Find provides a mechanism for looking through the bucket via
|
|
// the set options. When a prefix is provided, the prefix is used to
|
|
// seek the bucket.
|
|
func (s *StoreBase) Find(ctx context.Context, tx Tx, opts FindOpts) error {
|
|
span, ctx := tracing.StartSpanFromContext(ctx)
|
|
defer span.Finish()
|
|
|
|
ctx, cancel := context.WithCancel(ctx)
|
|
defer cancel()
|
|
|
|
cur, err := s.bucketCursor(ctx, tx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
iter := &iterator{
|
|
cursor: cur,
|
|
descending: opts.Descending,
|
|
limit: opts.Limit,
|
|
offset: opts.Offset,
|
|
prefix: opts.Prefix,
|
|
decodeFn: s.DecodeEntFn,
|
|
filterFn: opts.FilterEntFn,
|
|
}
|
|
|
|
for k, v, err := iter.Next(ctx); k != nil; k, v, err = iter.Next(ctx) {
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := opts.CaptureFn(k, v); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// FindEnt returns the decoded entity body via the provided entity.
|
|
// An example entity should not include a Body, but rather the ID,
|
|
// Name, or OrgID.
|
|
func (s *StoreBase) FindEnt(ctx context.Context, tx Tx, ent Entity) (interface{}, error) {
|
|
span, ctx := tracing.StartSpanFromContext(ctx)
|
|
defer span.Finish()
|
|
|
|
encodedID, err := s.EntKey(ctx, ent)
|
|
if err != nil {
|
|
// TODO: fix this error up
|
|
return nil, err
|
|
}
|
|
|
|
body, err := s.bucketGet(ctx, tx, encodedID)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return s.decodeEnt(ctx, body)
|
|
}
|
|
|
|
type (
|
|
putOption struct {
|
|
isNew bool
|
|
isUpdate bool
|
|
}
|
|
|
|
// PutOptionFn provides a hint to the store to make some guarantees about the
|
|
// put action. I.e. If it is new, then will validate there is no existing entity
|
|
// by the given PK.
|
|
PutOptionFn func(o *putOption) error
|
|
)
|
|
|
|
// PutNew will create an entity that is not does not already exist. Guarantees uniqueness
|
|
// by the store's uniqueness guarantees.
|
|
func PutNew() PutOptionFn {
|
|
return func(o *putOption) error {
|
|
o.isNew = true
|
|
return nil
|
|
}
|
|
}
|
|
|
|
// PutUpdate will update an entity that must already exist.
|
|
func PutUpdate() PutOptionFn {
|
|
return func(o *putOption) error {
|
|
o.isUpdate = true
|
|
return nil
|
|
}
|
|
}
|
|
|
|
// Put will persist the entity.
|
|
func (s *StoreBase) Put(ctx context.Context, tx Tx, ent Entity, opts ...PutOptionFn) error {
|
|
span, ctx := tracing.StartSpanFromContext(ctx)
|
|
defer span.Finish()
|
|
|
|
var opt putOption
|
|
for _, o := range opts {
|
|
if err := o(&opt); err != nil {
|
|
return &influxdb.Error{
|
|
Code: influxdb.EConflict,
|
|
Err: err,
|
|
}
|
|
}
|
|
}
|
|
|
|
if err := s.putValidate(ctx, tx, ent, opt); err != nil {
|
|
return err
|
|
}
|
|
|
|
encodedID, err := s.EntKey(ctx, ent)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
body, err := s.encodeEnt(ctx, ent, s.EncodeEntBodyFn)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return s.bucketPut(ctx, tx, encodedID, body)
|
|
}
|
|
|
|
func (s *StoreBase) putValidate(ctx context.Context, tx Tx, ent Entity, opt putOption) error {
|
|
if !opt.isUpdate && !opt.isNew {
|
|
return nil
|
|
}
|
|
|
|
_, err := s.FindEnt(ctx, tx, ent)
|
|
if opt.isNew {
|
|
if err == nil || influxdb.ErrorCode(err) != influxdb.ENotFound {
|
|
return &influxdb.Error{
|
|
Code: influxdb.EConflict,
|
|
Msg: fmt.Sprintf("%s is not unique", s.Resource),
|
|
Err: err,
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
return err
|
|
}
|
|
|
|
func (s *StoreBase) bucket(ctx context.Context, tx Tx) (Bucket, error) {
|
|
span, _ := tracing.StartSpanFromContext(ctx)
|
|
defer span.Finish()
|
|
|
|
bkt, err := tx.Bucket(s.BktName)
|
|
if err != nil {
|
|
return nil, &influxdb.Error{
|
|
Code: influxdb.EInternal,
|
|
Msg: fmt.Sprintf("unexpected error retrieving bucket %q; Err %v", string(s.BktName), err),
|
|
Err: err,
|
|
}
|
|
}
|
|
return bkt, nil
|
|
}
|
|
|
|
func (s *StoreBase) bucketCursor(ctx context.Context, tx Tx) (Cursor, error) {
|
|
span, _ := tracing.StartSpanFromContext(ctx)
|
|
defer span.Finish()
|
|
|
|
b, err := s.bucket(ctx, tx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
cur, err := b.Cursor()
|
|
if err != nil {
|
|
return nil, &influxdb.Error{
|
|
Code: influxdb.EInternal,
|
|
Msg: fmt.Sprintf("failed to retrieve cursor"),
|
|
Err: err,
|
|
}
|
|
}
|
|
return cur, nil
|
|
}
|
|
|
|
func (s *StoreBase) bucketDelete(ctx context.Context, tx Tx, key []byte) error {
|
|
span, _ := tracing.StartSpanFromContext(ctx)
|
|
defer span.Finish()
|
|
|
|
b, err := s.bucket(ctx, tx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = b.Delete(key)
|
|
if err == nil {
|
|
return nil
|
|
}
|
|
|
|
iErr := &influxdb.Error{
|
|
Code: influxdb.EInternal,
|
|
Err: err,
|
|
}
|
|
if IsNotFound(err) {
|
|
iErr.Code = influxdb.ENotFound
|
|
iErr.Msg = fmt.Sprintf("%s does exist for key: %q", s.Resource, string(key))
|
|
}
|
|
return iErr
|
|
}
|
|
|
|
func (s *StoreBase) bucketGet(ctx context.Context, tx Tx, key []byte) ([]byte, error) {
|
|
span, _ := tracing.StartSpanFromContext(ctx)
|
|
defer span.Finish()
|
|
|
|
b, err := s.bucket(ctx, tx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
body, err := b.Get(key)
|
|
if IsNotFound(err) {
|
|
return nil, &influxdb.Error{
|
|
Code: influxdb.ENotFound,
|
|
Msg: fmt.Sprintf("%s not found for key %q", s.Resource, string(key)),
|
|
}
|
|
}
|
|
if err != nil {
|
|
return nil, &influxdb.Error{
|
|
Code: influxdb.EInternal,
|
|
Err: err,
|
|
}
|
|
}
|
|
|
|
return body, nil
|
|
}
|
|
|
|
func (s *StoreBase) bucketPut(ctx context.Context, tx Tx, key, body []byte) error {
|
|
span, _ := tracing.StartSpanFromContext(ctx)
|
|
defer span.Finish()
|
|
|
|
b, err := s.bucket(ctx, tx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := b.Put(key, body); err != nil {
|
|
return &influxdb.Error{
|
|
Code: influxdb.EInternal,
|
|
Err: err,
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (s *StoreBase) decodeEnt(ctx context.Context, body []byte) (interface{}, error) {
|
|
span, _ := tracing.StartSpanFromContext(ctx)
|
|
defer span.Finish()
|
|
|
|
_, v, err := s.DecodeEntFn([]byte{}, body) // ignore key here
|
|
if err != nil {
|
|
return nil, &influxdb.Error{
|
|
Code: influxdb.EInternal,
|
|
Msg: fmt.Sprintf("failed to decode %s body", s.Resource),
|
|
Err: err,
|
|
}
|
|
}
|
|
return v, nil
|
|
}
|
|
|
|
func (s *StoreBase) encodeEnt(ctx context.Context, ent Entity, fn EncodeEntFn) ([]byte, error) {
|
|
span, _ := tracing.StartSpanFromContext(ctx)
|
|
defer span.Finish()
|
|
|
|
if fn == nil {
|
|
return nil, &influxdb.Error{
|
|
Code: influxdb.EInvalid,
|
|
Msg: fmt.Sprintf("no key was provided for %s", s.Resource),
|
|
}
|
|
}
|
|
|
|
encoded, field, err := fn(ent)
|
|
if err != nil {
|
|
return encoded, &influxdb.Error{
|
|
Code: influxdb.EInvalid,
|
|
Msg: fmt.Sprintf("provided %s %s is an invalid format", s.Resource, field),
|
|
Err: err,
|
|
}
|
|
}
|
|
return encoded, nil
|
|
}
|
|
|
|
type iterator struct {
|
|
cursor Cursor
|
|
|
|
counter int
|
|
descending bool
|
|
limit int
|
|
offset int
|
|
prefix []byte
|
|
|
|
nextFn func() (key, val []byte)
|
|
|
|
decodeFn func(key, val []byte) (k []byte, decodedVal interface{}, err error)
|
|
filterFn FilterFn
|
|
}
|
|
|
|
func (i *iterator) Next(ctx context.Context) (key []byte, val interface{}, err error) {
|
|
span, _ := tracing.StartSpanFromContext(ctx)
|
|
defer span.Finish()
|
|
|
|
if i.limit > 0 && i.counter >= i.limit+i.offset {
|
|
return nil, nil, nil
|
|
}
|
|
|
|
var k, vRaw []byte
|
|
switch {
|
|
case i.nextFn != nil:
|
|
k, vRaw = i.nextFn()
|
|
case len(i.prefix) > 0:
|
|
k, vRaw = i.cursor.Seek(i.prefix)
|
|
i.nextFn = i.cursor.Next
|
|
case i.descending:
|
|
k, vRaw = i.cursor.Last()
|
|
i.nextFn = i.cursor.Prev
|
|
default:
|
|
k, vRaw = i.cursor.First()
|
|
i.nextFn = i.cursor.Next
|
|
}
|
|
|
|
k, decodedVal, err := i.decodeFn(k, vRaw)
|
|
for ; ; k, decodedVal, err = i.decodeFn(i.nextFn()) {
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
if i.isNext(k, decodedVal) {
|
|
break
|
|
}
|
|
}
|
|
return k, decodedVal, nil
|
|
}
|
|
|
|
func (i *iterator) isNext(k []byte, v interface{}) bool {
|
|
if len(k) == 0 {
|
|
return true
|
|
}
|
|
|
|
if i.filterFn != nil && !i.filterFn(k, v) {
|
|
return false
|
|
}
|
|
|
|
// increase counter here since the entity is a valid ent
|
|
// and counts towards the total the user is looking for
|
|
// i.e. limit = 5 => 5 valid ents
|
|
// i.e. offset = 5 => return valid ents after seeing 5 valid ents
|
|
i.counter++
|
|
|
|
if i.limit > 0 && i.counter >= i.limit+i.offset {
|
|
return true
|
|
}
|
|
if i.offset > 0 && i.counter <= i.offset {
|
|
return false
|
|
}
|
|
return true
|
|
}
|
|
|
|
func IsErrUnexpectedDecodeVal(ok bool) error {
|
|
if ok {
|
|
return nil
|
|
}
|
|
return errors.New("unexpected value decoded")
|
|
}
|