2021-04-13 09:47:02 +08:00
|
|
|
package dataservice
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2021-04-16 16:30:55 +08:00
|
|
|
"math"
|
2021-04-13 09:47:02 +08:00
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
"github.com/zilliztech/milvus-distributed/internal/msgstream"
|
|
|
|
"github.com/zilliztech/milvus-distributed/internal/proto/commonpb"
|
|
|
|
"github.com/zilliztech/milvus-distributed/internal/proto/datapb"
|
2021-04-16 16:30:55 +08:00
|
|
|
"github.com/zilliztech/milvus-distributed/internal/proto/internalpb"
|
2021-04-13 09:47:02 +08:00
|
|
|
"github.com/zilliztech/milvus-distributed/internal/types"
|
2021-04-16 16:30:55 +08:00
|
|
|
"github.com/zilliztech/milvus-distributed/internal/util/typeutil"
|
2021-04-13 09:47:02 +08:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestRegisterNode(t *testing.T) {
|
2021-04-16 16:30:55 +08:00
|
|
|
svr := newTestServer(t)
|
|
|
|
defer closeTestServer(t, svr)
|
|
|
|
t.Run("register node", func(t *testing.T) {
|
|
|
|
resp, err := svr.RegisterNode(context.TODO(), &datapb.RegisterNodeRequest{
|
|
|
|
Base: &commonpb.MsgBase{
|
|
|
|
MsgType: 0,
|
|
|
|
MsgID: 0,
|
|
|
|
Timestamp: 0,
|
|
|
|
SourceID: 1000,
|
|
|
|
},
|
|
|
|
Address: &commonpb.Address{
|
|
|
|
Ip: "localhost",
|
|
|
|
Port: 1000,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.EqualValues(t, commonpb.ErrorCode_Success, resp.Status.ErrorCode)
|
|
|
|
assert.EqualValues(t, Params.DataNodeNum, svr.cluster.GetNumOfNodes())
|
|
|
|
assert.EqualValues(t, []int64{1000}, svr.cluster.GetNodeIDs())
|
|
|
|
})
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetSegmentInfoChannel(t *testing.T) {
|
|
|
|
svr := newTestServer(t)
|
|
|
|
defer closeTestServer(t, svr)
|
|
|
|
t.Run("get segment info channel", func(t *testing.T) {
|
|
|
|
resp, err := svr.GetSegmentInfoChannel(context.TODO())
|
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.EqualValues(t, commonpb.ErrorCode_Success, resp.Status.ErrorCode)
|
|
|
|
assert.EqualValues(t, Params.SegmentInfoChannelName, resp.Value)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetInsertChannels(t *testing.T) {
|
|
|
|
svr := newTestServer(t)
|
|
|
|
defer closeTestServer(t, svr)
|
|
|
|
t.Run("get insert channels", func(t *testing.T) {
|
|
|
|
resp, err := svr.GetInsertChannels(context.TODO(), &datapb.GetInsertChannelsRequest{
|
|
|
|
Base: &commonpb.MsgBase{
|
|
|
|
MsgType: 0,
|
|
|
|
MsgID: 0,
|
|
|
|
Timestamp: 0,
|
|
|
|
SourceID: 1000,
|
|
|
|
},
|
|
|
|
DbID: 0,
|
|
|
|
CollectionID: 0,
|
|
|
|
})
|
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.EqualValues(t, commonpb.ErrorCode_Success, resp.Status.ErrorCode)
|
|
|
|
assert.EqualValues(t, svr.getInsertChannels(), resp.Values)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAssignSegmentID(t *testing.T) {
|
|
|
|
svr := newTestServer(t)
|
|
|
|
defer closeTestServer(t, svr)
|
|
|
|
schema := newTestSchema()
|
|
|
|
svr.meta.AddCollection(&datapb.CollectionInfo{
|
|
|
|
ID: 0,
|
|
|
|
Schema: schema,
|
|
|
|
Partitions: []int64{},
|
|
|
|
})
|
|
|
|
recordSize, err := typeutil.EstimateSizePerRecord(schema)
|
|
|
|
assert.Nil(t, err)
|
|
|
|
maxCount := int(Params.SegmentSize * 1024 * 1024 / float64(recordSize))
|
|
|
|
|
|
|
|
cases := []struct {
|
|
|
|
Description string
|
|
|
|
CollectionID UniqueID
|
|
|
|
PartitionID UniqueID
|
|
|
|
ChannelName string
|
|
|
|
Count uint32
|
|
|
|
IsSuccess bool
|
|
|
|
}{
|
|
|
|
{"assign segment normally", 0, 0, "channel0", 1000, true},
|
|
|
|
{"assign segment with unexisted collection", 1, 0, "channel0", 1000, false},
|
|
|
|
{"assign with max count", 0, 0, "channel0", uint32(maxCount), true},
|
|
|
|
{"assign with max uint32 count", 0, 0, "channel1", math.MaxUint32, false},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range cases {
|
|
|
|
t.Run(test.Description, func(t *testing.T) {
|
|
|
|
req := &datapb.SegmentIDRequest{
|
|
|
|
Count: test.Count,
|
|
|
|
ChannelName: test.ChannelName,
|
|
|
|
CollectionID: test.CollectionID,
|
|
|
|
PartitionID: test.PartitionID,
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, err := svr.AssignSegmentID(context.TODO(), &datapb.AssignSegmentIDRequest{
|
|
|
|
NodeID: 0,
|
|
|
|
PeerRole: "",
|
|
|
|
SegmentIDRequests: []*datapb.SegmentIDRequest{req},
|
|
|
|
})
|
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.EqualValues(t, 1, len(resp.SegIDAssignments))
|
|
|
|
assign := resp.SegIDAssignments[0]
|
|
|
|
if test.IsSuccess {
|
|
|
|
assert.EqualValues(t, commonpb.ErrorCode_Success, assign.Status.ErrorCode)
|
|
|
|
assert.EqualValues(t, test.CollectionID, assign.CollectionID)
|
|
|
|
assert.EqualValues(t, test.PartitionID, assign.PartitionID)
|
|
|
|
assert.EqualValues(t, test.ChannelName, assign.ChannelName)
|
|
|
|
assert.EqualValues(t, test.Count, assign.Count)
|
|
|
|
} else {
|
|
|
|
assert.NotEqualValues(t, commonpb.ErrorCode_Success, assign.Status.ErrorCode)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestShowSegments(t *testing.T) {
|
|
|
|
svr := newTestServer(t)
|
|
|
|
defer closeTestServer(t, svr)
|
|
|
|
segments := []struct {
|
|
|
|
id UniqueID
|
|
|
|
collectionID UniqueID
|
|
|
|
partitionID UniqueID
|
|
|
|
}{
|
|
|
|
{0, 0, 0},
|
|
|
|
{1, 0, 0},
|
|
|
|
{2, 0, 1},
|
|
|
|
{3, 1, 1},
|
|
|
|
}
|
|
|
|
for _, segment := range segments {
|
|
|
|
err := svr.meta.AddSegment(&datapb.SegmentInfo{
|
|
|
|
ID: segment.id,
|
|
|
|
CollectionID: segment.collectionID,
|
|
|
|
PartitionID: segment.partitionID,
|
|
|
|
})
|
|
|
|
assert.Nil(t, err)
|
|
|
|
}
|
|
|
|
cases := []struct {
|
|
|
|
description string
|
|
|
|
collectionID UniqueID
|
|
|
|
partitionID UniqueID
|
|
|
|
expected []UniqueID
|
|
|
|
}{
|
|
|
|
{"show segments normally", 0, 0, []UniqueID{0, 1}},
|
|
|
|
{"show non-existed segments", 1, 2, []UniqueID{}},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range cases {
|
|
|
|
t.Run(test.description, func(t *testing.T) {
|
|
|
|
resp, err := svr.ShowSegments(context.TODO(), &datapb.ShowSegmentsRequest{
|
|
|
|
Base: &commonpb.MsgBase{
|
|
|
|
MsgType: 0,
|
|
|
|
MsgID: 0,
|
|
|
|
Timestamp: 0,
|
|
|
|
SourceID: 0,
|
|
|
|
},
|
|
|
|
CollectionID: test.collectionID,
|
|
|
|
PartitionID: test.partitionID,
|
|
|
|
DbID: 0,
|
|
|
|
})
|
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.EqualValues(t, commonpb.ErrorCode_Success, resp.Status.ErrorCode)
|
2021-04-19 11:12:56 +08:00
|
|
|
assert.ElementsMatch(t, test.expected, resp.SegmentIDs)
|
2021-04-16 16:30:55 +08:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestFlush(t *testing.T) {
|
|
|
|
svr := newTestServer(t)
|
|
|
|
defer closeTestServer(t, svr)
|
|
|
|
schema := newTestSchema()
|
|
|
|
err := svr.meta.AddCollection(&datapb.CollectionInfo{
|
|
|
|
ID: 0,
|
|
|
|
Schema: schema,
|
|
|
|
Partitions: []int64{},
|
|
|
|
})
|
|
|
|
assert.Nil(t, err)
|
|
|
|
segments := []struct {
|
|
|
|
id UniqueID
|
|
|
|
collectionID UniqueID
|
|
|
|
}{
|
|
|
|
{1, 0},
|
|
|
|
{2, 0},
|
|
|
|
}
|
|
|
|
for _, segment := range segments {
|
|
|
|
err = svr.segAllocator.OpenSegment(context.TODO(), &datapb.SegmentInfo{
|
|
|
|
ID: segment.id,
|
|
|
|
CollectionID: segment.collectionID,
|
|
|
|
PartitionID: 0,
|
|
|
|
State: commonpb.SegmentState_Growing,
|
|
|
|
})
|
|
|
|
assert.Nil(t, err)
|
|
|
|
}
|
|
|
|
resp, err := svr.Flush(context.TODO(), &datapb.FlushRequest{
|
|
|
|
Base: &commonpb.MsgBase{
|
|
|
|
MsgType: commonpb.MsgType_Flush,
|
|
|
|
MsgID: 0,
|
|
|
|
Timestamp: 0,
|
|
|
|
SourceID: 0,
|
|
|
|
},
|
|
|
|
DbID: 0,
|
|
|
|
CollectionID: 0,
|
|
|
|
})
|
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.EqualValues(t, commonpb.ErrorCode_Success, resp.ErrorCode)
|
|
|
|
ids, err := svr.segAllocator.GetSealedSegments(context.TODO())
|
|
|
|
assert.Nil(t, err)
|
2021-04-19 11:12:56 +08:00
|
|
|
assert.ElementsMatch(t, ids, []UniqueID{1, 2})
|
2021-04-16 16:30:55 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetComponentStates(t *testing.T) {
|
|
|
|
svr := newTestServer(t)
|
|
|
|
defer closeTestServer(t, svr)
|
|
|
|
cli := newMockDataNodeClient(1)
|
|
|
|
err := cli.Init()
|
|
|
|
assert.Nil(t, err)
|
|
|
|
err = cli.Start()
|
|
|
|
assert.Nil(t, err)
|
|
|
|
|
|
|
|
err = svr.cluster.Register(&dataNode{
|
|
|
|
id: 1,
|
|
|
|
address: struct {
|
|
|
|
ip string
|
|
|
|
port int64
|
|
|
|
}{
|
|
|
|
ip: "",
|
|
|
|
port: 0,
|
|
|
|
},
|
|
|
|
client: cli,
|
|
|
|
channelNum: 0,
|
|
|
|
})
|
|
|
|
assert.Nil(t, err)
|
|
|
|
|
|
|
|
resp, err := svr.GetComponentStates(context.TODO())
|
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.EqualValues(t, commonpb.ErrorCode_Success, resp.Status.ErrorCode)
|
|
|
|
assert.EqualValues(t, internalpb.StateCode_Healthy, resp.State.StateCode)
|
|
|
|
assert.EqualValues(t, 1, len(resp.SubcomponentStates))
|
|
|
|
assert.EqualValues(t, internalpb.StateCode_Healthy, resp.SubcomponentStates[0].StateCode)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetTimeTickChannel(t *testing.T) {
|
|
|
|
svr := newTestServer(t)
|
|
|
|
defer closeTestServer(t, svr)
|
|
|
|
resp, err := svr.GetTimeTickChannel(context.TODO())
|
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.EqualValues(t, commonpb.ErrorCode_Success, resp.Status.ErrorCode)
|
|
|
|
assert.EqualValues(t, Params.TimeTickChannelName, resp.Value)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetStatisticsChannel(t *testing.T) {
|
|
|
|
svr := newTestServer(t)
|
|
|
|
defer closeTestServer(t, svr)
|
|
|
|
resp, err := svr.GetStatisticsChannel(context.TODO())
|
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.EqualValues(t, commonpb.ErrorCode_Success, resp.Status.ErrorCode)
|
|
|
|
assert.EqualValues(t, Params.StatisticsChannelName, resp.Value)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetSegmentStates(t *testing.T) {
|
|
|
|
svr := newTestServer(t)
|
|
|
|
defer closeTestServer(t, svr)
|
|
|
|
err := svr.meta.AddSegment(&datapb.SegmentInfo{
|
|
|
|
ID: 1000,
|
|
|
|
CollectionID: 100,
|
|
|
|
PartitionID: 0,
|
|
|
|
InsertChannel: "",
|
|
|
|
OpenTime: 0,
|
|
|
|
SealedTime: 0,
|
|
|
|
FlushedTime: 0,
|
|
|
|
NumRows: 0,
|
|
|
|
MemSize: 0,
|
|
|
|
State: commonpb.SegmentState_Growing,
|
|
|
|
StartPosition: &internalpb.MsgPosition{
|
|
|
|
ChannelName: "",
|
|
|
|
MsgID: []byte{},
|
|
|
|
MsgGroup: "",
|
|
|
|
Timestamp: 0,
|
|
|
|
},
|
|
|
|
EndPosition: &internalpb.MsgPosition{
|
|
|
|
ChannelName: "",
|
|
|
|
MsgID: []byte{},
|
|
|
|
MsgGroup: "",
|
|
|
|
Timestamp: 0,
|
|
|
|
},
|
|
|
|
})
|
|
|
|
assert.Nil(t, err)
|
|
|
|
|
|
|
|
cases := []struct {
|
|
|
|
description string
|
|
|
|
id UniqueID
|
|
|
|
expected bool
|
|
|
|
expectedState commonpb.SegmentState
|
|
|
|
}{
|
|
|
|
{"get existed segment", 1000, true, commonpb.SegmentState_Growing},
|
|
|
|
{"get non-existed segment", 10, false, commonpb.SegmentState_Growing},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range cases {
|
|
|
|
t.Run(test.description, func(t *testing.T) {
|
|
|
|
resp, err := svr.GetSegmentStates(context.TODO(), &datapb.GetSegmentStatesRequest{
|
|
|
|
Base: &commonpb.MsgBase{
|
|
|
|
MsgType: 0,
|
|
|
|
MsgID: 0,
|
|
|
|
Timestamp: 0,
|
|
|
|
SourceID: 0,
|
|
|
|
},
|
|
|
|
SegmentIDs: []int64{test.id},
|
|
|
|
})
|
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.EqualValues(t, commonpb.ErrorCode_Success, resp.Status.ErrorCode)
|
|
|
|
assert.EqualValues(t, 1, len(resp.States))
|
|
|
|
if test.expected {
|
|
|
|
assert.EqualValues(t, commonpb.ErrorCode_Success, resp.States[0].Status.ErrorCode)
|
|
|
|
assert.EqualValues(t, test.expectedState, resp.States[0].State)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func newTestServer(t *testing.T) *Server {
|
2021-04-13 09:47:02 +08:00
|
|
|
Params.Init()
|
|
|
|
Params.DataNodeNum = 1
|
|
|
|
var err error
|
|
|
|
factory := msgstream.NewPmsFactory()
|
|
|
|
m := map[string]interface{}{
|
|
|
|
"pulsarAddress": Params.PulsarAddress,
|
|
|
|
"receiveBufSize": 1024,
|
|
|
|
"pulsarBufSize": 1024,
|
|
|
|
}
|
|
|
|
err = factory.SetParams(m)
|
|
|
|
assert.Nil(t, err)
|
|
|
|
svr, err := CreateServer(context.TODO(), factory)
|
|
|
|
assert.Nil(t, err)
|
|
|
|
ms := newMockMasterService()
|
|
|
|
err = ms.Init()
|
|
|
|
assert.Nil(t, err)
|
|
|
|
err = ms.Start()
|
|
|
|
assert.Nil(t, err)
|
|
|
|
defer ms.Stop()
|
|
|
|
svr.SetMasterClient(ms)
|
|
|
|
svr.createDataNodeClient = func(addr string) types.DataNode {
|
|
|
|
return newMockDataNodeClient(0)
|
|
|
|
}
|
|
|
|
assert.Nil(t, err)
|
|
|
|
err = svr.Init()
|
|
|
|
assert.Nil(t, err)
|
|
|
|
err = svr.Start()
|
|
|
|
assert.Nil(t, err)
|
2021-04-16 16:30:55 +08:00
|
|
|
return svr
|
|
|
|
}
|
|
|
|
|
|
|
|
func closeTestServer(t *testing.T, svr *Server) {
|
|
|
|
err := svr.Stop()
|
|
|
|
assert.Nil(t, err)
|
|
|
|
err = svr.CleanMeta()
|
|
|
|
assert.Nil(t, err)
|
2021-04-13 09:47:02 +08:00
|
|
|
}
|