2021-04-19 05:47:10 +00:00
|
|
|
// Copyright (C) 2019-2020 Zilliz. All rights reserved.
|
|
|
|
//
|
|
|
|
// 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.
|
|
|
|
|
2021-04-12 01:18:43 +00:00
|
|
|
package querynode
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"math/rand"
|
2021-04-16 06:40:33 +00:00
|
|
|
"strconv"
|
|
|
|
"strings"
|
2021-04-12 01:18:43 +00:00
|
|
|
|
|
|
|
"go.uber.org/zap"
|
|
|
|
|
2021-04-22 06:45:57 +00:00
|
|
|
"github.com/milvus-io/milvus/internal/log"
|
|
|
|
"github.com/milvus-io/milvus/internal/msgstream"
|
|
|
|
"github.com/milvus-io/milvus/internal/proto/internalpb"
|
|
|
|
queryPb "github.com/milvus-io/milvus/internal/proto/querypb"
|
2021-04-12 01:18:43 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type task interface {
|
|
|
|
ID() UniqueID // return ReqID
|
|
|
|
SetID(uid UniqueID) // set ReqID
|
|
|
|
Timestamp() Timestamp
|
|
|
|
PreExecute(ctx context.Context) error
|
|
|
|
Execute(ctx context.Context) error
|
|
|
|
PostExecute(ctx context.Context) error
|
|
|
|
WaitToFinish() error
|
|
|
|
Notify(err error)
|
|
|
|
OnEnqueue() error
|
|
|
|
}
|
|
|
|
|
|
|
|
type baseTask struct {
|
|
|
|
done chan error
|
|
|
|
ctx context.Context
|
|
|
|
id UniqueID
|
|
|
|
}
|
|
|
|
|
2021-04-16 06:40:33 +00:00
|
|
|
type watchDmChannelsTask struct {
|
|
|
|
baseTask
|
|
|
|
req *queryPb.WatchDmChannelsRequest
|
|
|
|
node *QueryNode
|
|
|
|
}
|
|
|
|
|
2021-04-12 01:18:43 +00:00
|
|
|
type loadSegmentsTask struct {
|
|
|
|
baseTask
|
|
|
|
req *queryPb.LoadSegmentsRequest
|
|
|
|
node *QueryNode
|
|
|
|
}
|
|
|
|
|
|
|
|
type releaseCollectionTask struct {
|
|
|
|
baseTask
|
|
|
|
req *queryPb.ReleaseCollectionRequest
|
|
|
|
node *QueryNode
|
|
|
|
}
|
|
|
|
|
|
|
|
type releasePartitionsTask struct {
|
|
|
|
baseTask
|
|
|
|
req *queryPb.ReleasePartitionsRequest
|
|
|
|
node *QueryNode
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *baseTask) ID() UniqueID {
|
|
|
|
return b.id
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *baseTask) SetID(uid UniqueID) {
|
|
|
|
b.id = uid
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *baseTask) WaitToFinish() error {
|
2021-04-16 06:40:33 +00:00
|
|
|
err := <-b.done
|
|
|
|
return err
|
2021-04-12 01:18:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (b *baseTask) Notify(err error) {
|
|
|
|
b.done <- err
|
|
|
|
}
|
|
|
|
|
2021-04-16 06:40:33 +00:00
|
|
|
// watchDmChannelsTask
|
|
|
|
func (w *watchDmChannelsTask) Timestamp() Timestamp {
|
|
|
|
if w.req.Base == nil {
|
|
|
|
log.Error("nil base req in watchDmChannelsTask", zap.Any("collectionID", w.req.CollectionID))
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
return w.req.Base.Timestamp
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *watchDmChannelsTask) OnEnqueue() error {
|
|
|
|
if w.req == nil || w.req.Base == nil {
|
|
|
|
w.SetID(rand.Int63n(100000000000))
|
|
|
|
} else {
|
|
|
|
w.SetID(w.req.Base.MsgID)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *watchDmChannelsTask) PreExecute(ctx context.Context) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *watchDmChannelsTask) Execute(ctx context.Context) error {
|
|
|
|
log.Debug("starting WatchDmChannels ...", zap.String("ChannelIDs", fmt.Sprintln(w.req.ChannelIDs)))
|
|
|
|
collectionID := w.req.CollectionID
|
2021-05-28 02:26:30 +00:00
|
|
|
ds, err := w.node.streaming.getDataSyncService(collectionID)
|
2021-04-16 06:40:33 +00:00
|
|
|
if err != nil || ds.dmStream == nil {
|
|
|
|
errMsg := "null data sync service or null data manipulation stream, collectionID = " + fmt.Sprintln(collectionID)
|
|
|
|
log.Error(errMsg)
|
|
|
|
return errors.New(errMsg)
|
|
|
|
}
|
|
|
|
|
|
|
|
switch t := ds.dmStream.(type) {
|
|
|
|
case *msgstream.MqTtMsgStream:
|
|
|
|
default:
|
|
|
|
_ = t
|
|
|
|
errMsg := "type assertion failed for dm message stream"
|
|
|
|
log.Error(errMsg)
|
|
|
|
return errors.New(errMsg)
|
|
|
|
}
|
|
|
|
|
|
|
|
getUniqueSubName := func() string {
|
|
|
|
prefixName := Params.MsgChannelSubName
|
|
|
|
return prefixName + "-" + strconv.FormatInt(collectionID, 10)
|
|
|
|
}
|
|
|
|
|
|
|
|
// add request channel
|
|
|
|
consumeChannels := w.req.ChannelIDs
|
|
|
|
toSeekInfo := make([]*internalpb.MsgPosition, 0)
|
|
|
|
toDirSubChannels := make([]string, 0)
|
|
|
|
|
|
|
|
consumeSubName := getUniqueSubName()
|
|
|
|
|
|
|
|
for _, info := range w.req.Infos {
|
|
|
|
if len(info.Pos.MsgID) == 0 {
|
|
|
|
toDirSubChannels = append(toDirSubChannels, info.ChannelID)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
info.Pos.MsgGroup = consumeSubName
|
|
|
|
toSeekInfo = append(toSeekInfo, info.Pos)
|
|
|
|
|
|
|
|
log.Debug("prevent inserting segments", zap.String("segmentIDs", fmt.Sprintln(info.ExcludedSegments)))
|
2021-05-28 02:26:30 +00:00
|
|
|
err := w.node.streaming.replica.addExcludedSegments(collectionID, info.ExcludedSegments)
|
2021-04-16 06:40:33 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Error(err.Error())
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ds.dmStream.AsConsumer(toDirSubChannels, consumeSubName)
|
|
|
|
for _, pos := range toSeekInfo {
|
2021-05-29 15:21:34 +00:00
|
|
|
err := ds.dmStream.Seek([]*internalpb.MsgPosition{pos})
|
2021-04-16 06:40:33 +00:00
|
|
|
if err != nil {
|
|
|
|
errMsg := "msgStream seek error :" + err.Error()
|
|
|
|
log.Error(errMsg)
|
|
|
|
return errors.New(errMsg)
|
|
|
|
}
|
|
|
|
}
|
2021-05-28 07:40:32 +00:00
|
|
|
|
|
|
|
collection, err := w.node.streaming.replica.getCollectionByID(collectionID)
|
|
|
|
if err != nil {
|
|
|
|
log.Error(err.Error())
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
collection.addWatchedDmChannels(w.req.ChannelIDs)
|
2021-04-16 06:40:33 +00:00
|
|
|
log.Debug("querynode AsConsumer: " + strings.Join(consumeChannels, ", ") + " : " + consumeSubName)
|
|
|
|
log.Debug("WatchDmChannels done", zap.String("ChannelIDs", fmt.Sprintln(w.req.ChannelIDs)))
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *watchDmChannelsTask) PostExecute(ctx context.Context) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-04-12 01:18:43 +00:00
|
|
|
// loadSegmentsTask
|
|
|
|
func (l *loadSegmentsTask) Timestamp() Timestamp {
|
2021-04-16 06:40:33 +00:00
|
|
|
if l.req.Base == nil {
|
|
|
|
log.Error("nil base req in loadSegmentsTask", zap.Any("collectionID", l.req.CollectionID))
|
|
|
|
return 0
|
|
|
|
}
|
2021-04-12 01:18:43 +00:00
|
|
|
return l.req.Base.Timestamp
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *loadSegmentsTask) OnEnqueue() error {
|
|
|
|
if l.req == nil || l.req.Base == nil {
|
|
|
|
l.SetID(rand.Int63n(100000000000))
|
|
|
|
} else {
|
|
|
|
l.SetID(l.req.Base.MsgID)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *loadSegmentsTask) PreExecute(ctx context.Context) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *loadSegmentsTask) Execute(ctx context.Context) error {
|
|
|
|
// TODO: support db
|
|
|
|
collectionID := l.req.CollectionID
|
|
|
|
partitionID := l.req.PartitionID
|
|
|
|
segmentIDs := l.req.SegmentIDs
|
|
|
|
fieldIDs := l.req.FieldIDs
|
|
|
|
schema := l.req.Schema
|
|
|
|
|
|
|
|
log.Debug("query node load segment", zap.String("loadSegmentRequest", fmt.Sprintln(l.req)))
|
|
|
|
|
2021-05-28 02:26:30 +00:00
|
|
|
hasCollectionInHistorical := l.node.historical.replica.hasCollection(collectionID)
|
|
|
|
hasPartitionInHistorical := l.node.historical.replica.hasPartition(partitionID)
|
|
|
|
if !hasCollectionInHistorical {
|
2021-04-12 01:18:43 +00:00
|
|
|
// loading init
|
2021-05-28 02:26:30 +00:00
|
|
|
err := l.node.historical.replica.addCollection(collectionID, schema)
|
2021-04-12 01:18:43 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-05-28 02:26:30 +00:00
|
|
|
|
|
|
|
hasCollectionInStreaming := l.node.streaming.replica.hasCollection(collectionID)
|
|
|
|
if !hasCollectionInStreaming {
|
|
|
|
err = l.node.streaming.replica.addCollection(collectionID, schema)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
l.node.streaming.replica.initExcludedSegments(collectionID)
|
2021-05-28 07:40:32 +00:00
|
|
|
newDS := newDataSyncService(l.node.queryNodeLoopCtx,
|
|
|
|
l.node.streaming.replica,
|
|
|
|
l.node.streaming.tSafeReplica,
|
|
|
|
l.node.msFactory,
|
|
|
|
collectionID)
|
2021-04-12 01:18:43 +00:00
|
|
|
// ignore duplicated dataSyncService error
|
2021-05-28 02:26:30 +00:00
|
|
|
_ = l.node.streaming.addDataSyncService(collectionID, newDS)
|
|
|
|
ds, err := l.node.streaming.getDataSyncService(collectionID)
|
2021-04-12 01:18:43 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
go ds.start()
|
|
|
|
l.node.searchService.startSearchCollection(collectionID)
|
|
|
|
}
|
2021-05-28 02:26:30 +00:00
|
|
|
if !hasPartitionInHistorical {
|
|
|
|
err := l.node.historical.replica.addPartition(collectionID, partitionID)
|
2021-04-12 01:18:43 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-05-28 02:26:30 +00:00
|
|
|
|
|
|
|
hasPartitionInStreaming := l.node.streaming.replica.hasPartition(partitionID)
|
|
|
|
if !hasPartitionInStreaming {
|
|
|
|
err = l.node.streaming.replica.addPartition(collectionID, partitionID)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
2021-04-12 01:18:43 +00:00
|
|
|
}
|
2021-05-28 02:26:30 +00:00
|
|
|
err := l.node.streaming.replica.enablePartition(partitionID)
|
2021-04-12 01:18:43 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(segmentIDs) == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-05-28 02:26:30 +00:00
|
|
|
err = l.node.historical.loadService.loadSegmentPassively(collectionID, partitionID, segmentIDs, fieldIDs)
|
2021-04-12 01:18:43 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Debug("LoadSegments done", zap.String("segmentIDs", fmt.Sprintln(l.req.SegmentIDs)))
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *loadSegmentsTask) PostExecute(ctx context.Context) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// releaseCollectionTask
|
|
|
|
func (r *releaseCollectionTask) Timestamp() Timestamp {
|
2021-04-16 06:40:33 +00:00
|
|
|
if r.req.Base == nil {
|
|
|
|
log.Error("nil base req in releaseCollectionTask", zap.Any("collectionID", r.req.CollectionID))
|
|
|
|
return 0
|
|
|
|
}
|
2021-04-12 01:18:43 +00:00
|
|
|
return r.req.Base.Timestamp
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *releaseCollectionTask) OnEnqueue() error {
|
|
|
|
if r.req == nil || r.req.Base == nil {
|
|
|
|
r.SetID(rand.Int63n(100000000000))
|
|
|
|
} else {
|
|
|
|
r.SetID(r.req.Base.MsgID)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *releaseCollectionTask) PreExecute(ctx context.Context) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *releaseCollectionTask) Execute(ctx context.Context) error {
|
2021-05-28 02:26:30 +00:00
|
|
|
ds, err := r.node.streaming.getDataSyncService(r.req.CollectionID)
|
2021-04-12 01:18:43 +00:00
|
|
|
if err == nil && ds != nil {
|
|
|
|
ds.close()
|
2021-05-28 02:26:30 +00:00
|
|
|
r.node.streaming.removeDataSyncService(r.req.CollectionID)
|
2021-05-28 07:40:32 +00:00
|
|
|
// TODO: remove and use vChannel
|
|
|
|
vChannel := collectionIDToChannel(r.req.CollectionID)
|
|
|
|
r.node.streaming.tSafeReplica.removeTSafe(vChannel)
|
2021-05-28 02:26:30 +00:00
|
|
|
r.node.streaming.replica.removeExcludedSegments(r.req.CollectionID)
|
2021-04-12 01:18:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if r.node.searchService.hasSearchCollection(r.req.CollectionID) {
|
|
|
|
r.node.searchService.stopSearchCollection(r.req.CollectionID)
|
|
|
|
}
|
|
|
|
|
2021-05-28 02:26:30 +00:00
|
|
|
hasCollectionInHistorical := r.node.historical.replica.hasCollection(r.req.CollectionID)
|
|
|
|
if hasCollectionInHistorical {
|
|
|
|
err := r.node.historical.replica.removeCollection(r.req.CollectionID)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
hasCollectionInStreaming := r.node.streaming.replica.hasCollection(r.req.CollectionID)
|
|
|
|
if hasCollectionInStreaming {
|
|
|
|
err := r.node.streaming.replica.removePartition(r.req.CollectionID)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-04-12 01:18:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
log.Debug("ReleaseCollection done", zap.Int64("collectionID", r.req.CollectionID))
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *releaseCollectionTask) PostExecute(ctx context.Context) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// releasePartitionsTask
|
|
|
|
func (r *releasePartitionsTask) Timestamp() Timestamp {
|
2021-04-16 06:40:33 +00:00
|
|
|
if r.req.Base == nil {
|
|
|
|
log.Error("nil base req in releasePartitionsTask", zap.Any("collectionID", r.req.CollectionID))
|
|
|
|
return 0
|
|
|
|
}
|
2021-04-12 01:18:43 +00:00
|
|
|
return r.req.Base.Timestamp
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *releasePartitionsTask) OnEnqueue() error {
|
|
|
|
if r.req == nil || r.req.Base == nil {
|
|
|
|
r.SetID(rand.Int63n(100000000000))
|
|
|
|
} else {
|
|
|
|
r.SetID(r.req.Base.MsgID)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *releasePartitionsTask) PreExecute(ctx context.Context) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *releasePartitionsTask) Execute(ctx context.Context) error {
|
|
|
|
for _, id := range r.req.PartitionIDs {
|
2021-05-28 02:26:30 +00:00
|
|
|
hasPartitionInHistorical := r.node.historical.replica.hasPartition(id)
|
|
|
|
if hasPartitionInHistorical {
|
|
|
|
err := r.node.historical.replica.removePartition(id)
|
|
|
|
if err != nil {
|
|
|
|
// not return, try to release all partitions
|
|
|
|
log.Error(err.Error())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
hasPartitionInStreaming := r.node.streaming.replica.hasPartition(id)
|
|
|
|
if hasPartitionInStreaming {
|
|
|
|
err := r.node.streaming.replica.removePartition(id)
|
|
|
|
if err != nil {
|
|
|
|
log.Error(err.Error())
|
|
|
|
}
|
2021-04-12 01:18:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *releasePartitionsTask) PostExecute(ctx context.Context) error {
|
|
|
|
return nil
|
|
|
|
}
|