2021-10-29 12:12:44 +08:00
|
|
|
// Licensed to the LF AI & Data foundation under one
|
|
|
|
// or more contributor license agreements. See the NOTICE file
|
|
|
|
// distributed with this work for additional information
|
|
|
|
// regarding copyright ownership. The ASF licenses this file
|
|
|
|
// to you under the Apache License, Version 2.0 (the
|
|
|
|
// "License"); you may not use this file except in compliance
|
2021-08-24 09:45:51 +08:00
|
|
|
// with the License. You may obtain a copy of the License at
|
|
|
|
//
|
2021-10-29 12:12:44 +08:00
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
2021-08-24 09:45:51 +08:00
|
|
|
//
|
2021-10-29 12:12:44 +08:00
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
2021-08-24 09:45:51 +08:00
|
|
|
|
|
|
|
package etcdkv_test
|
|
|
|
|
|
|
|
import (
|
2022-03-07 18:01:58 +08:00
|
|
|
"errors"
|
2022-04-28 11:17:47 +08:00
|
|
|
"io/ioutil"
|
2021-08-24 09:45:51 +08:00
|
|
|
"os"
|
|
|
|
"testing"
|
2022-03-25 11:03:25 +08:00
|
|
|
"time"
|
2021-08-24 09:45:51 +08:00
|
|
|
|
2022-03-07 18:01:58 +08:00
|
|
|
"github.com/milvus-io/milvus/internal/kv"
|
2021-09-03 19:35:47 +08:00
|
|
|
"github.com/milvus-io/milvus/internal/util/metricsinfo"
|
|
|
|
|
2021-08-24 09:45:51 +08:00
|
|
|
embed_etcd_kv "github.com/milvus-io/milvus/internal/kv/etcd"
|
|
|
|
"github.com/milvus-io/milvus/internal/util/paramtable"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
clientv3 "go.etcd.io/etcd/client/v3"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestEmbedEtcd(te *testing.T) {
|
2021-09-03 19:35:47 +08:00
|
|
|
os.Setenv(metricsinfo.DeployModeEnvKey, metricsinfo.StandaloneDeployMode)
|
2022-02-08 20:57:47 +08:00
|
|
|
param := new(paramtable.ServiceParam)
|
2021-08-24 09:45:51 +08:00
|
|
|
param.Init()
|
|
|
|
param.BaseTable.Save("etcd.use.embed", "true")
|
2021-08-27 10:13:57 +08:00
|
|
|
param.BaseTable.Save("etcd.config.path", "../../../configs/advanced/etcd.yaml")
|
2022-04-28 11:17:47 +08:00
|
|
|
|
|
|
|
dir, err := ioutil.TempDir(os.TempDir(), "kv_etcd")
|
|
|
|
assert.Nil(te, err)
|
|
|
|
param.BaseTable.Save("etcd.data.dir", dir)
|
|
|
|
|
2022-02-07 10:09:45 +08:00
|
|
|
param.EtcdCfg.LoadCfgToMemory()
|
2022-04-28 11:17:47 +08:00
|
|
|
|
2021-08-24 09:45:51 +08:00
|
|
|
//clean up data
|
|
|
|
defer func() {
|
2022-04-28 11:17:47 +08:00
|
|
|
err = os.RemoveAll(dir)
|
2021-08-24 09:45:51 +08:00
|
|
|
}()
|
2022-04-28 11:17:47 +08:00
|
|
|
|
2021-08-24 09:45:51 +08:00
|
|
|
te.Run("EtcdKV SaveAndLoad", func(t *testing.T) {
|
|
|
|
rootPath := "/etcd/test/root/saveandload"
|
2022-02-07 10:09:45 +08:00
|
|
|
metaKv, err := embed_etcd_kv.NewMetaKvFactory(rootPath, ¶m.EtcdCfg)
|
2021-08-24 09:45:51 +08:00
|
|
|
require.NoError(te, err)
|
|
|
|
assert.NotNil(te, metaKv)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
defer metaKv.Close()
|
|
|
|
defer metaKv.RemoveWithPrefix("")
|
|
|
|
|
|
|
|
saveAndLoadTests := []struct {
|
|
|
|
key string
|
|
|
|
value string
|
|
|
|
}{
|
|
|
|
{"test1", "value1"},
|
|
|
|
{"test2", "value2"},
|
|
|
|
{"test1/a", "value_a"},
|
|
|
|
{"test1/b", "value_b"},
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, test := range saveAndLoadTests {
|
|
|
|
if i < 4 {
|
|
|
|
err = metaKv.Save(test.key, test.value)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
val, err := metaKv.Load(test.key)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, test.value, val)
|
|
|
|
}
|
|
|
|
|
|
|
|
invalidLoadTests := []struct {
|
|
|
|
invalidKey string
|
|
|
|
}{
|
|
|
|
{"t"},
|
|
|
|
{"a"},
|
|
|
|
{"test1a"},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range invalidLoadTests {
|
|
|
|
val, err := metaKv.Load(test.invalidKey)
|
|
|
|
assert.Error(t, err)
|
|
|
|
assert.Zero(t, val)
|
|
|
|
}
|
|
|
|
|
|
|
|
loadPrefixTests := []struct {
|
|
|
|
prefix string
|
|
|
|
|
|
|
|
expectedKeys []string
|
|
|
|
expectedValues []string
|
|
|
|
expectedError error
|
|
|
|
}{
|
|
|
|
{"test", []string{
|
|
|
|
metaKv.GetPath("test1"),
|
|
|
|
metaKv.GetPath("test2"),
|
|
|
|
metaKv.GetPath("test1/a"),
|
|
|
|
metaKv.GetPath("test1/b")}, []string{"value1", "value2", "value_a", "value_b"}, nil},
|
|
|
|
{"test1", []string{
|
|
|
|
metaKv.GetPath("test1"),
|
|
|
|
metaKv.GetPath("test1/a"),
|
|
|
|
metaKv.GetPath("test1/b")}, []string{"value1", "value_a", "value_b"}, nil},
|
|
|
|
{"test2", []string{metaKv.GetPath("test2")}, []string{"value2"}, nil},
|
|
|
|
{"", []string{
|
|
|
|
metaKv.GetPath("test1"),
|
|
|
|
metaKv.GetPath("test2"),
|
|
|
|
metaKv.GetPath("test1/a"),
|
|
|
|
metaKv.GetPath("test1/b")}, []string{"value1", "value2", "value_a", "value_b"}, nil},
|
|
|
|
{"test1/a", []string{metaKv.GetPath("test1/a")}, []string{"value_a"}, nil},
|
|
|
|
{"a", []string{}, []string{}, nil},
|
|
|
|
{"root", []string{}, []string{}, nil},
|
|
|
|
{"/etcd/test/root", []string{}, []string{}, nil},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range loadPrefixTests {
|
|
|
|
actualKeys, actualValues, err := metaKv.LoadWithPrefix(test.prefix)
|
|
|
|
assert.ElementsMatch(t, test.expectedKeys, actualKeys)
|
|
|
|
assert.ElementsMatch(t, test.expectedValues, actualValues)
|
|
|
|
assert.Equal(t, test.expectedError, err)
|
|
|
|
|
|
|
|
actualKeys, actualValues, versions, err := metaKv.LoadWithPrefix2(test.prefix)
|
|
|
|
assert.ElementsMatch(t, test.expectedKeys, actualKeys)
|
|
|
|
assert.ElementsMatch(t, test.expectedValues, actualValues)
|
|
|
|
assert.NotZero(t, versions)
|
|
|
|
assert.Equal(t, test.expectedError, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
removeTests := []struct {
|
|
|
|
validKey string
|
|
|
|
invalidKey string
|
|
|
|
}{
|
|
|
|
{"test1", "abc"},
|
|
|
|
{"test1/a", "test1/lskfjal"},
|
|
|
|
{"test1/b", "test1/b"},
|
|
|
|
{"test2", "-"},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range removeTests {
|
|
|
|
err = metaKv.Remove(test.validKey)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
_, err = metaKv.Load(test.validKey)
|
|
|
|
assert.Error(t, err)
|
|
|
|
|
|
|
|
err = metaKv.Remove(test.validKey)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
err = metaKv.Remove(test.invalidKey)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
2022-03-04 15:14:01 +08:00
|
|
|
te.Run("EtcdKV SaveAndLoadBytes", func(t *testing.T) {
|
|
|
|
rootPath := "/etcd/test/root/saveandloadbytes"
|
|
|
|
_metaKv, err := embed_etcd_kv.NewMetaKvFactory(rootPath, ¶m.EtcdCfg)
|
|
|
|
metaKv := _metaKv.(*embed_etcd_kv.EmbedEtcdKV)
|
|
|
|
require.NoError(te, err)
|
|
|
|
assert.NotNil(te, metaKv)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
defer metaKv.Close()
|
|
|
|
defer metaKv.RemoveWithPrefix("")
|
|
|
|
|
|
|
|
saveAndLoadTests := []struct {
|
|
|
|
key string
|
|
|
|
value []byte
|
|
|
|
}{
|
|
|
|
{"test1", []byte("value1")},
|
|
|
|
{"test2", []byte("value2")},
|
|
|
|
{"test1/a", []byte("value_a")},
|
|
|
|
{"test1/b", []byte("value_b")},
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, test := range saveAndLoadTests {
|
|
|
|
if i < 4 {
|
|
|
|
err = metaKv.SaveBytes(test.key, test.value)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
val, err := metaKv.LoadBytes(test.key)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, test.value, val)
|
|
|
|
}
|
|
|
|
|
|
|
|
invalidLoadTests := []struct {
|
|
|
|
invalidKey string
|
|
|
|
}{
|
|
|
|
{"t"},
|
|
|
|
{"a"},
|
|
|
|
{"test1a"},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range invalidLoadTests {
|
|
|
|
val, err := metaKv.LoadBytes(test.invalidKey)
|
|
|
|
assert.Error(t, err)
|
|
|
|
assert.Zero(t, val)
|
|
|
|
}
|
|
|
|
|
|
|
|
loadPrefixTests := []struct {
|
|
|
|
prefix string
|
|
|
|
|
|
|
|
expectedKeys []string
|
|
|
|
expectedValues [][]byte
|
|
|
|
expectedError error
|
|
|
|
}{
|
|
|
|
{"test", []string{
|
|
|
|
metaKv.GetPath("test1"),
|
|
|
|
metaKv.GetPath("test2"),
|
|
|
|
metaKv.GetPath("test1/a"),
|
|
|
|
metaKv.GetPath("test1/b")}, [][]byte{[]byte("value1"), []byte("value2"), []byte("value_a"), []byte("value_b")}, nil},
|
|
|
|
{"test1", []string{
|
|
|
|
metaKv.GetPath("test1"),
|
|
|
|
metaKv.GetPath("test1/a"),
|
|
|
|
metaKv.GetPath("test1/b")}, [][]byte{[]byte("value1"), []byte("value_a"), []byte("value_b")}, nil},
|
|
|
|
{"test2", []string{metaKv.GetPath("test2")}, [][]byte{[]byte("value2")}, nil},
|
|
|
|
{"", []string{
|
|
|
|
metaKv.GetPath("test1"),
|
|
|
|
metaKv.GetPath("test2"),
|
|
|
|
metaKv.GetPath("test1/a"),
|
|
|
|
metaKv.GetPath("test1/b")}, [][]byte{[]byte("value1"), []byte("value2"), []byte("value_a"), []byte("value_b")}, nil},
|
|
|
|
{"test1/a", []string{metaKv.GetPath("test1/a")}, [][]byte{[]byte("value_a")}, nil},
|
|
|
|
{"a", []string{}, [][]byte{}, nil},
|
|
|
|
{"root", []string{}, [][]byte{}, nil},
|
|
|
|
{"/etcd/test/root", []string{}, [][]byte{}, nil},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range loadPrefixTests {
|
|
|
|
actualKeys, actualValues, err := metaKv.LoadBytesWithPrefix(test.prefix)
|
|
|
|
assert.ElementsMatch(t, test.expectedKeys, actualKeys)
|
|
|
|
assert.ElementsMatch(t, test.expectedValues, actualValues)
|
|
|
|
assert.Equal(t, test.expectedError, err)
|
|
|
|
|
|
|
|
actualKeys, actualValues, versions, err := metaKv.LoadBytesWithPrefix2(test.prefix)
|
|
|
|
assert.ElementsMatch(t, test.expectedKeys, actualKeys)
|
|
|
|
assert.ElementsMatch(t, test.expectedValues, actualValues)
|
|
|
|
assert.NotZero(t, versions)
|
|
|
|
assert.Equal(t, test.expectedError, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
removeTests := []struct {
|
|
|
|
validKey string
|
|
|
|
invalidKey string
|
|
|
|
}{
|
|
|
|
{"test1", "abc"},
|
|
|
|
{"test1/a", "test1/lskfjal"},
|
|
|
|
{"test1/b", "test1/b"},
|
|
|
|
{"test2", "-"},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range removeTests {
|
|
|
|
err = metaKv.Remove(test.validKey)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
_, err = metaKv.Load(test.validKey)
|
|
|
|
assert.Error(t, err)
|
|
|
|
|
|
|
|
err = metaKv.Remove(test.validKey)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
err = metaKv.Remove(test.invalidKey)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
2021-08-24 09:45:51 +08:00
|
|
|
te.Run("EtcdKV LoadWithRevision", func(t *testing.T) {
|
|
|
|
rootPath := "/etcd/test/root/LoadWithRevision"
|
2022-02-07 10:09:45 +08:00
|
|
|
metaKv, err := embed_etcd_kv.NewMetaKvFactory(rootPath, ¶m.EtcdCfg)
|
2021-08-24 09:45:51 +08:00
|
|
|
assert.Nil(t, err)
|
|
|
|
|
|
|
|
defer metaKv.Close()
|
|
|
|
defer metaKv.RemoveWithPrefix("")
|
|
|
|
|
|
|
|
prepareKV := []struct {
|
|
|
|
inKey string
|
|
|
|
inValue string
|
|
|
|
}{
|
|
|
|
{"a", "a_version1"},
|
|
|
|
{"b", "b_version2"},
|
|
|
|
{"a", "a_version3"},
|
|
|
|
{"c", "c_version4"},
|
|
|
|
{"a/suba", "a_version5"},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range prepareKV {
|
|
|
|
err = metaKv.Save(test.inKey, test.inValue)
|
|
|
|
require.NoError(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
loadWithRevisionTests := []struct {
|
|
|
|
inKey string
|
|
|
|
|
|
|
|
expectedKeyNo int
|
|
|
|
expectedValues []string
|
|
|
|
}{
|
|
|
|
{"a", 2, []string{"a_version3", "a_version5"}},
|
|
|
|
{"b", 1, []string{"b_version2"}},
|
|
|
|
{"c", 1, []string{"c_version4"}},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range loadWithRevisionTests {
|
|
|
|
keys, values, revision, err := metaKv.LoadWithRevision(test.inKey)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, test.expectedKeyNo, len(keys))
|
|
|
|
assert.ElementsMatch(t, test.expectedValues, values)
|
|
|
|
assert.NotZero(t, revision)
|
|
|
|
}
|
|
|
|
|
|
|
|
})
|
|
|
|
|
2022-03-04 15:14:01 +08:00
|
|
|
te.Run("EtcdKV LoadBytesWithRevision", func(t *testing.T) {
|
|
|
|
rootPath := "/etcd/test/root/LoadBytesWithRevision"
|
|
|
|
_metaKv, err := embed_etcd_kv.NewMetaKvFactory(rootPath, ¶m.EtcdCfg)
|
|
|
|
metaKv := _metaKv.(*embed_etcd_kv.EmbedEtcdKV)
|
|
|
|
assert.Nil(t, err)
|
|
|
|
|
|
|
|
defer metaKv.Close()
|
|
|
|
defer metaKv.RemoveWithPrefix("")
|
|
|
|
|
|
|
|
prepareKV := []struct {
|
|
|
|
inKey string
|
|
|
|
inValue []byte
|
|
|
|
}{
|
|
|
|
{"a", []byte("a_version1")},
|
|
|
|
{"b", []byte("b_version2")},
|
|
|
|
{"a", []byte("a_version3")},
|
|
|
|
{"c", []byte("c_version4")},
|
|
|
|
{"a/suba", []byte("a_version5")},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range prepareKV {
|
|
|
|
err = metaKv.SaveBytes(test.inKey, test.inValue)
|
|
|
|
require.NoError(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
loadWithRevisionTests := []struct {
|
|
|
|
inKey string
|
|
|
|
|
|
|
|
expectedKeyNo int
|
|
|
|
expectedValues [][]byte
|
|
|
|
}{
|
|
|
|
{"a", 2, [][]byte{[]byte("a_version3"), []byte("a_version5")}},
|
|
|
|
{"b", 1, [][]byte{[]byte("b_version2")}},
|
|
|
|
{"c", 1, [][]byte{[]byte("c_version4")}},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range loadWithRevisionTests {
|
|
|
|
keys, values, revision, err := metaKv.LoadBytesWithRevision(test.inKey)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, test.expectedKeyNo, len(keys))
|
|
|
|
assert.ElementsMatch(t, test.expectedValues, values)
|
|
|
|
assert.NotZero(t, revision)
|
|
|
|
}
|
|
|
|
|
|
|
|
})
|
|
|
|
|
2021-08-24 09:45:51 +08:00
|
|
|
te.Run("EtcdKV MultiSaveAndMultiLoad", func(t *testing.T) {
|
|
|
|
rootPath := "/etcd/test/root/multi_save_and_multi_load"
|
2022-02-07 10:09:45 +08:00
|
|
|
metaKv, err := embed_etcd_kv.NewMetaKvFactory(rootPath, ¶m.EtcdCfg)
|
2021-08-24 09:45:51 +08:00
|
|
|
assert.Nil(t, err)
|
|
|
|
|
|
|
|
defer metaKv.Close()
|
|
|
|
defer metaKv.RemoveWithPrefix("")
|
|
|
|
|
|
|
|
multiSaveTests := map[string]string{
|
|
|
|
"key_1": "value_1",
|
|
|
|
"key_2": "value_2",
|
|
|
|
"key_3/a": "value_3a",
|
|
|
|
"multikey_1": "multivalue_1",
|
|
|
|
"multikey_2": "multivalue_2",
|
|
|
|
"_": "other",
|
|
|
|
}
|
|
|
|
|
|
|
|
err = metaKv.MultiSave(multiSaveTests)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
for k, v := range multiSaveTests {
|
|
|
|
actualV, err := metaKv.Load(k)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, v, actualV)
|
|
|
|
}
|
|
|
|
|
|
|
|
multiLoadTests := []struct {
|
|
|
|
inputKeys []string
|
|
|
|
expectedValues []string
|
|
|
|
}{
|
|
|
|
{[]string{"key_1"}, []string{"value_1"}},
|
|
|
|
{[]string{"key_1", "key_2", "key_3/a"}, []string{"value_1", "value_2", "value_3a"}},
|
|
|
|
{[]string{"multikey_1", "multikey_2"}, []string{"multivalue_1", "multivalue_2"}},
|
|
|
|
{[]string{"_"}, []string{"other"}},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range multiLoadTests {
|
|
|
|
vs, err := metaKv.MultiLoad(test.inputKeys)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, test.expectedValues, vs)
|
|
|
|
}
|
|
|
|
|
|
|
|
invalidMultiLoad := []struct {
|
|
|
|
invalidKeys []string
|
|
|
|
expectedValues []string
|
|
|
|
}{
|
|
|
|
{[]string{"a", "key_1"}, []string{"", "value_1"}},
|
|
|
|
{[]string{".....", "key_1"}, []string{"", "value_1"}},
|
|
|
|
{[]string{"*********"}, []string{""}},
|
|
|
|
{[]string{"key_1", "1"}, []string{"value_1", ""}},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range invalidMultiLoad {
|
|
|
|
vs, err := metaKv.MultiLoad(test.invalidKeys)
|
|
|
|
assert.Error(t, err)
|
|
|
|
assert.Equal(t, test.expectedValues, vs)
|
|
|
|
}
|
|
|
|
|
|
|
|
removeWithPrefixTests := []string{
|
|
|
|
"key_1",
|
|
|
|
"multi",
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, k := range removeWithPrefixTests {
|
|
|
|
err = metaKv.RemoveWithPrefix(k)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
ks, vs, err := metaKv.LoadWithPrefix(k)
|
|
|
|
assert.Empty(t, ks)
|
|
|
|
assert.Empty(t, vs)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
multiRemoveTests := []string{
|
|
|
|
"key_2",
|
|
|
|
"key_3/a",
|
|
|
|
"multikey_2",
|
|
|
|
"_",
|
|
|
|
}
|
|
|
|
|
|
|
|
err = metaKv.MultiRemove(multiRemoveTests)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
ks, vs, err := metaKv.LoadWithPrefix("")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Empty(t, ks)
|
|
|
|
assert.Empty(t, vs)
|
|
|
|
|
|
|
|
multiSaveAndRemoveTests := []struct {
|
|
|
|
multiSaves map[string]string
|
|
|
|
multiRemoves []string
|
|
|
|
}{
|
|
|
|
{map[string]string{"key_1": "value_1"}, []string{}},
|
|
|
|
{map[string]string{"key_2": "value_2"}, []string{"key_1"}},
|
|
|
|
{map[string]string{"key_3/a": "value_3a"}, []string{"key_2"}},
|
|
|
|
{map[string]string{"multikey_1": "multivalue_1"}, []string{}},
|
|
|
|
{map[string]string{"multikey_2": "multivalue_2"}, []string{"multikey_1", "key_3/a"}},
|
|
|
|
{make(map[string]string), []string{"multikey_2"}},
|
|
|
|
}
|
|
|
|
for _, test := range multiSaveAndRemoveTests {
|
|
|
|
err = metaKv.MultiSaveAndRemove(test.multiSaves, test.multiRemoves)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
ks, vs, err = metaKv.LoadWithPrefix("")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Empty(t, ks)
|
|
|
|
assert.Empty(t, vs)
|
|
|
|
})
|
|
|
|
|
2022-03-04 15:14:01 +08:00
|
|
|
te.Run("EtcdKV MultiSaveAndMultiLoadBytes", func(t *testing.T) {
|
|
|
|
rootPath := "/etcd/test/root/multi_save_and_multi_load"
|
|
|
|
_metaKv, err := embed_etcd_kv.NewMetaKvFactory(rootPath, ¶m.EtcdCfg)
|
|
|
|
metaKv := _metaKv.(*embed_etcd_kv.EmbedEtcdKV)
|
|
|
|
assert.Nil(t, err)
|
|
|
|
|
|
|
|
defer metaKv.Close()
|
|
|
|
defer metaKv.RemoveWithPrefix("")
|
|
|
|
|
|
|
|
multiSaveTests := map[string][]byte{
|
|
|
|
"key_1": []byte("value_1"),
|
|
|
|
"key_2": []byte("value_2"),
|
|
|
|
"key_3/a": []byte("value_3a"),
|
|
|
|
"multikey_1": []byte("multivalue_1"),
|
|
|
|
"multikey_2": []byte("multivalue_2"),
|
|
|
|
"_": []byte("other"),
|
|
|
|
}
|
|
|
|
|
|
|
|
err = metaKv.MultiSaveBytes(multiSaveTests)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
for k, v := range multiSaveTests {
|
|
|
|
actualV, err := metaKv.LoadBytes(k)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, v, actualV)
|
|
|
|
}
|
|
|
|
|
|
|
|
multiLoadTests := []struct {
|
|
|
|
inputKeys []string
|
|
|
|
expectedValues [][]byte
|
|
|
|
}{
|
|
|
|
{[]string{"key_1"}, [][]byte{[]byte("value_1")}},
|
|
|
|
{[]string{"key_1", "key_2", "key_3/a"}, [][]byte{[]byte("value_1"), []byte("value_2"), []byte("value_3a")}},
|
|
|
|
{[]string{"multikey_1", "multikey_2"}, [][]byte{[]byte("multivalue_1"), []byte("multivalue_2")}},
|
|
|
|
{[]string{"_"}, [][]byte{[]byte("other")}},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range multiLoadTests {
|
|
|
|
vs, err := metaKv.MultiLoadBytes(test.inputKeys)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, test.expectedValues, vs)
|
|
|
|
}
|
|
|
|
|
|
|
|
invalidMultiLoad := []struct {
|
|
|
|
invalidKeys []string
|
|
|
|
expectedValues [][]byte
|
|
|
|
}{
|
|
|
|
{[]string{"a", "key_1"}, [][]byte{[]byte(""), []byte("value_1")}},
|
|
|
|
{[]string{".....", "key_1"}, [][]byte{[]byte(""), []byte("value_1")}},
|
|
|
|
{[]string{"*********"}, [][]byte{[]byte("")}},
|
|
|
|
{[]string{"key_1", "1"}, [][]byte{[]byte("value_1"), []byte("")}},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range invalidMultiLoad {
|
|
|
|
vs, err := metaKv.MultiLoadBytes(test.invalidKeys)
|
|
|
|
assert.Error(t, err)
|
|
|
|
assert.Equal(t, test.expectedValues, vs)
|
|
|
|
}
|
|
|
|
|
|
|
|
removeWithPrefixTests := []string{
|
|
|
|
"key_1",
|
|
|
|
"multi",
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, k := range removeWithPrefixTests {
|
|
|
|
err = metaKv.RemoveWithPrefix(k)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
ks, vs, err := metaKv.LoadBytesWithPrefix(k)
|
|
|
|
assert.Empty(t, ks)
|
|
|
|
assert.Empty(t, vs)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
multiRemoveTests := []string{
|
|
|
|
"key_2",
|
|
|
|
"key_3/a",
|
|
|
|
"multikey_2",
|
|
|
|
"_",
|
|
|
|
}
|
|
|
|
|
|
|
|
err = metaKv.MultiRemove(multiRemoveTests)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
ks, vs, err := metaKv.LoadBytesWithPrefix("")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Empty(t, ks)
|
|
|
|
assert.Empty(t, vs)
|
|
|
|
|
|
|
|
multiSaveAndRemoveTests := []struct {
|
|
|
|
multiSaves map[string][]byte
|
|
|
|
multiRemoves []string
|
|
|
|
}{
|
|
|
|
{map[string][]byte{"key_1": []byte("value_1")}, []string{}},
|
|
|
|
{map[string][]byte{"key_2": []byte("value_2")}, []string{"key_1"}},
|
|
|
|
{map[string][]byte{"key_3/a": []byte("value_3a")}, []string{"key_2"}},
|
|
|
|
{map[string][]byte{"multikey_1": []byte("multivalue_1")}, []string{}},
|
|
|
|
{map[string][]byte{"multikey_2": []byte("multivalue_2")}, []string{"multikey_1", "key_3/a"}},
|
|
|
|
{map[string][]byte{}, []string{"multikey_2"}},
|
|
|
|
}
|
|
|
|
for _, test := range multiSaveAndRemoveTests {
|
|
|
|
err = metaKv.MultiSaveBytesAndRemove(test.multiSaves, test.multiRemoves)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
ks, vs, err = metaKv.LoadBytesWithPrefix("")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Empty(t, ks)
|
|
|
|
assert.Empty(t, vs)
|
|
|
|
})
|
|
|
|
|
2021-08-24 09:45:51 +08:00
|
|
|
te.Run("EtcdKV MultiRemoveWithPrefix", func(t *testing.T) {
|
|
|
|
rootPath := "/etcd/test/root/multi_remove_with_prefix"
|
2022-02-07 10:09:45 +08:00
|
|
|
metaKv, err := embed_etcd_kv.NewMetaKvFactory(rootPath, ¶m.EtcdCfg)
|
2021-08-24 09:45:51 +08:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
defer metaKv.Close()
|
|
|
|
defer metaKv.RemoveWithPrefix("")
|
|
|
|
|
|
|
|
prepareTests := map[string]string{
|
|
|
|
"x/abc/1": "1",
|
|
|
|
"x/abc/2": "2",
|
|
|
|
"x/def/1": "10",
|
|
|
|
"x/def/2": "20",
|
|
|
|
"x/den/1": "100",
|
|
|
|
"x/den/2": "200",
|
|
|
|
}
|
|
|
|
|
|
|
|
err = metaKv.MultiSave(prepareTests)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
multiRemoveWithPrefixTests := []struct {
|
|
|
|
prefix []string
|
|
|
|
|
|
|
|
testKey string
|
|
|
|
expectedValue string
|
|
|
|
}{
|
|
|
|
{[]string{"x/abc"}, "x/abc/1", ""},
|
|
|
|
{[]string{}, "x/abc/2", ""},
|
|
|
|
{[]string{}, "x/def/1", "10"},
|
|
|
|
{[]string{}, "x/def/2", "20"},
|
|
|
|
{[]string{}, "x/den/1", "100"},
|
|
|
|
{[]string{}, "x/den/2", "200"},
|
|
|
|
{[]string{}, "not-exist", ""},
|
|
|
|
{[]string{"x/def", "x/den"}, "x/def/1", ""},
|
|
|
|
{[]string{}, "x/def/1", ""},
|
|
|
|
{[]string{}, "x/def/2", ""},
|
|
|
|
{[]string{}, "x/den/1", ""},
|
|
|
|
{[]string{}, "x/den/2", ""},
|
|
|
|
{[]string{}, "not-exist", ""},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range multiRemoveWithPrefixTests {
|
|
|
|
if len(test.prefix) > 0 {
|
|
|
|
err = metaKv.MultiRemoveWithPrefix(test.prefix)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
v, _ := metaKv.Load(test.testKey)
|
|
|
|
assert.Equal(t, test.expectedValue, v)
|
|
|
|
}
|
|
|
|
|
|
|
|
k, v, err := metaKv.LoadWithPrefix("/")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Zero(t, len(k))
|
|
|
|
assert.Zero(t, len(v))
|
|
|
|
|
|
|
|
// MultiSaveAndRemoveWithPrefix
|
|
|
|
err = metaKv.MultiSave(prepareTests)
|
|
|
|
require.NoError(t, err)
|
|
|
|
multiSaveAndRemoveWithPrefixTests := []struct {
|
|
|
|
multiSave map[string]string
|
|
|
|
prefix []string
|
|
|
|
|
|
|
|
loadPrefix string
|
|
|
|
lengthBeforeRemove int
|
|
|
|
lengthAfterRemove int
|
|
|
|
}{
|
|
|
|
{map[string]string{}, []string{"x/abc", "x/def", "x/den"}, "x", 6, 0},
|
|
|
|
{map[string]string{"y/a": "vvv", "y/b": "vvv"}, []string{}, "y", 0, 2},
|
|
|
|
{map[string]string{"y/c": "vvv"}, []string{}, "y", 2, 3},
|
|
|
|
{map[string]string{"p/a": "vvv"}, []string{"y/a", "y"}, "y", 3, 0},
|
|
|
|
{map[string]string{}, []string{"p"}, "p", 1, 0},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range multiSaveAndRemoveWithPrefixTests {
|
|
|
|
k, _, err = metaKv.LoadWithPrefix(test.loadPrefix)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, test.lengthBeforeRemove, len(k))
|
|
|
|
|
|
|
|
err = metaKv.MultiSaveAndRemoveWithPrefix(test.multiSave, test.prefix)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
k, _, err = metaKv.LoadWithPrefix(test.loadPrefix)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, test.lengthAfterRemove, len(k))
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
2022-03-04 15:14:01 +08:00
|
|
|
te.Run("EtcdKV MultiRemoveWithPrefixBytes", func(t *testing.T) {
|
|
|
|
rootPath := "/etcd/test/root/multi_remove_with_prefix_bytes"
|
|
|
|
_metaKv, err := embed_etcd_kv.NewMetaKvFactory(rootPath, ¶m.EtcdCfg)
|
|
|
|
metaKv := _metaKv.(*embed_etcd_kv.EmbedEtcdKV)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
defer metaKv.Close()
|
|
|
|
defer metaKv.RemoveWithPrefix("")
|
|
|
|
|
|
|
|
prepareTests := map[string][]byte{
|
|
|
|
"x/abc/1": []byte("1"),
|
|
|
|
"x/abc/2": []byte("2"),
|
|
|
|
"x/def/1": []byte("10"),
|
|
|
|
"x/def/2": []byte("20"),
|
|
|
|
"x/den/1": []byte("100"),
|
|
|
|
"x/den/2": []byte("200"),
|
|
|
|
}
|
|
|
|
|
|
|
|
err = metaKv.MultiSaveBytes(prepareTests)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
multiRemoveWithPrefixTests := []struct {
|
|
|
|
prefix []string
|
|
|
|
|
|
|
|
testKey string
|
|
|
|
expectedValue []byte
|
|
|
|
}{
|
|
|
|
{[]string{"x/abc"}, "x/abc/1", nil},
|
|
|
|
{[]string{}, "x/abc/2", nil},
|
|
|
|
{[]string{}, "x/def/1", []byte("10")},
|
|
|
|
{[]string{}, "x/def/2", []byte("20")},
|
|
|
|
{[]string{}, "x/den/1", []byte("100")},
|
|
|
|
{[]string{}, "x/den/2", []byte("200")},
|
|
|
|
{[]string{}, "not-exist", nil},
|
|
|
|
{[]string{"x/def", "x/den"}, "x/def/1", nil},
|
|
|
|
{[]string{}, "x/def/1", nil},
|
|
|
|
{[]string{}, "x/def/2", nil},
|
|
|
|
{[]string{}, "x/den/1", nil},
|
|
|
|
{[]string{}, "x/den/2", nil},
|
|
|
|
{[]string{}, "not-exist", nil},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range multiRemoveWithPrefixTests {
|
|
|
|
if len(test.prefix) > 0 {
|
|
|
|
err = metaKv.MultiRemoveWithPrefix(test.prefix)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
v, _ := metaKv.LoadBytes(test.testKey)
|
|
|
|
assert.Equal(t, test.expectedValue, v)
|
|
|
|
}
|
|
|
|
|
|
|
|
k, v, err := metaKv.LoadBytesWithPrefix("/")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Zero(t, len(k))
|
|
|
|
assert.Zero(t, len(v))
|
|
|
|
|
|
|
|
// MultiSaveAndRemoveWithPrefix
|
|
|
|
err = metaKv.MultiSaveBytes(prepareTests)
|
|
|
|
require.NoError(t, err)
|
|
|
|
multiSaveAndRemoveWithPrefixTests := []struct {
|
|
|
|
multiSave map[string][]byte
|
|
|
|
prefix []string
|
|
|
|
|
|
|
|
loadPrefix string
|
|
|
|
lengthBeforeRemove int
|
|
|
|
lengthAfterRemove int
|
|
|
|
}{
|
|
|
|
{map[string][]byte{}, []string{"x/abc", "x/def", "x/den"}, "x", 6, 0},
|
|
|
|
{map[string][]byte{"y/a": []byte("vvv"), "y/b": []byte("vvv")}, []string{}, "y", 0, 2},
|
|
|
|
{map[string][]byte{"y/c": []byte("vvv")}, []string{}, "y", 2, 3},
|
|
|
|
{map[string][]byte{"p/a": []byte("vvv")}, []string{"y/a", "y"}, "y", 3, 0},
|
|
|
|
{map[string][]byte{}, []string{"p"}, "p", 1, 0},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range multiSaveAndRemoveWithPrefixTests {
|
|
|
|
k, _, err = metaKv.LoadBytesWithPrefix(test.loadPrefix)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, test.lengthBeforeRemove, len(k))
|
|
|
|
|
|
|
|
err = metaKv.MultiSaveBytesAndRemoveWithPrefix(test.multiSave, test.prefix)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
k, _, err = metaKv.LoadBytesWithPrefix(test.loadPrefix)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, test.lengthAfterRemove, len(k))
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
2021-08-24 09:45:51 +08:00
|
|
|
te.Run("EtcdKV Watch", func(t *testing.T) {
|
|
|
|
rootPath := "/etcd/test/root/watch"
|
2022-02-07 10:09:45 +08:00
|
|
|
metaKv, err := embed_etcd_kv.NewMetaKvFactory(rootPath, ¶m.EtcdCfg)
|
2021-08-24 09:45:51 +08:00
|
|
|
assert.Nil(t, err)
|
|
|
|
|
|
|
|
defer metaKv.Close()
|
|
|
|
defer metaKv.RemoveWithPrefix("")
|
|
|
|
|
|
|
|
ch := metaKv.Watch("x")
|
|
|
|
resp := <-ch
|
|
|
|
assert.True(t, resp.Created)
|
|
|
|
|
|
|
|
ch = metaKv.WatchWithPrefix("x")
|
|
|
|
resp = <-ch
|
|
|
|
assert.True(t, resp.Created)
|
|
|
|
})
|
|
|
|
|
|
|
|
te.Run("Etcd Revision", func(t *testing.T) {
|
|
|
|
rootPath := "/etcd/test/root/watch"
|
2022-02-07 10:09:45 +08:00
|
|
|
metaKv, err := embed_etcd_kv.NewMetaKvFactory(rootPath, ¶m.EtcdCfg)
|
2021-08-24 09:45:51 +08:00
|
|
|
assert.Nil(t, err)
|
|
|
|
|
|
|
|
defer metaKv.Close()
|
|
|
|
defer metaKv.RemoveWithPrefix("")
|
|
|
|
|
|
|
|
revisionTests := []struct {
|
|
|
|
inKey string
|
|
|
|
fistValue string
|
|
|
|
secondValue string
|
|
|
|
}{
|
|
|
|
{"a", "v1", "v11"},
|
|
|
|
{"y", "v2", "v22"},
|
|
|
|
{"z", "v3", "v33"},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range revisionTests {
|
|
|
|
err = metaKv.Save(test.inKey, test.fistValue)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
_, _, revision, _ := metaKv.LoadWithRevision(test.inKey)
|
|
|
|
ch := metaKv.WatchWithRevision(test.inKey, revision+1)
|
|
|
|
|
|
|
|
err = metaKv.Save(test.inKey, test.secondValue)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
resp := <-ch
|
|
|
|
assert.Equal(t, 1, len(resp.Events))
|
|
|
|
assert.Equal(t, test.secondValue, string(resp.Events[0].Kv.Value))
|
|
|
|
assert.Equal(t, revision+1, resp.Header.Revision)
|
|
|
|
}
|
|
|
|
|
2022-03-07 18:01:58 +08:00
|
|
|
var compareErr *kv.CompareFailedError
|
2021-08-24 09:45:51 +08:00
|
|
|
err = metaKv.CompareVersionAndSwap("a/b/c", 0, "1")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
value, err := metaKv.Load("a/b/c")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, value, "1")
|
|
|
|
|
|
|
|
err = metaKv.CompareVersionAndSwap("a/b/c", 0, "1")
|
|
|
|
assert.Error(t, err)
|
2022-03-07 18:01:58 +08:00
|
|
|
assert.True(t, errors.As(err, &compareErr))
|
2021-08-24 09:45:51 +08:00
|
|
|
|
|
|
|
err = metaKv.CompareValueAndSwap("a/b/c", "1", "2")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
err = metaKv.CompareValueAndSwap("a/b/c", "1", "2")
|
|
|
|
assert.Error(t, err)
|
2022-03-07 18:01:58 +08:00
|
|
|
assert.True(t, errors.As(err, &compareErr))
|
2021-08-24 09:45:51 +08:00
|
|
|
})
|
|
|
|
|
2022-03-04 15:14:01 +08:00
|
|
|
te.Run("Etcd Revision Bytes", func(t *testing.T) {
|
|
|
|
rootPath := "/etcd/test/root/revision_bytes"
|
|
|
|
_metaKv, err := embed_etcd_kv.NewMetaKvFactory(rootPath, ¶m.EtcdCfg)
|
|
|
|
metaKv := _metaKv.(*embed_etcd_kv.EmbedEtcdKV)
|
|
|
|
assert.Nil(t, err)
|
|
|
|
|
|
|
|
defer metaKv.Close()
|
|
|
|
defer metaKv.RemoveWithPrefix("")
|
|
|
|
|
|
|
|
revisionTests := []struct {
|
|
|
|
inKey string
|
|
|
|
fistValue []byte
|
|
|
|
secondValue []byte
|
|
|
|
}{
|
|
|
|
{"a", []byte("v1"), []byte("v11")},
|
|
|
|
{"y", []byte("v2"), []byte("v22")},
|
|
|
|
{"z", []byte("v3"), []byte("v33")},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range revisionTests {
|
|
|
|
err = metaKv.SaveBytes(test.inKey, test.fistValue)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
_, _, revision, _ := metaKv.LoadBytesWithRevision(test.inKey)
|
|
|
|
ch := metaKv.WatchWithRevision(test.inKey, revision+1)
|
|
|
|
|
|
|
|
err = metaKv.SaveBytes(test.inKey, test.secondValue)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
resp := <-ch
|
|
|
|
assert.Equal(t, 1, len(resp.Events))
|
|
|
|
assert.Equal(t, test.secondValue, resp.Events[0].Kv.Value)
|
|
|
|
assert.Equal(t, revision+1, resp.Header.Revision)
|
|
|
|
}
|
|
|
|
|
2022-03-07 18:01:58 +08:00
|
|
|
var compareErr *kv.CompareFailedError
|
2022-03-04 15:14:01 +08:00
|
|
|
err = metaKv.CompareVersionAndSwapBytes("a/b/c", 0, []byte("1"))
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
value, err := metaKv.LoadBytes("a/b/c")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, value, []byte("1"))
|
|
|
|
|
|
|
|
err = metaKv.CompareVersionAndSwapBytes("a/b/c", 0, []byte("1"))
|
|
|
|
assert.Error(t, err)
|
2022-03-07 18:01:58 +08:00
|
|
|
assert.True(t, errors.As(err, &compareErr))
|
2022-03-04 15:14:01 +08:00
|
|
|
|
|
|
|
err = metaKv.CompareValueAndSwapBytes("a/b/c", []byte("1"), []byte("2"))
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
err = metaKv.CompareValueAndSwapBytes("a/b/c", []byte("1"), []byte("2"))
|
|
|
|
assert.Error(t, err)
|
2022-03-07 18:01:58 +08:00
|
|
|
assert.True(t, errors.As(err, &compareErr))
|
|
|
|
|
2022-03-04 15:14:01 +08:00
|
|
|
})
|
|
|
|
|
2021-08-24 09:45:51 +08:00
|
|
|
te.Run("Etcd Lease", func(t *testing.T) {
|
|
|
|
rootPath := "/etcd/test/root/lease"
|
2022-02-07 10:09:45 +08:00
|
|
|
metaKv, err := embed_etcd_kv.NewMetaKvFactory(rootPath, ¶m.EtcdCfg)
|
2021-08-24 09:45:51 +08:00
|
|
|
assert.Nil(t, err)
|
|
|
|
|
|
|
|
defer metaKv.Close()
|
|
|
|
defer metaKv.RemoveWithPrefix("")
|
|
|
|
|
|
|
|
leaseID, err := metaKv.Grant(10)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
metaKv.KeepAlive(leaseID)
|
|
|
|
|
|
|
|
tests := map[string]string{
|
|
|
|
"a/b": "v1",
|
|
|
|
"a/b/c": "v2",
|
|
|
|
"x": "v3",
|
|
|
|
}
|
|
|
|
|
|
|
|
for k, v := range tests {
|
2022-03-25 11:03:25 +08:00
|
|
|
// SaveWithIgnoreLease must be used when the key already exists.
|
|
|
|
err = metaKv.SaveWithIgnoreLease(k, v)
|
|
|
|
assert.Error(t, err)
|
|
|
|
|
2021-08-24 09:45:51 +08:00
|
|
|
err = metaKv.SaveWithLease(k, v, leaseID)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
err = metaKv.SaveWithLease(k, v, clientv3.LeaseID(999))
|
|
|
|
assert.Error(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
})
|
2022-03-04 15:14:01 +08:00
|
|
|
|
2022-03-25 11:03:25 +08:00
|
|
|
te.Run("Etcd Lease Ignore", func(t *testing.T) {
|
|
|
|
rootPath := "/etcd/test/root/lease_ignore"
|
|
|
|
metaKv, err := embed_etcd_kv.NewMetaKvFactory(rootPath, ¶m.EtcdCfg)
|
2022-03-04 15:14:01 +08:00
|
|
|
assert.Nil(t, err)
|
|
|
|
|
|
|
|
defer metaKv.Close()
|
|
|
|
defer metaKv.RemoveWithPrefix("")
|
|
|
|
|
2022-03-25 11:03:25 +08:00
|
|
|
tests := map[string]string{
|
|
|
|
"a/b": "v1",
|
|
|
|
"a/b/c": "v2",
|
|
|
|
"x": "v3",
|
2022-03-04 15:14:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
for k, v := range tests {
|
2022-03-25 11:03:25 +08:00
|
|
|
leaseID, err := metaKv.Grant(1)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
err = metaKv.SaveWithLease(k, v, leaseID)
|
2022-03-04 15:14:01 +08:00
|
|
|
assert.NoError(t, err)
|
|
|
|
|
2022-03-25 11:03:25 +08:00
|
|
|
err = metaKv.SaveWithIgnoreLease(k, "updated_"+v)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
// Record should be updated correctly.
|
|
|
|
value, err := metaKv.Load(k)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, "updated_"+v, value)
|
|
|
|
|
|
|
|
// Let the lease expire. 3 seconds should be pretty safe.
|
|
|
|
time.Sleep(3 * time.Second)
|
|
|
|
|
|
|
|
// Updated record should still expire with lease.
|
|
|
|
_, err = metaKv.Load(k)
|
2022-03-04 15:14:01 +08:00
|
|
|
assert.Error(t, err)
|
|
|
|
}
|
|
|
|
})
|
2021-08-24 09:45:51 +08:00
|
|
|
}
|