milvus/internal/querynode/shard_segment_detector_test.go

304 lines
7.2 KiB
Go

package querynode
import (
"context"
"fmt"
"path"
"testing"
"time"
"github.com/golang/protobuf/proto"
"github.com/milvus-io/milvus/internal/proto/querypb"
"github.com/milvus-io/milvus/internal/util/funcutil"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
"go.etcd.io/etcd/server/v3/etcdserver/api/v3client"
)
func TestEtcdShardSegmentDetector_watch(t *testing.T) {
client := v3client.New(embedetcdServer.Server)
defer client.Close()
type testCase struct {
name string
oldRecords map[string]*querypb.SegmentInfo
updateRecords map[string]*querypb.SegmentInfo
delRecords []string
expectInitEvents []segmentEvent
expectUpdateEvents []segmentEvent
oldGarbage map[string]string
newGarbage map[string]string
collectionID int64
replicaID int64
channel string
}
cases := []testCase{
{
name: "normal init",
collectionID: 1,
replicaID: 1,
channel: "dml_1_1_v0",
oldRecords: map[string]*querypb.SegmentInfo{
"segment_1": {
CollectionID: 1,
SegmentID: 1,
NodeID: 1,
DmChannel: "dml_1_1_v0",
ReplicaIds: []int64{1, 2},
NodeIds: []UniqueID{1},
},
},
oldGarbage: map[string]string{
"noice1": string([]byte{23, 21, 20}),
},
expectInitEvents: []segmentEvent{
{
eventType: segmentAdd,
segmentID: 1,
nodeIDs: []int64{1},
state: segmentStateLoaded,
},
},
},
{
name: "normal init with other segments",
collectionID: 1,
replicaID: 1,
channel: "dml_1_1_v0",
oldRecords: map[string]*querypb.SegmentInfo{
"segment_1": {
CollectionID: 1,
SegmentID: 1,
NodeID: 1,
DmChannel: "dml_1_1_v0",
ReplicaIds: []int64{1, 2},
NodeIds: []UniqueID{1},
},
"segment_2": {
CollectionID: 1,
SegmentID: 2,
NodeID: 1,
DmChannel: "dml_1_1_v1",
ReplicaIds: []int64{1, 2},
NodeIds: []UniqueID{1},
},
"segment_3": {
CollectionID: 2,
SegmentID: 3,
NodeID: 2,
DmChannel: "dml_3_2_v0",
ReplicaIds: []int64{1, 2},
NodeIds: []UniqueID{2},
},
"segment_4": { // may not happen
CollectionID: 1,
SegmentID: 4,
NodeID: 1,
DmChannel: "dml_1_1_v0",
ReplicaIds: []int64{2},
NodeIds: []UniqueID{1},
},
},
expectInitEvents: []segmentEvent{
{
eventType: segmentAdd,
segmentID: 1,
nodeIDs: []int64{1},
state: segmentStateLoaded,
},
},
},
{
name: "normal add segment",
collectionID: 1,
replicaID: 1,
channel: "dml_1_1_v0",
updateRecords: map[string]*querypb.SegmentInfo{
"segment_1": {
CollectionID: 1,
SegmentID: 1,
NodeIds: []int64{1},
DmChannel: "dml_1_1_v0",
ReplicaIds: []int64{1, 2},
},
},
oldGarbage: map[string]string{
"noice1": string([]byte{23, 21, 20}),
},
expectUpdateEvents: []segmentEvent{
{
eventType: segmentAdd,
segmentID: 1,
nodeIDs: []int64{1},
state: segmentStateLoaded,
},
},
},
{
name: "normal add segment with other replica",
collectionID: 1,
replicaID: 1,
channel: "dml_1_1_v0",
updateRecords: map[string]*querypb.SegmentInfo{
"segment_1": {
CollectionID: 1,
SegmentID: 1,
NodeIds: []int64{1},
DmChannel: "dml_1_1_v0",
ReplicaIds: []int64{1, 2},
},
"segment_2": {
CollectionID: 1,
SegmentID: 2,
NodeIds: []int64{2},
DmChannel: "dml_2_1_v1",
ReplicaIds: []int64{2},
},
},
newGarbage: map[string]string{
"noice1": string([]byte{23, 21, 20}),
},
expectUpdateEvents: []segmentEvent{
{
eventType: segmentAdd,
segmentID: 1,
nodeIDs: []int64{1},
state: segmentStateLoaded,
},
},
},
{
name: "normal remove segment",
collectionID: 1,
replicaID: 1,
channel: "dml_1_1_v0",
oldRecords: map[string]*querypb.SegmentInfo{
"segment_1": {
CollectionID: 1,
SegmentID: 1,
NodeIds: []int64{1},
DmChannel: "dml_1_1_v0",
ReplicaIds: []int64{1, 2},
},
},
oldGarbage: map[string]string{
"noice1": string([]byte{23, 21, 20}),
},
expectInitEvents: []segmentEvent{
{
eventType: segmentAdd,
segmentID: 1,
nodeIDs: []int64{1},
state: segmentStateLoaded,
},
},
delRecords: []string{
"segment_1", "noice1",
},
expectUpdateEvents: []segmentEvent{
{
eventType: segmentDel,
segmentID: 1,
nodeIDs: []int64{1},
state: segmentStateOffline,
},
},
},
{
name: "normal remove segment with other replica",
collectionID: 1,
replicaID: 1,
channel: "dml_1_1_v0",
oldRecords: map[string]*querypb.SegmentInfo{
"segment_1": {
CollectionID: 1,
SegmentID: 1,
NodeIds: []int64{1},
DmChannel: "dml_1_1_v0",
ReplicaIds: []int64{1, 2},
},
"segment_2": {
CollectionID: 1,
SegmentID: 2,
NodeIds: []int64{2},
DmChannel: "dml_2_1_v1",
ReplicaIds: []int64{2},
},
},
expectInitEvents: []segmentEvent{
{
eventType: segmentAdd,
segmentID: 1,
nodeIDs: []int64{1},
state: segmentStateLoaded,
},
},
delRecords: []string{
"segment_1", "segment_2",
},
expectUpdateEvents: []segmentEvent{
{
eventType: segmentDel,
segmentID: 1,
nodeIDs: []int64{1},
state: segmentStateOffline,
},
},
},
}
for _, tc := range cases {
t.Run(tc.name, func(t *testing.T) {
suffix := funcutil.RandomString(6)
rootPath := fmt.Sprintf("qn_shard_segment_detector_watch_%s", suffix)
ctx := context.Background()
// put existing records
for key, info := range tc.oldRecords {
bs, err := proto.Marshal(info)
require.NoError(t, err)
_, err = client.Put(ctx, path.Join(rootPath, key), string(bs))
require.NoError(t, err)
}
// put garbage data
for key, value := range tc.oldGarbage {
_, err := client.Put(ctx, path.Join(rootPath, key), value)
require.NoError(t, err)
}
sd := NewEtcdShardSegmentDetector(client, rootPath)
segments, ch := sd.watchSegments(tc.collectionID, tc.replicaID, tc.channel)
assert.ElementsMatch(t, tc.expectInitEvents, segments)
// update etcd kvs to generate events
go func() {
for key, info := range tc.updateRecords {
bs, err := proto.Marshal(info)
require.NoError(t, err)
_, err = client.Put(ctx, path.Join(rootPath, key), string(bs))
require.NoError(t, err)
}
for _, k := range tc.delRecords {
_, err := client.Delete(ctx, path.Join(rootPath, k))
require.NoError(t, err)
}
for key, value := range tc.newGarbage {
_, err := client.Put(ctx, path.Join(rootPath, key), value)
require.NoError(t, err)
}
// need a way to detect event processed
time.Sleep(time.Millisecond * 100)
sd.Close()
}()
var newEvents []segmentEvent
for evt := range ch {
newEvents = append(newEvents, evt)
}
assert.ElementsMatch(t, tc.expectUpdateEvents, newEvents)
})
}
}