2021-01-16 02:12:14 +00:00
|
|
|
package querynode
|
2020-11-09 08:27:11 +00:00
|
|
|
|
|
|
|
import (
|
2021-01-11 10:35:54 +00:00
|
|
|
"context"
|
2020-11-09 08:27:11 +00:00
|
|
|
"fmt"
|
|
|
|
"log"
|
|
|
|
"sync"
|
2020-11-12 04:04:12 +00:00
|
|
|
|
|
|
|
"github.com/zilliztech/milvus-distributed/internal/proto/commonpb"
|
2020-11-09 08:27:11 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type insertNode struct {
|
2021-01-15 07:28:54 +00:00
|
|
|
baseNode
|
2020-12-08 06:41:04 +00:00
|
|
|
replica collectionReplica
|
2020-11-09 08:27:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type InsertData struct {
|
2021-01-11 10:35:54 +00:00
|
|
|
insertContext map[int64]context.Context
|
2020-11-09 08:27:11 +00:00
|
|
|
insertIDs map[UniqueID][]UniqueID
|
|
|
|
insertTimestamps map[UniqueID][]Timestamp
|
|
|
|
insertRecords map[UniqueID][]*commonpb.Blob
|
|
|
|
insertOffset map[UniqueID]int64
|
|
|
|
}
|
|
|
|
|
|
|
|
func (iNode *insertNode) Name() string {
|
|
|
|
return "iNode"
|
|
|
|
}
|
|
|
|
|
|
|
|
func (iNode *insertNode) Operate(in []*Msg) []*Msg {
|
|
|
|
// fmt.Println("Do insertNode operation")
|
|
|
|
|
|
|
|
if len(in) != 1 {
|
2020-11-12 04:04:12 +00:00
|
|
|
log.Println("Invalid operate message input in insertNode, input length = ", len(in))
|
2020-11-09 08:27:11 +00:00
|
|
|
// TODO: add error handling
|
|
|
|
}
|
|
|
|
|
|
|
|
iMsg, ok := (*in[0]).(*insertMsg)
|
|
|
|
if !ok {
|
|
|
|
log.Println("type assertion failed for insertMsg")
|
|
|
|
// TODO: add error handling
|
|
|
|
}
|
|
|
|
|
|
|
|
insertData := InsertData{
|
|
|
|
insertIDs: make(map[int64][]int64),
|
|
|
|
insertTimestamps: make(map[int64][]uint64),
|
|
|
|
insertRecords: make(map[int64][]*commonpb.Blob),
|
|
|
|
insertOffset: make(map[int64]int64),
|
|
|
|
}
|
|
|
|
|
|
|
|
// 1. hash insertMessages to insertData
|
|
|
|
for _, task := range iMsg.insertMessages {
|
2020-11-13 07:17:18 +00:00
|
|
|
insertData.insertIDs[task.SegmentID] = append(insertData.insertIDs[task.SegmentID], task.RowIDs...)
|
|
|
|
insertData.insertTimestamps[task.SegmentID] = append(insertData.insertTimestamps[task.SegmentID], task.Timestamps...)
|
|
|
|
insertData.insertRecords[task.SegmentID] = append(insertData.insertRecords[task.SegmentID], task.RowData...)
|
2020-11-16 09:01:10 +00:00
|
|
|
|
|
|
|
// check if segment exists, if not, create this segment
|
2020-12-08 06:41:04 +00:00
|
|
|
if !iNode.replica.hasSegment(task.SegmentID) {
|
2021-02-03 10:12:48 +00:00
|
|
|
err := iNode.replica.addSegment(task.SegmentID, task.PartitionID, task.CollectionID, segTypeGrowing)
|
2020-11-16 09:01:10 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Println(err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
2020-11-09 08:27:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// 2. do preInsert
|
|
|
|
for segmentID := range insertData.insertRecords {
|
2020-12-08 06:41:04 +00:00
|
|
|
var targetSegment, err = iNode.replica.getSegmentByID(segmentID)
|
2020-11-09 08:27:11 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Println("preInsert failed")
|
|
|
|
// TODO: add error handling
|
|
|
|
}
|
|
|
|
|
|
|
|
var numOfRecords = len(insertData.insertRecords[segmentID])
|
|
|
|
if targetSegment != nil {
|
|
|
|
var offset = targetSegment.segmentPreInsert(numOfRecords)
|
|
|
|
insertData.insertOffset[segmentID] = offset
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// 3. do insert
|
|
|
|
wg := sync.WaitGroup{}
|
|
|
|
for segmentID := range insertData.insertRecords {
|
|
|
|
wg.Add(1)
|
2021-02-04 11:34:35 +00:00
|
|
|
go iNode.insert(&insertData, segmentID, &wg)
|
2020-11-09 08:27:11 +00:00
|
|
|
}
|
|
|
|
wg.Wait()
|
|
|
|
|
|
|
|
var res Msg = &serviceTimeMsg{
|
2021-01-06 08:44:12 +00:00
|
|
|
gcRecord: iMsg.gcRecord,
|
2020-11-09 08:27:11 +00:00
|
|
|
timeRange: iMsg.timeRange,
|
|
|
|
}
|
|
|
|
return []*Msg{&res}
|
|
|
|
}
|
|
|
|
|
2021-02-04 11:34:35 +00:00
|
|
|
func (iNode *insertNode) insert(insertData *InsertData, segmentID int64, wg *sync.WaitGroup) {
|
2020-12-08 06:41:04 +00:00
|
|
|
var targetSegment, err = iNode.replica.getSegmentByID(segmentID)
|
2020-11-09 08:27:11 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Println("cannot find segment:", segmentID)
|
|
|
|
// TODO: add error handling
|
2020-11-26 08:01:31 +00:00
|
|
|
wg.Done()
|
2020-11-09 08:27:11 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
ids := insertData.insertIDs[segmentID]
|
|
|
|
timestamps := insertData.insertTimestamps[segmentID]
|
|
|
|
records := insertData.insertRecords[segmentID]
|
|
|
|
offsets := insertData.insertOffset[segmentID]
|
|
|
|
|
|
|
|
err = targetSegment.segmentInsert(offsets, &ids, ×tamps, &records)
|
|
|
|
if err != nil {
|
2020-11-26 08:01:31 +00:00
|
|
|
log.Println(err)
|
2020-11-09 08:27:11 +00:00
|
|
|
// TODO: add error handling
|
2020-11-26 08:01:31 +00:00
|
|
|
wg.Done()
|
2020-11-09 08:27:11 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
fmt.Println("Do insert done, len = ", len(insertData.insertIDs[segmentID]))
|
|
|
|
wg.Done()
|
|
|
|
}
|
|
|
|
|
2020-12-08 06:41:04 +00:00
|
|
|
func newInsertNode(replica collectionReplica) *insertNode {
|
2020-12-10 08:31:09 +00:00
|
|
|
maxQueueLength := Params.FlowGraphMaxQueueLength
|
|
|
|
maxParallelism := Params.FlowGraphMaxParallelism
|
2020-11-18 09:32:52 +00:00
|
|
|
|
2021-01-15 07:28:54 +00:00
|
|
|
baseNode := baseNode{}
|
2020-11-09 08:27:11 +00:00
|
|
|
baseNode.SetMaxQueueLength(maxQueueLength)
|
|
|
|
baseNode.SetMaxParallelism(maxParallelism)
|
|
|
|
|
|
|
|
return &insertNode{
|
2021-01-15 07:28:54 +00:00
|
|
|
baseNode: baseNode,
|
2020-11-17 12:00:23 +00:00
|
|
|
replica: replica,
|
2020-11-09 08:27:11 +00:00
|
|
|
}
|
|
|
|
}
|