Files
tidb/pkg/expression/expression_test.go
2025-05-06 14:03:42 +00:00

326 lines
12 KiB
Go

// Copyright 2017 PingCAP, Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package expression
import (
"testing"
"github.com/pingcap/errors"
"github.com/pingcap/tidb/pkg/expression/exprstatic"
"github.com/pingcap/tidb/pkg/meta/model"
"github.com/pingcap/tidb/pkg/parser/ast"
"github.com/pingcap/tidb/pkg/parser/mysql"
"github.com/pingcap/tidb/pkg/types"
"github.com/pingcap/tidb/pkg/util/chunk"
"github.com/stretchr/testify/require"
)
func TestNewValuesFunc(t *testing.T) {
ctx := createContext(t)
res := NewValuesFunc(ctx, 0, types.NewFieldType(mysql.TypeLonglong))
require.Equal(t, "values", res.FuncName.O)
require.Equal(t, mysql.TypeLonglong, res.RetType.GetType())
_, ok := res.Function.(*builtinValuesIntSig)
require.True(t, ok)
}
func TestEvaluateExprWithNull(t *testing.T) {
ctx := createContext(t)
tblInfo := newTestTableBuilder("").add("col0", mysql.TypeLonglong, 0).add("col1", mysql.TypeLonglong, 0).build()
schema := tableInfoToSchemaForTest(tblInfo)
col0 := schema.Columns[0]
col1 := schema.Columns[1]
schema.Columns = schema.Columns[:1]
innerIfNull, err := newFunctionForTest(ctx, ast.Ifnull, col1, NewOne())
require.NoError(t, err)
outerIfNull, err := newFunctionForTest(ctx, ast.Ifnull, col0, innerIfNull)
require.NoError(t, err)
res, err := EvaluateExprWithNull(ctx, schema, outerIfNull, true)
require.Nil(t, err)
require.Equal(t, "ifnull(Column#1, 1)", res.StringWithCtx(ctx, errors.RedactLogDisable))
require.Equal(t, "ifnull(Column#1, ?)", res.StringWithCtx(ctx, errors.RedactLogEnable))
require.Equal(t, "ifnull(Column#1, ‹1›)", res.StringWithCtx(ctx, errors.RedactLogMarker))
schema.Columns = append(schema.Columns, col1)
// ifnull(null, ifnull(null, 1))
res, err = EvaluateExprWithNull(ctx, schema, outerIfNull, true)
require.Nil(t, err)
require.True(t, res.Equal(ctx, NewOne()))
}
func TestEvaluateExprWithNullMeetError(t *testing.T) {
ctx := createContext(t)
tblInfo := newTestTableBuilder("").add("col0", mysql.TypeLonglong, 0).add("col1", mysql.TypeLonglong, 0).build()
schema := tableInfoToSchemaForTest(tblInfo)
col0 := schema.Columns[0]
col1 := schema.Columns[1]
schema.Columns = schema.Columns[:1]
innerFunc, err := newFunctionForTest(ctx, ast.Ifnull, col1, NewOne())
require.NoError(t, err)
// rename the function name to make it invalid, so that the inner function will meet an error
innerFunc.(*ScalarFunction).FuncName.L = "invalid"
outerIfNull, err := newFunctionForTest(ctx, ast.Ifnull, col0, innerFunc)
require.NoError(t, err)
// the inner function has an error
_, err = EvaluateExprWithNull(ctx, schema, outerIfNull, true)
require.NotNil(t, err)
// check in NullRejectCheck ctx
_, err = EvaluateExprWithNull(ctx.GetNullRejectCheckExprCtx(), schema, outerIfNull, true)
require.NotNil(t, err)
}
func TestEvaluateExprWithNullAndParameters(t *testing.T) {
ctx := createContext(t)
tblInfo := newTestTableBuilder("").add("col0", mysql.TypeLonglong, 0).build()
schema := tableInfoToSchemaForTest(tblInfo)
col0 := schema.Columns[0]
ctx.GetSessionVars().StmtCtx.EnablePlanCache()
// cases for parameters
ltWithoutParam, err := newFunctionForTest(ctx, ast.LT, col0, NewOne())
require.NoError(t, err)
res, err := EvaluateExprWithNull(ctx, schema, ltWithoutParam, true)
require.Nil(t, err)
require.True(t, res.Equal(ctx, NewNull())) // the expression is evaluated to null
param := NewOne()
param.ParamMarker = &ParamMarker{order: 0}
ctx.GetSessionVars().PlanCacheParams.Append(types.NewIntDatum(10))
ltWithParam, err := newFunctionForTest(ctx, ast.LT, col0, param)
require.NoError(t, err)
res, err = EvaluateExprWithNull(ctx, schema, ltWithParam, true)
require.Nil(t, err)
_, isConst := res.(*Constant)
require.True(t, isConst) // this expression is evaluated and skip-plan cache flag is set.
require.True(t, !ctx.GetSessionVars().StmtCtx.UseCache())
}
func TestEvaluateExprWithNullNoChangeRetType(t *testing.T) {
ctx := createContext(t)
tblInfo := newTestTableBuilder("").add("col_str", mysql.TypeString, 0).build()
schema := tableInfoToSchemaForTest(tblInfo)
castStrAsJSON := BuildCastFunction(ctx, schema.Columns[0], types.NewFieldType(mysql.TypeJSON))
jsonConstant := &Constant{Value: types.NewDatum("123"), RetType: types.NewFieldType(mysql.TypeJSON)}
// initially has ParseToJSONFlag
flagInCast := castStrAsJSON.(*ScalarFunction).RetType.GetFlag()
require.True(t, mysql.HasParseToJSONFlag(flagInCast))
// cast's ParseToJSONFlag removed by `DisableParseJSONFlag4Expr`
eq, err := newFunctionForTest(ctx, ast.EQ, jsonConstant, castStrAsJSON)
require.NoError(t, err)
flagInCast = eq.(*ScalarFunction).GetArgs()[1].(*ScalarFunction).RetType.GetFlag()
require.False(t, mysql.HasParseToJSONFlag(flagInCast))
// after EvaluateExprWithNull, this flag should be still false
EvaluateExprWithNull(ctx, schema, eq, true)
flagInCast = eq.(*ScalarFunction).GetArgs()[1].(*ScalarFunction).RetType.GetFlag()
require.False(t, mysql.HasParseToJSONFlag(flagInCast))
}
func TestConstant(t *testing.T) {
ctx := createContext(t)
require.False(t, NewZero().IsCorrelated())
require.Equal(t, ConstStrict, NewZero().ConstLevel())
require.True(t, NewZero().Decorrelate(nil).Equal(ctx, NewZero()))
require.Equal(t, []byte{0x0, 0x8, 0x0}, NewZero().HashCode())
require.False(t, NewZero().Equal(ctx, NewOne()))
}
func TestIsBinaryLiteral(t *testing.T) {
col := &Column{RetType: types.NewFieldType(mysql.TypeEnum)}
require.False(t, IsBinaryLiteral(col))
col.RetType.SetType(mysql.TypeSet)
require.False(t, IsBinaryLiteral(col))
col.RetType.SetType(mysql.TypeBit)
require.False(t, IsBinaryLiteral(col))
col.RetType.SetType(mysql.TypeDuration)
require.False(t, IsBinaryLiteral(col))
con := &Constant{RetType: types.NewFieldType(mysql.TypeVarString), Value: types.NewBinaryLiteralDatum([]byte{byte(0), byte(1)})}
require.True(t, IsBinaryLiteral(con))
con.Value = types.NewIntDatum(1)
require.False(t, IsBinaryLiteral(col))
}
func TestConstLevel(t *testing.T) {
ctxConst := NewZero()
ctxConst.DeferredExpr = newFunctionWithMockCtx(ast.UnixTimestamp)
ctx := exprstatic.NewEvalContext()
for _, c := range []struct {
exp Expression
level ConstLevel
}{
{newFunctionWithMockCtx(ast.Rand), ConstNone},
{newFunctionWithMockCtx(ast.UUID), ConstNone},
{newFunctionWithMockCtx(ast.GetParam, NewOne()), ConstNone},
{newFunctionWithMockCtx(ast.Abs, NewOne()), ConstStrict},
{newFunctionWithMockCtx(ast.Abs, newColumn(1)), ConstNone},
{newFunctionWithMockCtx(ast.Plus, NewOne(), NewOne()), ConstStrict},
{newFunctionWithMockCtx(ast.Plus, newColumn(1), NewOne()), ConstNone},
{newFunctionWithMockCtx(ast.Plus, NewOne(), newColumn(1)), ConstNone},
{newFunctionWithMockCtx(ast.Plus, NewOne(), newColumn(1)), ConstNone},
{newFunctionWithMockCtx(ast.Plus, NewOne(), ctxConst), ConstOnlyInContext},
} {
require.Equal(t, c.level, c.exp.ConstLevel(), c.exp.StringWithCtx(ctx, errors.RedactLogDisable))
}
}
func TestVectorizable(t *testing.T) {
exprs := make([]Expression, 0, 4)
sf := newFunctionWithMockCtx(ast.Rand)
column := &Column{
UniqueID: 0,
RetType: types.NewFieldType(mysql.TypeLonglong),
}
exprs = append(exprs, sf)
exprs = append(exprs, NewOne())
exprs = append(exprs, NewNull())
exprs = append(exprs, column)
require.True(t, Vectorizable(exprs))
column0 := &Column{
UniqueID: 1,
RetType: types.NewFieldType(mysql.TypeString),
}
column1 := &Column{
UniqueID: 2,
RetType: types.NewFieldType(mysql.TypeString),
}
column2 := &Column{
UniqueID: 3,
RetType: types.NewFieldType(mysql.TypeLonglong),
}
exprs = exprs[:0]
sf = newFunctionWithMockCtx(ast.SetVar, column0, column1)
exprs = append(exprs, sf)
require.False(t, Vectorizable(exprs))
exprs = exprs[:0]
sf = newFunctionWithMockCtx(ast.GetVar, column0)
exprs = append(exprs, sf)
require.False(t, Vectorizable(exprs))
exprs = exprs[:0]
sf = newFunctionWithMockCtx(ast.NextVal, column0)
exprs = append(exprs, sf)
sf = newFunctionWithMockCtx(ast.LastVal, column0)
exprs = append(exprs, sf)
sf = newFunctionWithMockCtx(ast.SetVal, column1, column2)
exprs = append(exprs, sf)
require.False(t, Vectorizable(exprs))
}
type testTableBuilder struct {
tableName string
columnNames []string
tps []byte
flags []uint
}
func newTestTableBuilder(tableName string) *testTableBuilder {
return &testTableBuilder{tableName: tableName}
}
func (builder *testTableBuilder) add(name string, tp byte, flag uint) *testTableBuilder {
builder.columnNames = append(builder.columnNames, name)
builder.tps = append(builder.tps, tp)
builder.flags = append(builder.flags, flag)
return builder
}
func (builder *testTableBuilder) build() *model.TableInfo {
ti := &model.TableInfo{
ID: 1,
Name: ast.NewCIStr(builder.tableName),
State: model.StatePublic,
}
for i, colName := range builder.columnNames {
tp := builder.tps[i]
fieldType := types.NewFieldType(tp)
flen, decimal := mysql.GetDefaultFieldLengthAndDecimal(tp)
fieldType.SetFlen(flen)
fieldType.SetDecimal(decimal)
charset, collate := types.DefaultCharsetForType(tp)
fieldType.SetCharset(charset)
fieldType.SetCollate(collate)
fieldType.SetFlag(builder.flags[i])
ti.Columns = append(ti.Columns, &model.ColumnInfo{
ID: int64(i + 1),
Name: ast.NewCIStr(colName),
Offset: i,
FieldType: *fieldType,
State: model.StatePublic,
})
}
return ti
}
func tableInfoToSchemaForTest(tableInfo *model.TableInfo) *Schema {
columns := tableInfo.Columns
schema := NewSchema(make([]*Column, 0, len(columns))...)
for i, col := range columns {
schema.Append(&Column{
UniqueID: int64(i),
ID: col.ID,
RetType: &col.FieldType,
})
}
return schema
}
func TestEvalExpr(t *testing.T) {
ctx := createContext(t)
eTypes := []types.EvalType{types.ETInt, types.ETReal, types.ETDecimal, types.ETString, types.ETTimestamp, types.ETDatetime, types.ETDuration}
tNames := []string{"int", "real", "decimal", "string", "timestamp", "datetime", "duration"}
for i := range tNames {
ft := eType2FieldType(eTypes[i])
colExpr := &Column{Index: 0, RetType: ft}
input := chunk.New([]*types.FieldType{ft}, 1024, 1024)
fillColumnWithGener(eTypes[i], input, 0, nil)
colBuf := chunk.NewColumn(ft, 1024)
colBuf2 := chunk.NewColumn(ft, 1024)
var err error
require.True(t, colExpr.Vectorized())
err = EvalExpr(ctx, false, colExpr, colExpr.GetType(ctx).EvalType(), input, colBuf)
require.NoError(t, err)
err = EvalExpr(ctx, true, colExpr, colExpr.GetType(ctx).EvalType(), input, colBuf2)
require.NoError(t, err)
for j := range 1024 {
isNull := colBuf.IsNull(j)
isNull2 := colBuf2.IsNull(j)
require.Equal(t, isNull2, isNull)
if isNull {
continue
}
require.Equal(t, string(colBuf2.GetRaw(j)), string(colBuf.GetRaw(j)))
}
}
}
func TestExpressionMemeoryUsage(t *testing.T) {
c1 := &Column{OrigName: "Origin"}
c2 := Column{OrigName: "OriginName"}
require.Greater(t, c2.MemoryUsage(), c1.MemoryUsage())
c1 = nil
require.Equal(t, c1.MemoryUsage(), int64(0))
c3 := Constant{Value: types.NewIntDatum(1)}
c4 := Constant{Value: types.NewStringDatum("11")}
require.Greater(t, c4.MemoryUsage(), c3.MemoryUsage())
}