mirror of
https://gitee.com/milvus-io/milvus.git
synced 2024-12-12 13:15:09 +08:00
e09dc3be58
issue: #34985 #35493 after querynode has been suspended, it's not allow to load segment/channel on it, which means the node is read only. to be compatible with resource group design, after query node has been suspend, we remove it from it's original resource group, make it a read only query node in replica. then two things will happens: 1. it's original resource group will be lacking of query nodes, query coord will assign new node to it. 2. querycoord will try to move out all segments/channels after querynode has been suspended Signed-off-by: Wei Liu <wei.liu@zilliz.com>
443 lines
16 KiB
Go
443 lines
16 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 querycoordv2
|
|
|
|
import (
|
|
"context"
|
|
|
|
"github.com/cockroachdb/errors"
|
|
"github.com/samber/lo"
|
|
"go.uber.org/zap"
|
|
|
|
"github.com/milvus-io/milvus-proto/go-api/v2/commonpb"
|
|
"github.com/milvus-io/milvus/internal/proto/querypb"
|
|
"github.com/milvus-io/milvus/internal/querycoordv2/meta"
|
|
"github.com/milvus-io/milvus/internal/querycoordv2/session"
|
|
"github.com/milvus-io/milvus/internal/querycoordv2/utils"
|
|
"github.com/milvus-io/milvus/pkg/log"
|
|
"github.com/milvus-io/milvus/pkg/util/merr"
|
|
"github.com/milvus-io/milvus/pkg/util/typeutil"
|
|
)
|
|
|
|
func (s *Server) ListCheckers(ctx context.Context, req *querypb.ListCheckersRequest) (*querypb.ListCheckersResponse, error) {
|
|
log := log.Ctx(ctx)
|
|
log.Info("list checkers request received")
|
|
if err := merr.CheckHealthy(s.State()); err != nil {
|
|
log.Warn("failed to list checkers", zap.Error(err))
|
|
return &querypb.ListCheckersResponse{
|
|
Status: merr.Status(err),
|
|
}, nil
|
|
}
|
|
checkers := s.checkerController.Checkers()
|
|
checkerIDSet := typeutil.NewSet(req.CheckerIDs...)
|
|
|
|
resp := &querypb.ListCheckersResponse{
|
|
Status: merr.Success(),
|
|
}
|
|
for _, checker := range checkers {
|
|
if checkerIDSet.Len() == 0 || checkerIDSet.Contain(int32(checker.ID())) {
|
|
resp.CheckerInfos = append(resp.CheckerInfos, &querypb.CheckerInfo{
|
|
Id: int32(checker.ID()),
|
|
Activated: checker.IsActive(),
|
|
Desc: checker.ID().String(),
|
|
Found: true,
|
|
})
|
|
checkerIDSet.Remove(int32(checker.ID()))
|
|
}
|
|
}
|
|
|
|
for _, id := range checkerIDSet.Collect() {
|
|
resp.CheckerInfos = append(resp.CheckerInfos, &querypb.CheckerInfo{
|
|
Id: id,
|
|
Found: false,
|
|
})
|
|
}
|
|
|
|
return resp, nil
|
|
}
|
|
|
|
func (s *Server) ActivateChecker(ctx context.Context, req *querypb.ActivateCheckerRequest) (*commonpb.Status, error) {
|
|
log := log.Ctx(ctx)
|
|
log.Info("activate checker request received")
|
|
if err := merr.CheckHealthy(s.State()); err != nil {
|
|
log.Warn("failed to activate checker", zap.Error(err))
|
|
return merr.Status(err), nil
|
|
}
|
|
if err := s.checkerController.Activate(utils.CheckerType(req.CheckerID)); err != nil {
|
|
log.Warn("failed to activate checker", zap.Error(err))
|
|
return merr.Status(merr.WrapErrServiceInternal(err.Error())), nil
|
|
}
|
|
return merr.Success(), nil
|
|
}
|
|
|
|
func (s *Server) DeactivateChecker(ctx context.Context, req *querypb.DeactivateCheckerRequest) (*commonpb.Status, error) {
|
|
log := log.Ctx(ctx)
|
|
log.Info("deactivate checker request received")
|
|
if err := merr.CheckHealthy(s.State()); err != nil {
|
|
log.Warn("failed to deactivate checker", zap.Error(err))
|
|
return merr.Status(err), nil
|
|
}
|
|
if err := s.checkerController.Deactivate(utils.CheckerType(req.CheckerID)); err != nil {
|
|
log.Warn("failed to deactivate checker", zap.Error(err))
|
|
return merr.Status(merr.WrapErrServiceInternal(err.Error())), nil
|
|
}
|
|
return merr.Success(), nil
|
|
}
|
|
|
|
// return all available node list, for each node, return it's (nodeID, ip_address)
|
|
func (s *Server) ListQueryNode(ctx context.Context, req *querypb.ListQueryNodeRequest) (*querypb.ListQueryNodeResponse, error) {
|
|
log := log.Ctx(ctx)
|
|
log.Info("ListQueryNode request received")
|
|
|
|
errMsg := "failed to list querynode state"
|
|
if err := merr.CheckHealthy(s.State()); err != nil {
|
|
log.Warn(errMsg, zap.Error(err))
|
|
return &querypb.ListQueryNodeResponse{
|
|
Status: merr.Status(errors.Wrap(err, errMsg)),
|
|
}, nil
|
|
}
|
|
|
|
nodes := lo.Map(s.nodeMgr.GetAll(), func(nodeInfo *session.NodeInfo, _ int) *querypb.NodeInfo {
|
|
return &querypb.NodeInfo{
|
|
ID: nodeInfo.ID(),
|
|
Address: nodeInfo.Addr(),
|
|
State: nodeInfo.GetState().String(),
|
|
}
|
|
})
|
|
|
|
return &querypb.ListQueryNodeResponse{
|
|
Status: merr.Success(),
|
|
NodeInfos: nodes,
|
|
}, nil
|
|
}
|
|
|
|
// return query node's data distribution, for given nodeID, return it's (channel_name_list, sealed_segment_list)
|
|
func (s *Server) GetQueryNodeDistribution(ctx context.Context, req *querypb.GetQueryNodeDistributionRequest) (*querypb.GetQueryNodeDistributionResponse, error) {
|
|
log := log.Ctx(ctx).With(zap.Int64("nodeID", req.GetNodeID()))
|
|
log.Info("GetQueryNodeDistribution request received")
|
|
|
|
errMsg := "failed to get query node distribution"
|
|
if err := merr.CheckHealthy(s.State()); err != nil {
|
|
log.Warn(errMsg, zap.Error(err))
|
|
return &querypb.GetQueryNodeDistributionResponse{
|
|
Status: merr.Status(errors.Wrap(err, errMsg)),
|
|
}, nil
|
|
}
|
|
|
|
if s.nodeMgr.Get(req.GetNodeID()) == nil {
|
|
err := merr.WrapErrNodeNotFound(req.GetNodeID(), errMsg)
|
|
log.Warn(errMsg, zap.Error(err))
|
|
return &querypb.GetQueryNodeDistributionResponse{
|
|
Status: merr.Status(err),
|
|
}, nil
|
|
}
|
|
|
|
segments := s.dist.SegmentDistManager.GetByFilter(meta.WithNodeID(req.GetNodeID()))
|
|
channels := s.dist.ChannelDistManager.GetByFilter(meta.WithNodeID2Channel(req.GetNodeID()))
|
|
return &querypb.GetQueryNodeDistributionResponse{
|
|
Status: merr.Success(),
|
|
ChannelNames: lo.Map(channels, func(c *meta.DmChannel, _ int) string { return c.GetChannelName() }),
|
|
SealedSegmentIDs: lo.Map(segments, func(s *meta.Segment, _ int) int64 { return s.GetID() }),
|
|
}, nil
|
|
}
|
|
|
|
// suspend background balance for all query node, include stopping balance and auto balance
|
|
func (s *Server) SuspendBalance(ctx context.Context, req *querypb.SuspendBalanceRequest) (*commonpb.Status, error) {
|
|
log := log.Ctx(ctx)
|
|
log.Info("SuspendBalance request received")
|
|
|
|
errMsg := "failed to suspend balance for all querynode"
|
|
if err := merr.CheckHealthy(s.State()); err != nil {
|
|
return merr.Status(err), nil
|
|
}
|
|
|
|
err := s.checkerController.Deactivate(utils.BalanceChecker)
|
|
if err != nil {
|
|
log.Warn(errMsg, zap.Error(err))
|
|
return merr.Status(err), nil
|
|
}
|
|
|
|
return merr.Success(), nil
|
|
}
|
|
|
|
// resume background balance for all query node, include stopping balance and auto balance
|
|
func (s *Server) ResumeBalance(ctx context.Context, req *querypb.ResumeBalanceRequest) (*commonpb.Status, error) {
|
|
log := log.Ctx(ctx)
|
|
|
|
log.Info("ResumeBalance request received")
|
|
|
|
errMsg := "failed to resume balance for all querynode"
|
|
if err := merr.CheckHealthy(s.State()); err != nil {
|
|
return merr.Status(err), nil
|
|
}
|
|
|
|
err := s.checkerController.Activate(utils.BalanceChecker)
|
|
if err != nil {
|
|
log.Warn(errMsg, zap.Error(err))
|
|
return merr.Status(err), nil
|
|
}
|
|
|
|
return merr.Success(), nil
|
|
}
|
|
|
|
// suspend node from resource operation, for given node, suspend load_segment/sub_channel operations
|
|
func (s *Server) SuspendNode(ctx context.Context, req *querypb.SuspendNodeRequest) (*commonpb.Status, error) {
|
|
log := log.Ctx(ctx)
|
|
|
|
log.Info("SuspendNode request received", zap.Int64("nodeID", req.GetNodeID()))
|
|
|
|
errMsg := "failed to suspend query node"
|
|
if err := merr.CheckHealthy(s.State()); err != nil {
|
|
log.Warn(errMsg, zap.Error(err))
|
|
return merr.Status(err), nil
|
|
}
|
|
|
|
if s.nodeMgr.Get(req.GetNodeID()) == nil {
|
|
err := merr.WrapErrNodeNotFound(req.GetNodeID(), errMsg)
|
|
log.Warn(errMsg, zap.Error(err))
|
|
return merr.Status(err), nil
|
|
}
|
|
|
|
s.meta.ResourceManager.HandleNodeDown(req.GetNodeID())
|
|
return merr.Success(), nil
|
|
}
|
|
|
|
// resume node from resource operation, for given node, resume load_segment/sub_channel operations
|
|
func (s *Server) ResumeNode(ctx context.Context, req *querypb.ResumeNodeRequest) (*commonpb.Status, error) {
|
|
log := log.Ctx(ctx)
|
|
log.Info("ResumeNode request received", zap.Int64("nodeID", req.GetNodeID()))
|
|
|
|
errMsg := "failed to resume query node"
|
|
if err := merr.CheckHealthy(s.State()); err != nil {
|
|
log.Warn(errMsg, zap.Error(err))
|
|
return merr.Status(errors.Wrap(err, errMsg)), nil
|
|
}
|
|
|
|
if s.nodeMgr.Get(req.GetNodeID()) == nil {
|
|
err := merr.WrapErrNodeNotFound(req.GetNodeID(), errMsg)
|
|
log.Warn(errMsg, zap.Error(err))
|
|
return merr.Status(err), nil
|
|
}
|
|
|
|
s.meta.ResourceManager.HandleNodeUp(req.GetNodeID())
|
|
|
|
return merr.Success(), nil
|
|
}
|
|
|
|
// transfer segment from source to target,
|
|
// if no segment_id specified, default to transfer all segment on the source node.
|
|
// if no target_nodeId specified, default to move segment to all other nodes
|
|
func (s *Server) TransferSegment(ctx context.Context, req *querypb.TransferSegmentRequest) (*commonpb.Status, error) {
|
|
log := log.Ctx(ctx)
|
|
|
|
log.Info("TransferSegment request received",
|
|
zap.Int64("source", req.GetSourceNodeID()),
|
|
zap.Int64("dest", req.GetTargetNodeID()),
|
|
zap.Int64("segment", req.GetSegmentID()))
|
|
|
|
if err := merr.CheckHealthy(s.State()); err != nil {
|
|
msg := "failed to load balance"
|
|
log.Warn(msg, zap.Error(err))
|
|
return merr.Status(errors.Wrap(err, msg)), nil
|
|
}
|
|
|
|
// check whether srcNode is healthy
|
|
srcNode := req.GetSourceNodeID()
|
|
if err := s.isStoppingNode(srcNode); err != nil {
|
|
err := merr.WrapErrNodeNotAvailable(srcNode, "the source node is invalid")
|
|
return merr.Status(err), nil
|
|
}
|
|
|
|
replicas := s.meta.ReplicaManager.GetByNode(req.GetSourceNodeID())
|
|
for _, replica := range replicas {
|
|
// when no dst node specified, default to use all other nodes in same
|
|
dstNodeSet := typeutil.NewUniqueSet()
|
|
if req.GetToAllNodes() {
|
|
dstNodeSet.Insert(replica.GetRWNodes()...)
|
|
} else {
|
|
// check whether dstNode is healthy
|
|
if err := s.isStoppingNode(req.GetTargetNodeID()); err != nil {
|
|
err := merr.WrapErrNodeNotAvailable(srcNode, "the target node is invalid")
|
|
return merr.Status(err), nil
|
|
}
|
|
dstNodeSet.Insert(req.GetTargetNodeID())
|
|
}
|
|
dstNodeSet.Remove(srcNode)
|
|
|
|
// check sealed segment list
|
|
segments := s.dist.SegmentDistManager.GetByFilter(meta.WithCollectionID(replica.GetCollectionID()), meta.WithNodeID(srcNode))
|
|
|
|
toBalance := typeutil.NewSet[*meta.Segment]()
|
|
if req.GetTransferAll() {
|
|
toBalance.Insert(segments...)
|
|
} else {
|
|
// check whether sealed segment exist
|
|
segment, ok := lo.Find(segments, func(s *meta.Segment) bool { return s.GetID() == req.GetSegmentID() })
|
|
if !ok {
|
|
err := merr.WrapErrSegmentNotFound(req.GetSegmentID(), "segment not found in source node")
|
|
return merr.Status(err), nil
|
|
}
|
|
|
|
existInTarget := s.targetMgr.GetSealedSegment(segment.GetCollectionID(), segment.GetID(), meta.CurrentTarget) != nil
|
|
if !existInTarget {
|
|
log.Info("segment doesn't exist in current target, skip it", zap.Int64("segmentID", req.GetSegmentID()))
|
|
} else {
|
|
toBalance.Insert(segment)
|
|
}
|
|
}
|
|
|
|
err := s.balanceSegments(ctx, replica.GetCollectionID(), replica, srcNode, dstNodeSet.Collect(), toBalance.Collect(), false, req.GetCopyMode())
|
|
if err != nil {
|
|
msg := "failed to balance segments"
|
|
log.Warn(msg, zap.Error(err))
|
|
return merr.Status(errors.Wrap(err, msg)), nil
|
|
}
|
|
}
|
|
return merr.Success(), nil
|
|
}
|
|
|
|
// transfer channel from source to target,
|
|
// if no channel_name specified, default to transfer all channel on the source node.
|
|
// if no target_nodeId specified, default to move channel to all other nodes
|
|
func (s *Server) TransferChannel(ctx context.Context, req *querypb.TransferChannelRequest) (*commonpb.Status, error) {
|
|
log := log.Ctx(ctx)
|
|
|
|
log.Info("TransferChannel request received",
|
|
zap.Int64("source", req.GetSourceNodeID()),
|
|
zap.Int64("dest", req.GetTargetNodeID()),
|
|
zap.String("channel", req.GetChannelName()))
|
|
|
|
if err := merr.CheckHealthy(s.State()); err != nil {
|
|
msg := "failed to load balance"
|
|
log.Warn(msg, zap.Error(err))
|
|
return merr.Status(errors.Wrap(err, msg)), nil
|
|
}
|
|
|
|
// check whether srcNode is healthy
|
|
srcNode := req.GetSourceNodeID()
|
|
if err := s.isStoppingNode(srcNode); err != nil {
|
|
err := merr.WrapErrNodeNotAvailable(srcNode, "the source node is invalid")
|
|
return merr.Status(err), nil
|
|
}
|
|
|
|
replicas := s.meta.ReplicaManager.GetByNode(req.GetSourceNodeID())
|
|
for _, replica := range replicas {
|
|
// when no dst node specified, default to use all other nodes in same
|
|
dstNodeSet := typeutil.NewUniqueSet()
|
|
if req.GetToAllNodes() {
|
|
dstNodeSet.Insert(replica.GetRWNodes()...)
|
|
} else {
|
|
// check whether dstNode is healthy
|
|
if err := s.isStoppingNode(req.GetTargetNodeID()); err != nil {
|
|
err := merr.WrapErrNodeNotAvailable(srcNode, "the target node is invalid")
|
|
return merr.Status(err), nil
|
|
}
|
|
dstNodeSet.Insert(req.GetTargetNodeID())
|
|
}
|
|
dstNodeSet.Remove(srcNode)
|
|
|
|
// check sealed segment list
|
|
channels := s.dist.ChannelDistManager.GetByCollectionAndFilter(replica.GetCollectionID(), meta.WithNodeID2Channel(srcNode))
|
|
toBalance := typeutil.NewSet[*meta.DmChannel]()
|
|
if req.GetTransferAll() {
|
|
toBalance.Insert(channels...)
|
|
} else {
|
|
// check whether sealed segment exist
|
|
channel, ok := lo.Find(channels, func(ch *meta.DmChannel) bool { return ch.GetChannelName() == req.GetChannelName() })
|
|
if !ok {
|
|
err := merr.WrapErrChannelNotFound(req.GetChannelName(), "channel not found in source node")
|
|
return merr.Status(err), nil
|
|
}
|
|
existInTarget := s.targetMgr.GetDmChannel(channel.GetCollectionID(), channel.GetChannelName(), meta.CurrentTarget) != nil
|
|
if !existInTarget {
|
|
log.Info("channel doesn't exist in current target, skip it", zap.String("channelName", channel.GetChannelName()))
|
|
} else {
|
|
toBalance.Insert(channel)
|
|
}
|
|
}
|
|
|
|
err := s.balanceChannels(ctx, replica.GetCollectionID(), replica, srcNode, dstNodeSet.Collect(), toBalance.Collect(), false, req.GetCopyMode())
|
|
if err != nil {
|
|
msg := "failed to balance channels"
|
|
log.Warn(msg, zap.Error(err))
|
|
return merr.Status(errors.Wrap(err, msg)), nil
|
|
}
|
|
}
|
|
return merr.Success(), nil
|
|
}
|
|
|
|
func (s *Server) CheckQueryNodeDistribution(ctx context.Context, req *querypb.CheckQueryNodeDistributionRequest) (*commonpb.Status, error) {
|
|
log := log.Ctx(ctx)
|
|
|
|
log.Info("CheckQueryNodeDistribution request received",
|
|
zap.Int64("source", req.GetSourceNodeID()),
|
|
zap.Int64("dest", req.GetTargetNodeID()))
|
|
|
|
errMsg := "failed to check query node distribution"
|
|
if err := merr.CheckHealthy(s.State()); err != nil {
|
|
log.Warn(errMsg, zap.Error(err))
|
|
return merr.Status(err), nil
|
|
}
|
|
|
|
sourceNode := s.nodeMgr.Get(req.GetSourceNodeID())
|
|
if sourceNode == nil {
|
|
err := merr.WrapErrNodeNotFound(req.GetSourceNodeID(), "source node not found")
|
|
log.Warn(errMsg, zap.Error(err))
|
|
return merr.Status(err), nil
|
|
}
|
|
|
|
targetNode := s.nodeMgr.Get(req.GetTargetNodeID())
|
|
if targetNode == nil {
|
|
err := merr.WrapErrNodeNotFound(req.GetTargetNodeID(), "target node not found")
|
|
log.Warn(errMsg, zap.Error(err))
|
|
return merr.Status(err), nil
|
|
}
|
|
|
|
// check channel list
|
|
channelOnSrc := s.dist.ChannelDistManager.GetByFilter(meta.WithNodeID2Channel(req.GetSourceNodeID()))
|
|
channelOnDst := s.dist.ChannelDistManager.GetByFilter(meta.WithNodeID2Channel(req.GetTargetNodeID()))
|
|
channelDstMap := lo.SliceToMap(channelOnDst, func(ch *meta.DmChannel) (string, *meta.DmChannel) {
|
|
return ch.GetChannelName(), ch
|
|
})
|
|
for _, ch := range channelOnSrc {
|
|
if s.targetMgr.GetDmChannel(ch.GetCollectionID(), ch.GetChannelName(), meta.CurrentTargetFirst) == nil {
|
|
continue
|
|
}
|
|
|
|
if _, ok := channelDstMap[ch.GetChannelName()]; !ok {
|
|
return merr.Status(merr.WrapErrChannelLack(ch.GetChannelName())), nil
|
|
}
|
|
}
|
|
|
|
// check whether all segment exist in source node has been loaded in target node
|
|
segmentOnSrc := s.dist.SegmentDistManager.GetByFilter(meta.WithNodeID(req.GetSourceNodeID()))
|
|
segmentOnDst := s.dist.SegmentDistManager.GetByFilter(meta.WithNodeID(req.GetTargetNodeID()))
|
|
segmentDstMap := lo.SliceToMap(segmentOnDst, func(s *meta.Segment) (int64, *meta.Segment) {
|
|
return s.GetID(), s
|
|
})
|
|
for _, segment := range segmentOnSrc {
|
|
if s.targetMgr.GetSealedSegment(segment.GetCollectionID(), segment.GetID(), meta.CurrentTargetFirst) == nil {
|
|
continue
|
|
}
|
|
|
|
if _, ok := segmentDstMap[segment.GetID()]; !ok {
|
|
return merr.Status(merr.WrapErrSegmentLack(segment.GetID())), nil
|
|
}
|
|
}
|
|
return merr.Success(), nil
|
|
}
|