mirror of https://github.com/milvus-io/milvus.git
247 lines
7.4 KiB
Go
247 lines
7.4 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 datacoord
|
|
|
|
import (
|
|
"path"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/golang/protobuf/proto"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"github.com/milvus-io/milvus/internal/proto/datapb"
|
|
)
|
|
|
|
func TestChannelStateTimer(t *testing.T) {
|
|
kv := getWatchKV(t)
|
|
defer kv.Close()
|
|
|
|
prefix := Params.CommonCfg.DataCoordWatchSubPath.GetValue()
|
|
|
|
t.Run("test getWatcher", func(t *testing.T) {
|
|
timer := newChannelStateTimer(kv)
|
|
|
|
etcdCh, timeoutCh := timer.getWatchers(prefix)
|
|
assert.NotNil(t, etcdCh)
|
|
assert.NotNil(t, timeoutCh)
|
|
|
|
timer.getWatchers(prefix)
|
|
assert.NotNil(t, etcdCh)
|
|
assert.NotNil(t, timeoutCh)
|
|
})
|
|
|
|
t.Run("test loadAllChannels", func(t *testing.T) {
|
|
defer kv.RemoveWithPrefix("")
|
|
timer := newChannelStateTimer(kv)
|
|
timer.loadAllChannels(1)
|
|
|
|
validWatchInfo := datapb.ChannelWatchInfo{
|
|
Vchan: &datapb.VchannelInfo{},
|
|
StartTs: time.Now().Unix(),
|
|
State: datapb.ChannelWatchState_ToWatch,
|
|
}
|
|
validData, err := proto.Marshal(&validWatchInfo)
|
|
require.NoError(t, err)
|
|
|
|
prefix = Params.CommonCfg.DataCoordWatchSubPath.GetValue()
|
|
prepareKvs := map[string]string{
|
|
path.Join(prefix, "1/channel-1"): "invalidWatchInfo",
|
|
path.Join(prefix, "1/channel-2"): string(validData),
|
|
path.Join(prefix, "2/channel-3"): string(validData),
|
|
}
|
|
|
|
err = kv.MultiSave(prepareKvs)
|
|
require.NoError(t, err)
|
|
|
|
tests := []struct {
|
|
inNodeID UniqueID
|
|
outLen int
|
|
}{
|
|
{1, 1},
|
|
{2, 1},
|
|
{3, 0},
|
|
}
|
|
|
|
for _, test := range tests {
|
|
infos, err := timer.loadAllChannels(test.inNodeID)
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, test.outLen, len(infos))
|
|
}
|
|
})
|
|
|
|
t.Run("test startOne", func(t *testing.T) {
|
|
normalTimeoutTs := 20 * time.Second
|
|
nowTimeoutTs := 1 * time.Millisecond
|
|
zeroTimeoutTs := 0 * time.Second
|
|
resetTimeoutTs := 30 * time.Second
|
|
tests := []struct {
|
|
channelName string
|
|
timeoutTs time.Duration
|
|
|
|
description string
|
|
}{
|
|
{"channel-1", normalTimeoutTs, "test stop"},
|
|
{"channel-2", nowTimeoutTs, "test timeout"},
|
|
{"channel-3", zeroTimeoutTs, "not start"},
|
|
{"channel-4", resetTimeoutTs, "reset timer"},
|
|
}
|
|
|
|
timer := newChannelStateTimer(kv)
|
|
|
|
_, timeoutCh := timer.getWatchers(prefix)
|
|
|
|
for _, test := range tests {
|
|
t.Run(test.description, func(t *testing.T) {
|
|
timer.startOne(datapb.ChannelWatchState_ToWatch, test.channelName, 1, test.timeoutTs)
|
|
if test.timeoutTs == nowTimeoutTs {
|
|
e := <-timeoutCh
|
|
assert.Equal(t, watchTimeoutAck, e.ackType)
|
|
assert.Equal(t, test.channelName, e.channelName)
|
|
} else if test.timeoutTs == resetTimeoutTs {
|
|
timer.resetIfExist(test.channelName, nowTimeoutTs)
|
|
e := <-timeoutCh
|
|
assert.Equal(t, watchTimeoutAck, e.ackType)
|
|
assert.Equal(t, test.channelName, e.channelName)
|
|
} else {
|
|
timer.stopIfExist(&ackEvent{watchSuccessAck, test.channelName, 1})
|
|
}
|
|
})
|
|
}
|
|
|
|
timer.startOne(datapb.ChannelWatchState_ToWatch, "channel-remove", 1, normalTimeoutTs)
|
|
timer.removeTimers([]string{"channel-remove"})
|
|
})
|
|
|
|
t.Run("test startOne no leaking issue 17335", func(t *testing.T) {
|
|
timer := newChannelStateTimer(kv)
|
|
|
|
timer.startOne(datapb.ChannelWatchState_ToRelease, "channel-1", 1, 20*time.Second)
|
|
stop, ok := timer.runningTimerStops.Get("channel-1")
|
|
require.True(t, ok)
|
|
|
|
timer.startOne(datapb.ChannelWatchState_ToWatch, "channel-1", 1, 20*time.Second)
|
|
_, ok = <-stop
|
|
assert.False(t, ok)
|
|
|
|
stop2, ok := timer.runningTimerStops.Get("channel-1")
|
|
assert.True(t, ok)
|
|
|
|
timer.removeTimers([]string{"channel-1"})
|
|
_, ok = <-stop2
|
|
assert.False(t, ok)
|
|
})
|
|
}
|
|
|
|
func TestChannelStateTimer_parses(t *testing.T) {
|
|
const (
|
|
ValidTest = true
|
|
InValidTest = false
|
|
)
|
|
|
|
t.Run("test parseWatchInfo", func(t *testing.T) {
|
|
validWatchInfo := datapb.ChannelWatchInfo{
|
|
Vchan: &datapb.VchannelInfo{},
|
|
StartTs: time.Now().Unix(),
|
|
State: datapb.ChannelWatchState_ToWatch,
|
|
}
|
|
validData, err := proto.Marshal(&validWatchInfo)
|
|
require.NoError(t, err)
|
|
|
|
invalidDataUnableToMarshal := []byte("invalidData")
|
|
|
|
invalidWatchInfoNilVchan := validWatchInfo
|
|
invalidWatchInfoNilVchan.Vchan = nil
|
|
invalidDataNilVchan, err := proto.Marshal(&invalidWatchInfoNilVchan)
|
|
require.NoError(t, err)
|
|
|
|
tests := []struct {
|
|
inKey string
|
|
inData []byte
|
|
|
|
isValid bool
|
|
description string
|
|
}{
|
|
{"key", validData, ValidTest, "test with valid watchInfo"},
|
|
{"key", invalidDataUnableToMarshal, InValidTest, "test with watchInfo unable to marshal"},
|
|
{"key", invalidDataNilVchan, InValidTest, "test with watchInfo with nil Vchan"},
|
|
}
|
|
|
|
for _, test := range tests {
|
|
t.Run(test.description, func(t *testing.T) {
|
|
info, err := parseWatchInfo(test.inKey, test.inData)
|
|
if test.isValid {
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, info)
|
|
assert.Equal(t, info.GetState(), validWatchInfo.GetState())
|
|
assert.Equal(t, info.GetStartTs(), validWatchInfo.GetStartTs())
|
|
} else {
|
|
assert.Nil(t, info)
|
|
assert.Error(t, err)
|
|
}
|
|
})
|
|
}
|
|
})
|
|
|
|
t.Run("test parseWatchInfo compatibility", func(t *testing.T) {
|
|
oldWatchInfo := datapb.ChannelWatchInfo{
|
|
Vchan: &datapb.VchannelInfo{
|
|
CollectionID: 1,
|
|
ChannelName: "delta-channel1",
|
|
UnflushedSegments: []*datapb.SegmentInfo{{ID: 1}},
|
|
FlushedSegments: []*datapb.SegmentInfo{{ID: 2}},
|
|
DroppedSegments: []*datapb.SegmentInfo{{ID: 3}},
|
|
UnflushedSegmentIds: []int64{1},
|
|
},
|
|
StartTs: time.Now().Unix(),
|
|
State: datapb.ChannelWatchState_ToWatch,
|
|
}
|
|
|
|
oldData, err := proto.Marshal(&oldWatchInfo)
|
|
assert.NoError(t, err)
|
|
newWatchInfo, err := parseWatchInfo("key", oldData)
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, []*datapb.SegmentInfo{}, newWatchInfo.GetVchan().GetUnflushedSegments())
|
|
assert.Equal(t, []*datapb.SegmentInfo{}, newWatchInfo.GetVchan().GetFlushedSegments())
|
|
assert.Equal(t, []*datapb.SegmentInfo{}, newWatchInfo.GetVchan().GetDroppedSegments())
|
|
assert.NotEmpty(t, newWatchInfo.GetVchan().GetUnflushedSegmentIds())
|
|
assert.NotEmpty(t, newWatchInfo.GetVchan().GetFlushedSegmentIds())
|
|
assert.NotEmpty(t, newWatchInfo.GetVchan().GetDroppedSegmentIds())
|
|
})
|
|
|
|
t.Run("test getAckType", func(t *testing.T) {
|
|
tests := []struct {
|
|
inState datapb.ChannelWatchState
|
|
outAckType ackType
|
|
}{
|
|
{datapb.ChannelWatchState_WatchSuccess, watchSuccessAck},
|
|
{datapb.ChannelWatchState_WatchFailure, watchFailAck},
|
|
{datapb.ChannelWatchState_ToWatch, watchTimeoutAck},
|
|
{datapb.ChannelWatchState_Uncomplete, watchTimeoutAck},
|
|
{datapb.ChannelWatchState_ReleaseSuccess, releaseSuccessAck},
|
|
{datapb.ChannelWatchState_ReleaseFailure, releaseFailAck},
|
|
{datapb.ChannelWatchState_ToRelease, releaseTimeoutAck},
|
|
{100, invalidAck},
|
|
}
|
|
|
|
for _, test := range tests {
|
|
assert.Equal(t, test.outAckType, getAckType(test.inState))
|
|
}
|
|
})
|
|
}
|