mirror of https://github.com/milvus-io/milvus.git
129 lines
3.4 KiB
Go
129 lines
3.4 KiB
Go
package datanode
|
|
|
|
import (
|
|
"context"
|
|
|
|
"github.com/golang/protobuf/proto"
|
|
"github.com/samber/lo"
|
|
"go.opentelemetry.io/otel/trace"
|
|
"go.uber.org/zap"
|
|
|
|
"github.com/milvus-io/milvus-proto/go-api/v2/commonpb"
|
|
"github.com/milvus-io/milvus-proto/go-api/v2/msgpb"
|
|
"github.com/milvus-io/milvus/internal/datanode/metacache"
|
|
"github.com/milvus-io/milvus/internal/datanode/writebuffer"
|
|
"github.com/milvus-io/milvus/pkg/log"
|
|
"github.com/milvus-io/milvus/pkg/mq/msgstream"
|
|
"github.com/milvus-io/milvus/pkg/util/paramtable"
|
|
)
|
|
|
|
type writeNode struct {
|
|
BaseNode
|
|
|
|
channelName string
|
|
wbManager writebuffer.BufferManager
|
|
updater statsUpdater
|
|
metacache metacache.MetaCache
|
|
}
|
|
|
|
func (wNode *writeNode) Operate(in []Msg) []Msg {
|
|
fgMsg := in[0].(*flowGraphMsg)
|
|
|
|
// close msg, ignore all data
|
|
if fgMsg.IsCloseMsg() {
|
|
return []Msg{fgMsg}
|
|
}
|
|
|
|
// replace pchannel with vchannel
|
|
startPositions := make([]*msgpb.MsgPosition, 0, len(fgMsg.startPositions))
|
|
for idx := range fgMsg.startPositions {
|
|
pos := proto.Clone(fgMsg.startPositions[idx]).(*msgpb.MsgPosition)
|
|
pos.ChannelName = wNode.channelName
|
|
startPositions = append(startPositions, pos)
|
|
}
|
|
fgMsg.startPositions = startPositions
|
|
endPositions := make([]*msgpb.MsgPosition, 0, len(fgMsg.endPositions))
|
|
for idx := range fgMsg.endPositions {
|
|
pos := proto.Clone(fgMsg.endPositions[idx]).(*msgpb.MsgPosition)
|
|
pos.ChannelName = wNode.channelName
|
|
endPositions = append(endPositions, pos)
|
|
}
|
|
fgMsg.endPositions = endPositions
|
|
|
|
if len(fgMsg.startPositions) == 0 {
|
|
return []Msg{}
|
|
}
|
|
if len(fgMsg.endPositions) == 0 {
|
|
return []Msg{}
|
|
}
|
|
|
|
var spans []trace.Span
|
|
for _, msg := range fgMsg.insertMessages {
|
|
ctx, sp := startTracer(msg, "WriteNode")
|
|
spans = append(spans, sp)
|
|
msg.SetTraceCtx(ctx)
|
|
}
|
|
defer func() {
|
|
for _, sp := range spans {
|
|
sp.End()
|
|
}
|
|
}()
|
|
|
|
start, end := fgMsg.startPositions[0], fgMsg.endPositions[0]
|
|
|
|
err := wNode.wbManager.BufferData(wNode.channelName, fgMsg.insertMessages, fgMsg.deleteMessages, start, end)
|
|
if err != nil {
|
|
log.Error("failed to buffer data", zap.Error(err))
|
|
panic(err)
|
|
}
|
|
|
|
stats := lo.FilterMap(
|
|
lo.Keys(lo.SliceToMap(fgMsg.insertMessages, func(msg *msgstream.InsertMsg) (int64, struct{}) { return msg.GetSegmentID(), struct{}{} })),
|
|
func(id int64, _ int) (*commonpb.SegmentStats, bool) {
|
|
segInfo, ok := wNode.metacache.GetSegmentByID(id)
|
|
if !ok {
|
|
log.Warn("segment not found for stats", zap.Int64("segment", id))
|
|
return nil, false
|
|
}
|
|
return &commonpb.SegmentStats{
|
|
SegmentID: id,
|
|
NumRows: segInfo.NumOfRows(),
|
|
}, true
|
|
})
|
|
|
|
wNode.updater.update(wNode.channelName, end.GetTimestamp(), stats)
|
|
|
|
res := flowGraphMsg{
|
|
timeRange: fgMsg.timeRange,
|
|
startPositions: fgMsg.startPositions,
|
|
endPositions: fgMsg.endPositions,
|
|
dropCollection: fgMsg.dropCollection,
|
|
}
|
|
|
|
if fgMsg.dropCollection {
|
|
wNode.wbManager.DropChannel(wNode.channelName)
|
|
}
|
|
|
|
// send delete msg to DeleteNode
|
|
return []Msg{&res}
|
|
}
|
|
|
|
func newWriteNode(
|
|
_ context.Context,
|
|
writeBufferManager writebuffer.BufferManager,
|
|
updater statsUpdater,
|
|
config *nodeConfig,
|
|
) *writeNode {
|
|
baseNode := BaseNode{}
|
|
baseNode.SetMaxQueueLength(paramtable.Get().DataNodeCfg.FlowGraphMaxQueueLength.GetAsInt32())
|
|
baseNode.SetMaxParallelism(paramtable.Get().DataNodeCfg.FlowGraphMaxParallelism.GetAsInt32())
|
|
|
|
return &writeNode{
|
|
BaseNode: baseNode,
|
|
channelName: config.vChannelName,
|
|
wbManager: writeBufferManager,
|
|
updater: updater,
|
|
metacache: config.metacache,
|
|
}
|
|
}
|