mirror of https://github.com/milvus-io/milvus.git
940 lines
28 KiB
Go
940 lines
28 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 (
|
|
"context"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/cockroachdb/errors"
|
|
"github.com/samber/lo"
|
|
"github.com/stretchr/testify/mock"
|
|
"github.com/stretchr/testify/suite"
|
|
|
|
"github.com/milvus-io/milvus/internal/datacoord/allocator"
|
|
"github.com/milvus-io/milvus/internal/datacoord/session"
|
|
"github.com/milvus-io/milvus/internal/metastore/kv/binlog"
|
|
"github.com/milvus-io/milvus/internal/metastore/kv/datacoord"
|
|
"github.com/milvus-io/milvus/internal/proto/datapb"
|
|
"github.com/milvus-io/milvus/pkg/util/metautil"
|
|
"github.com/milvus-io/milvus/pkg/util/paramtable"
|
|
"github.com/milvus-io/milvus/pkg/util/typeutil"
|
|
)
|
|
|
|
func TestCompactionPlanHandlerSuite(t *testing.T) {
|
|
suite.Run(t, new(CompactionPlanHandlerSuite))
|
|
}
|
|
|
|
type CompactionPlanHandlerSuite struct {
|
|
suite.Suite
|
|
|
|
mockMeta *MockCompactionMeta
|
|
mockAlloc *allocator.MockAllocator
|
|
mockCm *MockChannelManager
|
|
mockSessMgr *session.MockDataNodeManager
|
|
handler *compactionPlanHandler
|
|
cluster Cluster
|
|
}
|
|
|
|
func (s *CompactionPlanHandlerSuite) SetupTest() {
|
|
s.mockMeta = NewMockCompactionMeta(s.T())
|
|
s.mockAlloc = allocator.NewMockAllocator(s.T())
|
|
s.mockCm = NewMockChannelManager(s.T())
|
|
s.mockSessMgr = session.NewMockDataNodeManager(s.T())
|
|
s.cluster = NewMockCluster(s.T())
|
|
s.handler = newCompactionPlanHandler(s.cluster, s.mockSessMgr, s.mockCm, s.mockMeta, s.mockAlloc, nil, nil)
|
|
}
|
|
|
|
func (s *CompactionPlanHandlerSuite) TestScheduleEmpty() {
|
|
s.SetupTest()
|
|
s.handler.schedule()
|
|
s.Empty(s.handler.executingTasks)
|
|
}
|
|
|
|
func (s *CompactionPlanHandlerSuite) generateInitTasksForSchedule() {
|
|
task1 := &mixCompactionTask{
|
|
plan: &datapb.CompactionPlan{PlanID: 1, Channel: "ch-1", Type: datapb.CompactionType_MixCompaction},
|
|
sessions: s.mockSessMgr,
|
|
meta: s.mockMeta,
|
|
}
|
|
task1.SetTask(&datapb.CompactionTask{
|
|
PlanID: 1,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
State: datapb.CompactionTaskState_pipelining,
|
|
Channel: "ch-1",
|
|
NodeID: 100,
|
|
})
|
|
|
|
task2 := &mixCompactionTask{
|
|
plan: &datapb.CompactionPlan{PlanID: 2, Channel: "ch-1", Type: datapb.CompactionType_MixCompaction},
|
|
sessions: s.mockSessMgr,
|
|
meta: s.mockMeta,
|
|
}
|
|
task2.SetTask(&datapb.CompactionTask{
|
|
PlanID: 2,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
State: datapb.CompactionTaskState_pipelining,
|
|
Channel: "ch-1",
|
|
NodeID: 100,
|
|
})
|
|
|
|
task3 := &mixCompactionTask{
|
|
plan: &datapb.CompactionPlan{PlanID: 3, Channel: "ch-2", Type: datapb.CompactionType_MixCompaction},
|
|
sessions: s.mockSessMgr,
|
|
meta: s.mockMeta,
|
|
}
|
|
task3.SetTask(&datapb.CompactionTask{
|
|
PlanID: 3,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
State: datapb.CompactionTaskState_pipelining,
|
|
Channel: "ch-2",
|
|
NodeID: 101,
|
|
})
|
|
|
|
task4 := &mixCompactionTask{
|
|
plan: &datapb.CompactionPlan{PlanID: 4, Channel: "ch-3", Type: datapb.CompactionType_Level0DeleteCompaction},
|
|
sessions: s.mockSessMgr,
|
|
meta: s.mockMeta,
|
|
}
|
|
task4.SetTask(&datapb.CompactionTask{
|
|
PlanID: 4,
|
|
Type: datapb.CompactionType_Level0DeleteCompaction,
|
|
State: datapb.CompactionTaskState_pipelining,
|
|
Channel: "ch-3",
|
|
NodeID: 102,
|
|
})
|
|
|
|
ret := []CompactionTask{task1, task2, task3, task4}
|
|
for _, t := range ret {
|
|
s.handler.restoreTask(t)
|
|
}
|
|
}
|
|
|
|
func (s *CompactionPlanHandlerSuite) TestScheduleNodeWith1ParallelTask() {
|
|
// dataNode 101's paralleTasks has 1 task running, not L0 task
|
|
tests := []struct {
|
|
description string
|
|
tasks []CompactionTask
|
|
plans []*datapb.CompactionPlan
|
|
expectedOut []UniqueID // planID
|
|
}{
|
|
{"with L0 tasks diff channel",
|
|
[]CompactionTask{
|
|
newL0CompactionTask(&datapb.CompactionTask{
|
|
PlanID: 10,
|
|
Type: datapb.CompactionType_Level0DeleteCompaction,
|
|
State: datapb.CompactionTaskState_pipelining,
|
|
Channel: "ch-10",
|
|
NodeID: 101,
|
|
}, nil, s.mockMeta, s.mockSessMgr),
|
|
newL0CompactionTask(&datapb.CompactionTask{
|
|
PlanID: 11,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
State: datapb.CompactionTaskState_pipelining,
|
|
Channel: "ch-11",
|
|
NodeID: 101,
|
|
}, nil, s.mockMeta, s.mockSessMgr),
|
|
},
|
|
[]*datapb.CompactionPlan{
|
|
{PlanID: 10, Channel: "ch-10", Type: datapb.CompactionType_Level0DeleteCompaction},
|
|
{PlanID: 11, Channel: "ch-11", Type: datapb.CompactionType_MixCompaction},
|
|
},
|
|
[]UniqueID{10, 11},
|
|
},
|
|
{"with L0 tasks same channel",
|
|
[]CompactionTask{
|
|
newMixCompactionTask(&datapb.CompactionTask{
|
|
PlanID: 11,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
State: datapb.CompactionTaskState_pipelining,
|
|
Channel: "ch-11",
|
|
NodeID: 101,
|
|
}, nil, s.mockMeta, s.mockSessMgr),
|
|
newL0CompactionTask(&datapb.CompactionTask{
|
|
PlanID: 10,
|
|
Type: datapb.CompactionType_Level0DeleteCompaction,
|
|
State: datapb.CompactionTaskState_pipelining,
|
|
Channel: "ch-11",
|
|
NodeID: 101,
|
|
}, nil, s.mockMeta, s.mockSessMgr),
|
|
},
|
|
[]*datapb.CompactionPlan{
|
|
{PlanID: 11, Channel: "ch-11", Type: datapb.CompactionType_MixCompaction},
|
|
{PlanID: 10, Channel: "ch-11", Type: datapb.CompactionType_Level0DeleteCompaction},
|
|
},
|
|
[]UniqueID{10},
|
|
},
|
|
{"without L0 tasks",
|
|
[]CompactionTask{
|
|
newMixCompactionTask(&datapb.CompactionTask{
|
|
PlanID: 14,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
State: datapb.CompactionTaskState_pipelining,
|
|
Channel: "ch-2",
|
|
NodeID: 101,
|
|
}, nil, s.mockMeta, s.mockSessMgr),
|
|
newMixCompactionTask(&datapb.CompactionTask{
|
|
PlanID: 13,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
State: datapb.CompactionTaskState_pipelining,
|
|
Channel: "ch-11",
|
|
NodeID: 101,
|
|
}, nil, s.mockMeta, s.mockSessMgr),
|
|
},
|
|
[]*datapb.CompactionPlan{
|
|
{PlanID: 14, Channel: "ch-2", Type: datapb.CompactionType_MixCompaction},
|
|
{PlanID: 13, Channel: "ch-11", Type: datapb.CompactionType_MixCompaction},
|
|
},
|
|
[]UniqueID{13, 14},
|
|
},
|
|
{"empty tasks",
|
|
[]CompactionTask{},
|
|
[]*datapb.CompactionPlan{},
|
|
[]UniqueID{}},
|
|
}
|
|
|
|
for _, test := range tests {
|
|
s.Run(test.description, func() {
|
|
s.SetupTest()
|
|
s.generateInitTasksForSchedule()
|
|
// submit the testing tasks
|
|
for i, t := range test.tasks {
|
|
t.SetPlan(test.plans[i])
|
|
s.handler.submitTask(t)
|
|
}
|
|
|
|
gotTasks := s.handler.schedule()
|
|
s.Equal(test.expectedOut, lo.Map(gotTasks, func(t CompactionTask, _ int) int64 {
|
|
return t.GetTaskProto().GetPlanID()
|
|
}))
|
|
})
|
|
}
|
|
}
|
|
|
|
func (s *CompactionPlanHandlerSuite) TestScheduleNodeWithL0Executing() {
|
|
// dataNode 102's paralleTasks has running L0 tasks
|
|
// nothing of the same channel will be able to schedule
|
|
tests := []struct {
|
|
description string
|
|
tasks []CompactionTask
|
|
plans []*datapb.CompactionPlan
|
|
expectedOut []UniqueID // planID
|
|
}{
|
|
{"with L0 tasks diff channel",
|
|
[]CompactionTask{
|
|
newL0CompactionTask(&datapb.CompactionTask{
|
|
PlanID: 10,
|
|
Type: datapb.CompactionType_Level0DeleteCompaction,
|
|
State: datapb.CompactionTaskState_pipelining,
|
|
Channel: "ch-10",
|
|
NodeID: 102,
|
|
}, nil, s.mockMeta, s.mockSessMgr),
|
|
newMixCompactionTask(&datapb.CompactionTask{
|
|
PlanID: 11,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
State: datapb.CompactionTaskState_pipelining,
|
|
Channel: "ch-11",
|
|
NodeID: 102,
|
|
}, nil, s.mockMeta, s.mockSessMgr),
|
|
},
|
|
[]*datapb.CompactionPlan{{}, {}},
|
|
[]UniqueID{10, 11},
|
|
},
|
|
{"with L0 tasks same channel", []CompactionTask{
|
|
newL0CompactionTask(&datapb.CompactionTask{
|
|
PlanID: 10,
|
|
Type: datapb.CompactionType_Level0DeleteCompaction,
|
|
State: datapb.CompactionTaskState_pipelining,
|
|
Channel: "ch-11",
|
|
NodeID: 102,
|
|
}, nil, s.mockMeta, s.mockSessMgr),
|
|
newMixCompactionTask(&datapb.CompactionTask{
|
|
PlanID: 11,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
State: datapb.CompactionTaskState_pipelining,
|
|
Channel: "ch-11",
|
|
NodeID: 102,
|
|
}, nil, s.mockMeta, s.mockSessMgr),
|
|
newMixCompactionTask(&datapb.CompactionTask{
|
|
PlanID: 13,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
State: datapb.CompactionTaskState_pipelining,
|
|
Channel: "ch-3",
|
|
NodeID: 102,
|
|
}, nil, s.mockMeta, s.mockSessMgr),
|
|
},
|
|
[]*datapb.CompactionPlan{
|
|
{PlanID: 10, Channel: "ch-3", Type: datapb.CompactionType_Level0DeleteCompaction},
|
|
{PlanID: 11, Channel: "ch-11", Type: datapb.CompactionType_MixCompaction},
|
|
{PlanID: 13, Channel: "ch-3", Type: datapb.CompactionType_MixCompaction},
|
|
},
|
|
[]UniqueID{10, 13},
|
|
},
|
|
{"with multiple L0 tasks same channel", []CompactionTask{
|
|
newL0CompactionTask(&datapb.CompactionTask{
|
|
PlanID: 10,
|
|
Type: datapb.CompactionType_Level0DeleteCompaction,
|
|
State: datapb.CompactionTaskState_pipelining,
|
|
Channel: "ch-11",
|
|
NodeID: 102,
|
|
}, nil, s.mockMeta, s.mockSessMgr),
|
|
newL0CompactionTask(&datapb.CompactionTask{
|
|
PlanID: 11,
|
|
Type: datapb.CompactionType_Level0DeleteCompaction,
|
|
State: datapb.CompactionTaskState_pipelining,
|
|
Channel: "ch-11",
|
|
NodeID: 102,
|
|
}, nil, s.mockMeta, s.mockSessMgr),
|
|
newL0CompactionTask(&datapb.CompactionTask{
|
|
PlanID: 12,
|
|
Type: datapb.CompactionType_Level0DeleteCompaction,
|
|
State: datapb.CompactionTaskState_pipelining,
|
|
Channel: "ch-11",
|
|
NodeID: 102,
|
|
}, nil, s.mockMeta, s.mockSessMgr),
|
|
},
|
|
[]*datapb.CompactionPlan{
|
|
{PlanID: 10, Channel: "ch-3", Type: datapb.CompactionType_Level0DeleteCompaction},
|
|
{PlanID: 11, Channel: "ch-3", Type: datapb.CompactionType_Level0DeleteCompaction},
|
|
{PlanID: 12, Channel: "ch-3", Type: datapb.CompactionType_Level0DeleteCompaction},
|
|
},
|
|
[]UniqueID{10, 11, 12},
|
|
},
|
|
{"without L0 tasks", []CompactionTask{
|
|
newMixCompactionTask(&datapb.CompactionTask{
|
|
PlanID: 14,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
Channel: "ch-3",
|
|
NodeID: 102,
|
|
}, nil, s.mockMeta, s.mockSessMgr),
|
|
newMixCompactionTask(&datapb.CompactionTask{
|
|
PlanID: 13,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
Channel: "ch-11",
|
|
NodeID: 102,
|
|
}, nil, s.mockMeta, s.mockSessMgr),
|
|
},
|
|
[]*datapb.CompactionPlan{
|
|
{PlanID: 14, Channel: "ch-3", Type: datapb.CompactionType_MixCompaction},
|
|
{},
|
|
},
|
|
[]UniqueID{13, 14}},
|
|
{"empty tasks", []CompactionTask{}, []*datapb.CompactionPlan{}, []UniqueID{}},
|
|
}
|
|
|
|
for _, test := range tests {
|
|
s.Run(test.description, func() {
|
|
s.SetupTest()
|
|
|
|
// submit the testing tasks
|
|
for _, t := range test.tasks {
|
|
s.handler.submitTask(t)
|
|
}
|
|
|
|
gotTasks := s.handler.schedule()
|
|
s.Equal(test.expectedOut, lo.Map(gotTasks, func(t CompactionTask, _ int) int64 {
|
|
return t.GetTaskProto().GetPlanID()
|
|
}))
|
|
})
|
|
}
|
|
}
|
|
|
|
func (s *CompactionPlanHandlerSuite) TestPickAnyNode() {
|
|
s.SetupTest()
|
|
nodeSlots := map[int64]int64{
|
|
100: 16,
|
|
101: 23,
|
|
}
|
|
|
|
task1 := newMixCompactionTask(&datapb.CompactionTask{
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
}, nil, nil, nil)
|
|
task1.slotUsage = paramtable.Get().DataCoordCfg.MixCompactionSlotUsage.GetAsInt64()
|
|
node, useSlot := s.handler.pickAnyNode(nodeSlots, task1)
|
|
s.Equal(int64(101), node)
|
|
nodeSlots[node] = nodeSlots[node] - useSlot
|
|
|
|
task2 := newMixCompactionTask(&datapb.CompactionTask{
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
}, nil, nil, nil)
|
|
task2.slotUsage = paramtable.Get().DataCoordCfg.MixCompactionSlotUsage.GetAsInt64()
|
|
|
|
node, useSlot = s.handler.pickAnyNode(nodeSlots, task2)
|
|
s.Equal(int64(100), node)
|
|
nodeSlots[node] = nodeSlots[node] - useSlot
|
|
|
|
task3 := newMixCompactionTask(&datapb.CompactionTask{
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
}, nil, nil, nil)
|
|
task3.slotUsage = paramtable.Get().DataCoordCfg.MixCompactionSlotUsage.GetAsInt64()
|
|
|
|
node, useSlot = s.handler.pickAnyNode(nodeSlots, task3)
|
|
s.Equal(int64(101), node)
|
|
nodeSlots[node] = nodeSlots[node] - useSlot
|
|
|
|
node, useSlot = s.handler.pickAnyNode(map[int64]int64{}, &mixCompactionTask{})
|
|
s.Equal(int64(NullNodeID), node)
|
|
}
|
|
|
|
func (s *CompactionPlanHandlerSuite) TestPickAnyNodeSlotUsageShouldNotBeZero() {
|
|
s.SetupTest()
|
|
nodeSlots := map[int64]int64{
|
|
100: 16,
|
|
101: 23,
|
|
}
|
|
task1 := newMixCompactionTask(&datapb.CompactionTask{
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
}, nil, nil, nil)
|
|
task1.slotUsage = 0
|
|
nodeID, useSlot := s.handler.pickAnyNode(nodeSlots, task1)
|
|
s.Equal(int64(NullNodeID), nodeID)
|
|
s.Equal(int64(0), useSlot)
|
|
}
|
|
|
|
func (s *CompactionPlanHandlerSuite) TestPickAnyNodeForClusteringTask() {
|
|
s.SetupTest()
|
|
nodeSlots := map[int64]int64{
|
|
100: 2,
|
|
101: 16,
|
|
102: 10,
|
|
}
|
|
executingTasks := make(map[int64]CompactionTask, 0)
|
|
|
|
task1 := newClusteringCompactionTask(&datapb.CompactionTask{
|
|
Type: datapb.CompactionType_ClusteringCompaction,
|
|
}, nil, nil, nil, nil, nil)
|
|
task1.slotUsage = paramtable.Get().DataCoordCfg.ClusteringCompactionSlotUsage.GetAsInt64()
|
|
|
|
task2 := newClusteringCompactionTask(&datapb.CompactionTask{
|
|
Type: datapb.CompactionType_ClusteringCompaction,
|
|
}, nil, nil, nil, nil, nil)
|
|
task2.slotUsage = paramtable.Get().DataCoordCfg.ClusteringCompactionSlotUsage.GetAsInt64()
|
|
|
|
executingTasks[1] = task1
|
|
executingTasks[2] = task2
|
|
s.handler.executingTasks = executingTasks
|
|
node, useSlot := s.handler.pickAnyNode(nodeSlots, task1)
|
|
s.Equal(int64(101), node)
|
|
nodeSlots[node] = nodeSlots[node] - useSlot
|
|
|
|
node, useSlot = s.handler.pickAnyNode(nodeSlots, task2)
|
|
s.Equal(int64(NullNodeID), node)
|
|
}
|
|
|
|
func (s *CompactionPlanHandlerSuite) TestPickShardNode() {
|
|
s.SetupTest()
|
|
nodeSlots := map[int64]int64{
|
|
100: 2,
|
|
101: 6,
|
|
}
|
|
|
|
t1 := newMixCompactionTask(&datapb.CompactionTask{
|
|
PlanID: 19530,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
Channel: "ch-01",
|
|
NodeID: 1,
|
|
}, nil, s.mockMeta, s.mockSessMgr)
|
|
t1.plan = &datapb.CompactionPlan{
|
|
PlanID: 19530,
|
|
Channel: "ch-01",
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
}
|
|
|
|
t2 := newMixCompactionTask(&datapb.CompactionTask{
|
|
PlanID: 19531,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
Channel: "ch-02",
|
|
NodeID: 1,
|
|
}, nil, s.mockMeta, s.mockSessMgr)
|
|
t2.plan = &datapb.CompactionPlan{
|
|
PlanID: 19531,
|
|
Channel: "ch-02",
|
|
Type: datapb.CompactionType_Level0DeleteCompaction,
|
|
}
|
|
|
|
s.mockCm.EXPECT().FindWatcher(mock.Anything).RunAndReturn(func(channel string) (int64, error) {
|
|
if channel == "ch-01" {
|
|
return 100, nil
|
|
}
|
|
if channel == "ch-02" {
|
|
return 101, nil
|
|
}
|
|
return 1, nil
|
|
}).Twice()
|
|
|
|
node := s.handler.pickShardNode(nodeSlots, t1)
|
|
s.Equal(int64(100), node)
|
|
|
|
node = s.handler.pickShardNode(nodeSlots, t2)
|
|
s.Equal(int64(101), node)
|
|
}
|
|
|
|
func (s *CompactionPlanHandlerSuite) TestRemoveTasksByChannel() {
|
|
s.SetupTest()
|
|
ch := "ch1"
|
|
|
|
t1 := newMixCompactionTask(&datapb.CompactionTask{
|
|
PlanID: 19530,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
Channel: ch,
|
|
NodeID: 1,
|
|
}, nil, s.mockMeta, s.mockSessMgr)
|
|
t1.plan = &datapb.CompactionPlan{
|
|
PlanID: 19530,
|
|
Channel: ch,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
}
|
|
|
|
t2 := newMixCompactionTask(&datapb.CompactionTask{
|
|
PlanID: 19531,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
Channel: ch,
|
|
NodeID: 1,
|
|
}, nil, s.mockMeta, s.mockSessMgr)
|
|
t2.plan = &datapb.CompactionPlan{
|
|
PlanID: 19531,
|
|
Channel: ch,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
}
|
|
|
|
s.handler.submitTask(t1)
|
|
s.handler.restoreTask(t2)
|
|
s.handler.removeTasksByChannel(ch)
|
|
}
|
|
|
|
func (s *CompactionPlanHandlerSuite) TestGetCompactionTask() {
|
|
s.SetupTest()
|
|
|
|
t1 := newMixCompactionTask(&datapb.CompactionTask{
|
|
TriggerID: 1,
|
|
PlanID: 1,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
Channel: "ch-01",
|
|
State: datapb.CompactionTaskState_executing,
|
|
}, nil, s.mockMeta, s.mockSessMgr)
|
|
t1.plan = &datapb.CompactionPlan{
|
|
PlanID: 1,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
Channel: "ch-01",
|
|
}
|
|
|
|
t2 := newMixCompactionTask(&datapb.CompactionTask{
|
|
TriggerID: 1,
|
|
PlanID: 2,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
Channel: "ch-01",
|
|
State: datapb.CompactionTaskState_completed,
|
|
}, nil, s.mockMeta, s.mockSessMgr)
|
|
t2.plan = &datapb.CompactionPlan{
|
|
PlanID: 2,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
Channel: "ch-01",
|
|
}
|
|
|
|
t3 := newL0CompactionTask(&datapb.CompactionTask{
|
|
TriggerID: 1,
|
|
PlanID: 3,
|
|
Type: datapb.CompactionType_Level0DeleteCompaction,
|
|
Channel: "ch-02",
|
|
State: datapb.CompactionTaskState_failed,
|
|
}, nil, s.mockMeta, s.mockSessMgr)
|
|
t3.plan = &datapb.CompactionPlan{
|
|
PlanID: 3,
|
|
Type: datapb.CompactionType_Level0DeleteCompaction,
|
|
Channel: "ch-02",
|
|
}
|
|
|
|
inTasks := map[int64]CompactionTask{
|
|
1: t1,
|
|
2: t2,
|
|
3: t3,
|
|
}
|
|
s.mockMeta.EXPECT().GetCompactionTasksByTriggerID(mock.Anything).RunAndReturn(func(i int64) []*datapb.CompactionTask {
|
|
var ret []*datapb.CompactionTask
|
|
for _, t := range inTasks {
|
|
if t.GetTaskProto().GetTriggerID() != i {
|
|
continue
|
|
}
|
|
ret = append(ret, t.ShadowClone())
|
|
}
|
|
return ret
|
|
})
|
|
|
|
for _, t := range inTasks {
|
|
s.handler.submitTask(t)
|
|
}
|
|
|
|
s.handler.schedule()
|
|
|
|
info := s.handler.getCompactionInfo(1)
|
|
s.Equal(1, info.completedCnt)
|
|
s.Equal(1, info.executingCnt)
|
|
s.Equal(1, info.failedCnt)
|
|
}
|
|
|
|
func (s *CompactionPlanHandlerSuite) TestExecCompactionPlan() {
|
|
s.SetupTest()
|
|
s.mockMeta.EXPECT().CheckAndSetSegmentsCompacting(mock.Anything).Return(true, true).Maybe()
|
|
s.mockMeta.EXPECT().SaveCompactionTask(mock.Anything).Return(nil)
|
|
handler := newCompactionPlanHandler(nil, s.mockSessMgr, s.mockCm, s.mockMeta, s.mockAlloc, nil, nil)
|
|
|
|
task := &datapb.CompactionTask{
|
|
TriggerID: 1,
|
|
PlanID: 1,
|
|
Channel: "ch-1",
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
}
|
|
err := handler.enqueueCompaction(task)
|
|
s.NoError(err)
|
|
t := handler.getCompactionTask(1)
|
|
s.NotNil(t)
|
|
task.PlanID = 2
|
|
err = s.handler.enqueueCompaction(task)
|
|
s.NoError(err)
|
|
}
|
|
|
|
func (s *CompactionPlanHandlerSuite) TestCheckCompaction() {
|
|
s.SetupTest()
|
|
|
|
s.mockSessMgr.EXPECT().GetCompactionPlanResult(UniqueID(111), int64(1)).Return(
|
|
&datapb.CompactionPlanResult{PlanID: 1, State: datapb.CompactionTaskState_executing}, nil).Once()
|
|
|
|
s.mockSessMgr.EXPECT().GetCompactionPlanResult(UniqueID(111), int64(2)).Return(
|
|
&datapb.CompactionPlanResult{
|
|
PlanID: 2,
|
|
State: datapb.CompactionTaskState_completed,
|
|
Segments: []*datapb.CompactionSegment{{PlanID: 2}},
|
|
}, nil).Once()
|
|
|
|
s.mockSessMgr.EXPECT().GetCompactionPlanResult(UniqueID(111), int64(6)).Return(
|
|
&datapb.CompactionPlanResult{
|
|
PlanID: 6,
|
|
Channel: "ch-2",
|
|
State: datapb.CompactionTaskState_completed,
|
|
Segments: []*datapb.CompactionSegment{{PlanID: 6}},
|
|
}, nil).Once()
|
|
|
|
s.mockSessMgr.EXPECT().DropCompactionPlan(mock.Anything, mock.Anything).Return(nil)
|
|
s.mockMeta.EXPECT().SetSegmentsCompacting(mock.Anything, mock.Anything).Return()
|
|
|
|
t1 := newMixCompactionTask(&datapb.CompactionTask{
|
|
PlanID: 1,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
TimeoutInSeconds: 1,
|
|
Channel: "ch-1",
|
|
State: datapb.CompactionTaskState_executing,
|
|
NodeID: 111,
|
|
}, nil, s.mockMeta, s.mockSessMgr)
|
|
t1.plan = &datapb.CompactionPlan{
|
|
PlanID: 1, Channel: "ch-1",
|
|
TimeoutInSeconds: 1,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
}
|
|
|
|
t2 := newMixCompactionTask(&datapb.CompactionTask{
|
|
PlanID: 2,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
Channel: "ch-1",
|
|
State: datapb.CompactionTaskState_executing,
|
|
NodeID: 111,
|
|
}, nil, s.mockMeta, s.mockSessMgr)
|
|
t2.plan = &datapb.CompactionPlan{
|
|
PlanID: 2,
|
|
Channel: "ch-1",
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
}
|
|
|
|
t3 := newMixCompactionTask(&datapb.CompactionTask{
|
|
PlanID: 3,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
Channel: "ch-1",
|
|
State: datapb.CompactionTaskState_timeout,
|
|
NodeID: 111,
|
|
}, nil, s.mockMeta, s.mockSessMgr)
|
|
t3.plan = &datapb.CompactionPlan{
|
|
PlanID: 3,
|
|
Channel: "ch-1",
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
}
|
|
|
|
t4 := newMixCompactionTask(&datapb.CompactionTask{
|
|
PlanID: 4,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
Channel: "ch-1",
|
|
State: datapb.CompactionTaskState_timeout,
|
|
NodeID: 111,
|
|
}, nil, s.mockMeta, s.mockSessMgr)
|
|
t4.plan = &datapb.CompactionPlan{
|
|
PlanID: 4,
|
|
Channel: "ch-1",
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
}
|
|
|
|
t6 := newMixCompactionTask(&datapb.CompactionTask{
|
|
PlanID: 6,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
Channel: "ch-2",
|
|
State: datapb.CompactionTaskState_executing,
|
|
NodeID: 111,
|
|
}, nil, s.mockMeta, s.mockSessMgr)
|
|
t6.plan = &datapb.CompactionPlan{
|
|
PlanID: 6,
|
|
Channel: "ch-2",
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
}
|
|
|
|
inTasks := map[int64]CompactionTask{
|
|
1: t1,
|
|
2: t2,
|
|
3: t3,
|
|
4: t4,
|
|
6: t6,
|
|
}
|
|
|
|
// s.mockSessMgr.EXPECT().SyncSegments(int64(111), mock.Anything).Return(nil)
|
|
// s.mockMeta.EXPECT().UpdateSegmentsInfo(mock.Anything).Return(nil)
|
|
s.mockMeta.EXPECT().SaveCompactionTask(mock.Anything).Return(nil)
|
|
s.mockMeta.EXPECT().CompleteCompactionMutation(mock.Anything, mock.Anything).RunAndReturn(
|
|
func(t *datapb.CompactionTask, result *datapb.CompactionPlanResult) ([]*SegmentInfo, *segMetricMutation, error) {
|
|
if t.GetPlanID() == 2 {
|
|
segment := NewSegmentInfo(&datapb.SegmentInfo{ID: 100})
|
|
return []*SegmentInfo{segment}, &segMetricMutation{}, nil
|
|
} else if t.GetPlanID() == 6 {
|
|
return nil, nil, errors.Errorf("intended error")
|
|
}
|
|
return nil, nil, errors.Errorf("unexpected error")
|
|
}).Twice()
|
|
|
|
for _, t := range inTasks {
|
|
s.handler.submitTask(t)
|
|
}
|
|
|
|
s.handler.schedule()
|
|
// time.Sleep(2 * time.Second)
|
|
s.handler.checkCompaction()
|
|
|
|
t := s.handler.getCompactionTask(1)
|
|
// timeout
|
|
s.Nil(t)
|
|
|
|
t = s.handler.getCompactionTask(2)
|
|
// completed
|
|
s.Nil(t)
|
|
|
|
t = s.handler.getCompactionTask(3)
|
|
s.Nil(t)
|
|
|
|
t = s.handler.getCompactionTask(4)
|
|
s.Nil(t)
|
|
|
|
t = s.handler.getCompactionTask(5)
|
|
// not exist
|
|
s.Nil(t)
|
|
|
|
t = s.handler.getCompactionTask(6)
|
|
s.Equal(datapb.CompactionTaskState_executing, t.GetTaskProto().GetState())
|
|
}
|
|
|
|
func (s *CompactionPlanHandlerSuite) TestCompactionGC() {
|
|
s.SetupTest()
|
|
inTasks := []*datapb.CompactionTask{
|
|
{
|
|
PlanID: 1,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
State: datapb.CompactionTaskState_completed,
|
|
StartTime: time.Now().Add(-time.Second * 100000).Unix(),
|
|
},
|
|
{
|
|
PlanID: 2,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
State: datapb.CompactionTaskState_cleaned,
|
|
StartTime: time.Now().Add(-time.Second * 100000).Unix(),
|
|
},
|
|
{
|
|
PlanID: 3,
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
State: datapb.CompactionTaskState_cleaned,
|
|
StartTime: time.Now().Unix(),
|
|
},
|
|
}
|
|
|
|
catalog := &datacoord.Catalog{MetaKv: NewMetaMemoryKV()}
|
|
compactionTaskMeta, err := newCompactionTaskMeta(context.TODO(), catalog)
|
|
s.NoError(err)
|
|
s.handler.meta = &meta{compactionTaskMeta: compactionTaskMeta}
|
|
for _, t := range inTasks {
|
|
s.handler.meta.SaveCompactionTask(t)
|
|
}
|
|
|
|
s.handler.cleanCompactionTaskMeta()
|
|
// two task should be cleaned, one remains
|
|
tasks := s.handler.meta.GetCompactionTaskMeta().GetCompactionTasks()
|
|
s.Equal(1, len(tasks))
|
|
}
|
|
|
|
func (s *CompactionPlanHandlerSuite) TestProcessCompleteCompaction() {
|
|
s.SetupTest()
|
|
|
|
// s.mockSessMgr.EXPECT().SyncSegments(mock.Anything, mock.Anything).Return(nil).Once()
|
|
s.mockMeta.EXPECT().SaveCompactionTask(mock.Anything).Return(nil)
|
|
s.mockMeta.EXPECT().SetSegmentsCompacting(mock.Anything, mock.Anything).Return().Once()
|
|
segment := NewSegmentInfo(&datapb.SegmentInfo{ID: 100})
|
|
s.mockMeta.EXPECT().CompleteCompactionMutation(mock.Anything, mock.Anything).Return(
|
|
[]*SegmentInfo{segment},
|
|
&segMetricMutation{}, nil).Once()
|
|
|
|
dataNodeID := UniqueID(111)
|
|
|
|
seg1 := &datapb.SegmentInfo{
|
|
ID: 1,
|
|
Binlogs: []*datapb.FieldBinlog{getFieldBinlogIDs(101, 1)},
|
|
Statslogs: []*datapb.FieldBinlog{getFieldBinlogIDs(101, 2)},
|
|
Deltalogs: []*datapb.FieldBinlog{getFieldBinlogIDs(101, 3)},
|
|
}
|
|
|
|
seg2 := &datapb.SegmentInfo{
|
|
ID: 2,
|
|
Binlogs: []*datapb.FieldBinlog{getFieldBinlogIDs(101, 4)},
|
|
Statslogs: []*datapb.FieldBinlog{getFieldBinlogIDs(101, 5)},
|
|
Deltalogs: []*datapb.FieldBinlog{getFieldBinlogIDs(101, 6)},
|
|
}
|
|
|
|
plan := &datapb.CompactionPlan{
|
|
PlanID: 1,
|
|
SegmentBinlogs: []*datapb.CompactionSegmentBinlogs{
|
|
{
|
|
SegmentID: seg1.ID,
|
|
FieldBinlogs: seg1.GetBinlogs(),
|
|
Field2StatslogPaths: seg1.GetStatslogs(),
|
|
Deltalogs: seg1.GetDeltalogs(),
|
|
},
|
|
{
|
|
SegmentID: seg2.ID,
|
|
FieldBinlogs: seg2.GetBinlogs(),
|
|
Field2StatslogPaths: seg2.GetStatslogs(),
|
|
Deltalogs: seg2.GetDeltalogs(),
|
|
},
|
|
},
|
|
Type: datapb.CompactionType_MixCompaction,
|
|
}
|
|
|
|
task := newMixCompactionTask(&datapb.CompactionTask{
|
|
PlanID: plan.GetPlanID(),
|
|
TriggerID: 1,
|
|
Type: plan.GetType(),
|
|
State: datapb.CompactionTaskState_executing,
|
|
NodeID: dataNodeID,
|
|
InputSegments: []UniqueID{1, 2},
|
|
}, nil, s.mockMeta, s.mockSessMgr)
|
|
|
|
compactionResult := datapb.CompactionPlanResult{
|
|
PlanID: 1,
|
|
State: datapb.CompactionTaskState_completed,
|
|
Segments: []*datapb.CompactionSegment{
|
|
{
|
|
SegmentID: 3,
|
|
NumOfRows: 15,
|
|
InsertLogs: []*datapb.FieldBinlog{getFieldBinlogIDs(101, 301)},
|
|
Field2StatslogPaths: []*datapb.FieldBinlog{getFieldBinlogIDs(101, 302)},
|
|
Deltalogs: []*datapb.FieldBinlog{getFieldBinlogIDs(101, 303)},
|
|
},
|
|
},
|
|
}
|
|
|
|
s.mockSessMgr.EXPECT().GetCompactionPlanResult(UniqueID(111), int64(1)).Return(&compactionResult, nil).Once()
|
|
s.mockSessMgr.EXPECT().DropCompactionPlan(mock.Anything, mock.Anything).Return(nil)
|
|
|
|
s.handler.submitTask(task)
|
|
s.handler.schedule()
|
|
err := s.handler.checkCompaction()
|
|
s.NoError(err)
|
|
}
|
|
|
|
func getFieldBinlogIDs(fieldID int64, logIDs ...int64) *datapb.FieldBinlog {
|
|
l := &datapb.FieldBinlog{
|
|
FieldID: fieldID,
|
|
Binlogs: make([]*datapb.Binlog, 0, len(logIDs)),
|
|
}
|
|
for _, id := range logIDs {
|
|
l.Binlogs = append(l.Binlogs, &datapb.Binlog{LogID: id})
|
|
}
|
|
err := binlog.CompressFieldBinlogs([]*datapb.FieldBinlog{l})
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return l
|
|
}
|
|
|
|
func getFieldBinlogPaths(fieldID int64, paths ...string) *datapb.FieldBinlog {
|
|
l := &datapb.FieldBinlog{
|
|
FieldID: fieldID,
|
|
Binlogs: make([]*datapb.Binlog, 0, len(paths)),
|
|
}
|
|
for _, path := range paths {
|
|
l.Binlogs = append(l.Binlogs, &datapb.Binlog{LogPath: path})
|
|
}
|
|
err := binlog.CompressFieldBinlogs([]*datapb.FieldBinlog{l})
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return l
|
|
}
|
|
|
|
func getFieldBinlogIDsWithEntry(fieldID int64, entry int64, logIDs ...int64) *datapb.FieldBinlog {
|
|
l := &datapb.FieldBinlog{
|
|
FieldID: fieldID,
|
|
Binlogs: make([]*datapb.Binlog, 0, len(logIDs)),
|
|
}
|
|
for _, id := range logIDs {
|
|
l.Binlogs = append(l.Binlogs, &datapb.Binlog{LogID: id, EntriesNum: entry})
|
|
}
|
|
err := binlog.CompressFieldBinlogs([]*datapb.FieldBinlog{l})
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return l
|
|
}
|
|
|
|
func getInsertLogPath(rootPath string, segmentID typeutil.UniqueID) string {
|
|
return metautil.BuildInsertLogPath(rootPath, 10, 100, segmentID, 1000, 10000)
|
|
}
|
|
|
|
func getStatsLogPath(rootPath string, segmentID typeutil.UniqueID) string {
|
|
return metautil.BuildStatsLogPath(rootPath, 10, 100, segmentID, 1000, 10000)
|
|
}
|
|
|
|
func getDeltaLogPath(rootPath string, segmentID typeutil.UniqueID) string {
|
|
return metautil.BuildDeltaLogPath(rootPath, 10, 100, segmentID, 10000)
|
|
}
|
|
|
|
func TestCheckDelay(t *testing.T) {
|
|
handler := &compactionPlanHandler{}
|
|
t1 := newMixCompactionTask(&datapb.CompactionTask{
|
|
StartTime: time.Now().Add(-100 * time.Minute).Unix(),
|
|
}, nil, nil, nil)
|
|
handler.checkDelay(t1)
|
|
t2 := newL0CompactionTask(&datapb.CompactionTask{
|
|
StartTime: time.Now().Add(-100 * time.Minute).Unix(),
|
|
}, nil, nil, nil)
|
|
handler.checkDelay(t2)
|
|
t3 := newClusteringCompactionTask(&datapb.CompactionTask{
|
|
StartTime: time.Now().Add(-100 * time.Minute).Unix(),
|
|
}, nil, nil, nil, nil, nil)
|
|
handler.checkDelay(t3)
|
|
}
|