2022-04-01 18:59:29 +08:00
|
|
|
package proxy
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
2022-06-23 10:46:13 +08:00
|
|
|
"github.com/milvus-io/milvus/internal/common"
|
|
|
|
|
2022-04-01 18:59:29 +08:00
|
|
|
"github.com/golang/protobuf/proto"
|
|
|
|
"github.com/stretchr/testify/assert"
|
2022-04-20 16:15:41 +08:00
|
|
|
"github.com/stretchr/testify/require"
|
2022-04-01 18:59:29 +08:00
|
|
|
|
|
|
|
"github.com/milvus-io/milvus/internal/proto/commonpb"
|
|
|
|
"github.com/milvus-io/milvus/internal/proto/internalpb"
|
|
|
|
"github.com/milvus-io/milvus/internal/proto/milvuspb"
|
|
|
|
"github.com/milvus-io/milvus/internal/proto/querypb"
|
|
|
|
"github.com/milvus-io/milvus/internal/proto/schemapb"
|
2022-06-02 12:16:03 +08:00
|
|
|
"github.com/milvus-io/milvus/internal/types"
|
2022-04-01 18:59:29 +08:00
|
|
|
|
|
|
|
"github.com/milvus-io/milvus/internal/util/funcutil"
|
|
|
|
"github.com/milvus-io/milvus/internal/util/typeutil"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestQueryTask_all(t *testing.T) {
|
|
|
|
Params.Init()
|
|
|
|
|
2022-04-20 16:15:41 +08:00
|
|
|
var (
|
|
|
|
err error
|
|
|
|
ctx = context.TODO()
|
|
|
|
|
|
|
|
rc = NewRootCoordMock()
|
|
|
|
qc = NewQueryCoordMock(withValidShardLeaders())
|
|
|
|
qn = &QueryNodeMock{}
|
|
|
|
|
|
|
|
shardsNum = int32(2)
|
|
|
|
collectionName = t.Name() + funcutil.GenRandomStr()
|
|
|
|
|
|
|
|
expr = fmt.Sprintf("%s > 0", testInt64Field)
|
|
|
|
hitNum = 10
|
2022-07-06 15:06:21 +08:00
|
|
|
|
|
|
|
errPolicy = func(context.Context, *shardClientMgr, func(context.Context, int64, types.QueryNode, []string) error, map[string][]nodeInfo) error {
|
|
|
|
return fmt.Errorf("fake error")
|
|
|
|
}
|
2022-04-20 16:15:41 +08:00
|
|
|
)
|
|
|
|
|
2022-06-02 12:16:03 +08:00
|
|
|
mockCreator := func(ctx context.Context, address string) (types.QueryNode, error) {
|
2022-04-20 16:15:41 +08:00
|
|
|
return qn, nil
|
|
|
|
}
|
|
|
|
|
2022-06-02 12:16:03 +08:00
|
|
|
mgr := newShardClientMgr(withShardClientCreator(mockCreator))
|
|
|
|
|
2022-04-01 18:59:29 +08:00
|
|
|
rc.Start()
|
|
|
|
defer rc.Stop()
|
2022-04-20 16:15:41 +08:00
|
|
|
qc.Start()
|
|
|
|
defer qc.Stop()
|
2022-04-01 18:59:29 +08:00
|
|
|
|
2022-08-04 11:04:34 +08:00
|
|
|
err = InitMetaCache(ctx, rc, qc, mgr)
|
2022-04-01 18:59:29 +08:00
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
fieldName2Types := map[string]schemapb.DataType{
|
|
|
|
testBoolField: schemapb.DataType_Bool,
|
|
|
|
testInt32Field: schemapb.DataType_Int32,
|
|
|
|
testInt64Field: schemapb.DataType_Int64,
|
|
|
|
testFloatField: schemapb.DataType_Float,
|
|
|
|
testDoubleField: schemapb.DataType_Double,
|
|
|
|
testFloatVecField: schemapb.DataType_FloatVector,
|
|
|
|
}
|
|
|
|
if enableMultipleVectorFields {
|
|
|
|
fieldName2Types[testBinaryVecField] = schemapb.DataType_BinaryVector
|
|
|
|
}
|
|
|
|
|
|
|
|
schema := constructCollectionSchemaByDataType(collectionName, fieldName2Types, testInt64Field, false)
|
|
|
|
marshaledSchema, err := proto.Marshal(schema)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
createColT := &createCollectionTask{
|
|
|
|
Condition: NewTaskCondition(ctx),
|
|
|
|
CreateCollectionRequest: &milvuspb.CreateCollectionRequest{
|
|
|
|
CollectionName: collectionName,
|
|
|
|
Schema: marshaledSchema,
|
|
|
|
ShardsNum: shardsNum,
|
|
|
|
},
|
|
|
|
ctx: ctx,
|
|
|
|
rootCoord: rc,
|
|
|
|
}
|
|
|
|
|
2022-04-20 16:15:41 +08:00
|
|
|
require.NoError(t, createColT.OnEnqueue())
|
|
|
|
require.NoError(t, createColT.PreExecute(ctx))
|
|
|
|
require.NoError(t, createColT.Execute(ctx))
|
|
|
|
require.NoError(t, createColT.PostExecute(ctx))
|
2022-04-01 18:59:29 +08:00
|
|
|
|
|
|
|
collectionID, err := globalMetaCache.GetCollectionID(ctx, collectionName)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
status, err := qc.LoadCollection(ctx, &querypb.LoadCollectionRequest{
|
|
|
|
Base: &commonpb.MsgBase{
|
2022-04-20 16:15:41 +08:00
|
|
|
MsgType: commonpb.MsgType_LoadCollection,
|
2022-04-24 22:03:44 +08:00
|
|
|
SourceID: Params.ProxyCfg.GetNodeID(),
|
2022-04-01 18:59:29 +08:00
|
|
|
},
|
|
|
|
CollectionID: collectionID,
|
|
|
|
})
|
2022-04-20 16:15:41 +08:00
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, commonpb.ErrorCode_Success, status.ErrorCode)
|
2022-04-01 18:59:29 +08:00
|
|
|
|
2022-04-20 16:15:41 +08:00
|
|
|
// test begins
|
2022-04-01 18:59:29 +08:00
|
|
|
task := &queryTask{
|
|
|
|
Condition: NewTaskCondition(ctx),
|
|
|
|
RetrieveRequest: &internalpb.RetrieveRequest{
|
|
|
|
Base: &commonpb.MsgBase{
|
2022-04-20 16:15:41 +08:00
|
|
|
MsgType: commonpb.MsgType_Retrieve,
|
2022-04-24 22:03:44 +08:00
|
|
|
SourceID: Params.ProxyCfg.GetNodeID(),
|
2022-04-01 18:59:29 +08:00
|
|
|
},
|
2022-04-20 16:15:41 +08:00
|
|
|
CollectionID: collectionID,
|
|
|
|
OutputFieldsId: make([]int64, len(fieldName2Types)),
|
2022-04-01 18:59:29 +08:00
|
|
|
},
|
2022-04-20 16:15:41 +08:00
|
|
|
ctx: ctx,
|
2022-04-01 18:59:29 +08:00
|
|
|
result: &milvuspb.QueryResults{
|
|
|
|
Status: &commonpb.Status{
|
|
|
|
ErrorCode: commonpb.ErrorCode_Success,
|
|
|
|
},
|
|
|
|
},
|
2022-04-20 16:15:41 +08:00
|
|
|
request: &milvuspb.QueryRequest{
|
2022-04-01 18:59:29 +08:00
|
|
|
Base: &commonpb.MsgBase{
|
2022-04-20 16:15:41 +08:00
|
|
|
MsgType: commonpb.MsgType_Retrieve,
|
2022-04-24 22:03:44 +08:00
|
|
|
SourceID: Params.ProxyCfg.GetNodeID(),
|
2022-04-01 18:59:29 +08:00
|
|
|
},
|
2022-04-20 16:15:41 +08:00
|
|
|
CollectionName: collectionName,
|
|
|
|
Expr: expr,
|
2022-04-01 18:59:29 +08:00
|
|
|
},
|
2022-07-06 15:06:21 +08:00
|
|
|
qc: qc,
|
|
|
|
shardMgr: mgr,
|
2022-04-01 18:59:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
assert.NoError(t, task.OnEnqueue())
|
|
|
|
|
|
|
|
// test query task with timeout
|
|
|
|
ctx1, cancel1 := context.WithTimeout(ctx, 10*time.Second)
|
|
|
|
defer cancel1()
|
|
|
|
// before preExecute
|
|
|
|
assert.Equal(t, typeutil.ZeroTimestamp, task.TimeoutTimestamp)
|
|
|
|
task.ctx = ctx1
|
|
|
|
assert.NoError(t, task.PreExecute(ctx))
|
|
|
|
// after preExecute
|
|
|
|
assert.Greater(t, task.TimeoutTimestamp, typeutil.ZeroTimestamp)
|
|
|
|
|
2022-07-06 15:06:21 +08:00
|
|
|
task.ctx = ctx
|
|
|
|
task.queryShardPolicy = errPolicy
|
|
|
|
assert.Error(t, task.Execute(ctx))
|
|
|
|
|
|
|
|
task.queryShardPolicy = mergeRoundRobinPolicy
|
2022-04-20 16:15:41 +08:00
|
|
|
result1 := &internalpb.RetrieveResults{
|
|
|
|
Base: &commonpb.MsgBase{MsgType: commonpb.MsgType_RetrieveResult},
|
|
|
|
Status: &commonpb.Status{
|
|
|
|
ErrorCode: commonpb.ErrorCode_Success,
|
|
|
|
},
|
|
|
|
Ids: &schemapb.IDs{
|
|
|
|
IdField: &schemapb.IDs_IntId{
|
|
|
|
IntId: &schemapb.LongArray{Data: generateInt64Array(hitNum)},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2022-06-23 10:46:13 +08:00
|
|
|
outputFieldIDs := make([]UniqueID, 0, len(fieldName2Types))
|
|
|
|
for i := 0; i < len(fieldName2Types); i++ {
|
|
|
|
outputFieldIDs = append(outputFieldIDs, int64(common.StartOfUserFieldID+i))
|
|
|
|
}
|
|
|
|
task.RetrieveRequest.OutputFieldsId = outputFieldIDs
|
2022-04-20 16:15:41 +08:00
|
|
|
for fieldName, dataType := range fieldName2Types {
|
2022-04-29 13:35:49 +08:00
|
|
|
result1.FieldsData = append(result1.FieldsData, generateFieldData(dataType, fieldName, hitNum))
|
2022-04-20 16:15:41 +08:00
|
|
|
}
|
|
|
|
|
2022-07-06 15:06:21 +08:00
|
|
|
task.ctx = ctx
|
|
|
|
qn.queryError = fmt.Errorf("mock error")
|
|
|
|
assert.Error(t, task.Execute(ctx))
|
|
|
|
|
|
|
|
qn.queryError = nil
|
|
|
|
qn.withQueryResult = &internalpb.RetrieveResults{
|
|
|
|
Status: &commonpb.Status{
|
|
|
|
ErrorCode: commonpb.ErrorCode_NotShardLeader,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
assert.Equal(t, task.Execute(ctx), errInvalidShardLeaders)
|
|
|
|
|
|
|
|
qn.withQueryResult = &internalpb.RetrieveResults{
|
|
|
|
Status: &commonpb.Status{
|
|
|
|
ErrorCode: commonpb.ErrorCode_UnexpectedError,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
assert.Error(t, task.Execute(ctx))
|
|
|
|
|
2022-04-20 16:15:41 +08:00
|
|
|
qn.withQueryResult = result1
|
|
|
|
|
2022-04-01 18:59:29 +08:00
|
|
|
assert.NoError(t, task.Execute(ctx))
|
|
|
|
|
2022-04-20 16:15:41 +08:00
|
|
|
assert.NoError(t, task.PostExecute(ctx))
|
2022-04-01 18:59:29 +08:00
|
|
|
}
|
2022-08-16 18:16:48 +08:00
|
|
|
|
|
|
|
func Test_translateToOutputFieldIDs(t *testing.T) {
|
|
|
|
type testCases struct {
|
|
|
|
name string
|
|
|
|
outputFields []string
|
|
|
|
schema *schemapb.CollectionSchema
|
|
|
|
expectedError bool
|
|
|
|
expectedIDs []int64
|
|
|
|
}
|
|
|
|
|
|
|
|
cases := []testCases{
|
|
|
|
{
|
|
|
|
name: "empty output fields",
|
|
|
|
outputFields: []string{},
|
|
|
|
schema: &schemapb.CollectionSchema{
|
|
|
|
Fields: []*schemapb.FieldSchema{
|
|
|
|
{
|
|
|
|
FieldID: common.RowIDField,
|
|
|
|
Name: common.RowIDFieldName,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
FieldID: 100,
|
|
|
|
Name: "ID",
|
|
|
|
IsPrimaryKey: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
FieldID: 101,
|
|
|
|
Name: "Vector",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
expectedError: false,
|
|
|
|
expectedIDs: []int64{100, 101},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "nil output fields",
|
|
|
|
outputFields: nil,
|
|
|
|
schema: &schemapb.CollectionSchema{
|
|
|
|
Fields: []*schemapb.FieldSchema{
|
|
|
|
{
|
|
|
|
FieldID: common.RowIDField,
|
|
|
|
Name: common.RowIDFieldName,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
FieldID: 100,
|
|
|
|
Name: "ID",
|
|
|
|
IsPrimaryKey: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
FieldID: 101,
|
|
|
|
Name: "Vector",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
expectedError: false,
|
|
|
|
expectedIDs: []int64{100, 101},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "full list",
|
|
|
|
outputFields: []string{"ID", "Vector"},
|
|
|
|
schema: &schemapb.CollectionSchema{
|
|
|
|
Fields: []*schemapb.FieldSchema{
|
|
|
|
{
|
|
|
|
FieldID: common.RowIDField,
|
|
|
|
Name: common.RowIDFieldName,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
FieldID: 100,
|
|
|
|
Name: "ID",
|
|
|
|
IsPrimaryKey: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
FieldID: 101,
|
|
|
|
Name: "Vector",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
expectedError: false,
|
|
|
|
expectedIDs: []int64{100, 101},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "vector only",
|
|
|
|
outputFields: []string{"Vector"},
|
|
|
|
schema: &schemapb.CollectionSchema{
|
|
|
|
Fields: []*schemapb.FieldSchema{
|
|
|
|
{
|
|
|
|
FieldID: common.RowIDField,
|
|
|
|
Name: common.RowIDFieldName,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
FieldID: 100,
|
|
|
|
Name: "ID",
|
|
|
|
IsPrimaryKey: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
FieldID: 101,
|
|
|
|
Name: "Vector",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
expectedError: false,
|
|
|
|
expectedIDs: []int64{101, 100},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "with field not exist",
|
|
|
|
outputFields: []string{"ID", "Vector", "Extra"},
|
|
|
|
schema: &schemapb.CollectionSchema{
|
|
|
|
Fields: []*schemapb.FieldSchema{
|
|
|
|
{
|
|
|
|
FieldID: common.RowIDField,
|
|
|
|
Name: common.RowIDFieldName,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
FieldID: 100,
|
|
|
|
Name: "ID",
|
|
|
|
IsPrimaryKey: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
FieldID: 101,
|
|
|
|
Name: "Vector",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
expectedError: true,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tc := range cases {
|
|
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
|
|
ids, err := translateToOutputFieldIDs(tc.outputFields, tc.schema)
|
|
|
|
if tc.expectedError {
|
|
|
|
assert.Error(t, err)
|
|
|
|
} else {
|
|
|
|
assert.NoError(t, err)
|
|
|
|
require.Equal(t, len(tc.expectedIDs), len(ids))
|
|
|
|
for idx, expectedID := range tc.expectedIDs {
|
|
|
|
assert.Equal(t, expectedID, ids[idx])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2022-09-01 18:54:58 +08:00
|
|
|
|
|
|
|
func TestTaskQuery_functions(t *testing.T) {
|
|
|
|
t.Run("test parseQueryParams", func(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
description string
|
|
|
|
|
|
|
|
inKey []string
|
|
|
|
inValue []string
|
|
|
|
|
|
|
|
expectErr bool
|
|
|
|
outLimit int64
|
|
|
|
outOffset int64
|
|
|
|
}{
|
|
|
|
{"empty input", []string{}, []string{}, false, 0, 0},
|
|
|
|
{"valid limit=1", []string{LimitKey}, []string{"1"}, false, 1, 0},
|
|
|
|
{"valid limit=1, offset=2", []string{LimitKey, OffsetKey}, []string{"1", "2"}, false, 1, 2},
|
|
|
|
{"valid no limit, offset=2", []string{OffsetKey}, []string{"2"}, false, 0, 0},
|
|
|
|
{"invalid limit str", []string{LimitKey}, []string{"a"}, true, 0, 0},
|
|
|
|
{"invalid limit zero", []string{LimitKey}, []string{"0"}, true, 0, 0},
|
|
|
|
{"invalid offset negative", []string{LimitKey, OffsetKey}, []string{"1", "-1"}, true, 0, 0},
|
|
|
|
{"invalid limit=16384 offset=16384", []string{LimitKey, OffsetKey}, []string{"16384", "16384"}, true, 0, 0},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, test := range tests {
|
|
|
|
t.Run(test.description, func(t *testing.T) {
|
|
|
|
var inParams []*commonpb.KeyValuePair
|
|
|
|
for i := range test.inKey {
|
|
|
|
inParams = append(inParams, &commonpb.KeyValuePair{
|
|
|
|
Key: test.inKey[i],
|
|
|
|
Value: test.inValue[i],
|
|
|
|
})
|
|
|
|
|
|
|
|
}
|
|
|
|
ret, err := parseQueryParams(inParams)
|
|
|
|
if test.expectErr {
|
|
|
|
assert.Error(t, err)
|
|
|
|
assert.Empty(t, ret)
|
|
|
|
} else {
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, test.outLimit, ret.limit)
|
|
|
|
assert.Equal(t, test.outOffset, ret.offset)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|