192 lines
7.0 KiB
Go
192 lines
7.0 KiB
Go
// Copyright 2023 PingCAP, Inc.
|
|
//
|
|
// Licensed 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 scheduler
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"testing"
|
|
|
|
"github.com/pingcap/errors"
|
|
"github.com/pingcap/failpoint"
|
|
"github.com/pingcap/tidb/pkg/disttask/framework/mock"
|
|
"github.com/pingcap/tidb/pkg/disttask/framework/proto"
|
|
"github.com/pingcap/tidb/pkg/util/cpu"
|
|
"github.com/stretchr/testify/require"
|
|
"go.uber.org/mock/gomock"
|
|
)
|
|
|
|
func TestMaintainLiveNodes(t *testing.T) {
|
|
ctrl := gomock.NewController(t)
|
|
defer ctrl.Finish()
|
|
mockTaskMgr := mock.NewMockTaskManager(ctrl)
|
|
|
|
require.NoError(t, failpoint.Enable("github.com/pingcap/tidb/pkg/disttask/framework/scheduler/mockTaskExecutorNodes", "return()"))
|
|
t.Cleanup(func() {
|
|
require.NoError(t, failpoint.Disable("github.com/pingcap/tidb/pkg/disttask/framework/scheduler/mockTaskExecutorNodes"))
|
|
})
|
|
|
|
MockServerInfo.Store(&[]string{":4000"})
|
|
|
|
nodeMgr := newNodeManager("")
|
|
ctx := context.Background()
|
|
mockTaskMgr.EXPECT().GetAllNodes(gomock.Any()).Return(nil, errors.New("mock error"))
|
|
nodeMgr.maintainLiveNodes(ctx, mockTaskMgr)
|
|
require.Empty(t, nodeMgr.prevLiveNodes)
|
|
require.True(t, ctrl.Satisfied())
|
|
// no change
|
|
mockTaskMgr.EXPECT().GetAllNodes(gomock.Any()).Return([]proto.ManagedNode{{ID: ":4000"}}, nil)
|
|
nodeMgr.maintainLiveNodes(ctx, mockTaskMgr)
|
|
require.Equal(t, map[string]struct{}{":4000": {}}, nodeMgr.prevLiveNodes)
|
|
require.True(t, ctrl.Satisfied())
|
|
// run again, return fast
|
|
nodeMgr.maintainLiveNodes(ctx, mockTaskMgr)
|
|
require.Equal(t, map[string]struct{}{":4000": {}}, nodeMgr.prevLiveNodes)
|
|
require.True(t, ctrl.Satisfied())
|
|
|
|
// scale out 1 node
|
|
MockServerInfo.Store(&[]string{":4000", ":4001"})
|
|
|
|
// fail on clean
|
|
mockTaskMgr.EXPECT().GetAllNodes(gomock.Any()).Return([]proto.ManagedNode{{ID: ":4000"}, {ID: ":4001"}, {ID: ":4002"}}, nil)
|
|
mockTaskMgr.EXPECT().DeleteDeadNodes(gomock.Any(), gomock.Any()).Return(errors.New("mock error"))
|
|
nodeMgr.maintainLiveNodes(ctx, mockTaskMgr)
|
|
require.Equal(t, map[string]struct{}{":4000": {}}, nodeMgr.prevLiveNodes)
|
|
require.True(t, ctrl.Satisfied())
|
|
// remove 1 node
|
|
mockTaskMgr.EXPECT().GetAllNodes(gomock.Any()).Return([]proto.ManagedNode{{ID: ":4000"}, {ID: ":4001"}, {ID: ":4002"}}, nil)
|
|
mockTaskMgr.EXPECT().DeleteDeadNodes(gomock.Any(), gomock.Any()).Return(nil)
|
|
nodeMgr.maintainLiveNodes(ctx, mockTaskMgr)
|
|
require.Equal(t, map[string]struct{}{":4000": {}, ":4001": {}}, nodeMgr.prevLiveNodes)
|
|
require.True(t, ctrl.Satisfied())
|
|
// run again, return fast
|
|
nodeMgr.maintainLiveNodes(ctx, mockTaskMgr)
|
|
require.Equal(t, map[string]struct{}{":4000": {}, ":4001": {}}, nodeMgr.prevLiveNodes)
|
|
require.True(t, ctrl.Satisfied())
|
|
|
|
// scale in 1 node
|
|
MockServerInfo.Store(&[]string{":4000"})
|
|
|
|
mockTaskMgr.EXPECT().GetAllNodes(gomock.Any()).Return([]proto.ManagedNode{{ID: ":4000"}, {ID: ":4001"}, {ID: ":4002"}}, nil)
|
|
mockTaskMgr.EXPECT().DeleteDeadNodes(gomock.Any(), gomock.Any()).Return(nil)
|
|
nodeMgr.maintainLiveNodes(ctx, mockTaskMgr)
|
|
require.Equal(t, map[string]struct{}{":4000": {}}, nodeMgr.prevLiveNodes)
|
|
require.True(t, ctrl.Satisfied())
|
|
// run again, return fast
|
|
nodeMgr.maintainLiveNodes(ctx, mockTaskMgr)
|
|
require.Equal(t, map[string]struct{}{":4000": {}}, nodeMgr.prevLiveNodes)
|
|
require.True(t, ctrl.Satisfied())
|
|
}
|
|
|
|
func TestMaintainManagedNodes(t *testing.T) {
|
|
ctrl := gomock.NewController(t)
|
|
defer ctrl.Finish()
|
|
ctx := context.Background()
|
|
mockTaskMgr := mock.NewMockTaskManager(ctrl)
|
|
nodeMgr := newNodeManager("")
|
|
|
|
slotMgr := newSlotManager()
|
|
mockTaskMgr.EXPECT().GetAllNodes(gomock.Any()).Return(nil, errors.New("mock error"))
|
|
nodeMgr.refreshNodes(ctx, mockTaskMgr, slotMgr)
|
|
require.Equal(t, cpu.GetCPUCount(), int(slotMgr.capacity.Load()))
|
|
require.Empty(t, nodeMgr.getNodes())
|
|
require.True(t, ctrl.Satisfied())
|
|
|
|
mockTaskMgr.EXPECT().GetAllNodes(gomock.Any()).Return([]proto.ManagedNode{
|
|
{ID: ":4000", CPUCount: 100},
|
|
{ID: ":4001", CPUCount: 100},
|
|
}, nil)
|
|
nodeMgr.refreshNodes(ctx, mockTaskMgr, slotMgr)
|
|
require.Equal(t, []proto.ManagedNode{{ID: ":4000", Role: "", CPUCount: 100}, {ID: ":4001", Role: "", CPUCount: 100}}, nodeMgr.getNodes())
|
|
require.Equal(t, 100, int(slotMgr.capacity.Load()))
|
|
require.True(t, ctrl.Satisfied())
|
|
mockTaskMgr.EXPECT().GetAllNodes(gomock.Any()).Return(nil, nil)
|
|
nodeMgr.refreshNodes(ctx, mockTaskMgr, slotMgr)
|
|
require.NotNil(t, nodeMgr.getNodes())
|
|
require.Empty(t, nodeMgr.getNodes())
|
|
require.Equal(t, 100, int(slotMgr.capacity.Load()))
|
|
require.True(t, ctrl.Satisfied())
|
|
}
|
|
|
|
type filterCase struct {
|
|
nodes []proto.ManagedNode
|
|
targetScope string
|
|
expectedNodes []string
|
|
}
|
|
|
|
func mockManagedNode(id string, role string) proto.ManagedNode {
|
|
return proto.ManagedNode{
|
|
ID: id,
|
|
Role: role,
|
|
CPUCount: 100,
|
|
}
|
|
}
|
|
|
|
func TestFilterByScope(t *testing.T) {
|
|
cases := []filterCase{
|
|
{
|
|
nodes: []proto.ManagedNode{mockManagedNode("1", "background"), mockManagedNode("2", "background"), mockManagedNode("3", "")},
|
|
targetScope: "",
|
|
expectedNodes: []string{"1", "2"},
|
|
},
|
|
{
|
|
nodes: []proto.ManagedNode{mockManagedNode("1", ""), mockManagedNode("2", ""), mockManagedNode("3", "")},
|
|
targetScope: "",
|
|
expectedNodes: []string{"1", "2", "3"},
|
|
},
|
|
{
|
|
nodes: []proto.ManagedNode{mockManagedNode("1", "1"), mockManagedNode("2", ""), mockManagedNode("3", "")},
|
|
targetScope: "",
|
|
expectedNodes: []string{"2", "3"},
|
|
},
|
|
{
|
|
nodes: []proto.ManagedNode{mockManagedNode("1", "1"), mockManagedNode("2", ""), mockManagedNode("3", "")},
|
|
targetScope: "1",
|
|
expectedNodes: []string{"1"},
|
|
},
|
|
{
|
|
nodes: []proto.ManagedNode{mockManagedNode("1", "1"), mockManagedNode("2", "2"), mockManagedNode("3", "2")},
|
|
targetScope: "1",
|
|
expectedNodes: []string{"1"},
|
|
},
|
|
{
|
|
nodes: []proto.ManagedNode{mockManagedNode("1", "1"), mockManagedNode("2", "2"), mockManagedNode("3", "3")},
|
|
targetScope: "2",
|
|
expectedNodes: []string{"2"},
|
|
},
|
|
{
|
|
nodes: []proto.ManagedNode{mockManagedNode("1", "1"), mockManagedNode("2", "2"), mockManagedNode("3", "background")},
|
|
targetScope: "background",
|
|
expectedNodes: []string{"3"},
|
|
},
|
|
{
|
|
nodes: []proto.ManagedNode{mockManagedNode("1", "1"), mockManagedNode("2", ""), mockManagedNode("3", "background")},
|
|
targetScope: "",
|
|
expectedNodes: []string{"3"},
|
|
},
|
|
{
|
|
nodes: []proto.ManagedNode{mockManagedNode("1", ""), mockManagedNode("2", ""), mockManagedNode("3", "")},
|
|
targetScope: "background",
|
|
expectedNodes: nil,
|
|
},
|
|
}
|
|
|
|
for i, cas := range cases {
|
|
t.Run(fmt.Sprintf("case-%d", i), func(t *testing.T) {
|
|
require.Equal(t, cas.expectedNodes, filterByScope(cas.nodes, cas.targetScope))
|
|
})
|
|
}
|
|
}
|