Files
tidb/pkg/expression/builtin_like_test.go

174 lines
5.1 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 (
"fmt"
"testing"
"github.com/pingcap/tidb/pkg/parser/ast"
"github.com/pingcap/tidb/pkg/parser/terror"
"github.com/pingcap/tidb/pkg/testkit/testutil"
"github.com/pingcap/tidb/pkg/types"
"github.com/pingcap/tidb/pkg/util/chunk"
"github.com/pingcap/tidb/pkg/util/collate"
"github.com/stretchr/testify/require"
)
func TestLike(t *testing.T) {
ctx := createContext(t)
tests := []struct {
input string
pattern string
match int
}{
{"a", "", 0},
{"a", "a", 1},
{"a", "b", 0},
{"aA", "Aa", 0},
{"aAb", `Aa%`, 0},
{"aAb", "aA_", 1},
{"baab", "b_%b", 1},
{"baab", "b%_b", 1},
{"bab", "b_%b", 1},
{"bab", "b%_b", 1},
{"bb", "b_%b", 0},
{"bb", "b%_b", 0},
{"baabccc", "b_%b%", 1},
{"a", `\a`, 1},
}
for _, tt := range tests {
comment := fmt.Sprintf(`for input = "%s", pattern = "%s"`, tt.input, tt.pattern)
fc := funcs[ast.Like]
f, err := fc.getFunction(ctx, datumsToConstants(types.MakeDatums(tt.input, tt.pattern, int('\\'))))
require.NoError(t, err, comment)
r, err := evalBuiltinFuncConcurrent(f, ctx, chunk.Row{})
require.NoError(t, err, comment)
testutil.DatumEqual(t, types.NewDatum(tt.match), r, comment)
}
}
func TestRegexp(t *testing.T) {
ctx := createContext(t)
tests := []struct {
pattern string
input string
match int64
err error
}{
{"^$", "a", 0, nil},
{"a", "a", 1, nil},
{"a", "b", 0, nil},
{"aA", "aA", 1, nil},
{".", "a", 1, nil},
{"^.$", "ab", 0, nil},
{"..", "b", 0, nil},
{".ab", "aab", 1, nil},
{".*", "abcd", 1, nil},
{"(", "", 0, ErrRegexp},
{"(*", "", 0, ErrRegexp},
{"[a", "", 0, ErrRegexp},
{"\\", "", 0, ErrRegexp},
}
for _, tt := range tests {
fc := funcs[ast.Regexp]
f, err := fc.getFunction(ctx, datumsToConstants(types.MakeDatums(tt.input, tt.pattern)))
require.NoError(t, err)
match, err := evalBuiltinFunc(f, ctx, chunk.Row{})
if tt.err == nil {
require.NoError(t, err)
testutil.DatumEqual(t, types.NewDatum(tt.match), match, fmt.Sprintf("%v", tt))
} else {
require.True(t, terror.ErrorEqual(err, tt.err))
}
}
}
func TestCILike(t *testing.T) {
ctx := createContext(t)
tests := []struct {
input string
pattern string
generalMatch int
unicodeMatch int
unicode0900Match int
}{
{"a", "", 0, 0, 0},
{"a", "a", 1, 1, 1},
{"a", "á", 1, 1, 1},
{"a", "b", 0, 0, 0},
{"aA", "Aa", 1, 1, 1},
{"áAb", `Aa%`, 1, 1, 1},
{"áAb", `%ab%`, 1, 1, 1},
{"áAb", `%ab`, 1, 1, 1},
{"ÀAb", "aA_", 1, 1, 1},
{"áééá", "a_%a", 1, 1, 1},
{"áééá", "a%_a", 1, 1, 1},
{"áéá", "a_%a", 1, 1, 1},
{"áéá", "a%_a", 1, 1, 1},
{"áá", "a_%a", 0, 0, 0},
{"áá", "a%_a", 0, 0, 0},
{"áééáííí", "a_%a%", 1, 1, 1},
{"数汉据字库", "数%据_库", 1, 1, 1},
// performs matching on a per-character basis
// https://dev.mysql.com/doc/refman/5.7/en/string-comparison-functions.html#operator_like
{"ß", "s%", 1, 0, 0},
{"ß", "%s", 1, 0, 0},
{"ß", "ss", 0, 0, 0},
{"ß", "s", 1, 0, 0},
{"ss", "%ß%", 1, 0, 0},
{"ß", "_", 1, 1, 1},
{"ß", "__", 0, 0, 0},
{"Ⱕ", "ⱕ", 0, 0, 1},
}
for _, tt := range tests {
comment := fmt.Sprintf(`for input = "%s", pattern = "%s"`, tt.input, tt.pattern)
fc := funcs[ast.Like]
inputs := datumsToConstants(types.MakeDatums(tt.input, tt.pattern, 0))
f, err := fc.getFunction(ctx, inputs)
require.NoError(t, err, comment)
f.setCollator(collate.GetCollator("utf8mb4_general_ci"))
r, err := evalBuiltinFunc(f, ctx, chunk.Row{})
require.NoError(t, err, comment)
testutil.DatumEqual(t, types.NewDatum(tt.generalMatch), r, comment)
}
for _, tt := range tests {
comment := fmt.Sprintf(`for input = "%s", pattern = "%s"`, tt.input, tt.pattern)
fc := funcs[ast.Like]
inputs := datumsToConstants(types.MakeDatums(tt.input, tt.pattern, 0))
f, err := fc.getFunction(ctx, inputs)
require.NoError(t, err, comment)
f.setCollator(collate.GetCollator("utf8mb4_unicode_ci"))
r, err := evalBuiltinFunc(f, ctx, chunk.Row{})
require.NoError(t, err, comment)
testutil.DatumEqual(t, types.NewDatum(tt.unicodeMatch), r, comment)
}
for _, tt := range tests {
comment := fmt.Sprintf(`for input = "%s", pattern = "%s"`, tt.input, tt.pattern)
fc := funcs[ast.Like]
inputs := datumsToConstants(types.MakeDatums(tt.input, tt.pattern, 0))
f, err := fc.getFunction(ctx, inputs)
require.NoError(t, err, comment)
f.setCollator(collate.GetCollator("utf8mb4_0900_ai_ci"))
r, err := evalBuiltinFunc(f, ctx, chunk.Row{})
require.NoError(t, err, comment)
testutil.DatumEqual(t, types.NewDatum(tt.unicode0900Match), r, comment)
}
}