Files
tidb/pkg/parser/terror/terror_test.go

174 lines
5.0 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,
// See the License for the specific language governing permissions and
// limitations under the License.
package terror
import (
"encoding/json"
"fmt"
"os"
"runtime"
"strings"
"testing"
"github.com/pingcap/errors"
"github.com/stretchr/testify/require"
)
func TestErrCode(t *testing.T) {
require.Equal(t, ErrCode(1), CodeMissConnectionID)
require.Equal(t, ErrCode(2), CodeResultUndetermined)
}
func TestTError(t *testing.T) {
require.NotEmpty(t, ClassParser.String())
require.NotEmpty(t, ClassOptimizer.String())
require.NotEmpty(t, ClassKV.String())
require.NotEmpty(t, ClassServer.String())
parserErr := ClassParser.New(ErrCode(100), "error 100")
require.NotEmpty(t, parserErr.Error())
require.True(t, ClassParser.EqualClass(parserErr))
require.False(t, ClassParser.NotEqualClass(parserErr))
require.False(t, ClassOptimizer.EqualClass(parserErr))
optimizerErr := ClassOptimizer.New(ErrCode(2), "abc")
require.False(t, ClassOptimizer.EqualClass(errors.New("abc")))
require.False(t, ClassOptimizer.EqualClass(nil))
require.True(t, optimizerErr.Equal(optimizerErr.GenWithStack("def")))
require.False(t, optimizerErr.Equal(nil))
require.False(t, optimizerErr.Equal(errors.New("abc")))
// Test case for FastGen.
require.True(t, optimizerErr.Equal(optimizerErr.FastGen("def")))
require.True(t, optimizerErr.Equal(optimizerErr.FastGen("def: %s", "def")))
kvErr := ClassKV.New(1062, "key already exist")
e := kvErr.FastGen("Duplicate entry '%d' for key 'PRIMARY'", 1)
require.Equal(t, "[kv:1062]Duplicate entry '1' for key 'PRIMARY'", e.Error())
sqlErr := ToSQLError(errors.Cause(e).(*Error))
require.Equal(t, "Duplicate entry '1' for key 'PRIMARY'", sqlErr.Message)
require.Equal(t, uint16(1062), sqlErr.Code)
err := errors.Trace(ErrCritical.GenWithStackByArgs("test"))
require.True(t, ErrCritical.Equal(err))
err = errors.Trace(ErrCritical)
require.True(t, ErrCritical.Equal(err))
}
func TestJson(t *testing.T) {
prevTErr := errors.Normalize("json test", errors.MySQLErrorCode(int(CodeExecResultIsEmpty)))
buf, err := json.Marshal(prevTErr)
require.NoError(t, err)
var curTErr errors.Error
err = json.Unmarshal(buf, &curTErr)
require.NoError(t, err)
isEqual := prevTErr.Equal(&curTErr)
require.True(t, isEqual)
}
var predefinedErr = ClassExecutor.New(ErrCode(123), "predefiend error")
func example() error {
err := call()
return errors.Trace(err)
}
func call() error {
return predefinedErr.GenWithStack("error message:%s", "abc")
}
func TestErrorEqual(t *testing.T) {
e1 := errors.New("test error")
require.NotNil(t, e1)
e2 := errors.Trace(e1)
require.NotNil(t, e2)
e3 := errors.Trace(e2)
require.NotNil(t, e3)
require.Equal(t, e1, errors.Cause(e2))
require.Equal(t, e1, errors.Cause(e3))
require.Equal(t, errors.Cause(e3), errors.Cause(e2))
e4 := errors.New("test error")
require.NotEqual(t, e1, errors.Cause(e4))
e5 := errors.Errorf("test error")
require.NotEqual(t, e1, errors.Cause(e5))
require.True(t, ErrorEqual(e1, e2))
require.True(t, ErrorEqual(e1, e3))
require.True(t, ErrorEqual(e1, e4))
require.True(t, ErrorEqual(e1, e5))
var e6 error
require.True(t, ErrorEqual(nil, nil))
require.True(t, ErrorNotEqual(e1, e6))
code1 := ErrCode(9001)
code2 := ErrCode(9002)
te1 := ClassParser.Synthesize(code1, "abc")
te3 := ClassKV.New(code1, "abc")
te4 := ClassKV.New(code2, "abc")
require.False(t, ErrorEqual(te1, te3))
require.False(t, ErrorEqual(te3, te4))
}
func TestLog(t *testing.T) {
err := fmt.Errorf("xxx")
Log(err)
}
func TestTraceAndLocation(t *testing.T) {
err := example()
stack := errors.ErrorStack(err)
lines := strings.Split(stack, "\n")
goroot := strings.ReplaceAll(runtime.GOROOT(), string(os.PathSeparator), "/")
var sysStack = 0
for _, line := range lines {
// When you run test case in the bazel. you will find the difference stack. It looks like this:
//
// ```go
// testing.tRunner
// GOROOT/src/testing/testing.go:1576
// runtime.goexit
// src/runtime/asm_arm64.s:1172
// ```
//
// but run with ```go test```. It looks like this:
//
// ```go
// testing.tRunner
// /Users/pingcap/.gvm/gos/go1.20.1/src/testing/testing.go:1576
// runtime.goexit
// /Users/pingcap/.gvm/gos/go1.20.1/src/runtime/asm_arm64.s:1172
// ```
//
// So we have to deal with these boundary conditions.
if strings.Contains(line, goroot) || strings.Contains(line, "src/runtime") {
sysStack++
}
}
require.Equalf(t, 9, len(lines)-(2*sysStack), "stack =\n%s", stack)
var containTerr bool
for _, v := range lines {
if strings.Contains(v, "terror_test.go") {
containTerr = true
break
}
}
require.True(t, containTerr)
}