mirror of
https://gitee.com/milvus-io/milvus.git
synced 2024-12-05 05:18:52 +08:00
48821690b3
Signed-off-by: bigsheeper <yihao.dai@zilliz.com>
153 lines
4.2 KiB
Go
153 lines
4.2 KiB
Go
package querynode
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
|
|
"github.com/zilliztech/milvus-distributed/internal/kv"
|
|
"github.com/zilliztech/milvus-distributed/internal/proto/commonpb"
|
|
"github.com/zilliztech/milvus-distributed/internal/proto/datapb"
|
|
"github.com/zilliztech/milvus-distributed/internal/proto/indexpb"
|
|
queryPb "github.com/zilliztech/milvus-distributed/internal/proto/querypb"
|
|
"github.com/zilliztech/milvus-distributed/internal/querynode/client"
|
|
"github.com/zilliztech/milvus-distributed/internal/storage"
|
|
)
|
|
|
|
type segmentManager struct {
|
|
replica collectionReplica
|
|
|
|
// TODO: replace by client instead of grpc client
|
|
dataClient datapb.DataServiceClient
|
|
indexBuilderClient indexpb.IndexServiceClient
|
|
|
|
queryNodeClient *client.Client
|
|
kv kv.Base // minio kv
|
|
iCodec storage.InsertCodec
|
|
}
|
|
|
|
func (s *segmentManager) loadSegment(segmentID UniqueID, fieldIDs *[]int64) error {
|
|
insertBinlogPathRequest := &datapb.InsertBinlogPathRequest{
|
|
SegmentID: segmentID,
|
|
}
|
|
|
|
pathResponse, err := s.dataClient.GetInsertBinlogPaths(context.TODO(), insertBinlogPathRequest)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if len(pathResponse.FieldIDs) != len(pathResponse.Paths) {
|
|
return errors.New("illegal InsertBinlogPathsResponse")
|
|
}
|
|
|
|
containsFunc := func(s []int64, e int64) bool {
|
|
for _, a := range s {
|
|
if a == e {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
for i, fieldID := range pathResponse.FieldIDs {
|
|
// filter out the needless fields
|
|
if !containsFunc(*fieldIDs, fieldID) {
|
|
continue
|
|
}
|
|
|
|
paths := pathResponse.Paths[i].Values
|
|
blobs := make([]*storage.Blob, 0)
|
|
for _, path := range paths {
|
|
binLog, err := s.kv.Load(path)
|
|
if err != nil {
|
|
// TODO: return or continue?
|
|
return err
|
|
}
|
|
blobs = append(blobs, &storage.Blob{
|
|
Key: "", // TODO: key???
|
|
Value: []byte(binLog),
|
|
})
|
|
}
|
|
_, _, insertData, err := s.iCodec.Deserialize(blobs)
|
|
if err != nil {
|
|
// TODO: return or continue
|
|
return err
|
|
}
|
|
if len(insertData.Data) != 1 {
|
|
return errors.New("we expect only one field in deserialized insert data")
|
|
}
|
|
|
|
for _, value := range insertData.Data {
|
|
switch fieldData := value.(type) {
|
|
case storage.BoolFieldData:
|
|
numRows := fieldData.NumRows
|
|
data := fieldData.Data
|
|
fmt.Println(numRows, data, fieldID)
|
|
// TODO: s.replica.addSegment()
|
|
case storage.Int8FieldData:
|
|
// TODO: s.replica.addSegment()
|
|
case storage.Int16FieldData:
|
|
// TODO: s.replica.addSegment()
|
|
case storage.Int32FieldData:
|
|
// TODO: s.replica.addSegment()
|
|
case storage.Int64FieldData:
|
|
// TODO: s.replica.addSegment()
|
|
case storage.FloatFieldData:
|
|
// TODO: s.replica.addSegment()
|
|
case storage.DoubleFieldData:
|
|
// TODO: s.replica.addSegment()
|
|
case storage.StringFieldData:
|
|
// TODO: s.replica.addSegment()
|
|
case storage.FloatVectorFieldData:
|
|
// segment to be loaded doesn't need vector field,
|
|
// so we ignore the type of vector field data
|
|
continue
|
|
case storage.BinaryVectorFieldData:
|
|
continue
|
|
default:
|
|
return errors.New("unexpected field data type")
|
|
}
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (s *segmentManager) loadIndex(segmentID UniqueID, indexID UniqueID) error {
|
|
indexFilePathRequest := &indexpb.IndexFilePathRequest{
|
|
IndexID: indexID,
|
|
}
|
|
pathResponse, err := s.indexBuilderClient.GetIndexFilePaths(context.TODO(), indexFilePathRequest)
|
|
if err != nil || pathResponse.Status.ErrorCode != commonpb.ErrorCode_SUCCESS {
|
|
return err
|
|
}
|
|
targetSegment, err := s.replica.getSegmentByID(segmentID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// get vector field ids from schema to load index
|
|
vecFieldIDs, err := s.replica.getVecFieldIDsBySegmentID(segmentID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for _, vecFieldID := range vecFieldIDs {
|
|
targetIndexParam, ok := targetSegment.indexParam[vecFieldID]
|
|
if !ok {
|
|
return errors.New(fmt.Sprint("cannot found index params in segment ", segmentID, " with field = ", vecFieldID))
|
|
}
|
|
err := s.queryNodeClient.LoadIndex(pathResponse.IndexFilePaths, segmentID, vecFieldID, "", targetIndexParam)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (s *segmentManager) releaseSegment(in *queryPb.ReleaseSegmentRequest) error {
|
|
// TODO: implement
|
|
// TODO: release specific field, we need segCore supply relevant interface
|
|
return nil
|
|
}
|