mirror of https://github.com/milvus-io/milvus.git
474 lines
11 KiB
Go
474 lines
11 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 kv
|
|
|
|
import (
|
|
"fmt"
|
|
"math/rand"
|
|
"testing"
|
|
"time"
|
|
|
|
etcdkv "github.com/milvus-io/milvus/internal/kv/etcd"
|
|
"github.com/milvus-io/milvus/internal/util/etcd"
|
|
"github.com/milvus-io/milvus/internal/util/typeutil"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
var (
|
|
snapshotPrefix = "snapshots"
|
|
)
|
|
|
|
func Test_binarySearchRecords(t *testing.T) {
|
|
type testcase struct {
|
|
records []tsv
|
|
ts typeutil.Timestamp
|
|
expected string
|
|
shouldFound bool
|
|
}
|
|
|
|
cases := []testcase{
|
|
{
|
|
records: []tsv{},
|
|
ts: 0,
|
|
expected: "",
|
|
shouldFound: false,
|
|
},
|
|
{
|
|
records: []tsv{
|
|
{
|
|
ts: 101,
|
|
value: "abc",
|
|
},
|
|
},
|
|
ts: 100,
|
|
expected: "",
|
|
shouldFound: false,
|
|
},
|
|
{
|
|
records: []tsv{
|
|
{
|
|
ts: 100,
|
|
value: "a",
|
|
},
|
|
},
|
|
ts: 100,
|
|
expected: "a",
|
|
shouldFound: true,
|
|
},
|
|
{
|
|
records: []tsv{
|
|
{
|
|
ts: 100,
|
|
value: "a",
|
|
},
|
|
{
|
|
ts: 200,
|
|
value: "b",
|
|
},
|
|
},
|
|
ts: 100,
|
|
expected: "a",
|
|
shouldFound: true,
|
|
},
|
|
{
|
|
records: []tsv{
|
|
{
|
|
ts: 100,
|
|
value: "a",
|
|
},
|
|
{
|
|
ts: 200,
|
|
value: "b",
|
|
},
|
|
{
|
|
ts: 300,
|
|
value: "c",
|
|
},
|
|
},
|
|
ts: 150,
|
|
expected: "a",
|
|
shouldFound: true,
|
|
},
|
|
{
|
|
records: []tsv{
|
|
{
|
|
ts: 100,
|
|
value: "a",
|
|
},
|
|
{
|
|
ts: 200,
|
|
value: "b",
|
|
},
|
|
{
|
|
ts: 300,
|
|
value: "c",
|
|
},
|
|
},
|
|
ts: 300,
|
|
expected: "c",
|
|
shouldFound: true,
|
|
},
|
|
{
|
|
records: []tsv{
|
|
{
|
|
ts: 100,
|
|
value: "a",
|
|
},
|
|
{
|
|
ts: 200,
|
|
value: "b",
|
|
},
|
|
{
|
|
ts: 300,
|
|
value: "c",
|
|
},
|
|
},
|
|
ts: 201,
|
|
expected: "b",
|
|
shouldFound: true,
|
|
},
|
|
{
|
|
records: []tsv{
|
|
{
|
|
ts: 100,
|
|
value: "a",
|
|
},
|
|
{
|
|
ts: 200,
|
|
value: "b",
|
|
},
|
|
{
|
|
ts: 300,
|
|
value: "c",
|
|
},
|
|
},
|
|
ts: 301,
|
|
expected: "c",
|
|
shouldFound: true,
|
|
},
|
|
}
|
|
for _, c := range cases {
|
|
result, found := binarySearchRecords(c.records, c.ts)
|
|
assert.Equal(t, c.expected, result)
|
|
assert.Equal(t, c.shouldFound, found)
|
|
}
|
|
}
|
|
|
|
func Test_ComposeIsTsKey(t *testing.T) {
|
|
sep := "_ts"
|
|
ss, err := NewSuffixSnapshot((*etcdkv.EtcdKV)(nil), sep, "", snapshotPrefix)
|
|
require.Nil(t, err)
|
|
type testcase struct {
|
|
key string
|
|
expected uint64
|
|
shouldFound bool
|
|
}
|
|
testCases := []testcase{
|
|
{
|
|
key: ss.composeTSKey("key", 100),
|
|
expected: 100,
|
|
shouldFound: true,
|
|
},
|
|
{
|
|
key: ss.composeTSKey("other-key", 65536),
|
|
expected: 65536,
|
|
shouldFound: true,
|
|
},
|
|
{
|
|
key: "snapshots/test/1000",
|
|
expected: 0,
|
|
shouldFound: false,
|
|
},
|
|
{
|
|
key: "snapshots",
|
|
expected: 0,
|
|
shouldFound: false,
|
|
},
|
|
}
|
|
for _, c := range testCases {
|
|
ts, found := ss.isTSKey(c.key)
|
|
assert.EqualValues(t, c.expected, ts)
|
|
assert.Equal(t, c.shouldFound, found)
|
|
}
|
|
}
|
|
|
|
func Test_SuffixSnaphotIsTSOfKey(t *testing.T) {
|
|
sep := "_ts"
|
|
ss, err := NewSuffixSnapshot((*etcdkv.EtcdKV)(nil), sep, "", snapshotPrefix)
|
|
require.Nil(t, err)
|
|
type testcase struct {
|
|
key string
|
|
target string
|
|
expected uint64
|
|
shouldFound bool
|
|
}
|
|
testCases := []testcase{
|
|
{
|
|
key: ss.composeTSKey("key", 100),
|
|
target: "key",
|
|
expected: 100,
|
|
shouldFound: true,
|
|
},
|
|
{
|
|
key: ss.composeTSKey("other-key", 65536),
|
|
target: "other-key",
|
|
expected: 65536,
|
|
shouldFound: true,
|
|
},
|
|
{
|
|
key: ss.composeTSKey("other-key", 65536),
|
|
target: "key",
|
|
expected: 0,
|
|
shouldFound: false,
|
|
},
|
|
{
|
|
key: "snapshots/test/1000",
|
|
expected: 0,
|
|
shouldFound: false,
|
|
},
|
|
{
|
|
key: "snapshots",
|
|
expected: 0,
|
|
shouldFound: false,
|
|
},
|
|
}
|
|
for _, c := range testCases {
|
|
ts, found := ss.isTSOfKey(c.key, c.target)
|
|
assert.EqualValues(t, c.expected, ts)
|
|
assert.Equal(t, c.shouldFound, found)
|
|
}
|
|
|
|
}
|
|
|
|
func Test_SuffixSnapshotLoad(t *testing.T) {
|
|
rand.Seed(time.Now().UnixNano())
|
|
randVal := rand.Int()
|
|
|
|
Params.Init()
|
|
rootPath := fmt.Sprintf("/test/meta/%d", randVal)
|
|
sep := "_ts"
|
|
|
|
etcdCli, err := etcd.GetEtcdClient(&Params.EtcdCfg)
|
|
require.Nil(t, err)
|
|
defer etcdCli.Close()
|
|
etcdkv := etcdkv.NewEtcdKV(etcdCli, rootPath)
|
|
defer etcdkv.Close()
|
|
|
|
var vtso typeutil.Timestamp
|
|
ftso := func() typeutil.Timestamp {
|
|
return vtso
|
|
}
|
|
|
|
ss, err := NewSuffixSnapshot(etcdkv, sep, rootPath, snapshotPrefix)
|
|
assert.Nil(t, err)
|
|
assert.NotNil(t, ss)
|
|
|
|
for i := 0; i < 20; i++ {
|
|
vtso = typeutil.Timestamp(100 + i*5)
|
|
ts := ftso()
|
|
err = ss.Save("key", fmt.Sprintf("value-%d", i), ts)
|
|
assert.Nil(t, err)
|
|
assert.Equal(t, vtso, ts)
|
|
}
|
|
for i := 0; i < 20; i++ {
|
|
val, err := ss.Load("key", typeutil.Timestamp(100+i*5+2))
|
|
t.Log("ts:", typeutil.Timestamp(100+i*5+2), i, val)
|
|
assert.Nil(t, err)
|
|
assert.Equal(t, fmt.Sprintf("value-%d", i), val)
|
|
}
|
|
val, err := ss.Load("key", 0)
|
|
assert.Nil(t, err)
|
|
assert.Equal(t, "value-19", val)
|
|
|
|
ss, err = NewSuffixSnapshot(etcdkv, sep, rootPath, snapshotPrefix)
|
|
assert.Nil(t, err)
|
|
assert.NotNil(t, ss)
|
|
|
|
for i := 0; i < 20; i++ {
|
|
val, err := ss.Load("key", typeutil.Timestamp(100+i*5+2))
|
|
assert.Nil(t, err)
|
|
assert.Equal(t, val, fmt.Sprintf("value-%d", i))
|
|
}
|
|
|
|
ss.RemoveWithPrefix("")
|
|
}
|
|
|
|
func Test_SuffixSnapshotMultiSave(t *testing.T) {
|
|
rand.Seed(time.Now().UnixNano())
|
|
randVal := rand.Int()
|
|
|
|
Params.Init()
|
|
rootPath := fmt.Sprintf("/test/meta/%d", randVal)
|
|
sep := "_ts"
|
|
etcdCli, err := etcd.GetEtcdClient(&Params.EtcdCfg)
|
|
require.Nil(t, err)
|
|
defer etcdCli.Close()
|
|
etcdkv := etcdkv.NewEtcdKV(etcdCli, rootPath)
|
|
defer etcdkv.Close()
|
|
|
|
var vtso typeutil.Timestamp
|
|
ftso := func() typeutil.Timestamp {
|
|
return vtso
|
|
}
|
|
|
|
ss, err := NewSuffixSnapshot(etcdkv, sep, rootPath, snapshotPrefix)
|
|
assert.Nil(t, err)
|
|
assert.NotNil(t, ss)
|
|
|
|
for i := 0; i < 20; i++ {
|
|
saves := map[string]string{"k1": fmt.Sprintf("v1-%d", i), "k2": fmt.Sprintf("v2-%d", i)}
|
|
vtso = typeutil.Timestamp(100 + i*5)
|
|
ts := ftso()
|
|
err = ss.MultiSave(saves, ts)
|
|
assert.Nil(t, err)
|
|
assert.Equal(t, vtso, ts)
|
|
}
|
|
for i := 0; i < 20; i++ {
|
|
keys, vals, err := ss.LoadWithPrefix("k", typeutil.Timestamp(100+i*5+2))
|
|
t.Log(i, keys, vals)
|
|
assert.Nil(t, err)
|
|
assert.Equal(t, len(keys), len(vals))
|
|
assert.Equal(t, len(keys), 2)
|
|
assert.Equal(t, keys[0], "k1")
|
|
assert.Equal(t, keys[1], "k2")
|
|
assert.Equal(t, vals[0], fmt.Sprintf("v1-%d", i))
|
|
assert.Equal(t, vals[1], fmt.Sprintf("v2-%d", i))
|
|
}
|
|
keys, vals, err := ss.LoadWithPrefix("k", 0)
|
|
assert.Nil(t, err)
|
|
assert.Equal(t, len(keys), len(vals))
|
|
assert.Equal(t, len(keys), 2)
|
|
assert.Equal(t, keys[0], "k1")
|
|
assert.Equal(t, keys[1], "k2")
|
|
assert.Equal(t, vals[0], "v1-19")
|
|
assert.Equal(t, vals[1], "v2-19")
|
|
|
|
ss, err = NewSuffixSnapshot(etcdkv, sep, rootPath, snapshotPrefix)
|
|
assert.Nil(t, err)
|
|
assert.NotNil(t, ss)
|
|
for i := 0; i < 20; i++ {
|
|
keys, vals, err := ss.LoadWithPrefix("k", typeutil.Timestamp(100+i*5+2))
|
|
assert.Nil(t, err)
|
|
assert.Equal(t, len(keys), len(vals))
|
|
assert.Equal(t, len(keys), 2)
|
|
assert.ElementsMatch(t, keys, []string{"k1", "k2"})
|
|
assert.ElementsMatch(t, vals, []string{fmt.Sprintf("v1-%d", i), fmt.Sprintf("v2-%d", i)})
|
|
}
|
|
// mix non ts k-v
|
|
err = ss.Save("kextra", "extra-value", 0)
|
|
assert.Nil(t, err)
|
|
keys, vals, err = ss.LoadWithPrefix("k", typeutil.Timestamp(300))
|
|
assert.Nil(t, err)
|
|
assert.Equal(t, len(keys), len(vals))
|
|
assert.Equal(t, len(keys), 3)
|
|
assert.ElementsMatch(t, keys, []string{"k1", "k2", "kextra"})
|
|
assert.ElementsMatch(t, vals, []string{"v1-19", "v2-19", "extra-value"})
|
|
|
|
// clean up
|
|
ss.RemoveWithPrefix("")
|
|
}
|
|
|
|
func Test_SuffixSnapshotMultiSaveAndRemoveWithPrefix(t *testing.T) {
|
|
rand.Seed(time.Now().UnixNano())
|
|
randVal := rand.Int()
|
|
|
|
Params.Init()
|
|
rootPath := fmt.Sprintf("/test/meta/%d", randVal)
|
|
sep := "_ts"
|
|
|
|
etcdCli, err := etcd.GetEtcdClient(&Params.EtcdCfg)
|
|
require.Nil(t, err)
|
|
defer etcdCli.Close()
|
|
etcdkv := etcdkv.NewEtcdKV(etcdCli, rootPath)
|
|
require.Nil(t, err)
|
|
defer etcdkv.Close()
|
|
|
|
var vtso typeutil.Timestamp
|
|
ftso := func() typeutil.Timestamp {
|
|
return vtso
|
|
}
|
|
|
|
ss, err := NewSuffixSnapshot(etcdkv, sep, rootPath, snapshotPrefix)
|
|
assert.Nil(t, err)
|
|
assert.NotNil(t, ss)
|
|
|
|
for i := 0; i < 20; i++ {
|
|
vtso = typeutil.Timestamp(100 + i*5)
|
|
ts := ftso()
|
|
err = ss.Save(fmt.Sprintf("kd-%04d", i), fmt.Sprintf("value-%d", i), ts)
|
|
assert.Nil(t, err)
|
|
assert.Equal(t, vtso, ts)
|
|
}
|
|
for i := 20; i < 40; i++ {
|
|
sm := map[string]string{"ks": fmt.Sprintf("value-%d", i)}
|
|
dm := []string{fmt.Sprintf("kd-%04d", i-20)}
|
|
vtso = typeutil.Timestamp(100 + i*5)
|
|
ts := ftso()
|
|
err = ss.MultiSaveAndRemoveWithPrefix(sm, dm, ts)
|
|
assert.Nil(t, err)
|
|
assert.Equal(t, vtso, ts)
|
|
}
|
|
for i := 0; i < 20; i++ {
|
|
val, err := ss.Load(fmt.Sprintf("kd-%04d", i), typeutil.Timestamp(100+i*5+2))
|
|
assert.Nil(t, err)
|
|
assert.Equal(t, fmt.Sprintf("value-%d", i), val)
|
|
_, vals, err := ss.LoadWithPrefix("kd-", typeutil.Timestamp(100+i*5+2))
|
|
assert.Nil(t, err)
|
|
assert.Equal(t, i+1, len(vals))
|
|
}
|
|
for i := 20; i < 40; i++ {
|
|
val, err := ss.Load("ks", typeutil.Timestamp(100+i*5+2))
|
|
assert.Nil(t, err)
|
|
assert.Equal(t, fmt.Sprintf("value-%d", i), val)
|
|
_, vals, err := ss.LoadWithPrefix("kd-", typeutil.Timestamp(100+i*5+2))
|
|
assert.Nil(t, err)
|
|
assert.Equal(t, 39-i, len(vals))
|
|
}
|
|
|
|
ss, err = NewSuffixSnapshot(etcdkv, sep, rootPath, snapshotPrefix)
|
|
assert.Nil(t, err)
|
|
assert.NotNil(t, ss)
|
|
|
|
for i := 0; i < 20; i++ {
|
|
val, err := ss.Load(fmt.Sprintf("kd-%04d", i), typeutil.Timestamp(100+i*5+2))
|
|
assert.Nil(t, err)
|
|
assert.Equal(t, fmt.Sprintf("value-%d", i), val)
|
|
_, vals, err := ss.LoadWithPrefix("kd-", typeutil.Timestamp(100+i*5+2))
|
|
assert.Nil(t, err)
|
|
assert.Equal(t, i+1, len(vals))
|
|
}
|
|
for i := 20; i < 40; i++ {
|
|
val, err := ss.Load("ks", typeutil.Timestamp(100+i*5+2))
|
|
assert.Nil(t, err)
|
|
assert.Equal(t, fmt.Sprintf("value-%d", i), val)
|
|
_, vals, err := ss.LoadWithPrefix("kd-", typeutil.Timestamp(100+i*5+2))
|
|
assert.Nil(t, err)
|
|
assert.Equal(t, 39-i, len(vals))
|
|
}
|
|
// try to load
|
|
_, err = ss.Load("kd-0000", 500)
|
|
assert.NotNil(t, err)
|
|
_, err = ss.Load("kd-0000", 0)
|
|
assert.NotNil(t, err)
|
|
_, err = ss.Load("kd-0000", 1)
|
|
assert.NotNil(t, err)
|
|
|
|
// cleanup
|
|
ss.MultiSaveAndRemoveWithPrefix(map[string]string{}, []string{""}, 0)
|
|
}
|