mirror of
https://gitee.com/milvus-io/milvus.git
synced 2025-12-06 17:18:35 +08:00
#44892 fix etcd request context contamination by RBAC auth info ``` When RBAC is enabled, Milvus uses the gRPC metadata library to inject RBAC authentication information into the request context (ctx). Since etcd’s authentication mechanism also relies on the same metadata library, if the same ctx is passed down to the etcd request, the RBAC auth info from Milvus contaminates the auth information used by etcd. This causes the etcd server to report an invalid auth token error when RBAC is enabled but etcd auth is disabled. ``` #43638 upgrade wp to v0.1.10 Signed-off-by: tinswzy <zhenyuan.wei@zilliz.com>
971 lines
25 KiB
Go
971 lines
25 KiB
Go
// 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
|
|
// with the License. You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// 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.
|
|
|
|
package etcdkv
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"os"
|
|
"path"
|
|
"sort"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/cockroachdb/errors"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
"github.com/stretchr/testify/suite"
|
|
clientv3 "go.etcd.io/etcd/client/v3"
|
|
"golang.org/x/exp/maps"
|
|
|
|
"github.com/milvus-io/milvus/pkg/v2/kv/predicates"
|
|
"github.com/milvus-io/milvus/pkg/v2/util/etcd"
|
|
"github.com/milvus-io/milvus/pkg/v2/util/funcutil"
|
|
"github.com/milvus-io/milvus/pkg/v2/util/paramtable"
|
|
)
|
|
|
|
var Params = paramtable.Get()
|
|
|
|
func TestMain(m *testing.M) {
|
|
paramtable.Init()
|
|
code := m.Run()
|
|
os.Exit(code)
|
|
}
|
|
|
|
type EtcdKVSuite struct {
|
|
suite.Suite
|
|
|
|
rootPath string
|
|
etcdCli *clientv3.Client
|
|
etcdKV *etcdKV
|
|
}
|
|
|
|
func (s *EtcdKVSuite) SetupSuite() {
|
|
etcdCli, err := etcd.GetEtcdClient(
|
|
Params.EtcdCfg.UseEmbedEtcd.GetAsBool(),
|
|
Params.EtcdCfg.EtcdUseSSL.GetAsBool(),
|
|
Params.EtcdCfg.Endpoints.GetAsStrings(),
|
|
Params.EtcdCfg.EtcdTLSCert.GetValue(),
|
|
Params.EtcdCfg.EtcdTLSKey.GetValue(),
|
|
Params.EtcdCfg.EtcdTLSCACert.GetValue(),
|
|
Params.EtcdCfg.EtcdTLSMinVersion.GetValue())
|
|
s.Require().NoError(err)
|
|
|
|
s.etcdCli = etcdCli
|
|
}
|
|
|
|
func (s *EtcdKVSuite) TearDownSuite() {
|
|
if s.etcdCli != nil {
|
|
s.etcdCli.Close()
|
|
}
|
|
}
|
|
|
|
func (s *EtcdKVSuite) SetupTest() {
|
|
s.rootPath = path.Join("unittest/etcdkv", funcutil.RandomString(8))
|
|
s.etcdKV = NewEtcdKV(s.etcdCli, s.rootPath)
|
|
}
|
|
|
|
func (s *EtcdKVSuite) TearDownTest() {
|
|
s.etcdKV.RemoveWithPrefix(context.TODO(), "")
|
|
s.etcdKV.Close()
|
|
}
|
|
|
|
func (s *EtcdKVSuite) TestSaveLoad() {
|
|
etcdKV := s.etcdKV
|
|
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 := etcdKV.Save(context.TODO(), test.key, test.value)
|
|
s.Require().NoError(err)
|
|
}
|
|
|
|
val, err := etcdKV.Load(context.TODO(), test.key)
|
|
s.Require().NoError(err)
|
|
s.Equal(test.value, val)
|
|
}
|
|
|
|
invalidLoadTests := []struct {
|
|
invalidKey string
|
|
}{
|
|
{"t"},
|
|
{"a"},
|
|
{"test1a"},
|
|
}
|
|
|
|
for _, test := range invalidLoadTests {
|
|
val, err := etcdKV.Load(context.TODO(), test.invalidKey)
|
|
s.Error(err)
|
|
s.Zero(val)
|
|
}
|
|
|
|
loadPrefixTests := []struct {
|
|
prefix string
|
|
|
|
expectedKeys []string
|
|
expectedValues []string
|
|
expectedError error
|
|
}{
|
|
{"test", []string{
|
|
etcdKV.GetPath("test1"),
|
|
etcdKV.GetPath("test2"),
|
|
etcdKV.GetPath("test1/a"),
|
|
etcdKV.GetPath("test1/b"),
|
|
}, []string{"value1", "value2", "value_a", "value_b"}, nil},
|
|
{"test1", []string{
|
|
etcdKV.GetPath("test1"),
|
|
etcdKV.GetPath("test1/a"),
|
|
etcdKV.GetPath("test1/b"),
|
|
}, []string{"value1", "value_a", "value_b"}, nil},
|
|
{"test2", []string{etcdKV.GetPath("test2")}, []string{"value2"}, nil},
|
|
{"", []string{
|
|
etcdKV.GetPath("test1"),
|
|
etcdKV.GetPath("test2"),
|
|
etcdKV.GetPath("test1/a"),
|
|
etcdKV.GetPath("test1/b"),
|
|
}, []string{"value1", "value2", "value_a", "value_b"}, nil},
|
|
{"test1/a", []string{etcdKV.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 := etcdKV.LoadWithPrefix(context.TODO(), test.prefix)
|
|
s.ElementsMatch(test.expectedKeys, actualKeys)
|
|
s.ElementsMatch(test.expectedValues, actualValues)
|
|
s.Equal(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 := etcdKV.Remove(context.TODO(), test.validKey)
|
|
s.NoError(err)
|
|
|
|
_, err = etcdKV.Load(context.TODO(), test.validKey)
|
|
s.Error(err)
|
|
|
|
err = etcdKV.Remove(context.TODO(), test.validKey)
|
|
s.NoError(err)
|
|
err = etcdKV.Remove(context.TODO(), test.invalidKey)
|
|
s.NoError(err)
|
|
}
|
|
}
|
|
|
|
func (s *EtcdKVSuite) TestSaveAndLoadBytes() {
|
|
etcdKV := s.etcdKV
|
|
|
|
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 := etcdKV.SaveBytes(context.TODO(), test.key, []byte(test.value))
|
|
s.Require().NoError(err)
|
|
}
|
|
|
|
val, err := etcdKV.LoadBytes(context.TODO(), test.key)
|
|
s.NoError(err)
|
|
s.Equal(test.value, string(val))
|
|
}
|
|
|
|
invalidLoadTests := []struct {
|
|
invalidKey string
|
|
}{
|
|
{"t"},
|
|
{"a"},
|
|
{"test1a"},
|
|
}
|
|
|
|
for _, test := range invalidLoadTests {
|
|
val, err := etcdKV.LoadBytes(context.TODO(), test.invalidKey)
|
|
s.Error(err)
|
|
s.Zero(string(val))
|
|
}
|
|
|
|
loadPrefixTests := []struct {
|
|
prefix string
|
|
|
|
expectedKeys []string
|
|
expectedValues []string
|
|
expectedError error
|
|
}{
|
|
{"test", []string{
|
|
etcdKV.GetPath("test1"),
|
|
etcdKV.GetPath("test2"),
|
|
etcdKV.GetPath("test1/a"),
|
|
etcdKV.GetPath("test1/b"),
|
|
}, []string{"value1", "value2", "value_a", "value_b"}, nil},
|
|
{"test1", []string{
|
|
etcdKV.GetPath("test1"),
|
|
etcdKV.GetPath("test1/a"),
|
|
etcdKV.GetPath("test1/b"),
|
|
}, []string{"value1", "value_a", "value_b"}, nil},
|
|
{"test2", []string{etcdKV.GetPath("test2")}, []string{"value2"}, nil},
|
|
{"", []string{
|
|
etcdKV.GetPath("test1"),
|
|
etcdKV.GetPath("test2"),
|
|
etcdKV.GetPath("test1/a"),
|
|
etcdKV.GetPath("test1/b"),
|
|
}, []string{"value1", "value2", "value_a", "value_b"}, nil},
|
|
{"test1/a", []string{etcdKV.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 := etcdKV.LoadBytesWithPrefix(context.TODO(), test.prefix)
|
|
actualStringValues := make([]string, len(actualValues))
|
|
for i := range actualValues {
|
|
actualStringValues[i] = string(actualValues[i])
|
|
}
|
|
s.ElementsMatch(test.expectedKeys, actualKeys)
|
|
s.ElementsMatch(test.expectedValues, actualStringValues)
|
|
s.Equal(test.expectedError, err)
|
|
|
|
actualKeys, actualValues, versions, err := etcdKV.LoadBytesWithPrefix2(context.TODO(), test.prefix)
|
|
actualStringValues = make([]string, len(actualValues))
|
|
for i := range actualValues {
|
|
actualStringValues[i] = string(actualValues[i])
|
|
}
|
|
s.ElementsMatch(test.expectedKeys, actualKeys)
|
|
s.ElementsMatch(test.expectedValues, actualStringValues)
|
|
s.NotZero(versions)
|
|
s.Equal(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 := etcdKV.Remove(context.TODO(), test.validKey)
|
|
s.NoError(err)
|
|
|
|
_, err = etcdKV.Load(context.TODO(), test.validKey)
|
|
s.Error(err)
|
|
|
|
err = etcdKV.Remove(context.TODO(), test.validKey)
|
|
s.NoError(err)
|
|
err = etcdKV.Remove(context.TODO(), test.invalidKey)
|
|
s.NoError(err)
|
|
}
|
|
}
|
|
|
|
func (s *EtcdKVSuite) TestLoadBytesWithRevision() {
|
|
etcdKV := s.etcdKV
|
|
|
|
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 := etcdKV.SaveBytes(context.TODO(), test.inKey, []byte(test.inValue))
|
|
s.NoError(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 := etcdKV.LoadBytesWithRevision(context.TODO(), test.inKey)
|
|
s.NoError(err)
|
|
s.Equal(test.expectedKeyNo, len(keys))
|
|
stringValues := make([]string, len(values))
|
|
for i := range values {
|
|
stringValues[i] = string(values[i])
|
|
}
|
|
s.ElementsMatch(test.expectedValues, stringValues)
|
|
s.NotZero(revision)
|
|
}
|
|
}
|
|
|
|
func (s *EtcdKVSuite) TestMultiSaveAndMultiLoad() {
|
|
etcdKV := s.etcdKV
|
|
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 := etcdKV.MultiSave(context.TODO(), multiSaveTests)
|
|
s.Require().NoError(err)
|
|
for k, v := range multiSaveTests {
|
|
actualV, err := etcdKV.Load(context.TODO(), k)
|
|
s.NoError(err)
|
|
s.Equal(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 := etcdKV.MultiLoad(context.TODO(), test.inputKeys)
|
|
s.NoError(err)
|
|
s.Equal(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 := etcdKV.MultiLoad(context.TODO(), test.invalidKeys)
|
|
s.Error(err)
|
|
s.Equal(test.expectedValues, vs)
|
|
}
|
|
|
|
removeWithPrefixTests := []string{
|
|
"key_1",
|
|
"multi",
|
|
}
|
|
|
|
for _, k := range removeWithPrefixTests {
|
|
err = etcdKV.RemoveWithPrefix(context.TODO(), k)
|
|
s.NoError(err)
|
|
|
|
ks, vs, err := etcdKV.LoadWithPrefix(context.TODO(), k)
|
|
s.Empty(ks)
|
|
s.Empty(vs)
|
|
s.NoError(err)
|
|
}
|
|
|
|
multiRemoveTests := []string{
|
|
"key_2",
|
|
"key_3/a",
|
|
"multikey_2",
|
|
"_",
|
|
}
|
|
|
|
err = etcdKV.MultiRemove(context.TODO(), multiRemoveTests)
|
|
s.NoError(err)
|
|
|
|
ks, vs, err := etcdKV.LoadWithPrefix(context.TODO(), "")
|
|
s.NoError(err)
|
|
s.Empty(ks)
|
|
s.Empty(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 = etcdKV.MultiSaveAndRemove(context.TODO(), test.multiSaves, test.multiRemoves)
|
|
s.NoError(err)
|
|
}
|
|
|
|
ks, vs, err = etcdKV.LoadWithPrefix(context.TODO(), "")
|
|
s.NoError(err)
|
|
s.Empty(ks)
|
|
s.Empty(vs)
|
|
}
|
|
|
|
func (s *EtcdKVSuite) TestMultiSaveBytesAndMultiLoadBytes() {
|
|
etcdKV := s.etcdKV
|
|
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",
|
|
}
|
|
|
|
multiSaveBytesTests := make(map[string][]byte)
|
|
for k, v := range multiSaveTests {
|
|
multiSaveBytesTests[k] = []byte(v)
|
|
}
|
|
|
|
err := etcdKV.MultiSaveBytes(context.TODO(), multiSaveBytesTests)
|
|
s.Require().NoError(err)
|
|
for k, v := range multiSaveTests {
|
|
actualV, err := etcdKV.LoadBytes(context.TODO(), k)
|
|
s.NoError(err)
|
|
s.Equal(v, string(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 := etcdKV.MultiLoadBytes(context.TODO(), test.inputKeys)
|
|
stringVs := make([]string, len(vs))
|
|
for i := range vs {
|
|
stringVs[i] = string(vs[i])
|
|
}
|
|
s.NoError(err)
|
|
s.Equal(test.expectedValues, stringVs)
|
|
}
|
|
|
|
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 := etcdKV.MultiLoadBytes(context.TODO(), test.invalidKeys)
|
|
stringVs := make([]string, len(vs))
|
|
for i := range vs {
|
|
stringVs[i] = string(vs[i])
|
|
}
|
|
s.Error(err)
|
|
s.Equal(test.expectedValues, stringVs)
|
|
}
|
|
|
|
removeWithPrefixTests := []string{
|
|
"key_1",
|
|
"multi",
|
|
}
|
|
|
|
for _, k := range removeWithPrefixTests {
|
|
err = etcdKV.RemoveWithPrefix(context.TODO(), k)
|
|
s.NoError(err)
|
|
|
|
ks, vs, err := etcdKV.LoadBytesWithPrefix(context.TODO(), k)
|
|
s.Empty(ks)
|
|
s.Empty(vs)
|
|
s.NoError(err)
|
|
}
|
|
|
|
multiRemoveTests := []string{
|
|
"key_2",
|
|
"key_3/a",
|
|
"multikey_2",
|
|
"_",
|
|
}
|
|
|
|
err = etcdKV.MultiRemove(context.TODO(), multiRemoveTests)
|
|
s.NoError(err)
|
|
|
|
ks, vs, err := etcdKV.LoadBytesWithPrefix(context.TODO(), "")
|
|
s.NoError(err)
|
|
s.Empty(ks)
|
|
s.Empty(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"}},
|
|
{make(map[string][]byte), []string{"multikey_2"}},
|
|
}
|
|
|
|
for _, test := range multiSaveAndRemoveTests {
|
|
err = etcdKV.MultiSaveBytesAndRemove(context.TODO(), test.multiSaves, test.multiRemoves)
|
|
s.NoError(err)
|
|
}
|
|
|
|
ks, vs, err = etcdKV.LoadBytesWithPrefix(context.TODO(), "")
|
|
s.NoError(err)
|
|
s.Empty(ks)
|
|
s.Empty(vs)
|
|
}
|
|
|
|
func (s *EtcdKVSuite) TestTxnWithPredicates() {
|
|
etcdKV := s.etcdKV
|
|
|
|
prepareKV := map[string]string{
|
|
"lease1": "1",
|
|
"lease2": "2",
|
|
}
|
|
|
|
err := etcdKV.MultiSave(context.TODO(), prepareKV)
|
|
s.Require().NoError(err)
|
|
|
|
badPredicate := predicates.NewMockPredicate(s.T())
|
|
badPredicate.EXPECT().Type().Return(0)
|
|
badPredicate.EXPECT().Target().Return(predicates.PredTargetValue)
|
|
|
|
multiSaveAndRemovePredTests := []struct {
|
|
tag string
|
|
multiSave map[string]string
|
|
preds []predicates.Predicate
|
|
expectSuccess bool
|
|
}{
|
|
{"predicate_ok", map[string]string{"a": "b"}, []predicates.Predicate{predicates.ValueEqual("lease1", "1")}, true},
|
|
{"predicate_fail", map[string]string{"a": "b"}, []predicates.Predicate{predicates.ValueEqual("lease1", "2")}, false},
|
|
{"bad_predicate", map[string]string{"a": "b"}, []predicates.Predicate{badPredicate}, false},
|
|
}
|
|
|
|
for _, test := range multiSaveAndRemovePredTests {
|
|
s.Run(test.tag, func() {
|
|
err := etcdKV.MultiSaveAndRemove(context.TODO(), test.multiSave, nil, test.preds...)
|
|
if test.expectSuccess {
|
|
s.NoError(err)
|
|
} else {
|
|
s.Error(err)
|
|
}
|
|
err = etcdKV.MultiSaveAndRemoveWithPrefix(context.TODO(), test.multiSave, nil, test.preds...)
|
|
if test.expectSuccess {
|
|
s.NoError(err)
|
|
} else {
|
|
s.Error(err)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func (s *EtcdKVSuite) TestMultiSaveAndRemoveWithPrefix() {
|
|
etcdKV := s.etcdKV
|
|
|
|
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",
|
|
}
|
|
|
|
// MultiSaveAndRemoveWithPrefix
|
|
err := etcdKV.MultiSave(context.TODO(), prepareTests)
|
|
s.Require().NoError(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 := etcdKV.LoadWithPrefix(context.TODO(), test.loadPrefix)
|
|
s.NoError(err)
|
|
s.Equal(test.lengthBeforeRemove, len(k))
|
|
|
|
err = etcdKV.MultiSaveAndRemoveWithPrefix(context.TODO(), test.multiSave, test.prefix)
|
|
s.NoError(err)
|
|
|
|
k, _, err = etcdKV.LoadWithPrefix(context.TODO(), test.loadPrefix)
|
|
s.NoError(err)
|
|
s.Equal(test.lengthAfterRemove, len(k))
|
|
}
|
|
}
|
|
|
|
func (s *EtcdKVSuite) TestWatch() {
|
|
etcdKV := s.etcdKV
|
|
|
|
ch := etcdKV.Watch(context.TODO(), "x")
|
|
resp := <-ch
|
|
s.True(resp.Created)
|
|
|
|
ch = etcdKV.WatchWithPrefix(context.TODO(), "x")
|
|
resp = <-ch
|
|
s.True(resp.Created)
|
|
}
|
|
|
|
func (s *EtcdKVSuite) TestRevisionBytes() {
|
|
etcdKV := s.etcdKV
|
|
|
|
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 := etcdKV.SaveBytes(context.TODO(), test.inKey, test.fistValue)
|
|
s.Require().NoError(err)
|
|
|
|
_, _, revision, _ := etcdKV.LoadBytesWithRevision(context.TODO(), test.inKey)
|
|
ch := etcdKV.WatchWithRevision(context.TODO(), test.inKey, revision+1)
|
|
|
|
err = etcdKV.SaveBytes(context.TODO(), test.inKey, test.secondValue)
|
|
s.Require().NoError(err)
|
|
|
|
resp := <-ch
|
|
s.Equal(1, len(resp.Events))
|
|
s.Equal(string(test.secondValue), string(resp.Events[0].Kv.Value))
|
|
s.Equal(revision+1, resp.Header.Revision)
|
|
}
|
|
|
|
success, err := etcdKV.CompareVersionAndSwapBytes(context.TODO(), "a/b/c", 0, []byte("1"))
|
|
s.NoError(err)
|
|
s.True(success)
|
|
|
|
value, err := etcdKV.LoadBytes(context.TODO(), "a/b/c")
|
|
s.NoError(err)
|
|
s.Equal(string(value), "1")
|
|
|
|
success, err = etcdKV.CompareVersionAndSwapBytes(context.TODO(), "a/b/c", 0, []byte("1"))
|
|
s.NoError(err)
|
|
s.False(success)
|
|
}
|
|
|
|
func TestEtcdKV(t *testing.T) {
|
|
suite.Run(t, new(EtcdKVSuite))
|
|
}
|
|
|
|
func Test_WalkWithPagination(t *testing.T) {
|
|
etcdCli, err := etcd.GetEtcdClient(
|
|
Params.EtcdCfg.UseEmbedEtcd.GetAsBool(),
|
|
Params.EtcdCfg.EtcdUseSSL.GetAsBool(),
|
|
Params.EtcdCfg.Endpoints.GetAsStrings(),
|
|
Params.EtcdCfg.EtcdTLSCert.GetValue(),
|
|
Params.EtcdCfg.EtcdTLSKey.GetValue(),
|
|
Params.EtcdCfg.EtcdTLSCACert.GetValue(),
|
|
Params.EtcdCfg.EtcdTLSMinVersion.GetValue())
|
|
defer etcdCli.Close()
|
|
assert.NoError(t, err)
|
|
|
|
rootPath := "/etcd/test/root/pagination"
|
|
etcdKV := NewEtcdKV(etcdCli, rootPath)
|
|
|
|
defer etcdKV.Close()
|
|
defer etcdKV.RemoveWithPrefix(context.TODO(), "")
|
|
|
|
kvs := map[string]string{
|
|
"A/100": "v1",
|
|
"AA/100": "v2",
|
|
"AB/100": "v3",
|
|
"AB/2/100": "v4",
|
|
"B/100": "v5",
|
|
}
|
|
|
|
err = etcdKV.MultiSave(context.TODO(), kvs)
|
|
assert.NoError(t, err)
|
|
for k, v := range kvs {
|
|
actualV, err := etcdKV.Load(context.TODO(), k)
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, v, actualV)
|
|
}
|
|
|
|
t.Run("apply function error ", func(t *testing.T) {
|
|
err = etcdKV.WalkWithPrefix(context.TODO(), "A", 5, func(key []byte, value []byte) error {
|
|
return errors.New("error")
|
|
})
|
|
assert.Error(t, err)
|
|
})
|
|
|
|
t.Run("get with non-exist prefix ", func(t *testing.T) {
|
|
err = etcdKV.WalkWithPrefix(context.TODO(), "non-exist-prefix", 5, func(key []byte, value []byte) error {
|
|
return nil
|
|
})
|
|
assert.NoError(t, err)
|
|
})
|
|
|
|
t.Run("with different pagination", func(t *testing.T) {
|
|
testFn := func(pagination int) {
|
|
expected := map[string]string{
|
|
"A/100": "v1",
|
|
"AA/100": "v2",
|
|
"AB/100": "v3",
|
|
"AB/2/100": "v4",
|
|
}
|
|
|
|
expectedSortedKey := maps.Keys(expected)
|
|
sort.Strings(expectedSortedKey)
|
|
|
|
ret := make(map[string]string)
|
|
actualSortedKey := make([]string, 0)
|
|
|
|
err = etcdKV.WalkWithPrefix(context.TODO(), "A", pagination, func(key []byte, value []byte) error {
|
|
k := string(key)
|
|
k = k[len(rootPath)+1:]
|
|
ret[k] = string(value)
|
|
actualSortedKey = append(actualSortedKey, k)
|
|
return nil
|
|
})
|
|
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, expected, ret, fmt.Errorf("pagination: %d", pagination))
|
|
assert.Equal(t, expectedSortedKey, actualSortedKey, fmt.Errorf("pagination: %d", pagination))
|
|
}
|
|
|
|
testFn(-100)
|
|
testFn(-1)
|
|
testFn(0)
|
|
testFn(1)
|
|
testFn(5)
|
|
testFn(100)
|
|
})
|
|
}
|
|
|
|
func TestElapse(t *testing.T) {
|
|
start := time.Now()
|
|
isElapse := CheckElapseAndWarn(context.TODO(), start, "err message")
|
|
assert.Equal(t, isElapse, false)
|
|
|
|
time.Sleep(2001 * time.Millisecond)
|
|
isElapse = CheckElapseAndWarn(context.TODO(), start, "err message")
|
|
assert.Equal(t, isElapse, true)
|
|
}
|
|
|
|
func TestCheckValueSizeAndWarn(t *testing.T) {
|
|
ret := CheckValueSizeAndWarn(context.TODO(), "k", "v")
|
|
assert.False(t, ret)
|
|
|
|
v := make([]byte, 1024000)
|
|
ret = CheckValueSizeAndWarn(context.TODO(), "k", v)
|
|
assert.True(t, ret)
|
|
}
|
|
|
|
func TestCheckTnxBytesValueSizeAndWarn(t *testing.T) {
|
|
kvs := make(map[string][]byte, 0)
|
|
kvs["k"] = []byte("v")
|
|
ret := CheckTnxBytesValueSizeAndWarn(context.TODO(), kvs)
|
|
assert.False(t, ret)
|
|
|
|
kvs["k"] = make([]byte, 1024000)
|
|
ret = CheckTnxBytesValueSizeAndWarn(context.TODO(), kvs)
|
|
assert.True(t, ret)
|
|
}
|
|
|
|
func TestCheckTnxStringValueSizeAndWarn(t *testing.T) {
|
|
kvs := make(map[string]string, 0)
|
|
kvs["k"] = "v"
|
|
ret := CheckTnxStringValueSizeAndWarn(context.TODO(), kvs)
|
|
assert.False(t, ret)
|
|
|
|
kvs["k1"] = funcutil.RandomString(1024000)
|
|
ret = CheckTnxStringValueSizeAndWarn(context.TODO(), kvs)
|
|
assert.True(t, ret)
|
|
}
|
|
|
|
func TestHas(t *testing.T) {
|
|
etcdCli, err := etcd.GetEtcdClient(
|
|
Params.EtcdCfg.UseEmbedEtcd.GetAsBool(),
|
|
Params.EtcdCfg.EtcdUseSSL.GetAsBool(),
|
|
Params.EtcdCfg.Endpoints.GetAsStrings(),
|
|
Params.EtcdCfg.EtcdTLSCert.GetValue(),
|
|
Params.EtcdCfg.EtcdTLSKey.GetValue(),
|
|
Params.EtcdCfg.EtcdTLSCACert.GetValue(),
|
|
Params.EtcdCfg.EtcdTLSMinVersion.GetValue())
|
|
defer etcdCli.Close()
|
|
assert.NoError(t, err)
|
|
rootPath := "/etcd/test/root/has"
|
|
kv := NewEtcdKV(etcdCli, rootPath)
|
|
err = kv.RemoveWithPrefix(context.TODO(), "")
|
|
require.NoError(t, err)
|
|
|
|
defer kv.Close()
|
|
defer kv.RemoveWithPrefix(context.TODO(), "")
|
|
|
|
has, err := kv.Has(context.TODO(), "key1")
|
|
assert.NoError(t, err)
|
|
assert.False(t, has)
|
|
|
|
err = kv.Save(context.TODO(), "key1", "value1")
|
|
assert.NoError(t, err)
|
|
|
|
has, err = kv.Has(context.TODO(), "key1")
|
|
assert.NoError(t, err)
|
|
assert.True(t, has)
|
|
|
|
err = kv.Remove(context.TODO(), "key1")
|
|
assert.NoError(t, err)
|
|
|
|
has, err = kv.Has(context.TODO(), "key1")
|
|
assert.NoError(t, err)
|
|
assert.False(t, has)
|
|
}
|
|
|
|
func TestHasPrefix(t *testing.T) {
|
|
etcdCli, err := etcd.GetEtcdClient(
|
|
Params.EtcdCfg.UseEmbedEtcd.GetAsBool(),
|
|
Params.EtcdCfg.EtcdUseSSL.GetAsBool(),
|
|
Params.EtcdCfg.Endpoints.GetAsStrings(),
|
|
Params.EtcdCfg.EtcdTLSCert.GetValue(),
|
|
Params.EtcdCfg.EtcdTLSKey.GetValue(),
|
|
Params.EtcdCfg.EtcdTLSCACert.GetValue(),
|
|
Params.EtcdCfg.EtcdTLSMinVersion.GetValue())
|
|
defer etcdCli.Close()
|
|
assert.NoError(t, err)
|
|
rootPath := "/etcd/test/root/hasprefix"
|
|
kv := NewEtcdKV(etcdCli, rootPath)
|
|
err = kv.RemoveWithPrefix(context.TODO(), "")
|
|
require.NoError(t, err)
|
|
|
|
defer kv.Close()
|
|
defer kv.RemoveWithPrefix(context.TODO(), "")
|
|
|
|
has, err := kv.HasPrefix(context.TODO(), "key")
|
|
assert.NoError(t, err)
|
|
assert.False(t, has)
|
|
|
|
err = kv.Save(context.TODO(), "key1", "value1")
|
|
assert.NoError(t, err)
|
|
|
|
has, err = kv.HasPrefix(context.TODO(), "key")
|
|
assert.NoError(t, err)
|
|
assert.True(t, has)
|
|
|
|
err = kv.Remove(context.TODO(), "key1")
|
|
assert.NoError(t, err)
|
|
|
|
has, err = kv.HasPrefix(context.TODO(), "key")
|
|
assert.NoError(t, err)
|
|
assert.False(t, has)
|
|
}
|
|
|
|
func TestRetrySuccess(t *testing.T) {
|
|
// Test case where the function succeeds on the first attempt
|
|
err := retry(defaultRetryCount, defaultRetryInterval, func() error {
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Fatalf("expected no error, got %v", err)
|
|
}
|
|
}
|
|
|
|
func TestRetryFailure(t *testing.T) {
|
|
// Test case where the function fails all attempts
|
|
expectedErr := errors.New("always fail")
|
|
err := retry(defaultRetryCount, defaultRetryInterval, func() error {
|
|
return expectedErr
|
|
})
|
|
if err == nil {
|
|
t.Fatalf("expected error, got nil")
|
|
}
|
|
if err != expectedErr {
|
|
t.Fatalf("expected %v, got %v", expectedErr, err)
|
|
}
|
|
}
|
|
|
|
func TestRetryEventuallySucceeds(t *testing.T) {
|
|
// Test case where the function fails the first two attempts and succeeds on the third
|
|
attempts := 0
|
|
err := retry(defaultRetryCount, defaultRetryInterval, func() error {
|
|
attempts++
|
|
if attempts < 3 {
|
|
return errors.New("temporary failure")
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
t.Fatalf("expected no error, got %v", err)
|
|
}
|
|
if attempts != 3 {
|
|
t.Fatalf("expected 3 attempts, got %d", attempts)
|
|
}
|
|
}
|
|
|
|
func TestRetryInterval(t *testing.T) {
|
|
// Test case to check if retry respects the interval
|
|
startTime := time.Now()
|
|
err := retry(defaultRetryCount, defaultRetryInterval, func() error {
|
|
return errors.New("fail")
|
|
})
|
|
elapsed := time.Since(startTime)
|
|
// expected (defaultRetryCount - 1) intervals of defaultRetryInterval
|
|
expectedMin := defaultRetryInterval * (defaultRetryCount - 1)
|
|
expectedMax := expectedMin + (50 * time.Millisecond) // Allow 50ms margin for timing precision
|
|
|
|
if err == nil {
|
|
t.Fatalf("expected error, got nil")
|
|
}
|
|
if elapsed < expectedMin || elapsed > expectedMax {
|
|
t.Fatalf("expected elapsed time around %v, got %v", expectedMin, elapsed)
|
|
}
|
|
}
|