877 lines
30 KiB
Go
877 lines
30 KiB
Go
// Copyright 2016 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 server
|
|
|
|
import (
|
|
"bufio"
|
|
"bytes"
|
|
"context"
|
|
"encoding/binary"
|
|
"fmt"
|
|
"io"
|
|
|
|
. "github.com/pingcap/check"
|
|
"github.com/pingcap/failpoint"
|
|
"github.com/pingcap/kvproto/pkg/metapb"
|
|
"github.com/pingcap/parser/model"
|
|
"github.com/pingcap/parser/mysql"
|
|
"github.com/pingcap/tidb/domain"
|
|
"github.com/pingcap/tidb/executor"
|
|
"github.com/pingcap/tidb/kv"
|
|
"github.com/pingcap/tidb/session"
|
|
"github.com/pingcap/tidb/sessionctx"
|
|
"github.com/pingcap/tidb/sessionctx/variable"
|
|
"github.com/pingcap/tidb/store/mockstore"
|
|
"github.com/pingcap/tidb/store/mockstore/unistore"
|
|
"github.com/pingcap/tidb/table"
|
|
"github.com/pingcap/tidb/util/arena"
|
|
"github.com/pingcap/tidb/util/testkit"
|
|
"github.com/pingcap/tidb/util/testleak"
|
|
"github.com/tikv/client-go/v2/mockstore/cluster"
|
|
)
|
|
|
|
type ConnTestSuite struct {
|
|
dom *domain.Domain
|
|
store kv.Storage
|
|
}
|
|
|
|
var _ = SerialSuites(&ConnTestSuite{})
|
|
|
|
func (ts *ConnTestSuite) SetUpSuite(c *C) {
|
|
testleak.BeforeTest()
|
|
var err error
|
|
ts.store, err = mockstore.NewMockStore(
|
|
mockstore.WithClusterInspector(func(c cluster.Cluster) {
|
|
mockCluster := c.(*unistore.Cluster)
|
|
_, _, region1 := mockstore.BootstrapWithSingleStore(c)
|
|
store := c.AllocID()
|
|
peer := c.AllocID()
|
|
mockCluster.AddStore(store, "tiflash0", &metapb.StoreLabel{Key: "engine", Value: "tiflash"})
|
|
mockCluster.AddPeer(region1, store, peer)
|
|
}),
|
|
mockstore.WithStoreType(mockstore.EmbedUnistore),
|
|
)
|
|
c.Assert(err, IsNil)
|
|
ts.dom, err = session.BootstrapSession(ts.store)
|
|
c.Assert(err, IsNil)
|
|
}
|
|
|
|
func (ts *ConnTestSuite) TearDownSuite(c *C) {
|
|
ts.dom.Close()
|
|
ts.store.Close()
|
|
testleak.AfterTest(c)()
|
|
}
|
|
|
|
func (ts *ConnTestSuite) TestMalformHandshakeHeader(c *C) {
|
|
c.Parallel()
|
|
data := []byte{0x00}
|
|
var p handshakeResponse41
|
|
_, err := parseHandshakeResponseHeader(context.Background(), &p, data)
|
|
c.Assert(err, NotNil)
|
|
}
|
|
|
|
func (ts *ConnTestSuite) TestParseHandshakeResponse(c *C) {
|
|
c.Parallel()
|
|
// test data from http://dev.mysql.com/doc/internals/en/connection-phase-packets.html#packet-Protocol::HandshakeResponse41
|
|
data := []byte{
|
|
0x85, 0xa2, 0x1e, 0x00, 0x00, 0x00, 0x00, 0x40, 0x08, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x72, 0x6f, 0x6f, 0x74, 0x00, 0x14, 0x22, 0x50, 0x79, 0xa2, 0x12, 0xd4,
|
|
0xe8, 0x82, 0xe5, 0xb3, 0xf4, 0x1a, 0x97, 0x75, 0x6b, 0xc8, 0xbe, 0xdb, 0x9f, 0x80, 0x6d, 0x79,
|
|
0x73, 0x71, 0x6c, 0x5f, 0x6e, 0x61, 0x74, 0x69, 0x76, 0x65, 0x5f, 0x70, 0x61, 0x73, 0x73, 0x77,
|
|
0x6f, 0x72, 0x64, 0x00, 0x61, 0x03, 0x5f, 0x6f, 0x73, 0x09, 0x64, 0x65, 0x62, 0x69, 0x61, 0x6e,
|
|
0x36, 0x2e, 0x30, 0x0c, 0x5f, 0x63, 0x6c, 0x69, 0x65, 0x6e, 0x74, 0x5f, 0x6e, 0x61, 0x6d, 0x65,
|
|
0x08, 0x6c, 0x69, 0x62, 0x6d, 0x79, 0x73, 0x71, 0x6c, 0x04, 0x5f, 0x70, 0x69, 0x64, 0x05, 0x32,
|
|
0x32, 0x33, 0x34, 0x34, 0x0f, 0x5f, 0x63, 0x6c, 0x69, 0x65, 0x6e, 0x74, 0x5f, 0x76, 0x65, 0x72,
|
|
0x73, 0x69, 0x6f, 0x6e, 0x08, 0x35, 0x2e, 0x36, 0x2e, 0x36, 0x2d, 0x6d, 0x39, 0x09, 0x5f, 0x70,
|
|
0x6c, 0x61, 0x74, 0x66, 0x6f, 0x72, 0x6d, 0x06, 0x78, 0x38, 0x36, 0x5f, 0x36, 0x34, 0x03, 0x66,
|
|
0x6f, 0x6f, 0x03, 0x62, 0x61, 0x72,
|
|
}
|
|
var p handshakeResponse41
|
|
offset, err := parseHandshakeResponseHeader(context.Background(), &p, data)
|
|
c.Assert(err, IsNil)
|
|
c.Assert(p.Capability&mysql.ClientConnectAtts, Equals, mysql.ClientConnectAtts)
|
|
err = parseHandshakeResponseBody(context.Background(), &p, data, offset)
|
|
c.Assert(err, IsNil)
|
|
eq := mapIdentical(p.Attrs, map[string]string{
|
|
"_client_version": "5.6.6-m9",
|
|
"_platform": "x86_64",
|
|
"foo": "bar",
|
|
"_os": "debian6.0",
|
|
"_client_name": "libmysql",
|
|
"_pid": "22344"})
|
|
c.Assert(eq, IsTrue)
|
|
|
|
data = []byte{
|
|
0x8d, 0xa6, 0x0f, 0x00, 0x00, 0x00, 0x00, 0x01, 0x08, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x70, 0x61, 0x6d, 0x00, 0x14, 0xab, 0x09, 0xee, 0xf6, 0xbc, 0xb1, 0x32,
|
|
0x3e, 0x61, 0x14, 0x38, 0x65, 0xc0, 0x99, 0x1d, 0x95, 0x7d, 0x75, 0xd4, 0x47, 0x74, 0x65, 0x73,
|
|
0x74, 0x00, 0x6d, 0x79, 0x73, 0x71, 0x6c, 0x5f, 0x6e, 0x61, 0x74, 0x69, 0x76, 0x65, 0x5f, 0x70,
|
|
0x61, 0x73, 0x73, 0x77, 0x6f, 0x72, 0x64, 0x00,
|
|
}
|
|
p = handshakeResponse41{}
|
|
offset, err = parseHandshakeResponseHeader(context.Background(), &p, data)
|
|
c.Assert(err, IsNil)
|
|
capability := mysql.ClientProtocol41 |
|
|
mysql.ClientPluginAuth |
|
|
mysql.ClientSecureConnection |
|
|
mysql.ClientConnectWithDB
|
|
c.Assert(p.Capability&capability, Equals, capability)
|
|
err = parseHandshakeResponseBody(context.Background(), &p, data, offset)
|
|
c.Assert(err, IsNil)
|
|
c.Assert(p.User, Equals, "pam")
|
|
c.Assert(p.DBName, Equals, "test")
|
|
|
|
// Test for compatibility of Protocol::HandshakeResponse320
|
|
data = []byte{
|
|
0x00, 0x80, 0x00, 0x00, 0x01, 0x72, 0x6f, 0x6f, 0x74, 0x00, 0x00,
|
|
}
|
|
p = handshakeResponse41{}
|
|
offset, err = parseOldHandshakeResponseHeader(context.Background(), &p, data)
|
|
c.Assert(err, IsNil)
|
|
capability = mysql.ClientProtocol41 |
|
|
mysql.ClientSecureConnection
|
|
c.Assert(p.Capability&capability, Equals, capability)
|
|
err = parseOldHandshakeResponseBody(context.Background(), &p, data, offset)
|
|
c.Assert(err, IsNil)
|
|
c.Assert(p.User, Equals, "root")
|
|
}
|
|
|
|
func (ts *ConnTestSuite) TestIssue1768(c *C) {
|
|
c.Parallel()
|
|
// this data is from captured handshake packet, using mysql client.
|
|
// TiDB should handle authorization correctly, even mysql client set
|
|
// the ClientPluginAuthLenencClientData capability.
|
|
data := []byte{
|
|
0x85, 0xa6, 0xff, 0x01, 0x00, 0x00, 0x00, 0x01, 0x21, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
0x00, 0x00, 0x00, 0x00, 0x74, 0x65, 0x73, 0x74, 0x00, 0x14, 0xe9, 0x7a, 0x2b, 0xec, 0x4a, 0xa8,
|
|
0xea, 0x67, 0x8a, 0xc2, 0x46, 0x4d, 0x32, 0xa4, 0xda, 0x39, 0x77, 0xe5, 0x61, 0x1a, 0x65, 0x03,
|
|
0x5f, 0x6f, 0x73, 0x05, 0x4c, 0x69, 0x6e, 0x75, 0x78, 0x0c, 0x5f, 0x63, 0x6c, 0x69, 0x65, 0x6e,
|
|
0x74, 0x5f, 0x6e, 0x61, 0x6d, 0x65, 0x08, 0x6c, 0x69, 0x62, 0x6d, 0x79, 0x73, 0x71, 0x6c, 0x04,
|
|
0x5f, 0x70, 0x69, 0x64, 0x04, 0x39, 0x30, 0x33, 0x30, 0x0f, 0x5f, 0x63, 0x6c, 0x69, 0x65, 0x6e,
|
|
0x74, 0x5f, 0x76, 0x65, 0x72, 0x73, 0x69, 0x6f, 0x6e, 0x06, 0x35, 0x2e, 0x37, 0x2e, 0x31, 0x34,
|
|
0x09, 0x5f, 0x70, 0x6c, 0x61, 0x74, 0x66, 0x6f, 0x72, 0x6d, 0x06, 0x78, 0x38, 0x36, 0x5f, 0x36,
|
|
0x34, 0x0c, 0x70, 0x72, 0x6f, 0x67, 0x72, 0x61, 0x6d, 0x5f, 0x6e, 0x61, 0x6d, 0x65, 0x05, 0x6d,
|
|
0x79, 0x73, 0x71, 0x6c,
|
|
}
|
|
p := handshakeResponse41{}
|
|
offset, err := parseHandshakeResponseHeader(context.Background(), &p, data)
|
|
c.Assert(err, IsNil)
|
|
c.Assert(p.Capability&mysql.ClientPluginAuthLenencClientData, Equals, mysql.ClientPluginAuthLenencClientData)
|
|
err = parseHandshakeResponseBody(context.Background(), &p, data, offset)
|
|
c.Assert(err, IsNil)
|
|
c.Assert(len(p.Auth) > 0, IsTrue)
|
|
}
|
|
|
|
func (ts *ConnTestSuite) TestAuthSwitchRequest(c *C) {
|
|
c.Parallel()
|
|
// this data is from a MySQL 8.0 client
|
|
data := []byte{
|
|
0x85, 0xa6, 0xff, 0x1, 0x0, 0x0, 0x0, 0x1, 0x21, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
|
|
0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x72, 0x6f,
|
|
0x6f, 0x74, 0x0, 0x0, 0x63, 0x61, 0x63, 0x68, 0x69, 0x6e, 0x67, 0x5f, 0x73, 0x68, 0x61,
|
|
0x32, 0x5f, 0x70, 0x61, 0x73, 0x73, 0x77, 0x6f, 0x72, 0x64, 0x0, 0x79, 0x4, 0x5f, 0x70,
|
|
0x69, 0x64, 0x5, 0x37, 0x37, 0x30, 0x38, 0x36, 0x9, 0x5f, 0x70, 0x6c, 0x61, 0x74, 0x66,
|
|
0x6f, 0x72, 0x6d, 0x6, 0x78, 0x38, 0x36, 0x5f, 0x36, 0x34, 0x3, 0x5f, 0x6f, 0x73, 0x5,
|
|
0x4c, 0x69, 0x6e, 0x75, 0x78, 0xc, 0x5f, 0x63, 0x6c, 0x69, 0x65, 0x6e, 0x74, 0x5f, 0x6e,
|
|
0x61, 0x6d, 0x65, 0x8, 0x6c, 0x69, 0x62, 0x6d, 0x79, 0x73, 0x71, 0x6c, 0x7, 0x6f, 0x73,
|
|
0x5f, 0x75, 0x73, 0x65, 0x72, 0xa, 0x6e, 0x75, 0x6c, 0x6c, 0x6e, 0x6f, 0x74, 0x6e, 0x69,
|
|
0x6c, 0xf, 0x5f, 0x63, 0x6c, 0x69, 0x65, 0x6e, 0x74, 0x5f, 0x76, 0x65, 0x72, 0x73, 0x69,
|
|
0x6f, 0x6e, 0x6, 0x38, 0x2e, 0x30, 0x2e, 0x32, 0x31, 0xc, 0x70, 0x72, 0x6f, 0x67, 0x72,
|
|
0x61, 0x6d, 0x5f, 0x6e, 0x61, 0x6d, 0x65, 0x5, 0x6d, 0x79, 0x73, 0x71, 0x6c,
|
|
}
|
|
|
|
var resp handshakeResponse41
|
|
pos, err := parseHandshakeResponseHeader(context.Background(), &resp, data)
|
|
c.Assert(err, IsNil)
|
|
err = parseHandshakeResponseBody(context.Background(), &resp, data, pos)
|
|
c.Assert(err, IsNil)
|
|
c.Assert(resp.AuthPlugin == "caching_sha2_password", IsTrue)
|
|
}
|
|
|
|
func (ts *ConnTestSuite) TestInitialHandshake(c *C) {
|
|
c.Parallel()
|
|
var outBuffer bytes.Buffer
|
|
cc := &clientConn{
|
|
connectionID: 1,
|
|
salt: []byte{0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x10, 0x11, 0x12, 0x13, 0x14},
|
|
server: &Server{
|
|
capability: defaultCapability,
|
|
},
|
|
pkt: &packetIO{
|
|
bufWriter: bufio.NewWriter(&outBuffer),
|
|
},
|
|
}
|
|
err := cc.writeInitialHandshake(context.TODO())
|
|
c.Assert(err, IsNil)
|
|
|
|
expected := new(bytes.Buffer)
|
|
expected.WriteByte(0x0a) // Protocol
|
|
expected.WriteString(mysql.ServerVersion) // Version
|
|
expected.WriteByte(0x00) // NULL
|
|
err = binary.Write(expected, binary.LittleEndian, uint32(1)) // Connection ID
|
|
c.Assert(err, IsNil)
|
|
expected.Write([]byte{0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x00}) // Salt
|
|
err = binary.Write(expected, binary.LittleEndian, uint16(defaultCapability&0xFFFF)) // Server Capability
|
|
c.Assert(err, IsNil)
|
|
expected.WriteByte(uint8(mysql.DefaultCollationID)) // Server Language
|
|
err = binary.Write(expected, binary.LittleEndian, mysql.ServerStatusAutocommit) // Server Status
|
|
c.Assert(err, IsNil)
|
|
err = binary.Write(expected, binary.LittleEndian, uint16((defaultCapability>>16)&0xFFFF)) // Extended Server Capability
|
|
c.Assert(err, IsNil)
|
|
expected.WriteByte(0x15) // Authentication Plugin Length
|
|
expected.Write([]byte{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}) // Unused
|
|
expected.Write([]byte{0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x10, 0x11, 0x12, 0x13, 0x14, 0x00}) // Salt
|
|
expected.WriteString("mysql_native_password") // Authentication Plugin
|
|
expected.WriteByte(0x00) // NULL
|
|
c.Assert(outBuffer.Bytes()[4:], DeepEquals, expected.Bytes())
|
|
}
|
|
|
|
type dispatchInput struct {
|
|
com byte
|
|
in []byte
|
|
err error
|
|
out []byte
|
|
}
|
|
|
|
func (ts *ConnTestSuite) TestDispatch(c *C) {
|
|
userData := append([]byte("root"), 0x0, 0x0)
|
|
userData = append(userData, []byte("test")...)
|
|
userData = append(userData, 0x0)
|
|
|
|
inputs := []dispatchInput{
|
|
{
|
|
com: mysql.ComSleep,
|
|
in: nil,
|
|
err: nil,
|
|
out: nil,
|
|
},
|
|
{
|
|
com: mysql.ComQuit,
|
|
in: nil,
|
|
err: io.EOF,
|
|
out: nil,
|
|
},
|
|
{
|
|
com: mysql.ComQuery,
|
|
in: []byte("do 1"),
|
|
err: nil,
|
|
out: []byte{0x3, 0x0, 0x0, 0x0, 0x0, 0x00, 0x0},
|
|
},
|
|
{
|
|
com: mysql.ComInitDB,
|
|
in: []byte("test"),
|
|
err: nil,
|
|
out: []byte{0x3, 0x0, 0x0, 0x1, 0x0, 0x0, 0x0},
|
|
},
|
|
{
|
|
com: mysql.ComPing,
|
|
in: nil,
|
|
err: nil,
|
|
out: []byte{0x3, 0x0, 0x0, 0x2, 0x0, 0x0, 0x0},
|
|
},
|
|
{
|
|
com: mysql.ComStmtPrepare,
|
|
in: []byte("select 1"),
|
|
err: nil,
|
|
out: []byte{
|
|
0xc, 0x0, 0x0, 0x3, 0x0, 0x1, 0x0, 0x0, 0x0, 0x1, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x18,
|
|
0x0, 0x0, 0x4, 0x3, 0x64, 0x65, 0x66, 0x0, 0x0, 0x0, 0x1, 0x31, 0x1, 0x31, 0xc, 0x3f,
|
|
0x0, 0x1, 0x0, 0x0, 0x0, 0x8, 0x81, 0x0, 0x0, 0x0, 0x0, 0x1, 0x0, 0x0, 0x5, 0xfe,
|
|
},
|
|
},
|
|
{
|
|
com: mysql.ComStmtExecute,
|
|
in: []byte{0x1, 0x0, 0x0, 0x0, 0x1, 0x0, 0x0, 0x1, 0x0},
|
|
err: nil,
|
|
out: []byte{
|
|
0x1, 0x0, 0x0, 0x6, 0x1, 0x18, 0x0, 0x0, 0x7, 0x3, 0x64, 0x65, 0x66, 0x0, 0x0, 0x0,
|
|
0x1, 0x31, 0x1, 0x31, 0xc, 0x3f, 0x0, 0x1, 0x0, 0x0, 0x0, 0x8, 0x81, 0x0, 0x0, 0x0,
|
|
0x0, 0x1, 0x0, 0x0, 0x8, 0xfe,
|
|
},
|
|
},
|
|
{
|
|
com: mysql.ComStmtFetch,
|
|
in: []byte{0x1, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0},
|
|
err: nil,
|
|
out: []byte{0x1, 0x0, 0x0, 0x9, 0xfe},
|
|
},
|
|
{
|
|
com: mysql.ComStmtReset,
|
|
in: []byte{0x1, 0x0, 0x0, 0x0},
|
|
err: nil,
|
|
out: []byte{0x3, 0x0, 0x0, 0xa, 0x0, 0x0, 0x0},
|
|
},
|
|
{
|
|
com: mysql.ComSetOption,
|
|
in: []byte{0x1, 0x0, 0x0, 0x0},
|
|
err: nil,
|
|
out: []byte{0x1, 0x0, 0x0, 0xb, 0xfe},
|
|
},
|
|
{
|
|
com: mysql.ComStmtClose,
|
|
in: []byte{0x1, 0x0, 0x0, 0x0},
|
|
err: nil,
|
|
out: []byte{},
|
|
},
|
|
{
|
|
com: mysql.ComFieldList,
|
|
in: []byte("t"),
|
|
err: nil,
|
|
out: []byte{
|
|
0x26, 0x0, 0x0, 0xc, 0x3, 0x64, 0x65, 0x66, 0x4, 0x74, 0x65, 0x73, 0x74, 0x1, 0x74,
|
|
0x1, 0x74, 0x1, 0x61, 0x1, 0x61, 0xc, 0x3f, 0x0, 0xb, 0x0, 0x0, 0x0, 0x3, 0x0, 0x0,
|
|
0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x1, 0x0, 0x0, 0xd, 0xfe,
|
|
},
|
|
},
|
|
{
|
|
com: mysql.ComChangeUser,
|
|
in: userData,
|
|
err: nil,
|
|
out: []byte{0x3, 0x0, 0x0, 0xe, 0x0, 0x0, 0x0},
|
|
},
|
|
{
|
|
com: mysql.ComRefresh, // flush privileges
|
|
in: []byte{0x01},
|
|
err: nil,
|
|
out: []byte{0x3, 0x0, 0x0, 0xf, 0x0, 0x0, 0x0, 0x3, 0x0, 0x0, 0x10, 0x0, 0x0, 0x0},
|
|
},
|
|
{
|
|
com: mysql.ComRefresh, // flush logs etc
|
|
in: []byte{0x02},
|
|
err: nil,
|
|
out: []byte{0x3, 0x0, 0x0, 0x11, 0x0, 0x0, 0x0},
|
|
},
|
|
{
|
|
com: mysql.ComResetConnection,
|
|
in: nil,
|
|
err: nil,
|
|
out: []byte{0x3, 0x0, 0x0, 0x12, 0x0, 0x0, 0x0},
|
|
},
|
|
}
|
|
|
|
ts.testDispatch(c, inputs, 0)
|
|
}
|
|
|
|
func (ts *ConnTestSuite) TestDispatchClientProtocol41(c *C) {
|
|
userData := append([]byte("root"), 0x0, 0x0)
|
|
userData = append(userData, []byte("test")...)
|
|
userData = append(userData, 0x0)
|
|
|
|
inputs := []dispatchInput{
|
|
{
|
|
com: mysql.ComSleep,
|
|
in: nil,
|
|
err: nil,
|
|
out: nil,
|
|
},
|
|
{
|
|
com: mysql.ComQuit,
|
|
in: nil,
|
|
err: io.EOF,
|
|
out: nil,
|
|
},
|
|
{
|
|
com: mysql.ComQuery,
|
|
in: []byte("do 1"),
|
|
err: nil,
|
|
out: []byte{0x7, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x2, 0x0, 0x0, 0x0},
|
|
},
|
|
{
|
|
com: mysql.ComInitDB,
|
|
in: []byte("test"),
|
|
err: nil,
|
|
out: []byte{0x7, 0x0, 0x0, 0x1, 0x0, 0x0, 0x0, 0x2, 0x0, 0x0, 0x0},
|
|
},
|
|
{
|
|
com: mysql.ComPing,
|
|
in: nil,
|
|
err: nil,
|
|
out: []byte{0x7, 0x0, 0x0, 0x2, 0x0, 0x0, 0x0, 0x2, 0x0, 0x0, 0x0},
|
|
},
|
|
{
|
|
com: mysql.ComStmtPrepare,
|
|
in: []byte("select 1"),
|
|
err: nil,
|
|
out: []byte{
|
|
0xc, 0x0, 0x0, 0x3, 0x0, 0x1, 0x0, 0x0, 0x0, 0x1, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x18,
|
|
0x0, 0x0, 0x4, 0x3, 0x64, 0x65, 0x66, 0x0, 0x0, 0x0, 0x1, 0x31, 0x1, 0x31, 0xc, 0x3f,
|
|
0x0, 0x1, 0x0, 0x0, 0x0, 0x8, 0x81, 0x0, 0x0, 0x0, 0x0, 0x5, 0x0, 0x0, 0x5, 0xfe,
|
|
0x0, 0x0, 0x2, 0x0,
|
|
},
|
|
},
|
|
{
|
|
com: mysql.ComStmtExecute,
|
|
in: []byte{0x1, 0x0, 0x0, 0x0, 0x1, 0x0, 0x0, 0x1, 0x0},
|
|
err: nil,
|
|
out: []byte{
|
|
0x1, 0x0, 0x0, 0x6, 0x1, 0x18, 0x0, 0x0, 0x7, 0x3, 0x64, 0x65, 0x66, 0x0, 0x0, 0x0,
|
|
0x1, 0x31, 0x1, 0x31, 0xc, 0x3f, 0x0, 0x1, 0x0, 0x0, 0x0, 0x8, 0x81, 0x0, 0x0, 0x0,
|
|
0x0, 0x5, 0x0, 0x0, 0x8, 0xfe, 0x0, 0x0, 0x42, 0x0,
|
|
},
|
|
},
|
|
{
|
|
com: mysql.ComStmtFetch,
|
|
in: []byte{0x1, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0},
|
|
err: nil,
|
|
out: []byte{0x5, 0x0, 0x0, 0x9, 0xfe, 0x0, 0x0, 0x82, 0x0},
|
|
},
|
|
{
|
|
com: mysql.ComStmtReset,
|
|
in: []byte{0x1, 0x0, 0x0, 0x0},
|
|
err: nil,
|
|
out: []byte{0x7, 0x0, 0x0, 0xa, 0x0, 0x0, 0x0, 0x2, 0x0, 0x0, 0x0},
|
|
},
|
|
{
|
|
com: mysql.ComSetOption,
|
|
in: []byte{0x1, 0x0, 0x0, 0x0},
|
|
err: nil,
|
|
out: []byte{0x5, 0x0, 0x0, 0xb, 0xfe, 0x0, 0x0, 0x2, 0x0},
|
|
},
|
|
{
|
|
com: mysql.ComStmtClose,
|
|
in: []byte{0x1, 0x0, 0x0, 0x0},
|
|
err: nil,
|
|
out: []byte{},
|
|
},
|
|
{
|
|
com: mysql.ComFieldList,
|
|
in: []byte("t"),
|
|
err: nil,
|
|
out: []byte{
|
|
0x26, 0x0, 0x0, 0xc, 0x3, 0x64, 0x65, 0x66, 0x4, 0x74, 0x65, 0x73, 0x74, 0x1, 0x74,
|
|
0x1, 0x74, 0x1, 0x61, 0x1, 0x61, 0xc, 0x3f, 0x0, 0xb, 0x0, 0x0, 0x0, 0x3, 0x0, 0x0,
|
|
0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x5, 0x0, 0x0, 0xd, 0xfe,
|
|
0x0, 0x0, 0x2, 0x0,
|
|
},
|
|
},
|
|
{
|
|
com: mysql.ComChangeUser,
|
|
in: userData,
|
|
err: nil,
|
|
out: []byte{0x7, 0x0, 0x0, 0xe, 0x0, 0x0, 0x0, 0x2, 0x0, 0x0, 0x0},
|
|
},
|
|
{
|
|
com: mysql.ComRefresh, // flush privileges
|
|
in: []byte{0x01},
|
|
err: nil,
|
|
out: []byte{0x7, 0x0, 0x0, 0xf, 0x0, 0x0, 0x0, 0x2, 0x0, 0x0, 0x0, 0x7, 0x0, 0x0, 0x10, 0x0, 0x0, 0x0, 0x2, 0x0, 0x0, 0x0},
|
|
},
|
|
{
|
|
com: mysql.ComRefresh, // flush logs etc
|
|
in: []byte{0x02},
|
|
err: nil,
|
|
out: []byte{0x7, 0x0, 0x0, 0x11, 0x0, 0x0, 0x0, 0x2, 0x0, 0x0, 0x0},
|
|
},
|
|
{
|
|
com: mysql.ComResetConnection,
|
|
in: nil,
|
|
err: nil,
|
|
out: []byte{0x7, 0x0, 0x0, 0x12, 0x0, 0x0, 0x0, 0x2, 0x0, 0x0, 0x0},
|
|
},
|
|
}
|
|
|
|
ts.testDispatch(c, inputs, mysql.ClientProtocol41)
|
|
}
|
|
|
|
func (ts *ConnTestSuite) testDispatch(c *C, inputs []dispatchInput, capability uint32) {
|
|
store, err := mockstore.NewMockStore()
|
|
c.Assert(err, IsNil)
|
|
defer func() {
|
|
err := store.Close()
|
|
c.Assert(err, IsNil)
|
|
}()
|
|
dom, err := session.BootstrapSession(store)
|
|
c.Assert(err, IsNil)
|
|
defer dom.Close()
|
|
|
|
se, err := session.CreateSession4Test(store)
|
|
c.Assert(err, IsNil)
|
|
tc := &TiDBContext{
|
|
Session: se,
|
|
stmts: make(map[int]*TiDBStatement),
|
|
}
|
|
_, err = se.Execute(context.Background(), "create table test.t(a int)")
|
|
c.Assert(err, IsNil)
|
|
_, err = se.Execute(context.Background(), "insert into test.t values (1)")
|
|
c.Assert(err, IsNil)
|
|
|
|
var outBuffer bytes.Buffer
|
|
tidbdrv := NewTiDBDriver(ts.store)
|
|
cfg := newTestConfig()
|
|
cfg.Port, cfg.Status.StatusPort = 0, 0
|
|
cfg.Status.ReportStatus = false
|
|
server, err := NewServer(cfg, tidbdrv)
|
|
|
|
c.Assert(err, IsNil)
|
|
defer server.Close()
|
|
|
|
cc := &clientConn{
|
|
connectionID: 1,
|
|
salt: []byte{0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x10, 0x11, 0x12, 0x13, 0x14},
|
|
server: server,
|
|
pkt: &packetIO{
|
|
bufWriter: bufio.NewWriter(&outBuffer),
|
|
},
|
|
collation: mysql.DefaultCollationID,
|
|
peerHost: "localhost",
|
|
alloc: arena.NewAllocator(512),
|
|
ctx: tc,
|
|
capability: capability,
|
|
}
|
|
for _, cs := range inputs {
|
|
inBytes := append([]byte{cs.com}, cs.in...)
|
|
err := cc.dispatch(context.Background(), inBytes)
|
|
c.Assert(err, Equals, cs.err)
|
|
if err == nil {
|
|
err = cc.flush(context.TODO())
|
|
c.Assert(err, IsNil)
|
|
c.Assert(outBuffer.Bytes(), DeepEquals, cs.out)
|
|
} else {
|
|
_ = cc.flush(context.TODO())
|
|
}
|
|
outBuffer.Reset()
|
|
}
|
|
}
|
|
|
|
func (ts *ConnTestSuite) TestGetSessionVarsWaitTimeout(c *C) {
|
|
c.Parallel()
|
|
se, err := session.CreateSession4Test(ts.store)
|
|
c.Assert(err, IsNil)
|
|
tc := &TiDBContext{
|
|
Session: se,
|
|
stmts: make(map[int]*TiDBStatement),
|
|
}
|
|
cc := &clientConn{
|
|
connectionID: 1,
|
|
server: &Server{
|
|
capability: defaultCapability,
|
|
},
|
|
ctx: tc,
|
|
}
|
|
c.Assert(cc.getSessionVarsWaitTimeout(context.Background()), Equals, uint64(0))
|
|
}
|
|
|
|
func mapIdentical(m1, m2 map[string]string) bool {
|
|
return mapBelong(m1, m2) && mapBelong(m2, m1)
|
|
}
|
|
|
|
func mapBelong(m1, m2 map[string]string) bool {
|
|
for k1, v1 := range m1 {
|
|
v2, ok := m2[k1]
|
|
if !ok && v1 != v2 {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
func (ts *ConnTestSuite) TestConnExecutionTimeout(c *C) {
|
|
// There is no underlying netCon, use failpoint to avoid panic
|
|
c.Assert(failpoint.Enable("github.com/pingcap/tidb/server/FakeClientConn", "return(1)"), IsNil)
|
|
|
|
c.Parallel()
|
|
se, err := session.CreateSession4Test(ts.store)
|
|
c.Assert(err, IsNil)
|
|
|
|
connID := uint64(1)
|
|
se.SetConnectionID(connID)
|
|
tc := &TiDBContext{
|
|
Session: se,
|
|
stmts: make(map[int]*TiDBStatement),
|
|
}
|
|
cc := &clientConn{
|
|
connectionID: connID,
|
|
server: &Server{
|
|
capability: defaultCapability,
|
|
},
|
|
ctx: tc,
|
|
alloc: arena.NewAllocator(32 * 1024),
|
|
}
|
|
srv := &Server{
|
|
clients: map[uint64]*clientConn{
|
|
connID: cc,
|
|
},
|
|
}
|
|
handle := ts.dom.ExpensiveQueryHandle().SetSessionManager(srv)
|
|
go handle.Run()
|
|
|
|
_, err = se.Execute(context.Background(), "use test;")
|
|
c.Assert(err, IsNil)
|
|
_, err = se.Execute(context.Background(), "CREATE TABLE testTable2 (id bigint PRIMARY KEY, age int)")
|
|
c.Assert(err, IsNil)
|
|
for i := 0; i < 10; i++ {
|
|
str := fmt.Sprintf("insert into testTable2 values(%d, %d)", i, i%80)
|
|
_, err = se.Execute(context.Background(), str)
|
|
c.Assert(err, IsNil)
|
|
}
|
|
|
|
_, err = se.Execute(context.Background(), "select SLEEP(1);")
|
|
c.Assert(err, IsNil)
|
|
|
|
_, err = se.Execute(context.Background(), "set @@max_execution_time = 500;")
|
|
c.Assert(err, IsNil)
|
|
|
|
err = cc.handleQuery(context.Background(), "select * FROM testTable2 WHERE SLEEP(1);")
|
|
c.Assert(err, IsNil)
|
|
|
|
_, err = se.Execute(context.Background(), "set @@max_execution_time = 1500;")
|
|
c.Assert(err, IsNil)
|
|
|
|
_, err = se.Execute(context.Background(), "set @@tidb_expensive_query_time_threshold = 1;")
|
|
c.Assert(err, IsNil)
|
|
|
|
records, err := se.Execute(context.Background(), "select SLEEP(2);")
|
|
c.Assert(err, IsNil)
|
|
tk := testkit.NewTestKit(c, ts.store)
|
|
tk.ResultSetToResult(records[0], Commentf("%v", records[0])).Check(testkit.Rows("1"))
|
|
|
|
_, err = se.Execute(context.Background(), "set @@max_execution_time = 0;")
|
|
c.Assert(err, IsNil)
|
|
|
|
err = cc.handleQuery(context.Background(), "select * FROM testTable2 WHERE SLEEP(1);")
|
|
c.Assert(err, IsNil)
|
|
|
|
err = cc.handleQuery(context.Background(), "select /*+ MAX_EXECUTION_TIME(100)*/ * FROM testTable2 WHERE SLEEP(1);")
|
|
c.Assert(err, IsNil)
|
|
|
|
c.Assert(failpoint.Disable("github.com/pingcap/tidb/server/FakeClientConn"), IsNil)
|
|
}
|
|
|
|
func (ts *ConnTestSuite) TestShutDown(c *C) {
|
|
cc := &clientConn{}
|
|
se, err := session.CreateSession4Test(ts.store)
|
|
c.Assert(err, IsNil)
|
|
cc.ctx = &TiDBContext{Session: se}
|
|
// set killed flag
|
|
cc.status = connStatusShutdown
|
|
// assert ErrQueryInterrupted
|
|
err = cc.handleQuery(context.Background(), "select 1")
|
|
c.Assert(err, Equals, executor.ErrQueryInterrupted)
|
|
}
|
|
|
|
func (ts *ConnTestSuite) TestShutdownOrNotify(c *C) {
|
|
c.Parallel()
|
|
se, err := session.CreateSession4Test(ts.store)
|
|
c.Assert(err, IsNil)
|
|
tc := &TiDBContext{
|
|
Session: se,
|
|
stmts: make(map[int]*TiDBStatement),
|
|
}
|
|
cc := &clientConn{
|
|
connectionID: 1,
|
|
server: &Server{
|
|
capability: defaultCapability,
|
|
},
|
|
status: connStatusWaitShutdown,
|
|
ctx: tc,
|
|
}
|
|
c.Assert(cc.ShutdownOrNotify(), IsFalse)
|
|
cc.status = connStatusReading
|
|
c.Assert(cc.ShutdownOrNotify(), IsTrue)
|
|
c.Assert(cc.status, Equals, connStatusShutdown)
|
|
cc.status = connStatusDispatching
|
|
c.Assert(cc.ShutdownOrNotify(), IsFalse)
|
|
c.Assert(cc.status, Equals, connStatusWaitShutdown)
|
|
}
|
|
|
|
type snapshotCache interface {
|
|
SnapCacheHitCount() int
|
|
}
|
|
|
|
func (ts *ConnTestSuite) TestPrefetchPointKeys(c *C) {
|
|
cc := &clientConn{
|
|
alloc: arena.NewAllocator(1024),
|
|
pkt: &packetIO{
|
|
bufWriter: bufio.NewWriter(bytes.NewBuffer(nil)),
|
|
},
|
|
}
|
|
tk := testkit.NewTestKitWithInit(c, ts.store)
|
|
cc.ctx = &TiDBContext{Session: tk.Se}
|
|
ctx := context.Background()
|
|
tk.Se.GetSessionVars().EnableClusteredIndex = variable.ClusteredIndexDefModeIntOnly
|
|
tk.MustExec("create table prefetch (a int, b int, c int, primary key (a, b))")
|
|
tk.MustExec("insert prefetch values (1, 1, 1), (2, 2, 2), (3, 3, 3)")
|
|
tk.MustExec("begin optimistic")
|
|
tk.MustExec("update prefetch set c = c + 1 where a = 2 and b = 2")
|
|
|
|
// enable multi-statement
|
|
capabilities := cc.ctx.GetSessionVars().ClientCapability
|
|
capabilities ^= mysql.ClientMultiStatements
|
|
cc.ctx.SetClientCapability(capabilities)
|
|
query := "update prefetch set c = c + 1 where a = 1 and b = 1;" +
|
|
"update prefetch set c = c + 1 where a = 2 and b = 2;" +
|
|
"update prefetch set c = c + 1 where a = 3 and b = 3;"
|
|
err := cc.handleQuery(ctx, query)
|
|
c.Assert(err, IsNil)
|
|
txn, err := tk.Se.Txn(false)
|
|
c.Assert(err, IsNil)
|
|
c.Assert(txn.Valid(), IsTrue)
|
|
snap := txn.GetSnapshot()
|
|
c.Assert(snap.(snapshotCache).SnapCacheHitCount(), Equals, 4)
|
|
tk.MustExec("commit")
|
|
tk.MustQuery("select * from prefetch").Check(testkit.Rows("1 1 2", "2 2 4", "3 3 4"))
|
|
|
|
tk.MustExec("begin pessimistic")
|
|
tk.MustExec("update prefetch set c = c + 1 where a = 2 and b = 2")
|
|
c.Assert(tk.Se.GetSessionVars().TxnCtx.PessimisticCacheHit, Equals, 1)
|
|
err = cc.handleQuery(ctx, query)
|
|
c.Assert(err, IsNil)
|
|
txn, err = tk.Se.Txn(false)
|
|
c.Assert(err, IsNil)
|
|
c.Assert(txn.Valid(), IsTrue)
|
|
c.Assert(tk.Se.GetSessionVars().TxnCtx.PessimisticCacheHit, Equals, 5)
|
|
tk.MustExec("commit")
|
|
tk.MustQuery("select * from prefetch").Check(testkit.Rows("1 1 3", "2 2 6", "3 3 5"))
|
|
}
|
|
|
|
func testGetTableByName(c *C, ctx sessionctx.Context, db, table string) table.Table {
|
|
dom := domain.GetDomain(ctx)
|
|
// Make sure the table schema is the new schema.
|
|
err := dom.Reload()
|
|
c.Assert(err, IsNil)
|
|
tbl, err := dom.InfoSchema().TableByName(model.NewCIStr(db), model.NewCIStr(table))
|
|
c.Assert(err, IsNil)
|
|
return tbl
|
|
}
|
|
|
|
func (ts *ConnTestSuite) TestTiFlashFallback(c *C) {
|
|
cc := &clientConn{
|
|
alloc: arena.NewAllocator(1024),
|
|
pkt: &packetIO{
|
|
bufWriter: bufio.NewWriter(bytes.NewBuffer(nil)),
|
|
},
|
|
}
|
|
tk := testkit.NewTestKitWithInit(c, ts.store)
|
|
cc.ctx = &TiDBContext{Session: tk.Se, stmts: make(map[int]*TiDBStatement)}
|
|
|
|
tk.MustExec("drop table if exists t")
|
|
tk.MustExec("create table t(a int not null primary key, b int not null)")
|
|
tk.MustExec("alter table t set tiflash replica 1")
|
|
tb := testGetTableByName(c, tk.Se, "test", "t")
|
|
err := domain.GetDomain(tk.Se).DDL().UpdateTableReplicaInfo(tk.Se, tb.Meta().ID, true)
|
|
c.Assert(err, IsNil)
|
|
|
|
dml := "insert into t values"
|
|
for i := 0; i < 50; i++ {
|
|
dml += fmt.Sprintf("(%v, 0)", i)
|
|
if i != 49 {
|
|
dml += ","
|
|
}
|
|
}
|
|
tk.MustExec(dml)
|
|
tk.MustQuery("select count(*) from t").Check(testkit.Rows("50"))
|
|
|
|
c.Assert(failpoint.Enable("github.com/pingcap/tidb/store/copr/ReduceCopNextMaxBackoff", `return(true)`), IsNil)
|
|
defer func() {
|
|
c.Assert(failpoint.Disable("github.com/pingcap/tidb/store/copr/ReduceCopNextMaxBackoff"), IsNil)
|
|
}()
|
|
|
|
c.Assert(failpoint.Enable("github.com/pingcap/tidb/store/mockstore/unistore/BatchCopRpcErrtiflash0", "return(\"tiflash0\")"), IsNil)
|
|
// test COM_STMT_EXECUTE
|
|
ctx := context.Background()
|
|
tk.MustExec("set @@tidb_allow_fallback_to_tikv='tiflash'")
|
|
tk.MustExec("set @@tidb_allow_mpp=OFF")
|
|
c.Assert(cc.handleStmtPrepare(ctx, "select sum(a) from t"), IsNil)
|
|
c.Assert(cc.handleStmtExecute(ctx, []byte{0x1, 0x0, 0x0, 0x0, 0x0, 0x1, 0x0, 0x0, 0x0}), IsNil)
|
|
tk.MustQuery("show warnings").Check(testkit.Rows("Error 9012 TiFlash server timeout"))
|
|
|
|
// test COM_STMT_FETCH (cursor mode)
|
|
c.Assert(cc.handleStmtExecute(ctx, []byte{0x1, 0x0, 0x0, 0x0, 0x1, 0x1, 0x0, 0x0, 0x0}), IsNil)
|
|
c.Assert(cc.handleStmtFetch(ctx, []byte{0x1, 0x0, 0x0, 0x0, 0x1, 0x0, 0x0, 0x0}), NotNil)
|
|
tk.MustExec("set @@tidb_allow_fallback_to_tikv=''")
|
|
c.Assert(cc.handleStmtExecute(ctx, []byte{0x1, 0x0, 0x0, 0x0, 0x0, 0x1, 0x0, 0x0, 0x0}), NotNil)
|
|
c.Assert(failpoint.Disable("github.com/pingcap/tidb/store/mockstore/unistore/BatchCopRpcErrtiflash0"), IsNil)
|
|
|
|
c.Assert(failpoint.Enable("github.com/pingcap/tidb/server/fetchNextErr", "return(\"firstNext\")"), IsNil)
|
|
// test COM_STMT_EXECUTE (cursor mode)
|
|
tk.MustExec("set @@tidb_allow_fallback_to_tikv='tiflash'")
|
|
c.Assert(cc.handleStmtExecute(ctx, []byte{0x1, 0x0, 0x0, 0x0, 0x1, 0x1, 0x0, 0x0, 0x0}), IsNil)
|
|
c.Assert(failpoint.Disable("github.com/pingcap/tidb/server/fetchNextErr"), IsNil)
|
|
|
|
// test that TiDB would not retry if the first execution already sends data to client
|
|
c.Assert(failpoint.Enable("github.com/pingcap/tidb/server/fetchNextErr", "return(\"secondNext\")"), IsNil)
|
|
tk.MustExec("set @@tidb_allow_fallback_to_tikv='tiflash'")
|
|
c.Assert(cc.handleQuery(ctx, "select * from t t1 join t t2 on t1.a = t2.a"), NotNil)
|
|
c.Assert(failpoint.Disable("github.com/pingcap/tidb/server/fetchNextErr"), IsNil)
|
|
|
|
// simple TiFlash query (unary + non-streaming)
|
|
tk.MustExec("set @@tidb_allow_batch_cop=0; set @@tidb_allow_mpp=0;")
|
|
c.Assert(failpoint.Enable("tikvclient/tikvStoreSendReqResult", "return(\"requestTiFlashError\")"), IsNil)
|
|
testFallbackWork(c, tk, cc, "select sum(a) from t")
|
|
c.Assert(failpoint.Disable("tikvclient/tikvStoreSendReqResult"), IsNil)
|
|
|
|
// TiFlash query based on batch cop (batch + streaming)
|
|
tk.MustExec("set @@tidb_allow_batch_cop=1; set @@tidb_allow_mpp=0;")
|
|
|
|
c.Assert(failpoint.Enable("github.com/pingcap/tidb/store/mockstore/unistore/BatchCopRpcErrtiflash0", "return(\"tiflash0\")"), IsNil)
|
|
testFallbackWork(c, tk, cc, "select count(*) from t")
|
|
c.Assert(failpoint.Disable("github.com/pingcap/tidb/store/mockstore/unistore/BatchCopRpcErrtiflash0"), IsNil)
|
|
|
|
c.Assert(failpoint.Enable("github.com/pingcap/tidb/store/mockstore/unistore/batchCopRecvTimeout", "return(true)"), IsNil)
|
|
testFallbackWork(c, tk, cc, "select count(*) from t")
|
|
c.Assert(failpoint.Disable("github.com/pingcap/tidb/store/mockstore/unistore/batchCopRecvTimeout"), IsNil)
|
|
|
|
// TiFlash MPP query (MPP + streaming)
|
|
tk.MustExec("set @@tidb_allow_batch_cop=0; set @@tidb_allow_mpp=1;")
|
|
|
|
c.Assert(failpoint.Enable("github.com/pingcap/tidb/store/mockstore/unistore/mppDispatchTimeout", "return(true)"), IsNil)
|
|
testFallbackWork(c, tk, cc, "select * from t t1 join t t2 on t1.a = t2.a")
|
|
c.Assert(failpoint.Disable("github.com/pingcap/tidb/store/mockstore/unistore/mppDispatchTimeout"), IsNil)
|
|
|
|
c.Assert(failpoint.Enable("github.com/pingcap/tidb/store/mockstore/unistore/mppRecvTimeout", "return(-1)"), IsNil)
|
|
testFallbackWork(c, tk, cc, "select * from t t1 join t t2 on t1.a = t2.a")
|
|
c.Assert(failpoint.Disable("github.com/pingcap/tidb/store/mockstore/unistore/mppRecvTimeout"), IsNil)
|
|
|
|
c.Assert(failpoint.Enable("github.com/pingcap/tidb/store/mockstore/unistore/establishMppConnectionErr", "return(true)"), IsNil)
|
|
testFallbackWork(c, tk, cc, "select * from t t1 join t t2 on t1.a = t2.a")
|
|
c.Assert(failpoint.Disable("github.com/pingcap/tidb/store/mockstore/unistore/establishMppConnectionErr"), IsNil)
|
|
}
|
|
|
|
func testFallbackWork(c *C, tk *testkit.TestKit, cc *clientConn, sql string) {
|
|
ctx := context.Background()
|
|
tk.MustExec("set @@tidb_allow_fallback_to_tikv=''")
|
|
c.Assert(tk.QueryToErr(sql), NotNil)
|
|
tk.MustExec("set @@tidb_allow_fallback_to_tikv='tiflash'")
|
|
|
|
c.Assert(cc.handleQuery(ctx, sql), IsNil)
|
|
tk.MustQuery("show warnings").Check(testkit.Rows("Error 9012 TiFlash server timeout"))
|
|
}
|
|
|
|
// For issue https://github.com/pingcap/tidb/issues/25069
|
|
func (ts *ConnTestSuite) TestShowErrors(c *C) {
|
|
cc := &clientConn{
|
|
alloc: arena.NewAllocator(1024),
|
|
pkt: &packetIO{
|
|
bufWriter: bufio.NewWriter(bytes.NewBuffer(nil)),
|
|
},
|
|
}
|
|
ctx := context.Background()
|
|
tk := testkit.NewTestKitWithInit(c, ts.store)
|
|
cc.ctx = &TiDBContext{Session: tk.Se, stmts: make(map[int]*TiDBStatement)}
|
|
|
|
err := cc.handleQuery(ctx, "create database if not exists test;")
|
|
c.Assert(err, IsNil)
|
|
err = cc.handleQuery(ctx, "use test;")
|
|
c.Assert(err, IsNil)
|
|
|
|
stmts, err := cc.ctx.Parse(ctx, "drop table idontexist")
|
|
c.Assert(err, IsNil)
|
|
_, err = cc.ctx.ExecuteStmt(ctx, stmts[0])
|
|
c.Assert(err, NotNil)
|
|
tk.MustQuery("show errors").Check(testkit.Rows("Error 1051 Unknown table 'test.idontexist'"))
|
|
}
|