mirror of
https://gitee.com/milvus-io/milvus.git
synced 2024-12-01 11:29:48 +08:00
1748c54fd7
Signed-off-by: Wei Liu <wei.liu@zilliz.com>
229 lines
6.2 KiB
Go
229 lines
6.2 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 meta
|
|
|
|
import (
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/suite"
|
|
|
|
"github.com/milvus-io/milvus/internal/proto/datapb"
|
|
"github.com/milvus-io/milvus/internal/proto/querypb"
|
|
"github.com/milvus-io/milvus/pkg/util/typeutil"
|
|
)
|
|
|
|
type ChannelDistManagerSuite struct {
|
|
suite.Suite
|
|
dist *ChannelDistManager
|
|
collection int64
|
|
nodes []int64
|
|
channels map[string]*DmChannel
|
|
}
|
|
|
|
func (suite *ChannelDistManagerSuite) SetupSuite() {
|
|
// Replica 0: 0, 2
|
|
// Replica 1: 1
|
|
suite.collection = 10
|
|
suite.nodes = []int64{0, 1, 2}
|
|
suite.channels = map[string]*DmChannel{
|
|
"dmc0": DmChannelFromVChannel(&datapb.VchannelInfo{
|
|
CollectionID: suite.collection,
|
|
ChannelName: "dmc0",
|
|
}),
|
|
"dmc1": DmChannelFromVChannel(&datapb.VchannelInfo{
|
|
CollectionID: suite.collection,
|
|
ChannelName: "dmc1",
|
|
}),
|
|
}
|
|
}
|
|
|
|
func (suite *ChannelDistManagerSuite) SetupTest() {
|
|
suite.dist = NewChannelDistManager()
|
|
// Distribution:
|
|
// node 0 contains channel dmc0
|
|
// node 1 contains channel dmc0, dmc1
|
|
// node 2 contains channel dmc1
|
|
suite.dist.Update(suite.nodes[0], suite.channels["dmc0"].Clone())
|
|
suite.dist.Update(suite.nodes[1], suite.channels["dmc0"].Clone(), suite.channels["dmc1"].Clone())
|
|
suite.dist.Update(suite.nodes[2], suite.channels["dmc1"].Clone())
|
|
}
|
|
|
|
func (suite *ChannelDistManagerSuite) TestGetBy() {
|
|
dist := suite.dist
|
|
|
|
// Test GetAll
|
|
channels := dist.GetAll()
|
|
suite.Len(channels, 4)
|
|
|
|
// Test GetByNode
|
|
for _, node := range suite.nodes {
|
|
channels := dist.GetByNode(node)
|
|
suite.AssertNode(channels, node)
|
|
}
|
|
|
|
// Test GetByCollection
|
|
channels = dist.GetByCollection(suite.collection)
|
|
suite.Len(channels, 4)
|
|
suite.AssertCollection(channels, suite.collection)
|
|
channels = dist.GetByCollection(-1)
|
|
suite.Len(channels, 0)
|
|
|
|
// Test GetByNodeAndCollection
|
|
// 1. Valid node and valid collection
|
|
for _, node := range suite.nodes {
|
|
channels := dist.GetByCollectionAndNode(suite.collection, node)
|
|
suite.AssertNode(channels, node)
|
|
suite.AssertCollection(channels, suite.collection)
|
|
}
|
|
|
|
// 2. Valid node and invalid collection
|
|
channels = dist.GetByCollectionAndNode(-1, suite.nodes[1])
|
|
suite.Len(channels, 0)
|
|
|
|
// 3. Invalid node and valid collection
|
|
channels = dist.GetByCollectionAndNode(suite.collection, -1)
|
|
suite.Len(channels, 0)
|
|
}
|
|
|
|
func (suite *ChannelDistManagerSuite) TestGetShardLeader() {
|
|
replicas := []*Replica{
|
|
NewReplica(
|
|
&querypb.Replica{
|
|
CollectionID: suite.collection,
|
|
},
|
|
typeutil.NewUniqueSet(suite.nodes[0], suite.nodes[2]),
|
|
),
|
|
NewReplica(
|
|
&querypb.Replica{
|
|
CollectionID: suite.collection,
|
|
},
|
|
typeutil.NewUniqueSet(suite.nodes[1]),
|
|
),
|
|
}
|
|
|
|
// Test on replica 0
|
|
leader0, ok := suite.dist.GetShardLeader(replicas[0], "dmc0")
|
|
suite.True(ok)
|
|
suite.Equal(suite.nodes[0], leader0)
|
|
leader1, ok := suite.dist.GetShardLeader(replicas[0], "dmc1")
|
|
suite.True(ok)
|
|
suite.Equal(suite.nodes[2], leader1)
|
|
|
|
// Test on replica 1
|
|
leader0, ok = suite.dist.GetShardLeader(replicas[1], "dmc0")
|
|
suite.True(ok)
|
|
suite.Equal(suite.nodes[1], leader0)
|
|
leader1, ok = suite.dist.GetShardLeader(replicas[1], "dmc1")
|
|
suite.True(ok)
|
|
suite.Equal(suite.nodes[1], leader1)
|
|
|
|
// Test no shard leader for given channel
|
|
_, ok = suite.dist.GetShardLeader(replicas[0], "invalid-shard")
|
|
suite.False(ok)
|
|
|
|
// Test on replica 0
|
|
leaders := suite.dist.GetShardLeadersByReplica(replicas[0])
|
|
suite.Len(leaders, 2)
|
|
suite.Equal(leaders["dmc0"], suite.nodes[0])
|
|
suite.Equal(leaders["dmc1"], suite.nodes[2])
|
|
|
|
// Test on replica 1
|
|
leaders = suite.dist.GetShardLeadersByReplica(replicas[1])
|
|
suite.Len(leaders, 2)
|
|
suite.Equal(leaders["dmc0"], suite.nodes[1])
|
|
suite.Equal(leaders["dmc1"], suite.nodes[1])
|
|
}
|
|
|
|
func (suite *ChannelDistManagerSuite) TestGetChannelDistByReplica() {
|
|
replica := NewReplica(
|
|
&querypb.Replica{
|
|
CollectionID: suite.collection,
|
|
},
|
|
typeutil.NewUniqueSet(11, 22, 33),
|
|
)
|
|
|
|
ch1 := &DmChannel{
|
|
VchannelInfo: &datapb.VchannelInfo{
|
|
CollectionID: suite.collection,
|
|
ChannelName: "test-channel1",
|
|
},
|
|
Node: 11,
|
|
Version: 1,
|
|
}
|
|
ch2 := &DmChannel{
|
|
VchannelInfo: &datapb.VchannelInfo{
|
|
CollectionID: suite.collection,
|
|
ChannelName: "test-channel1",
|
|
},
|
|
Node: 22,
|
|
Version: 1,
|
|
}
|
|
ch3 := &DmChannel{
|
|
VchannelInfo: &datapb.VchannelInfo{
|
|
CollectionID: suite.collection,
|
|
ChannelName: "test-channel2",
|
|
},
|
|
Node: 33,
|
|
Version: 1,
|
|
}
|
|
suite.dist.Update(11, ch1)
|
|
suite.dist.Update(22, ch2)
|
|
suite.dist.Update(33, ch3)
|
|
|
|
dist := suite.dist.GetChannelDistByReplica(replica)
|
|
suite.Len(dist["test-channel1"], 2)
|
|
suite.Len(dist["test-channel2"], 1)
|
|
}
|
|
|
|
func (suite *ChannelDistManagerSuite) AssertNames(channels []*DmChannel, names ...string) bool {
|
|
for _, channel := range channels {
|
|
hasChannel := false
|
|
for _, name := range names {
|
|
if channel.ChannelName == name {
|
|
hasChannel = true
|
|
break
|
|
}
|
|
}
|
|
if !suite.True(hasChannel, "channel %v not in the given expected list %+v", channel.ChannelName, names) {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
func (suite *ChannelDistManagerSuite) AssertNode(channels []*DmChannel, node int64) bool {
|
|
for _, channel := range channels {
|
|
if !suite.Equal(node, channel.Node) {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
func (suite *ChannelDistManagerSuite) AssertCollection(channels []*DmChannel, collection int64) bool {
|
|
for _, channel := range channels {
|
|
if !suite.Equal(collection, channel.GetCollectionID()) {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
func TestChannelDistManager(t *testing.T) {
|
|
suite.Run(t, new(ChannelDistManagerSuite))
|
|
}
|