mirror of https://github.com/milvus-io/milvus.git
464 lines
13 KiB
Go
464 lines
13 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 indexnode
|
|
|
|
import (
|
|
"context"
|
|
"time"
|
|
|
|
"go.uber.org/zap"
|
|
|
|
"github.com/milvus-io/milvus-proto/go-api/v2/commonpb"
|
|
"github.com/milvus-io/milvus/internal/proto/datapb"
|
|
"github.com/milvus-io/milvus/internal/proto/indexpb"
|
|
"github.com/milvus-io/milvus/pkg/common"
|
|
"github.com/milvus-io/milvus/pkg/log"
|
|
)
|
|
|
|
type indexTaskInfo struct {
|
|
cancel context.CancelFunc
|
|
state commonpb.IndexState
|
|
fileKeys []string
|
|
serializedSize uint64
|
|
failReason string
|
|
currentIndexVersion int32
|
|
indexStoreVersion int64
|
|
|
|
// task statistics
|
|
statistic *indexpb.JobInfo
|
|
}
|
|
|
|
func (i *IndexNode) loadOrStoreIndexTask(ClusterID string, buildID UniqueID, info *indexTaskInfo) *indexTaskInfo {
|
|
i.stateLock.Lock()
|
|
defer i.stateLock.Unlock()
|
|
key := taskKey{ClusterID: ClusterID, TaskID: buildID}
|
|
oldInfo, ok := i.indexTasks[key]
|
|
if ok {
|
|
return oldInfo
|
|
}
|
|
i.indexTasks[key] = info
|
|
return nil
|
|
}
|
|
|
|
func (i *IndexNode) loadIndexTaskState(ClusterID string, buildID UniqueID) commonpb.IndexState {
|
|
key := taskKey{ClusterID: ClusterID, TaskID: buildID}
|
|
i.stateLock.Lock()
|
|
defer i.stateLock.Unlock()
|
|
task, ok := i.indexTasks[key]
|
|
if !ok {
|
|
return commonpb.IndexState_IndexStateNone
|
|
}
|
|
return task.state
|
|
}
|
|
|
|
func (i *IndexNode) storeIndexTaskState(ClusterID string, buildID UniqueID, state commonpb.IndexState, failReason string) {
|
|
key := taskKey{ClusterID: ClusterID, TaskID: buildID}
|
|
i.stateLock.Lock()
|
|
defer i.stateLock.Unlock()
|
|
if task, ok := i.indexTasks[key]; ok {
|
|
log.Debug("IndexNode store task state", zap.String("clusterID", ClusterID), zap.Int64("buildID", buildID),
|
|
zap.String("state", state.String()), zap.String("fail reason", failReason))
|
|
task.state = state
|
|
task.failReason = failReason
|
|
}
|
|
}
|
|
|
|
func (i *IndexNode) foreachIndexTaskInfo(fn func(ClusterID string, buildID UniqueID, info *indexTaskInfo)) {
|
|
i.stateLock.Lock()
|
|
defer i.stateLock.Unlock()
|
|
for key, info := range i.indexTasks {
|
|
fn(key.ClusterID, key.TaskID, info)
|
|
}
|
|
}
|
|
|
|
func (i *IndexNode) storeIndexFilesAndStatistic(
|
|
ClusterID string,
|
|
buildID UniqueID,
|
|
fileKeys []string,
|
|
serializedSize uint64,
|
|
currentIndexVersion int32,
|
|
) {
|
|
key := taskKey{ClusterID: ClusterID, TaskID: buildID}
|
|
i.stateLock.Lock()
|
|
defer i.stateLock.Unlock()
|
|
if info, ok := i.indexTasks[key]; ok {
|
|
info.fileKeys = common.CloneStringList(fileKeys)
|
|
info.serializedSize = serializedSize
|
|
info.currentIndexVersion = currentIndexVersion
|
|
return
|
|
}
|
|
}
|
|
|
|
func (i *IndexNode) storeIndexFilesAndStatisticV2(
|
|
ClusterID string,
|
|
buildID UniqueID,
|
|
fileKeys []string,
|
|
serializedSize uint64,
|
|
currentIndexVersion int32,
|
|
indexStoreVersion int64,
|
|
) {
|
|
key := taskKey{ClusterID: ClusterID, TaskID: buildID}
|
|
i.stateLock.Lock()
|
|
defer i.stateLock.Unlock()
|
|
if info, ok := i.indexTasks[key]; ok {
|
|
info.fileKeys = common.CloneStringList(fileKeys)
|
|
info.serializedSize = serializedSize
|
|
info.currentIndexVersion = currentIndexVersion
|
|
info.indexStoreVersion = indexStoreVersion
|
|
return
|
|
}
|
|
}
|
|
|
|
func (i *IndexNode) deleteIndexTaskInfos(ctx context.Context, keys []taskKey) []*indexTaskInfo {
|
|
i.stateLock.Lock()
|
|
defer i.stateLock.Unlock()
|
|
deleted := make([]*indexTaskInfo, 0, len(keys))
|
|
for _, key := range keys {
|
|
info, ok := i.indexTasks[key]
|
|
if ok {
|
|
deleted = append(deleted, info)
|
|
delete(i.indexTasks, key)
|
|
log.Ctx(ctx).Info("delete task infos",
|
|
zap.String("cluster_id", key.ClusterID), zap.Int64("build_id", key.TaskID))
|
|
}
|
|
}
|
|
return deleted
|
|
}
|
|
|
|
func (i *IndexNode) deleteAllIndexTasks() []*indexTaskInfo {
|
|
i.stateLock.Lock()
|
|
deletedTasks := i.indexTasks
|
|
i.indexTasks = make(map[taskKey]*indexTaskInfo)
|
|
i.stateLock.Unlock()
|
|
|
|
deleted := make([]*indexTaskInfo, 0, len(deletedTasks))
|
|
for _, info := range deletedTasks {
|
|
deleted = append(deleted, info)
|
|
}
|
|
return deleted
|
|
}
|
|
|
|
type analyzeTaskInfo struct {
|
|
cancel context.CancelFunc
|
|
state indexpb.JobState
|
|
failReason string
|
|
centroidsFile string
|
|
}
|
|
|
|
func (i *IndexNode) loadOrStoreAnalyzeTask(clusterID string, taskID UniqueID, info *analyzeTaskInfo) *analyzeTaskInfo {
|
|
i.stateLock.Lock()
|
|
defer i.stateLock.Unlock()
|
|
key := taskKey{ClusterID: clusterID, TaskID: taskID}
|
|
oldInfo, ok := i.analyzeTasks[key]
|
|
if ok {
|
|
return oldInfo
|
|
}
|
|
i.analyzeTasks[key] = info
|
|
return nil
|
|
}
|
|
|
|
func (i *IndexNode) loadAnalyzeTaskState(clusterID string, taskID UniqueID) indexpb.JobState {
|
|
key := taskKey{ClusterID: clusterID, TaskID: taskID}
|
|
i.stateLock.Lock()
|
|
defer i.stateLock.Unlock()
|
|
task, ok := i.analyzeTasks[key]
|
|
if !ok {
|
|
return indexpb.JobState_JobStateNone
|
|
}
|
|
return task.state
|
|
}
|
|
|
|
func (i *IndexNode) storeAnalyzeTaskState(clusterID string, taskID UniqueID, state indexpb.JobState, failReason string) {
|
|
key := taskKey{ClusterID: clusterID, TaskID: taskID}
|
|
i.stateLock.Lock()
|
|
defer i.stateLock.Unlock()
|
|
if task, ok := i.analyzeTasks[key]; ok {
|
|
log.Info("IndexNode store analyze task state", zap.String("clusterID", clusterID), zap.Int64("TaskID", taskID),
|
|
zap.String("state", state.String()), zap.String("fail reason", failReason))
|
|
task.state = state
|
|
task.failReason = failReason
|
|
}
|
|
}
|
|
|
|
func (i *IndexNode) foreachAnalyzeTaskInfo(fn func(clusterID string, taskID UniqueID, info *analyzeTaskInfo)) {
|
|
i.stateLock.Lock()
|
|
defer i.stateLock.Unlock()
|
|
for key, info := range i.analyzeTasks {
|
|
fn(key.ClusterID, key.TaskID, info)
|
|
}
|
|
}
|
|
|
|
func (i *IndexNode) storeAnalyzeFilesAndStatistic(
|
|
ClusterID string,
|
|
taskID UniqueID,
|
|
centroidsFile string,
|
|
) {
|
|
key := taskKey{ClusterID: ClusterID, TaskID: taskID}
|
|
i.stateLock.Lock()
|
|
defer i.stateLock.Unlock()
|
|
if info, ok := i.analyzeTasks[key]; ok {
|
|
info.centroidsFile = centroidsFile
|
|
return
|
|
}
|
|
}
|
|
|
|
func (i *IndexNode) getAnalyzeTaskInfo(clusterID string, taskID UniqueID) *analyzeTaskInfo {
|
|
i.stateLock.Lock()
|
|
defer i.stateLock.Unlock()
|
|
|
|
if info, ok := i.analyzeTasks[taskKey{ClusterID: clusterID, TaskID: taskID}]; ok {
|
|
return &analyzeTaskInfo{
|
|
cancel: info.cancel,
|
|
state: info.state,
|
|
failReason: info.failReason,
|
|
centroidsFile: info.centroidsFile,
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (i *IndexNode) deleteAnalyzeTaskInfos(ctx context.Context, keys []taskKey) []*analyzeTaskInfo {
|
|
i.stateLock.Lock()
|
|
defer i.stateLock.Unlock()
|
|
deleted := make([]*analyzeTaskInfo, 0, len(keys))
|
|
for _, key := range keys {
|
|
info, ok := i.analyzeTasks[key]
|
|
if ok {
|
|
deleted = append(deleted, info)
|
|
delete(i.analyzeTasks, key)
|
|
log.Ctx(ctx).Info("delete analyze task infos",
|
|
zap.String("clusterID", key.ClusterID), zap.Int64("TaskID", key.TaskID))
|
|
}
|
|
}
|
|
return deleted
|
|
}
|
|
|
|
func (i *IndexNode) deleteAllAnalyzeTasks() []*analyzeTaskInfo {
|
|
i.stateLock.Lock()
|
|
deletedTasks := i.analyzeTasks
|
|
i.analyzeTasks = make(map[taskKey]*analyzeTaskInfo)
|
|
i.stateLock.Unlock()
|
|
|
|
deleted := make([]*analyzeTaskInfo, 0, len(deletedTasks))
|
|
for _, info := range deletedTasks {
|
|
deleted = append(deleted, info)
|
|
}
|
|
return deleted
|
|
}
|
|
|
|
func (i *IndexNode) hasInProgressTask() bool {
|
|
i.stateLock.Lock()
|
|
defer i.stateLock.Unlock()
|
|
for _, info := range i.indexTasks {
|
|
if info.state == commonpb.IndexState_InProgress {
|
|
return true
|
|
}
|
|
}
|
|
|
|
for _, info := range i.analyzeTasks {
|
|
if info.state == indexpb.JobState_JobStateInProgress {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
func (i *IndexNode) waitTaskFinish() {
|
|
if !i.hasInProgressTask() {
|
|
return
|
|
}
|
|
|
|
gracefulTimeout := &Params.IndexNodeCfg.GracefulStopTimeout
|
|
ticker := time.NewTicker(time.Second)
|
|
defer ticker.Stop()
|
|
|
|
timeoutCtx, cancel := context.WithTimeout(i.loopCtx, gracefulTimeout.GetAsDuration(time.Second))
|
|
defer cancel()
|
|
for {
|
|
select {
|
|
case <-ticker.C:
|
|
if !i.hasInProgressTask() {
|
|
return
|
|
}
|
|
case <-timeoutCtx.Done():
|
|
log.Warn("timeout, the index node has some progress task")
|
|
for _, info := range i.indexTasks {
|
|
if info.state == commonpb.IndexState_InProgress {
|
|
log.Warn("progress task", zap.Any("info", info))
|
|
}
|
|
}
|
|
for _, info := range i.analyzeTasks {
|
|
if info.state == indexpb.JobState_JobStateInProgress {
|
|
log.Warn("progress task", zap.Any("info", info))
|
|
}
|
|
}
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
type statsTaskInfo struct {
|
|
cancel context.CancelFunc
|
|
state indexpb.JobState
|
|
failReason string
|
|
collID UniqueID
|
|
partID UniqueID
|
|
segID UniqueID
|
|
insertChannel string
|
|
numRows int64
|
|
insertLogs []*datapb.FieldBinlog
|
|
statsLogs []*datapb.FieldBinlog
|
|
textStatsLogs map[int64]*datapb.TextIndexStats
|
|
bm25Logs []*datapb.FieldBinlog
|
|
}
|
|
|
|
func (i *IndexNode) loadOrStoreStatsTask(clusterID string, taskID UniqueID, info *statsTaskInfo) *statsTaskInfo {
|
|
i.stateLock.Lock()
|
|
defer i.stateLock.Unlock()
|
|
key := taskKey{ClusterID: clusterID, TaskID: taskID}
|
|
oldInfo, ok := i.statsTasks[key]
|
|
if ok {
|
|
return oldInfo
|
|
}
|
|
i.statsTasks[key] = info
|
|
return nil
|
|
}
|
|
|
|
func (i *IndexNode) getStatsTaskState(clusterID string, taskID UniqueID) indexpb.JobState {
|
|
key := taskKey{ClusterID: clusterID, TaskID: taskID}
|
|
i.stateLock.Lock()
|
|
defer i.stateLock.Unlock()
|
|
task, ok := i.statsTasks[key]
|
|
if !ok {
|
|
return indexpb.JobState_JobStateNone
|
|
}
|
|
return task.state
|
|
}
|
|
|
|
func (i *IndexNode) storeStatsTaskState(clusterID string, taskID UniqueID, state indexpb.JobState, failReason string) {
|
|
key := taskKey{ClusterID: clusterID, TaskID: taskID}
|
|
i.stateLock.Lock()
|
|
defer i.stateLock.Unlock()
|
|
if task, ok := i.statsTasks[key]; ok {
|
|
log.Info("IndexNode store stats task state", zap.String("clusterID", clusterID), zap.Int64("TaskID", taskID),
|
|
zap.String("state", state.String()), zap.String("fail reason", failReason))
|
|
task.state = state
|
|
task.failReason = failReason
|
|
}
|
|
}
|
|
|
|
func (i *IndexNode) storePKSortStatsResult(
|
|
ClusterID string,
|
|
taskID UniqueID,
|
|
collID UniqueID,
|
|
partID UniqueID,
|
|
segID UniqueID,
|
|
channel string,
|
|
numRows int64,
|
|
insertLogs []*datapb.FieldBinlog,
|
|
statsLogs []*datapb.FieldBinlog,
|
|
bm25Logs []*datapb.FieldBinlog,
|
|
) {
|
|
key := taskKey{ClusterID: ClusterID, TaskID: taskID}
|
|
i.stateLock.Lock()
|
|
defer i.stateLock.Unlock()
|
|
if info, ok := i.statsTasks[key]; ok {
|
|
info.collID = collID
|
|
info.partID = partID
|
|
info.segID = segID
|
|
info.insertChannel = channel
|
|
info.numRows = numRows
|
|
info.insertLogs = insertLogs
|
|
info.statsLogs = statsLogs
|
|
info.bm25Logs = bm25Logs
|
|
return
|
|
}
|
|
}
|
|
|
|
func (i *IndexNode) storeStatsTextIndexResult(
|
|
ClusterID string,
|
|
taskID UniqueID,
|
|
collID UniqueID,
|
|
partID UniqueID,
|
|
segID UniqueID,
|
|
channel string,
|
|
texIndexLogs map[int64]*datapb.TextIndexStats,
|
|
) {
|
|
key := taskKey{ClusterID: ClusterID, TaskID: taskID}
|
|
i.stateLock.Lock()
|
|
defer i.stateLock.Unlock()
|
|
if info, ok := i.statsTasks[key]; ok {
|
|
info.textStatsLogs = texIndexLogs
|
|
info.segID = segID
|
|
info.collID = collID
|
|
info.partID = partID
|
|
info.insertChannel = channel
|
|
}
|
|
}
|
|
|
|
func (i *IndexNode) getStatsTaskInfo(clusterID string, taskID UniqueID) *statsTaskInfo {
|
|
i.stateLock.Lock()
|
|
defer i.stateLock.Unlock()
|
|
|
|
if info, ok := i.statsTasks[taskKey{ClusterID: clusterID, TaskID: taskID}]; ok {
|
|
return &statsTaskInfo{
|
|
cancel: info.cancel,
|
|
state: info.state,
|
|
failReason: info.failReason,
|
|
collID: info.collID,
|
|
partID: info.partID,
|
|
segID: info.segID,
|
|
insertChannel: info.insertChannel,
|
|
numRows: info.numRows,
|
|
insertLogs: info.insertLogs,
|
|
statsLogs: info.statsLogs,
|
|
textStatsLogs: info.textStatsLogs,
|
|
bm25Logs: info.bm25Logs,
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (i *IndexNode) deleteStatsTaskInfos(ctx context.Context, keys []taskKey) []*statsTaskInfo {
|
|
i.stateLock.Lock()
|
|
defer i.stateLock.Unlock()
|
|
deleted := make([]*statsTaskInfo, 0, len(keys))
|
|
for _, key := range keys {
|
|
info, ok := i.statsTasks[key]
|
|
if ok {
|
|
deleted = append(deleted, info)
|
|
delete(i.statsTasks, key)
|
|
log.Ctx(ctx).Info("delete stats task infos",
|
|
zap.String("clusterID", key.ClusterID), zap.Int64("TaskID", key.TaskID))
|
|
}
|
|
}
|
|
return deleted
|
|
}
|
|
|
|
func (i *IndexNode) deleteAllStatsTasks() []*statsTaskInfo {
|
|
i.stateLock.Lock()
|
|
deletedTasks := i.statsTasks
|
|
i.statsTasks = make(map[taskKey]*statsTaskInfo)
|
|
i.stateLock.Unlock()
|
|
|
|
deleted := make([]*statsTaskInfo, 0, len(deletedTasks))
|
|
for _, info := range deletedTasks {
|
|
deleted = append(deleted, info)
|
|
}
|
|
return deleted
|
|
}
|