302 lines
9.4 KiB
Go
302 lines
9.4 KiB
Go
package kv_test
|
|
|
|
import (
|
|
"context"
|
|
"testing"
|
|
|
|
"github.com/influxdata/influxdb/v2"
|
|
"github.com/influxdata/influxdb/v2/kv"
|
|
"github.com/influxdata/influxdb/v2/kv/migration"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func TestIndexStore(t *testing.T) {
|
|
newStoreBase := func(resource string, bktName []byte, encKeyFn, encBodyFn kv.EncodeEntFn, decFn kv.DecodeBucketValFn, decToEntFn kv.ConvertValToEntFn) *kv.StoreBase {
|
|
return kv.NewStoreBase(resource, bktName, encKeyFn, encBodyFn, decFn, decToEntFn)
|
|
}
|
|
|
|
newFooIndexStore := func(t *testing.T, bktSuffix string) (*kv.IndexStore, func(), kv.Store) {
|
|
t.Helper()
|
|
|
|
kvStoreStore, done, err := NewTestBoltStore(t)
|
|
require.NoError(t, err)
|
|
|
|
const resource = "foo"
|
|
|
|
bucketName := []byte("foo_ent_" + bktSuffix)
|
|
indexBucketName := []byte("foo_idx+" + bktSuffix)
|
|
|
|
ctx := context.Background()
|
|
if err := migration.CreateBuckets("add foo buckets", bucketName, indexBucketName).Up(ctx, kvStoreStore); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
indexStore := &kv.IndexStore{
|
|
Resource: resource,
|
|
EntStore: newStoreBase(resource, bucketName, kv.EncIDKey, kv.EncBodyJSON, decJSONFooFn, decFooEntFn),
|
|
IndexStore: kv.NewOrgNameKeyStore(resource, indexBucketName, false),
|
|
}
|
|
|
|
return indexStore, done, kvStoreStore
|
|
}
|
|
|
|
t.Run("Put", func(t *testing.T) {
|
|
t.Run("basic", func(t *testing.T) {
|
|
indexStore, done, kvStore := newFooIndexStore(t, "put")
|
|
defer done()
|
|
|
|
expected := testPutBase(t, kvStore, indexStore, indexStore.EntStore.BktName)
|
|
|
|
key, err := indexStore.IndexStore.EntKey(context.TODO(), kv.Entity{
|
|
UniqueKey: kv.Encode(kv.EncID(expected.OrgID), kv.EncString(expected.Name)),
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
rawIndex := getEntRaw(t, kvStore, indexStore.IndexStore.BktName, key)
|
|
assert.Equal(t, encodeID(t, expected.ID), rawIndex)
|
|
})
|
|
|
|
t.Run("new entity", func(t *testing.T) {
|
|
indexStore, done, kvStore := newFooIndexStore(t, "put")
|
|
defer done()
|
|
|
|
expected := foo{ID: 3, OrgID: 33, Name: "333"}
|
|
update(t, kvStore, func(tx kv.Tx) error {
|
|
ent := newFooEnt(expected.ID, expected.OrgID, expected.Name)
|
|
return indexStore.Put(context.TODO(), tx, ent, kv.PutNew())
|
|
})
|
|
|
|
key, err := indexStore.IndexStore.EntKey(context.TODO(), kv.Entity{
|
|
UniqueKey: kv.Encode(kv.EncID(expected.OrgID), kv.EncString(expected.Name)),
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
rawIndex := getEntRaw(t, kvStore, indexStore.IndexStore.BktName, key)
|
|
assert.Equal(t, encodeID(t, expected.ID), rawIndex)
|
|
})
|
|
|
|
t.Run("updating entity that doesn't exist returns not found error", func(t *testing.T) {
|
|
indexStore, done, kvStore := newFooIndexStore(t, "put")
|
|
defer done()
|
|
|
|
expected := testPutBase(t, kvStore, indexStore, indexStore.EntStore.BktName)
|
|
|
|
err := kvStore.Update(context.Background(), func(tx kv.Tx) error {
|
|
ent := newFooEnt(33333, expected.OrgID, "safe name")
|
|
return indexStore.Put(context.TODO(), tx, ent, kv.PutUpdate())
|
|
})
|
|
require.Error(t, err)
|
|
assert.Equal(t, influxdb.ENotFound, influxdb.ErrorCode(err))
|
|
})
|
|
|
|
t.Run("updating entity with no naming collision succeeds", func(t *testing.T) {
|
|
indexStore, done, kvStore := newFooIndexStore(t, "put")
|
|
defer done()
|
|
|
|
expected := testPutBase(t, kvStore, indexStore, indexStore.EntStore.BktName)
|
|
|
|
update(t, kvStore, func(tx kv.Tx) error {
|
|
entCopy := newFooEnt(expected.ID, expected.OrgID, "safe name")
|
|
return indexStore.Put(context.TODO(), tx, entCopy, kv.PutUpdate())
|
|
})
|
|
|
|
err := kvStore.View(context.TODO(), func(tx kv.Tx) error {
|
|
_, err := indexStore.FindEnt(context.TODO(), tx, kv.Entity{
|
|
PK: kv.EncID(expected.ID),
|
|
UniqueKey: kv.Encode(kv.EncID(expected.OrgID), kv.EncString(expected.Name)),
|
|
})
|
|
return err
|
|
})
|
|
require.NoError(t, err)
|
|
})
|
|
|
|
t.Run("updating an existing entity to a new unique identifier should delete the existing unique key", func(t *testing.T) {
|
|
indexStore, done, kvStore := newFooIndexStore(t, "put")
|
|
defer done()
|
|
|
|
expected := testPutBase(t, kvStore, indexStore, indexStore.EntStore.BktName)
|
|
|
|
update(t, kvStore, func(tx kv.Tx) error {
|
|
entCopy := newFooEnt(expected.ID, expected.OrgID, "safe name")
|
|
return indexStore.Put(context.TODO(), tx, entCopy, kv.PutUpdate())
|
|
})
|
|
|
|
update(t, kvStore, func(tx kv.Tx) error {
|
|
ent := newFooEnt(33, expected.OrgID, expected.Name)
|
|
return indexStore.Put(context.TODO(), tx, ent, kv.PutNew())
|
|
})
|
|
})
|
|
|
|
t.Run("error cases", func(t *testing.T) {
|
|
t.Run("new entity conflicts with existing", func(t *testing.T) {
|
|
indexStore, done, kvStore := newFooIndexStore(t, "put")
|
|
defer done()
|
|
|
|
expected := testPutBase(t, kvStore, indexStore, indexStore.EntStore.BktName)
|
|
|
|
err := kvStore.Update(context.TODO(), func(tx kv.Tx) error {
|
|
entCopy := newFooEnt(expected.ID, expected.OrgID, expected.Name)
|
|
return indexStore.Put(context.TODO(), tx, entCopy, kv.PutNew())
|
|
})
|
|
require.Error(t, err)
|
|
assert.Equal(t, influxdb.EConflict, influxdb.ErrorCode(err))
|
|
})
|
|
|
|
t.Run("updating entity that does not exist", func(t *testing.T) {
|
|
indexStore, done, kvStore := newFooIndexStore(t, "put")
|
|
defer done()
|
|
|
|
expected := testPutBase(t, kvStore, indexStore, indexStore.EntStore.BktName)
|
|
|
|
update(t, kvStore, func(tx kv.Tx) error {
|
|
ent := newFooEnt(9000, expected.OrgID, "name1")
|
|
return indexStore.Put(context.TODO(), tx, ent, kv.PutNew())
|
|
})
|
|
|
|
err := kvStore.Update(context.TODO(), func(tx kv.Tx) error {
|
|
// ent by id does not exist
|
|
entCopy := newFooEnt(333, expected.OrgID, "name1")
|
|
return indexStore.Put(context.TODO(), tx, entCopy, kv.PutUpdate())
|
|
})
|
|
require.Error(t, err)
|
|
assert.Equal(t, influxdb.ENotFound, influxdb.ErrorCode(err), "got: "+err.Error())
|
|
})
|
|
|
|
t.Run("updating entity that does collides with an existing entity", func(t *testing.T) {
|
|
indexStore, done, kvStore := newFooIndexStore(t, "put")
|
|
defer done()
|
|
|
|
expected := testPutBase(t, kvStore, indexStore, indexStore.EntStore.BktName)
|
|
|
|
update(t, kvStore, func(tx kv.Tx) error {
|
|
ent := newFooEnt(9000, expected.OrgID, "name1")
|
|
return indexStore.Put(context.TODO(), tx, ent, kv.PutNew())
|
|
})
|
|
|
|
err := kvStore.Update(context.TODO(), func(tx kv.Tx) error {
|
|
// name conflicts
|
|
entCopy := newFooEnt(expected.ID, expected.OrgID, "name1")
|
|
return indexStore.Put(context.TODO(), tx, entCopy, kv.PutUpdate())
|
|
})
|
|
require.Error(t, err)
|
|
assert.Equal(t, influxdb.EConflict, influxdb.ErrorCode(err))
|
|
assert.Contains(t, err.Error(), "update conflicts")
|
|
})
|
|
})
|
|
})
|
|
|
|
t.Run("DeleteEnt", func(t *testing.T) {
|
|
indexStore, done, kvStore := newFooIndexStore(t, "delete_ent")
|
|
defer done()
|
|
|
|
expected := testDeleteEntBase(t, kvStore, indexStore)
|
|
|
|
err := kvStore.View(context.TODO(), func(tx kv.Tx) error {
|
|
_, err := indexStore.IndexStore.FindEnt(context.TODO(), tx, kv.Entity{
|
|
UniqueKey: expected.UniqueKey,
|
|
})
|
|
return err
|
|
})
|
|
isNotFoundErr(t, err)
|
|
})
|
|
|
|
t.Run("Delete", func(t *testing.T) {
|
|
fn := func(t *testing.T, suffix string) (storeBase, func(), kv.Store) {
|
|
return newFooIndexStore(t, suffix)
|
|
}
|
|
|
|
testDeleteBase(t, fn, func(t *testing.T, kvStore kv.Store, base storeBase, foosLeft []foo) {
|
|
var expectedIndexIDs []interface{}
|
|
for _, ent := range foosLeft {
|
|
expectedIndexIDs = append(expectedIndexIDs, ent.ID)
|
|
}
|
|
|
|
indexStore, ok := base.(*kv.IndexStore)
|
|
require.True(t, ok)
|
|
|
|
// next to verify they are not within the index store
|
|
var actualIDs []interface{}
|
|
view(t, kvStore, func(tx kv.Tx) error {
|
|
return indexStore.IndexStore.Find(context.TODO(), tx, kv.FindOpts{
|
|
CaptureFn: func(key []byte, decodedVal interface{}) error {
|
|
actualIDs = append(actualIDs, decodedVal)
|
|
return nil
|
|
},
|
|
})
|
|
})
|
|
|
|
assert.Equal(t, expectedIndexIDs, actualIDs)
|
|
})
|
|
})
|
|
|
|
t.Run("FindEnt", func(t *testing.T) {
|
|
t.Run("by ID", func(t *testing.T) {
|
|
base, done, kvStoreStore := newFooIndexStore(t, "find_ent")
|
|
defer done()
|
|
testFindEnt(t, kvStoreStore, base)
|
|
})
|
|
|
|
t.Run("find by name", func(t *testing.T) {
|
|
base, done, kvStore := newFooIndexStore(t, "find_ent")
|
|
defer done()
|
|
|
|
expected := newFooEnt(1, 9000, "foo_1")
|
|
seedEnts(t, kvStore, base, expected)
|
|
|
|
var actual interface{}
|
|
view(t, kvStore, func(tx kv.Tx) error {
|
|
f, err := base.FindEnt(context.TODO(), tx, kv.Entity{
|
|
UniqueKey: expected.UniqueKey,
|
|
})
|
|
actual = f
|
|
return err
|
|
})
|
|
|
|
assert.Equal(t, expected.Body, actual)
|
|
})
|
|
})
|
|
|
|
t.Run("Find", func(t *testing.T) {
|
|
t.Run("base", func(t *testing.T) {
|
|
fn := func(t *testing.T, suffix string) (storeBase, func(), kv.Store) {
|
|
return newFooIndexStore(t, suffix)
|
|
}
|
|
|
|
testFind(t, fn)
|
|
})
|
|
|
|
t.Run("with entity filter", func(t *testing.T) {
|
|
base, done, kvStore := newFooIndexStore(t, "find_index_search")
|
|
defer done()
|
|
|
|
expectedEnts := []kv.Entity{
|
|
newFooEnt(1, 9000, "foo_0"),
|
|
newFooEnt(2, 9001, "foo_1"),
|
|
newFooEnt(3, 9003, "foo_2"),
|
|
}
|
|
|
|
seedEnts(t, kvStore, base, expectedEnts...)
|
|
|
|
var actuals []interface{}
|
|
view(t, kvStore, func(tx kv.Tx) error {
|
|
return base.Find(context.TODO(), tx, kv.FindOpts{
|
|
FilterEntFn: func(key []byte, decodedVal interface{}) bool {
|
|
return decodedVal.(foo).ID < 3
|
|
},
|
|
CaptureFn: func(key []byte, decodedVal interface{}) error {
|
|
actuals = append(actuals, decodedVal)
|
|
return nil
|
|
},
|
|
})
|
|
})
|
|
|
|
expected := []interface{}{
|
|
expectedEnts[0].Body,
|
|
expectedEnts[1].Body,
|
|
}
|
|
assert.Equal(t, expected, actuals)
|
|
})
|
|
})
|
|
}
|