milvus/internal/parser/planparserv2/check_identical_test.go

365 lines
9.7 KiB
Go

package planparserv2
import (
"testing"
"github.com/stretchr/testify/assert"
"github.com/milvus-io/milvus/internal/proto/planpb"
"github.com/milvus-io/milvus/pkg/util/typeutil"
)
func TestCheckIdentical(t *testing.T) {
schema := newTestSchema()
helper, err := typeutil.CreateSchemaHelper(schema)
assert.NoError(t, err)
exprStr1Arr := []string{
`not (((Int64Field > 0) and (FloatField <= 20.0)) or ((Int32Field in [1, 2, 3]) and (VarCharField < "str")))`,
`f1()`,
}
exprStr2Arr := []string{
`Int32Field in [1, 2, 3]`,
`f2(Int32Field, Int64Field)`,
}
for i := range exprStr1Arr {
exprStr1 := exprStr1Arr[i]
exprStr2 := exprStr2Arr[i]
expr1, err := ParseExpr(helper, exprStr1, nil)
assert.NoError(t, err)
expr2, err := ParseExpr(helper, exprStr2, nil)
assert.NoError(t, err)
assert.True(t, CheckPredicatesIdentical(expr1, expr1))
assert.True(t, CheckPredicatesIdentical(expr2, expr2))
assert.False(t, CheckPredicatesIdentical(expr1, expr2))
}
}
func TestCheckQueryInfoIdentical(t *testing.T) {
type args struct {
info1 *planpb.QueryInfo
info2 *planpb.QueryInfo
}
tests := []struct {
name string
args args
want bool
}{
{
args: args{
info1: &planpb.QueryInfo{Topk: 1},
info2: &planpb.QueryInfo{Topk: 2},
},
want: false,
},
{
args: args{
info1: &planpb.QueryInfo{Topk: 1, MetricType: "L2"},
info2: &planpb.QueryInfo{Topk: 1, MetricType: "IP"},
},
want: false,
},
{
args: args{
info1: &planpb.QueryInfo{Topk: 1, MetricType: "L2", SearchParams: `{"nprobe": 10}`},
info2: &planpb.QueryInfo{Topk: 1, MetricType: "L2", SearchParams: ""},
},
want: false,
},
{
args: args{
info1: &planpb.QueryInfo{Topk: 1, MetricType: "L2", SearchParams: `{"nprobe": 10}`, RoundDecimal: 5},
info2: &planpb.QueryInfo{Topk: 1, MetricType: "L2", SearchParams: `{"nprobe": 10}`, RoundDecimal: 6},
},
want: false,
},
{
args: args{
info1: &planpb.QueryInfo{Topk: 1, MetricType: "L2", SearchParams: `{"nprobe": 10}`, RoundDecimal: 6},
info2: &planpb.QueryInfo{Topk: 1, MetricType: "L2", SearchParams: `{"nprobe": 10}`, RoundDecimal: 6},
},
want: true,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
if got := CheckQueryInfoIdentical(tt.args.info1, tt.args.info2); got != tt.want {
t.Errorf("CheckQueryInfoIdentical() = %v, want %v\ninfo1:\n%v\ninfo2:\n%v", got, tt.want, tt.args.info1, tt.args.info2)
}
})
}
}
func TestCheckVectorANNSIdentical(t *testing.T) {
type args struct {
node1 *planpb.VectorANNS
node2 *planpb.VectorANNS
}
tests := []struct {
name string
args args
want bool
}{
{
args: args{
node1: &planpb.VectorANNS{VectorType: planpb.VectorType_BinaryVector},
node2: &planpb.VectorANNS{VectorType: planpb.VectorType_FloatVector},
},
want: false,
},
{
args: args{
node1: &planpb.VectorANNS{VectorType: planpb.VectorType_FloatVector, FieldId: 100},
node2: &planpb.VectorANNS{VectorType: planpb.VectorType_FloatVector, FieldId: 101},
},
want: false,
},
{
args: args{
node1: &planpb.VectorANNS{VectorType: planpb.VectorType_FloatVector, FieldId: 100, PlaceholderTag: "$0"},
node2: &planpb.VectorANNS{VectorType: planpb.VectorType_FloatVector, FieldId: 100, PlaceholderTag: "$1"},
},
want: false,
},
{
args: args{
node1: &planpb.VectorANNS{VectorType: planpb.VectorType_FloatVector, FieldId: 100, PlaceholderTag: "$0", QueryInfo: &planpb.QueryInfo{Topk: 100}},
node2: &planpb.VectorANNS{VectorType: planpb.VectorType_FloatVector, FieldId: 100, PlaceholderTag: "$0", QueryInfo: &planpb.QueryInfo{Topk: 10}},
},
want: false,
},
{
args: args{
node1: &planpb.VectorANNS{
VectorType: planpb.VectorType_FloatVector, FieldId: 100, PlaceholderTag: "$0", QueryInfo: &planpb.QueryInfo{Topk: 1, MetricType: "L2", SearchParams: `{"nprobe": 10}`, RoundDecimal: 6},
Predicates: &planpb.Expr{
Expr: &planpb.Expr_ColumnExpr{
ColumnExpr: &planpb.ColumnExpr{
Info: &planpb.ColumnInfo{},
},
},
},
},
node2: &planpb.VectorANNS{
VectorType: planpb.VectorType_FloatVector, FieldId: 100, PlaceholderTag: "$0", QueryInfo: &planpb.QueryInfo{Topk: 1, MetricType: "L2", SearchParams: `{"nprobe": 10}`, RoundDecimal: 6},
Predicates: &planpb.Expr{
Expr: &planpb.Expr_ValueExpr{
ValueExpr: &planpb.ValueExpr{Value: NewInt(100)},
},
},
},
},
want: false,
},
{
args: args{
node1: &planpb.VectorANNS{
VectorType: planpb.VectorType_FloatVector, FieldId: 100, PlaceholderTag: "$0", QueryInfo: &planpb.QueryInfo{Topk: 1, MetricType: "L2", SearchParams: `{"nprobe": 10}`, RoundDecimal: 6},
Predicates: &planpb.Expr{
Expr: &planpb.Expr_ValueExpr{
ValueExpr: &planpb.ValueExpr{Value: NewInt(100)},
},
},
},
node2: &planpb.VectorANNS{
VectorType: planpb.VectorType_FloatVector, FieldId: 100, PlaceholderTag: "$0", QueryInfo: &planpb.QueryInfo{Topk: 1, MetricType: "L2", SearchParams: `{"nprobe": 10}`, RoundDecimal: 6},
Predicates: &planpb.Expr{
Expr: &planpb.Expr_ValueExpr{
ValueExpr: &planpb.ValueExpr{Value: NewInt(100)},
},
},
},
},
want: true,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
if got := CheckVectorANNSIdentical(tt.args.node1, tt.args.node2); got != tt.want {
t.Errorf("CheckVectorANNSIdentical() = %v, want %v\nnode1:\n%v\nnode2:\n%v", got, tt.want, tt.args.node1, tt.args.node2)
}
})
}
}
func TestCheckPlanNodeIdentical(t *testing.T) {
type args struct {
node1 *planpb.PlanNode
node2 *planpb.PlanNode
}
tests := []struct {
name string
args args
want bool
}{
{
args: args{
node1: &planpb.PlanNode{
Node: &planpb.PlanNode_VectorAnns{},
OutputFieldIds: []int64{100, 101},
},
node2: &planpb.PlanNode{
Node: &planpb.PlanNode_VectorAnns{},
OutputFieldIds: []int64{100},
},
},
want: false,
},
{
args: args{
node1: &planpb.PlanNode{
Node: &planpb.PlanNode_VectorAnns{
VectorAnns: &planpb.VectorANNS{
VectorType: planpb.VectorType_BinaryVector,
},
},
OutputFieldIds: []int64{100},
},
node2: &planpb.PlanNode{
Node: &planpb.PlanNode_VectorAnns{
VectorAnns: &planpb.VectorANNS{
VectorType: planpb.VectorType_FloatVector,
},
},
OutputFieldIds: []int64{100},
},
},
want: false,
},
{
args: args{
node1: &planpb.PlanNode{
Node: &planpb.PlanNode_VectorAnns{
VectorAnns: &planpb.VectorANNS{
VectorType: planpb.VectorType_FloatVector, FieldId: 100, PlaceholderTag: "$0", QueryInfo: &planpb.QueryInfo{Topk: 1, MetricType: "L2", SearchParams: `{"nprobe": 10}`, RoundDecimal: 6},
Predicates: &planpb.Expr{
Expr: &planpb.Expr_ValueExpr{
ValueExpr: &planpb.ValueExpr{Value: NewInt(100)},
},
},
},
},
OutputFieldIds: []int64{100},
},
node2: &planpb.PlanNode{
Node: &planpb.PlanNode_VectorAnns{
VectorAnns: &planpb.VectorANNS{
VectorType: planpb.VectorType_FloatVector, FieldId: 100, PlaceholderTag: "$0", QueryInfo: &planpb.QueryInfo{Topk: 1, MetricType: "L2", SearchParams: `{"nprobe": 10}`, RoundDecimal: 6},
Predicates: &planpb.Expr{
Expr: &planpb.Expr_ValueExpr{
ValueExpr: &planpb.ValueExpr{Value: NewInt(100)},
},
},
},
},
OutputFieldIds: []int64{100},
},
},
want: true,
},
{
args: args{
node1: &planpb.PlanNode{
Node: &planpb.PlanNode_VectorAnns{},
OutputFieldIds: []int64{100, 101},
},
node2: &planpb.PlanNode{
Node: &planpb.PlanNode_Predicates{},
},
},
want: false,
},
{
args: args{
node1: &planpb.PlanNode{
Node: &planpb.PlanNode_Predicates{},
OutputFieldIds: []int64{100, 101},
},
node2: &planpb.PlanNode{
Node: &planpb.PlanNode_Predicates{},
OutputFieldIds: []int64{100},
},
},
want: false,
},
{
args: args{
node1: &planpb.PlanNode{
Node: &planpb.PlanNode_Predicates{
Predicates: &planpb.Expr{
Expr: &planpb.Expr_ColumnExpr{
ColumnExpr: &planpb.ColumnExpr{
Info: &planpb.ColumnInfo{},
},
},
},
},
OutputFieldIds: []int64{100, 101},
},
node2: &planpb.PlanNode{
Node: &planpb.PlanNode_Predicates{
Predicates: &planpb.Expr{
Expr: &planpb.Expr_ValueExpr{
ValueExpr: &planpb.ValueExpr{Value: NewInt(100)},
},
},
},
OutputFieldIds: []int64{100, 101},
},
},
want: false,
},
{
args: args{
node1: &planpb.PlanNode{
Node: &planpb.PlanNode_Predicates{
Predicates: &planpb.Expr{
Expr: &planpb.Expr_ValueExpr{
ValueExpr: &planpb.ValueExpr{Value: NewInt(100)},
},
},
},
OutputFieldIds: []int64{100, 101},
},
node2: &planpb.PlanNode{
Node: &planpb.PlanNode_Predicates{
Predicates: &planpb.Expr{
Expr: &planpb.Expr_ValueExpr{
ValueExpr: &planpb.ValueExpr{Value: NewInt(100)},
},
},
},
OutputFieldIds: []int64{100, 101},
},
},
want: true,
},
{
args: args{
node1: &planpb.PlanNode{
Node: &planpb.PlanNode_Predicates{},
},
node2: &planpb.PlanNode{
Node: &planpb.PlanNode_VectorAnns{},
},
},
want: false,
},
{
args: args{
node1: nil,
},
want: false,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
if got := CheckPlanNodeIdentical(tt.args.node1, tt.args.node2); got != tt.want {
t.Errorf("CheckPlanNodeIdentical() = %v, want %v\nnode1:\n%v\nnode2:\n%v", got, tt.want, tt.args.node1, tt.args.node2)
}
})
}
}