Files
tidb/pkg/expression/scalar_function_test.go
2025-02-13 11:22:15 +00:00

200 lines
6.3 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/tidb/pkg/parser/ast"
"github.com/pingcap/tidb/pkg/parser/mysql"
"github.com/pingcap/tidb/pkg/planner/cascades/base"
"github.com/pingcap/tidb/pkg/types"
"github.com/pingcap/tidb/pkg/util/chunk"
"github.com/pingcap/tidb/pkg/util/mock"
"github.com/stretchr/testify/require"
)
func TestExpressionSemanticEqual(t *testing.T) {
a := &Column{
UniqueID: 1,
RetType: types.NewFieldType(mysql.TypeDouble),
}
b := &Column{
UniqueID: 2,
RetType: types.NewFieldType(mysql.TypeLong),
}
// order sensitive cases
// a < b; b > a
sf1 := newFunctionWithMockCtx(ast.LT, a, b)
sf2 := newFunctionWithMockCtx(ast.GT, b, a)
require.True(t, ExpressionsSemanticEqual(sf1, sf2))
// a > b; b < a
sf3 := newFunctionWithMockCtx(ast.GT, a, b)
sf4 := newFunctionWithMockCtx(ast.LT, b, a)
require.True(t, ExpressionsSemanticEqual(sf3, sf4))
// a<=b; b>=a
sf5 := newFunctionWithMockCtx(ast.LE, a, b)
sf6 := newFunctionWithMockCtx(ast.GE, b, a)
require.True(t, ExpressionsSemanticEqual(sf5, sf6))
// a>=b; b<=a
sf7 := newFunctionWithMockCtx(ast.GE, a, b)
sf8 := newFunctionWithMockCtx(ast.LE, b, a)
require.True(t, ExpressionsSemanticEqual(sf7, sf8))
// not(a<b); a >= b
sf9 := newFunctionWithMockCtx(ast.UnaryNot, sf1)
require.True(t, ExpressionsSemanticEqual(sf9, sf7))
// a < b; not(a>=b)
sf10 := newFunctionWithMockCtx(ast.UnaryNot, sf7)
require.True(t, ExpressionsSemanticEqual(sf1, sf10))
// order insensitive cases
// a + b; b + a
p1 := newFunctionWithMockCtx(ast.Plus, a, b)
p2 := newFunctionWithMockCtx(ast.Plus, b, a)
require.True(t, ExpressionsSemanticEqual(p1, p2))
// a * b; b * a
m1 := newFunctionWithMockCtx(ast.Mul, a, b)
m2 := newFunctionWithMockCtx(ast.Mul, b, a)
require.True(t, ExpressionsSemanticEqual(m1, m2))
// a = b; b = a
e1 := newFunctionWithMockCtx(ast.EQ, a, b)
e2 := newFunctionWithMockCtx(ast.EQ, b, a)
require.True(t, ExpressionsSemanticEqual(e1, e2))
// a = b AND b + a; a + b AND b = a
a1 := newFunctionWithMockCtx(ast.LogicAnd, e1, p2)
a2 := newFunctionWithMockCtx(ast.LogicAnd, p1, e2)
require.True(t, ExpressionsSemanticEqual(a1, a2))
// a * b OR a + b; b + a OR b * a
o1 := newFunctionWithMockCtx(ast.LogicOr, m1, p1)
o2 := newFunctionWithMockCtx(ast.LogicOr, p2, m2)
require.True(t, ExpressionsSemanticEqual(o1, o2))
}
func TestScalarFunction(t *testing.T) {
ctx := mock.NewContext()
a := &Column{
UniqueID: 1,
RetType: types.NewFieldType(mysql.TypeDouble),
}
sf := newFunctionWithMockCtx(ast.LT, a, NewOne())
require.False(t, sf.IsCorrelated())
require.Equal(t, ConstNone, sf.ConstLevel())
require.True(t, sf.Decorrelate(nil).Equal(ctx, sf))
require.EqualValues(t, []byte{0x3, 0x4, 0x6c, 0x74, 0x1, 0x80, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x1, 0x0, 0x5, 0xbf, 0xf0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0}, sf.HashCode())
sf = NewValuesFunc(ctx, 0, types.NewFieldType(mysql.TypeLonglong))
newSf, ok := sf.Clone().(*ScalarFunction)
require.True(t, ok)
require.Equal(t, "values", newSf.FuncName.O)
require.Equal(t, mysql.TypeLonglong, newSf.RetType.GetType())
require.Equal(t, sf.Coercibility(), newSf.Coercibility())
require.Equal(t, sf.Repertoire(), newSf.Repertoire())
_, ok = newSf.Function.(*builtinValuesIntSig)
require.True(t, ok)
}
func TestIssue23309(t *testing.T) {
a := &Column{
UniqueID: 1,
RetType: types.NewFieldType(mysql.TypeDouble),
}
a.RetType.SetFlag(a.RetType.GetFlag() | mysql.NotNullFlag)
null := NewNull()
null.RetType = types.NewFieldType(mysql.TypeNull)
sf, _ := newFunctionWithMockCtx(ast.NE, a, null).(*ScalarFunction)
v, err := sf.GetArgs()[1].Eval(mock.NewContext(), chunk.Row{})
require.NoError(t, err)
require.True(t, v.IsNull())
ctx := createContext(t)
require.False(t, mysql.HasNotNullFlag(sf.GetArgs()[1].GetType(ctx).GetFlag()))
}
func TestScalarFuncs2Exprs(t *testing.T) {
ctx := mock.NewContext()
a := &Column{
UniqueID: 1,
RetType: types.NewFieldType(mysql.TypeDouble),
}
sf0, _ := newFunctionWithMockCtx(ast.LT, a, NewZero()).(*ScalarFunction)
sf1, _ := newFunctionWithMockCtx(ast.LT, a, NewOne()).(*ScalarFunction)
funcs := []*ScalarFunction{sf0, sf1}
exprs := ScalarFuncs2Exprs(funcs)
for i := range exprs {
require.True(t, exprs[i].Equal(ctx, funcs[i]))
}
}
func TestScalarFunctionHash64Equals(t *testing.T) {
a := &Column{
UniqueID: 1,
RetType: types.NewFieldType(mysql.TypeDouble),
}
sf0, _ := newFunctionWithMockCtx(ast.LT, a, NewZero()).(*ScalarFunction)
sf1, _ := newFunctionWithMockCtx(ast.LT, a, NewZero()).(*ScalarFunction)
hasher1 := base.NewHashEqualer()
hasher2 := base.NewHashEqualer()
sf0.Hash64(hasher1)
sf1.Hash64(hasher2)
require.Equal(t, hasher1.Sum64(), hasher2.Sum64())
require.True(t, sf0.Equals(sf1))
// change the func name
sf2, _ := newFunctionWithMockCtx(ast.GT, a, NewZero()).(*ScalarFunction)
hasher2.Reset()
sf2.Hash64(hasher2)
require.NotEqual(t, hasher1.Sum64(), hasher2.Sum64())
require.False(t, sf0.Equals(sf2))
// change the args
sf3, _ := newFunctionWithMockCtx(ast.LT, a, NewOne()).(*ScalarFunction)
hasher2.Reset()
sf3.Hash64(hasher2)
require.NotEqual(t, hasher1.Sum64(), hasher2.Sum64())
require.False(t, sf0.Equals(sf3))
// change the ret type
sf4, _ := newFunctionWithMockCtx(ast.LT, a, NewZero()).(*ScalarFunction)
sf4.RetType = types.NewFieldType(mysql.TypeLong)
hasher2.Reset()
sf4.Hash64(hasher2)
require.NotEqual(t, hasher1.Sum64(), hasher2.Sum64())
require.False(t, sf0.Equals(sf4))
}
// To test that when argument number is 0, unix_timestamp can not be pushed down to tikv
func TestForbidUnixTimestampPushdown(t *testing.T) {
ctx := mock.NewContext()
fc := &unixTimestampFunctionClass{baseFunctionClass{ast.UnixTimestamp, 0, 1}}
bt, err := fc.getFunction(ctx, nil)
require.NoError(t, err)
sf := &ScalarFunction{
Function: bt,
}
require.False(t, scalarExprSupportedByTiKV(ctx, sf))
}