Current state: Under Discussion
ISSUE: #6299
PRs: #6570
Keywords: Query / Search / Vector
Released: Milvus 2.0rc3
Summary
Using minimal memory consumption, let `search` or `query` operation support to return vector raw data in output fields.
Motivation
In Milvus 2.0rc1, operations like search or query do not support return vector raw data in output fields. This is from the consideration of memory consumption,
vector field with big dimension will occupy hundreds of times of memory comparing with scalar field. So in general load_collection or load_partition only load
scalar fields' raw data into memory. Vector fields' raw data is loaded into memory only in 3 cases:
- steaming segment
- vector field's index type is FLAT
- vector field's index has not been created
Design Details
在 search / query 结束后,再分析 output_fields 里是否包含向量列,若包含,则加载结果 IDs 所在 segment 向量列,通过结果 IDs 对应的 offset 得到对应向量数据。
- Add new field VectorFieldInfo into segment struct to record vector field related information
type VectorFieldInfo struct { mu sync.RWMutex fieldID UniqueID fieldBinlog *datapb.FieldBinlog rowDataInMemory bool rawData map[string]storage.FieldData // map[binlogPath]FieldData } type Segment struct { ... ... vectorFieldInfos map[UniqueID]*VectorFieldInfo }
- Add new interface in segment
// based on result.Offset, get vector raw data from fieldInfo, // then fill vector raw data into result func (s *Segment) fillRetrieveResults(result *segcorepb.RetrieveResults, fieldInfo *VectorFieldInfo) error
- Add new interface in segment_loader
// load vector field's data from info.fieldBinlog, save the raw data into info.rawData func (loader *segmentLoader) loadSegmentVectorFieldsData(info *VectorFieldInfo) error {
- Add new interface in query_collection
// load vector fields' data, and fill in result, if // 1). result is not empty // 2). output field contain vector // 3). vector field has not been loaded into memory func (q *queryCollection) fillVectorOutputFieldsIfNeeded(msg queryMsg, segment *Segment, result *segcorepb.RetrieveResults) error
Original vector data storage public interface and struct
Public Interfaces
```go
type FileManager interface {
GetFile(path string) (string, error)
PutFile(path string, content []byte) error
Exist(path string) bool
ReadFile(path string) []byte
}
```
A VectorFileManager implements FileManager interface.
```go
type VectorFileManager struct {
localFileManager FileManager
remoteFileManager FileManager
insertCodec *InsertCodec
}
```
localFileManager is responsible to local file manager. And can be implements with golang os library.
remoteFileManager is responsible for cloud storage or remote server storage, and will be implemented with minio client now.
When the offset of vector is obtained, we can get origin vector data from local vector data file.
Test Plan
Do query / search (with vector field in output fields) in all kinds of combinations of following scenarios, check the correctness of result.
- float vector or binary vector
- with/wo index
- all kinds of index type