fix wrong sequence of timetick stats on proxy #21855 (#22685)

Signed-off-by: MrPresent-Han <jamesharden11122@gmail.com>
pull/22745/head
MrPresent-Han 2023-03-14 15:43:54 +08:00 committed by GitHub
parent 6f6bd98c27
commit 89618b7e12
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 55 additions and 84 deletions

View File

@ -184,17 +184,6 @@ func (m *mockDmlTask) getChannels() ([]vChan, error) {
return m.vchans, nil return m.vchans, nil
} }
func (m *mockDmlTask) getPChanStats() (map[pChan]pChanStatistics, error) {
ret := make(map[pChan]pChanStatistics)
for _, pchan := range m.pchans {
ret[pchan] = pChanStatistics{
minTs: m.ts,
maxTs: m.ts,
}
}
return ret, nil
}
func newMockDmlTask(ctx context.Context) *mockDmlTask { func newMockDmlTask(ctx context.Context) *mockDmlTask {
shardNum := 2 shardNum := 2

View File

@ -106,7 +106,6 @@ type task interface {
type dmlTask interface { type dmlTask interface {
task task
getChannels() ([]pChan, error) getChannels() ([]pChan, error)
getPChanStats() (map[pChan]pChanStatistics, error)
} }
type BaseInsertTask = msgstream.InsertMsg type BaseInsertTask = msgstream.InsertMsg

View File

@ -78,26 +78,6 @@ func (dt *deleteTask) OnEnqueue() error {
return nil return nil
} }
func (dt *deleteTask) getPChanStats() (map[pChan]pChanStatistics, error) {
ret := make(map[pChan]pChanStatistics)
channels, err := dt.getChannels()
if err != nil {
return ret, err
}
beginTs := dt.BeginTs()
endTs := dt.EndTs()
for _, channel := range channels {
ret[channel] = pChanStatistics{
minTs: beginTs,
maxTs: endTs,
}
}
return ret, nil
}
func (dt *deleteTask) getChannels() ([]pChan, error) { func (dt *deleteTask) getChannels() ([]pChan, error) {
collID, err := globalMetaCache.GetCollectionID(dt.ctx, dt.deleteMsg.CollectionName) collID, err := globalMetaCache.GetCollectionID(dt.ctx, dt.deleteMsg.CollectionName)
if err != nil { if err != nil {

View File

@ -69,26 +69,6 @@ func (it *insertTask) EndTs() Timestamp {
return it.insertMsg.EndTimestamp return it.insertMsg.EndTimestamp
} }
func (it *insertTask) getPChanStats() (map[pChan]pChanStatistics, error) {
ret := make(map[pChan]pChanStatistics)
channels, err := it.getChannels()
if err != nil {
return ret, err
}
beginTs := it.BeginTs()
endTs := it.EndTs()
for _, channel := range channels {
ret[channel] = pChanStatistics{
minTs: beginTs,
maxTs: endTs,
}
}
return ret, nil
}
func (it *insertTask) getChannels() ([]pChan, error) { func (it *insertTask) getChannels() ([]pChan, error) {
collID, err := globalMetaCache.GetCollectionID(it.ctx, it.insertMsg.CollectionName) collID, err := globalMetaCache.GetCollectionID(it.ctx, it.insertMsg.CollectionName)
if err != nil { if err != nil {

View File

@ -228,16 +228,22 @@ type dmTaskQueue struct {
func (queue *dmTaskQueue) Enqueue(t task) error { func (queue *dmTaskQueue) Enqueue(t task) error {
queue.statsLock.Lock() queue.statsLock.Lock()
defer queue.statsLock.Unlock() defer queue.statsLock.Unlock()
err := queue.addPChanStats(t) //1. preAdd will check whether provided task is valid or addable
//and get the current pChannels for this dmTask
pChannels, dmt, err := queue.preAddPChanStats(t)
if err != nil { if err != nil {
return err return err
} }
//2. enqueue dml task
err = queue.baseTaskQueue.Enqueue(t) err = queue.baseTaskQueue.Enqueue(t)
if err != nil { if err != nil {
queue.popPChanStats(t)
return err return err
} }
//3. if preAdd succeed, commit will use pChannels got previously when preAdding and will definitely succeed
queue.commitPChanStats(dmt, pChannels)
//there's indeed a possibility that the collection info cache was expired after preAddPChanStats
//but considering root coord knows everything about meta modification, invalid stats appended after the meta changed
//will be discarded by root coord and will not lead to inconsistent state
return nil return nil
} }
@ -258,38 +264,51 @@ func (queue *dmTaskQueue) PopActiveTask(taskID UniqueID) task {
return t return t
} }
func (queue *dmTaskQueue) addPChanStats(t task) error { func (queue *dmTaskQueue) preAddPChanStats(t task) ([]pChan, dmlTask, error) {
if dmT, ok := t.(dmlTask); ok { if dmT, ok := t.(dmlTask); ok {
stats, err := dmT.getPChanStats() channels, err := dmT.getChannels()
if err != nil { if err != nil {
log.Warn("Proxy dmTaskQueue addPChanStats", zap.Any("tID", t.ID()), log.Warn("Proxy dmTaskQueue preAddPChanStats getChannels failed", zap.Any("tID", t.ID()),
zap.Any("stats", stats), zap.Error(err)) zap.Error(err))
return err return nil, nil, err
} }
for cName, stat := range stats { return channels, dmT, nil
info, ok := queue.pChanStatisticsInfos[cName] }
return nil, nil, fmt.Errorf("proxy preAddPChanStats reflect to dmlTask failed, tID:%v", t.ID())
}
func (queue *dmTaskQueue) commitPChanStats(dmt dmlTask, pChannels []pChan) {
//1. prepare new stat for all pChannels
newStats := make(map[pChan]pChanStatistics)
beginTs := dmt.BeginTs()
endTs := dmt.EndTs()
for _, channel := range pChannels {
newStats[channel] = pChanStatistics{
minTs: beginTs,
maxTs: endTs,
}
}
//2. update stats for all pChannels
for cName, newStat := range newStats {
currentStat, ok := queue.pChanStatisticsInfos[cName]
if !ok { if !ok {
info = &pChanStatInfo{ currentStat = &pChanStatInfo{
pChanStatistics: stat, pChanStatistics: newStat,
tsSet: map[Timestamp]struct{}{ tsSet: map[Timestamp]struct{}{
stat.minTs: {}, newStat.minTs: {},
}, },
} }
queue.pChanStatisticsInfos[cName] = info queue.pChanStatisticsInfos[cName] = currentStat
} else { } else {
if info.minTs > stat.minTs { if currentStat.minTs > newStat.minTs {
queue.pChanStatisticsInfos[cName].minTs = stat.minTs queue.pChanStatisticsInfos[cName].minTs = newStat.minTs
} }
if info.maxTs < stat.maxTs { if currentStat.maxTs < newStat.maxTs {
queue.pChanStatisticsInfos[cName].maxTs = stat.maxTs queue.pChanStatisticsInfos[cName].maxTs = newStat.maxTs
} }
queue.pChanStatisticsInfos[cName].tsSet[info.minTs] = struct{}{} queue.pChanStatisticsInfos[cName].tsSet[currentStat.minTs] = struct{}{}
} }
} }
} else {
return fmt.Errorf("proxy addUnissuedTask reflect to dmlTask failed, tID:%v", t.ID())
}
return nil
} }
func (queue *dmTaskQueue) popPChanStats(t task) error { func (queue *dmTaskQueue) popPChanStats(t task) error {

View File

@ -193,11 +193,15 @@ func TestDmTaskQueue_Basic(t *testing.T) {
assert.True(t, queue.utEmpty()) assert.True(t, queue.utEmpty())
assert.False(t, queue.utFull()) assert.False(t, queue.utFull())
//test wrong task type
dqlTask := newDefaultMockDqlTask()
err = queue.Enqueue(dqlTask)
assert.NotNil(t, err)
st := newDefaultMockDmlTask() st := newDefaultMockDmlTask()
stID := st.ID() stID := st.ID()
// no task in queue // no task in queue
unissuedTask = queue.FrontUnissuedTask() unissuedTask = queue.FrontUnissuedTask()
assert.Nil(t, unissuedTask) assert.Nil(t, unissuedTask)