Files
tidb/expression/builtin_miscellaneous_test.go
2021-12-14 18:06:36 +08:00

580 lines
16 KiB
Go

// Copyright 2021 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 (
"math"
"strings"
"testing"
"time"
"github.com/pingcap/tidb/parser/ast"
"github.com/pingcap/tidb/parser/mysql"
"github.com/pingcap/tidb/testkit/trequire"
"github.com/pingcap/tidb/types"
"github.com/pingcap/tidb/util/chunk"
"github.com/stretchr/testify/require"
)
func TestInetAton(t *testing.T) {
ctx := createContext(t)
tbl := []struct {
Input interface{}
Expected interface{}
}{
{"", nil},
{nil, nil},
{"255.255.255.255", 4294967295},
{"0.0.0.0", 0},
{"127.0.0.1", 2130706433},
{"0.0.0.256", nil},
{"113.14.22.3", 1896748547},
{"127", 127},
{"127.255", 2130706687},
{"127,256", nil},
{"127.2.1", 2130837505},
{"123.2.1.", nil},
{"127.0.0.1.1", nil},
}
dtbl := tblToDtbl(tbl)
fc := funcs[ast.InetAton]
for _, tt := range dtbl {
f, err := fc.getFunction(ctx, datumsToConstants(tt["Input"]))
require.NoError(t, err)
d, err := evalBuiltinFunc(f, chunk.Row{})
require.NoError(t, err)
trequire.DatumEqual(t, tt["Expected"][0], d)
}
}
func TestIsIPv4(t *testing.T) {
ctx := createContext(t)
tests := []struct {
ip string
expect interface{}
}{
{"192.168.1.1", 1},
{"255.255.255.255", 1},
{"10.t.255.255", 0},
{"10.1.2.3.4", 0},
{"2001:250:207:0:0:eef2::1", 0},
{"::ffff:1.2.3.4", 0},
{"1...1", 0},
{"192.168.1.", 0},
{".168.1.2", 0},
{"168.1.2", 0},
{"1.2.3.4.5", 0},
}
fc := funcs[ast.IsIPv4]
for _, test := range tests {
ip := types.NewStringDatum(test.ip)
f, err := fc.getFunction(ctx, datumsToConstants([]types.Datum{ip}))
require.NoError(t, err)
result, err := evalBuiltinFunc(f, chunk.Row{})
require.NoError(t, err)
trequire.DatumEqual(t, types.NewDatum(test.expect), result)
}
// test NULL input for is_ipv4
var argNull types.Datum
f, _ := fc.getFunction(ctx, datumsToConstants([]types.Datum{argNull}))
r, err := evalBuiltinFunc(f, chunk.Row{})
require.NoError(t, err)
trequire.DatumEqual(t, types.NewDatum(0), r)
}
func TestIsUUID(t *testing.T) {
ctx := createContext(t)
tests := []struct {
uuid string
expect interface{}
}{
{"6ccd780c-baba-1026-9564-5b8c656024db", 1},
{"6CCD780C-BABA-1026-9564-5B8C656024DB", 1},
{"6ccd780cbaba102695645b8c656024db", 1},
{"{6ccd780c-baba-1026-9564-5b8c656024db}", 1},
{"6ccd780c-baba-1026-9564-5b8c6560", 0},
{"6CCD780C-BABA-1026-9564-5B8C656024DQ", 0},
// This is a bug in google/uuid#60
{"{99a9ad03-5298-11ec-8f5c-00ff90147ac3*", 1},
// This is a format google/uuid support, while mysql doesn't
{"urn:uuid:99a9ad03-5298-11ec-8f5c-00ff90147ac3", 1},
}
fc := funcs[ast.IsUUID]
for _, test := range tests {
uuid := types.NewStringDatum(test.uuid)
f, err := fc.getFunction(ctx, datumsToConstants([]types.Datum{uuid}))
require.NoError(t, err)
result, err := evalBuiltinFunc(f, chunk.Row{})
require.NoError(t, err)
trequire.DatumEqual(t, types.NewDatum(test.expect), result)
}
var argNull types.Datum
f, _ := fc.getFunction(ctx, datumsToConstants([]types.Datum{argNull}))
r, err := evalBuiltinFunc(f, chunk.Row{})
require.NoError(t, err)
require.True(t, r.IsNull())
}
func TestUUID(t *testing.T) {
ctx := createContext(t)
f, err := newFunctionForTest(ctx, ast.UUID)
require.NoError(t, err)
d, err := f.Eval(chunk.Row{})
require.NoError(t, err)
parts := strings.Split(d.GetString(), "-")
require.Equal(t, 5, len(parts))
for i, p := range parts {
switch i {
case 0:
require.Equal(t, 8, len(p))
case 1:
require.Equal(t, 4, len(p))
case 2:
require.Equal(t, 4, len(p))
case 3:
require.Equal(t, 4, len(p))
case 4:
require.Equal(t, 12, len(p))
}
}
_, err = funcs[ast.UUID].getFunction(ctx, datumsToConstants(nil))
require.NoError(t, err)
}
func TestAnyValue(t *testing.T) {
ctx := createContext(t)
tbl := []struct {
arg interface{}
ret interface{}
}{
{nil, nil},
{1234, 1234},
{-0x99, -0x99},
{3.1415926, 3.1415926},
{"Hello, World", "Hello, World"},
}
for _, tt := range tbl {
fc := funcs[ast.AnyValue]
f, err := fc.getFunction(ctx, datumsToConstants(types.MakeDatums(tt.arg)))
require.NoError(t, err)
r, err := evalBuiltinFunc(f, chunk.Row{})
require.NoError(t, err)
trequire.DatumEqual(t, types.NewDatum(tt.ret), r)
}
}
func TestIsIPv6(t *testing.T) {
ctx := createContext(t)
tests := []struct {
ip string
expect interface{}
}{
{"2001:250:207:0:0:eef2::1", 1},
{"2001:0250:0207:0001:0000:0000:0000:ff02", 1},
{"2001:250:207::eff2::1,", 0},
{"192.168.1.1", 0},
{"::ffff:1.2.3.4", 1},
}
fc := funcs[ast.IsIPv6]
for _, test := range tests {
ip := types.NewStringDatum(test.ip)
f, err := fc.getFunction(ctx, datumsToConstants([]types.Datum{ip}))
require.NoError(t, err)
result, err := evalBuiltinFunc(f, chunk.Row{})
require.NoError(t, err)
trequire.DatumEqual(t, types.NewDatum(test.expect), result)
}
// test NULL input for is_ipv6
var argNull types.Datum
f, _ := fc.getFunction(ctx, datumsToConstants([]types.Datum{argNull}))
r, err := evalBuiltinFunc(f, chunk.Row{})
require.NoError(t, err)
trequire.DatumEqual(t, types.NewDatum(0), r)
}
func TestInetNtoa(t *testing.T) {
ctx := createContext(t)
tests := []struct {
ip int
expect interface{}
}{
{167773449, "10.0.5.9"},
{2063728641, "123.2.0.1"},
{0, "0.0.0.0"},
{545460846593, nil},
{-1, nil},
{math.MaxUint32, "255.255.255.255"},
}
fc := funcs[ast.InetNtoa]
for _, test := range tests {
ip := types.NewDatum(test.ip)
f, err := fc.getFunction(ctx, datumsToConstants([]types.Datum{ip}))
require.NoError(t, err)
result, err := evalBuiltinFunc(f, chunk.Row{})
require.NoError(t, err)
trequire.DatumEqual(t, types.NewDatum(test.expect), result)
}
var argNull types.Datum
f, _ := fc.getFunction(ctx, datumsToConstants([]types.Datum{argNull}))
r, err := evalBuiltinFunc(f, chunk.Row{})
require.NoError(t, err)
require.True(t, r.IsNull())
}
func TestInet6NtoA(t *testing.T) {
ctx := createContext(t)
tests := []struct {
ip []byte
expect interface{}
}{
// Success cases
{[]byte{0x00, 0x00, 0x00, 0x00}, "0.0.0.0"},
{[]byte{0x0A, 0x00, 0x05, 0x09}, "10.0.5.9"},
{[]byte{0xFD, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x5A, 0x55, 0xCA, 0xFF, 0xFE,
0xFA, 0x90, 0x89}, "fdfe::5a55:caff:fefa:9089"},
{[]byte{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0x01,
0x02, 0x03, 0x04}, "::ffff:1.2.3.4"},
{[]byte{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xFF,
0xFF, 0xFF, 0xFF}, "::ffff:255.255.255.255"},
// Fail cases
{[]byte{}, nil}, // missing bytes
{[]byte{0x0A, 0x00, 0x05}, nil}, // missing a byte ipv4
{[]byte{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xFF,
0xFF, 0xFF}, nil}, // missing a byte ipv6
}
fc := funcs[ast.Inet6Ntoa]
for _, test := range tests {
ip := types.NewDatum(test.ip)
f, err := fc.getFunction(ctx, datumsToConstants([]types.Datum{ip}))
require.NoError(t, err)
result, err := evalBuiltinFunc(f, chunk.Row{})
require.NoError(t, err)
trequire.DatumEqual(t, types.NewDatum(test.expect), result)
}
var argNull types.Datum
f, _ := fc.getFunction(ctx, datumsToConstants([]types.Datum{argNull}))
r, err := evalBuiltinFunc(f, chunk.Row{})
require.NoError(t, err)
require.True(t, r.IsNull())
}
func TestInet6AtoN(t *testing.T) {
ctx := createContext(t)
tests := []struct {
ip string
expect interface{}
}{
{"0.0.0.0", []byte{0x00, 0x00, 0x00, 0x00}},
{"10.0.5.9", []byte{0x0A, 0x00, 0x05, 0x09}},
{"fdfe::5a55:caff:fefa:9089", []byte{0xFD, 0xFE, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x5A, 0x55, 0xCA, 0xFF, 0xFE, 0xFA, 0x90, 0x89}},
{"::ffff:1.2.3.4", []byte{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0x01, 0x02, 0x03, 0x04}},
{"", nil},
{"Not IP address", nil},
{"::ffff:255.255.255.255", []byte{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF}},
}
fc := funcs[ast.Inet6Aton]
for _, test := range tests {
ip := types.NewDatum(test.ip)
f, err := fc.getFunction(ctx, datumsToConstants([]types.Datum{ip}))
require.NoError(t, err)
result, err := evalBuiltinFunc(f, chunk.Row{})
require.NoError(t, err)
trequire.DatumEqual(t, types.NewDatum(test.expect), result)
}
var argNull types.Datum
f, _ := fc.getFunction(ctx, datumsToConstants([]types.Datum{argNull}))
r, err := evalBuiltinFunc(f, chunk.Row{})
require.NoError(t, err)
require.True(t, r.IsNull())
}
func TestIsIPv4Mapped(t *testing.T) {
ctx := createContext(t)
tests := []struct {
ip []byte
expect interface{}
}{
{[]byte{}, 0},
{[]byte{0x10, 0x10, 0x10, 0x10}, 0},
{[]byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0xff, 0xff, 0x1, 0x2, 0x3, 0x4}, 1},
{[]byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x1, 0xff, 0xff, 0x1, 0x2, 0x3, 0x4}, 0},
{[]byte{0x0, 0x1, 0x2, 0x3, 0x4, 0x5, 0x6}, 0},
}
fc := funcs[ast.IsIPv4Mapped]
for _, test := range tests {
ip := types.NewDatum(test.ip)
f, err := fc.getFunction(ctx, datumsToConstants([]types.Datum{ip}))
require.NoError(t, err)
result, err := evalBuiltinFunc(f, chunk.Row{})
require.NoError(t, err)
trequire.DatumEqual(t, types.NewDatum(test.expect), result)
}
var argNull types.Datum
f, _ := fc.getFunction(ctx, datumsToConstants([]types.Datum{argNull}))
r, err := evalBuiltinFunc(f, chunk.Row{})
require.NoError(t, err)
trequire.DatumEqual(t, types.NewDatum(int64(0)), r)
}
func TestIsIPv4Compat(t *testing.T) {
ctx := createContext(t)
tests := []struct {
ip []byte
expect interface{}
}{
{[]byte{}, 0},
{[]byte{0x10, 0x10, 0x10, 0x10}, 0},
{[]byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x1, 0x2, 0x3, 0x4}, 1},
{[]byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x1, 0x0, 0x0, 0x1, 0x2, 0x3, 0x4}, 0},
{[]byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x1, 0xff, 0xff, 0x1, 0x2, 0x3, 0x4}, 0},
{[]byte{0x0, 0x1, 0x2, 0x3, 0x4, 0x5, 0x6}, 0},
}
fc := funcs[ast.IsIPv4Compat]
for _, test := range tests {
ip := types.NewDatum(test.ip)
f, err := fc.getFunction(ctx, datumsToConstants([]types.Datum{ip}))
require.NoError(t, err)
result, err := evalBuiltinFunc(f, chunk.Row{})
require.NoError(t, err)
trequire.DatumEqual(t, types.NewDatum(test.expect), result)
}
var argNull types.Datum
f, _ := fc.getFunction(ctx, datumsToConstants([]types.Datum{argNull}))
r, err := evalBuiltinFunc(f, chunk.Row{})
require.NoError(t, err)
trequire.DatumEqual(t, types.NewDatum(0), r)
}
func TestNameConst(t *testing.T) {
ctx := createContext(t)
dec := types.NewDecFromFloatForTest(123.123)
tm := types.NewTime(types.FromGoTime(time.Now()), mysql.TypeDatetime, 6)
du := types.Duration{Duration: 12*time.Hour + 1*time.Minute + 1*time.Second, Fsp: types.DefaultFsp}
cases := []struct {
colName string
arg interface{}
isNil bool
asserts func(d types.Datum)
}{
{"test_int", 3, false, func(d types.Datum) {
require.Equal(t, int64(3), d.GetInt64())
}},
{"test_float", 3.14159, false, func(d types.Datum) {
require.Equal(t, 3.14159, d.GetFloat64())
}},
{"test_string", "TiDB", false, func(d types.Datum) {
require.Equal(t, "TiDB", d.GetString())
}},
{"test_null", nil, true, func(d types.Datum) {
require.Equal(t, types.KindNull, d.Kind())
}},
{"test_decimal", dec, false, func(d types.Datum) {
require.Equal(t, dec.String(), d.GetMysqlDecimal().String())
}},
{"test_time", tm, false, func(d types.Datum) {
require.Equal(t, tm.String(), d.GetMysqlTime().String())
}},
{"test_duration", du, false, func(d types.Datum) {
require.Equal(t, du.String(), d.GetMysqlDuration().String())
}},
}
for _, c := range cases {
f, err := newFunctionForTest(ctx, ast.NameConst, primitiveValsToConstants(ctx, []interface{}{c.colName, c.arg})...)
require.NoError(t, err)
d, err := f.Eval(chunk.Row{})
require.NoError(t, err)
c.asserts(d)
}
}
func TestUUIDToBin(t *testing.T) {
ctx := createContext(t)
tests := []struct {
args []interface{}
expect interface{}
isNil bool
getWarning bool
getError bool
}{
{
[]interface{}{"6ccd780c-baba-1026-9564-5b8c656024db"},
[]byte{0x6C, 0xCD, 0x78, 0x0C, 0xBA, 0xBA, 0x10, 0x26, 0x95, 0x64, 0x5B, 0x8C, 0x65, 0x60, 0x24, 0xDB},
false,
false,
false,
},
{
[]interface{}{"6CCD780C-BABA-1026-9564-5B8C656024DB"},
[]byte{0x6C, 0xCD, 0x78, 0x0C, 0xBA, 0xBA, 0x10, 0x26, 0x95, 0x64, 0x5B, 0x8C, 0x65, 0x60, 0x24, 0xDB},
false,
false,
false,
},
{
[]interface{}{"6ccd780cbaba102695645b8c656024db"},
[]byte{0x6C, 0xCD, 0x78, 0x0C, 0xBA, 0xBA, 0x10, 0x26, 0x95, 0x64, 0x5B, 0x8C, 0x65, 0x60, 0x24, 0xDB},
false,
false,
false,
},
{
[]interface{}{"{6ccd780c-baba-1026-9564-5b8c656024db}"},
[]byte{0x6C, 0xCD, 0x78, 0x0C, 0xBA, 0xBA, 0x10, 0x26, 0x95, 0x64, 0x5B, 0x8C, 0x65, 0x60, 0x24, 0xDB},
false,
false,
false,
},
{
[]interface{}{"6ccd780c-baba-1026-9564-5b8c656024db", 0},
[]byte{0x6C, 0xCD, 0x78, 0x0C, 0xBA, 0xBA, 0x10, 0x26, 0x95, 0x64, 0x5B, 0x8C, 0x65, 0x60, 0x24, 0xDB},
false,
false,
false,
},
{
[]interface{}{"6ccd780c-baba-1026-9564-5b8c656024db", 1},
[]byte{0x10, 0x26, 0xBA, 0xBA, 0x6C, 0xCD, 0x78, 0x0C, 0x95, 0x64, 0x5B, 0x8C, 0x65, 0x60, 0x24, 0xDB},
false,
false,
false,
},
{
[]interface{}{"6ccd780c-baba-1026-9564-5b8c656024db", "a"},
[]byte{0x6C, 0xCD, 0x78, 0x0C, 0xBA, 0xBA, 0x10, 0x26, 0x95, 0x64, 0x5B, 0x8C, 0x65, 0x60, 0x24, 0xDB},
false,
true,
false,
},
{
[]interface{}{"6ccd780c-baba-1026-9564-5b8c6560"},
[]byte{},
false,
false,
true,
},
{
[]interface{}{nil},
[]byte{},
true,
false,
false,
},
}
for _, test := range tests {
preWarningCnt := ctx.GetSessionVars().StmtCtx.WarningCount()
f, err := newFunctionForTest(ctx, ast.UUIDToBin, primitiveValsToConstants(ctx, test.args)...)
require.NoError(t, err)
result, err := f.Eval(chunk.Row{})
if test.getError {
require.Error(t, err)
} else if test.getWarning {
require.NoError(t, err)
require.Equal(t, preWarningCnt+1, ctx.GetSessionVars().StmtCtx.WarningCount())
} else {
require.NoError(t, err)
if test.isNil {
require.Equal(t, types.KindNull, result.Kind())
} else {
trequire.DatumEqual(t, types.NewDatum(test.expect), result)
}
}
}
_, err := funcs[ast.UUIDToBin].getFunction(ctx, []Expression{NewZero()})
require.NoError(t, err)
}
func TestBinToUUID(t *testing.T) {
ctx := createContext(t)
tests := []struct {
args []interface{}
expect string
isNil bool
getWarning bool
getError bool
}{
{
[]interface{}{[]byte{0x6C, 0xCD, 0x78, 0x0C, 0xBA, 0xBA, 0x10, 0x26, 0x95, 0x64, 0x5B, 0x8C, 0x65, 0x60, 0x24, 0xDB}},
"6ccd780c-baba-1026-9564-5b8c656024db",
false,
false,
false,
},
{
[]interface{}{[]byte{0x6C, 0xCD, 0x78, 0x0C, 0xBA, 0xBA, 0x10, 0x26, 0x95, 0x64, 0x5B, 0x8C, 0x65, 0x60, 0x24, 0xDB}, 1},
"baba1026-780c-6ccd-9564-5b8c656024db",
false,
false,
false,
},
{
[]interface{}{[]byte{0x6C, 0xCD, 0x78, 0x0C, 0xBA, 0xBA, 0x10, 0x26, 0x95, 0x64, 0x5B, 0x8C, 0x65, 0x60, 0x24, 0xDB}, "a"},
"6ccd780c-baba-1026-9564-5b8c656024db",
false,
true,
false,
},
{
[]interface{}{[]byte{0x6C, 0xCD, 0x78, 0x0C, 0xBA, 0xBA, 0x10, 0x26, 0x95, 0x64, 0x5B, 0x8C, 0x65, 0x60}},
"",
false,
false,
true,
},
{
[]interface{}{nil},
"",
true,
false,
false,
},
}
for _, test := range tests {
preWarningCnt := ctx.GetSessionVars().StmtCtx.WarningCount()
f, err := newFunctionForTest(ctx, ast.BinToUUID, primitiveValsToConstants(ctx, test.args)...)
require.NoError(t, err)
result, err := f.Eval(chunk.Row{})
if test.getError {
require.Error(t, err)
} else if test.getWarning {
require.NoError(t, err)
require.Equal(t, preWarningCnt+1, ctx.GetSessionVars().StmtCtx.WarningCount())
} else {
require.NoError(t, err)
if test.isNil {
require.Equal(t, types.KindNull, result.Kind())
} else {
require.Equal(t, test.expect, result.GetString())
}
}
}
_, err := funcs[ast.BinToUUID].getFunction(ctx, []Expression{NewZero()})
require.NoError(t, err)
}