Files
tidb/session/schema_test.go

883 lines
28 KiB
Go

// Copyright 2022 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 session_test
import (
"context"
"fmt"
"sync/atomic"
"testing"
"time"
"github.com/pingcap/failpoint"
"github.com/pingcap/tidb/config"
"github.com/pingcap/tidb/domain"
"github.com/pingcap/tidb/kv"
"github.com/pingcap/tidb/parser/model"
"github.com/pingcap/tidb/parser/terror"
plannercore "github.com/pingcap/tidb/planner/core"
"github.com/pingcap/tidb/session"
"github.com/pingcap/tidb/sessionctx/variable"
"github.com/pingcap/tidb/store/mockstore"
"github.com/pingcap/tidb/tablecodec"
"github.com/pingcap/tidb/testkit"
"github.com/stretchr/testify/require"
"github.com/tikv/client-go/v2/testutils"
)
func skipIfWithRealTiKV(t *testing.T) {
if *withTiKV {
t.Skip("Schema tests has nothing to do with real tikv scenario")
}
}
func createMockStoreForSchemaTest(t *testing.T, opts ...mockstore.MockTiKVStoreOption) (kv.Storage, func()) {
store, err := mockstore.NewMockStore(opts...)
require.NoError(t, err)
session.DisableStats4Test()
dom, err := session.BootstrapSession(store)
require.NoError(t, err)
dom.SetStatsUpdating(true)
clean := func() {
dom.Close()
require.NoError(t, store.Close())
}
return store, clean
}
func TestPrepareStmtCommitWhenSchemaChanged(t *testing.T) {
skipIfWithRealTiKV(t)
store, clean := createMockStoreForSchemaTest(t)
defer clean()
tk1 := testkit.NewTestKit(t, store)
tk2 := testkit.NewTestKit(t, store)
tk1.MustExec("use test")
tk2.MustExec("use test")
tk1.MustExec("create table t (a int, b int)")
tk2.MustExec("prepare stmt from 'insert into t values (?, ?)'")
tk2.MustExec("set @a = 1")
// Commit find unrelated schema change.
tk2.MustExec("begin")
tk1.MustExec("create table t1 (id int)")
tk2.MustExec("execute stmt using @a, @a")
tk2.MustExec("commit")
tk2.MustExec("set @@tidb_disable_txn_auto_retry = 0")
tk2.MustExec("begin")
tk1.MustExec("alter table t drop column b")
tk2.MustExec("execute stmt using @a, @a")
err := tk2.ExecToErr("commit")
require.True(t, terror.ErrorEqual(err, plannercore.ErrWrongValueCountOnRow), fmt.Sprintf("err %v", err))
}
func TestCommitWhenSchemaChanged(t *testing.T) {
skipIfWithRealTiKV(t)
store, clean := createMockStoreForSchemaTest(t)
defer clean()
tk1 := testkit.NewTestKit(t, store)
tk2 := testkit.NewTestKit(t, store)
tk1.MustExec("use test")
tk2.MustExec("use test")
tk1.MustExec("create table t (a int, b int)")
tk2.MustExec("set @@tidb_disable_txn_auto_retry = 0")
tk2.MustExec("begin")
tk2.MustExec("insert into t values (1, 1)")
tk1.MustExec("alter table t drop column b")
// When tk2 commit, it will find schema already changed.
tk2.MustExec("insert into t values (4, 4)")
err := tk2.ExecToErr("commit")
require.True(t, terror.ErrorEqual(err, plannercore.ErrWrongValueCountOnRow), fmt.Sprintf("err %v", err))
}
func TestRetrySchemaChangeForEmptyChange(t *testing.T) {
skipIfWithRealTiKV(t)
store, clean := createMockStoreForSchemaTest(t)
defer clean()
tk1 := testkit.NewTestKit(t, store)
tk2 := testkit.NewTestKit(t, store)
tk1.MustExec("use test")
tk2.MustExec("use test")
tk1.MustExec("create table t (i int)")
tk1.MustExec("create table t1 (i int)")
tk1.MustExec("begin")
tk2.MustExec("alter table t add j int")
tk1.MustExec("select * from t for update")
tk1.MustExec("update t set i = -i")
tk1.MustExec("delete from t")
tk1.MustExec("insert into t1 values (1)")
tk1.MustExec("commit")
// TODO remove this enable after fixing table delta map.
tk1.MustExec("set tidb_enable_amend_pessimistic_txn = 1")
tk1.MustExec("begin pessimistic")
tk2.MustExec("alter table t add k int")
tk1.MustExec("select * from t for update")
tk1.MustExec("update t set i = -i")
tk1.MustExec("delete from t")
tk1.MustExec("insert into t1 values (1)")
tk1.MustExec("commit")
}
func TestRetrySchemaChange(t *testing.T) {
skipIfWithRealTiKV(t)
store, clean := createMockStoreForSchemaTest(t)
defer clean()
tk1 := testkit.NewTestKit(t, store)
tk2 := testkit.NewTestKit(t, store)
tk1.MustExec("use test")
tk2.MustExec("use test")
tk1.MustExec("create table t (a int primary key, b int)")
tk1.MustExec("insert into t values (1, 1)")
tk2.MustExec("set @@tidb_disable_txn_auto_retry = 0")
tk2.MustExec("begin")
tk2.MustExec("update t set b = 5 where a = 1")
tk1.MustExec("alter table t add index b_i (b)")
run := false
hook := func() {
if !run {
tk1.MustExec("update t set b = 3 where a = 1")
run = true
}
}
// In order to cover a bug that statement history is not updated during retry.
// See https://github.com/pingcap/tidb/pull/5202
// Step1: when tk2 commit, it find schema changed and retry().
// Step2: during retry, hook() is called, tk1 update primary key.
// Step3: tk2 continue commit in retry() meet a retryable error(write conflict), retry again.
// Step4: tk2 retry() success, if it use the stale statement, data and index will inconsistent.
fpName := "github.com/pingcap/tidb/session/preCommitHook"
require.NoError(t, failpoint.Enable(fpName, "return"))
defer func() { require.NoError(t, failpoint.Disable(fpName)) }()
ctx := context.WithValue(context.Background(), "__preCommitHook", hook)
require.NoError(t, tk2.Session().CommitTxn(ctx))
tk1.MustQuery("select * from t where t.b = 5").Check(testkit.Rows("1 5"))
}
func TestRetryMissingUnionScan(t *testing.T) {
skipIfWithRealTiKV(t)
store, clean := createMockStoreForSchemaTest(t)
defer clean()
tk1 := testkit.NewTestKit(t, store)
tk2 := testkit.NewTestKit(t, store)
tk1.MustExec("use test")
tk2.MustExec("use test")
tk1.MustExec("create table t (a int primary key, b int unique, c int)")
tk1.MustExec("insert into t values (1, 1, 1)")
tk2.MustExec("set @@tidb_disable_txn_auto_retry = 0")
tk2.MustExec("begin")
tk2.MustExec("update t set b = 1, c = 2 where b = 2")
tk2.MustExec("update t set b = 1, c = 2 where a = 1")
// Create a conflict to reproduces the bug that the second update statement in retry
// has a dirty table but doesn't use UnionScan.
tk1.MustExec("update t set b = 2 where a = 1")
tk2.MustExec("commit")
}
func TestTableReaderChunk(t *testing.T) {
skipIfWithRealTiKV(t)
// Since normally a single region mock tikv only returns one partial result we need to manually split the
// table to test multiple chunks.
var cluster testutils.Cluster
store, clean := testkit.CreateMockStore(t, mockstore.WithClusterInspector(func(c testutils.Cluster) {
mockstore.BootstrapWithSingleStore(c)
cluster = c
}))
defer clean()
tk := testkit.NewTestKit(t, store)
tk.MustExec("use test")
tk.MustExec("create table chk (a int)")
for i := 0; i < 100; i++ {
tk.MustExec(fmt.Sprintf("insert chk values (%d)", i))
}
tbl, err := domain.GetDomain(tk.Session()).InfoSchema().TableByName(model.NewCIStr("test"), model.NewCIStr("chk"))
require.NoError(t, err)
tableStart := tablecodec.GenTableRecordPrefix(tbl.Meta().ID)
cluster.SplitKeys(tableStart, tableStart.PrefixNext(), 10)
tk.Session().GetSessionVars().SetDistSQLScanConcurrency(1)
tk.MustExec("set tidb_init_chunk_size = 2")
defer func() {
tk.MustExec(fmt.Sprintf("set tidb_init_chunk_size = %d", variable.DefInitChunkSize))
}()
rs, err := tk.Exec("select * from chk")
require.NoError(t, err)
defer func() { require.NoError(t, rs.Close()) }()
req := rs.NewChunk(nil)
var count int
var numChunks int
for {
err = rs.Next(context.TODO(), req)
require.NoError(t, err)
numRows := req.NumRows()
if numRows == 0 {
break
}
for i := 0; i < numRows; i++ {
require.Equal(t, int64(count), req.GetRow(i).GetInt64(0))
count++
}
numChunks++
}
require.Equal(t, 100, count)
// FIXME: revert this result to new group value after distsql can handle initChunkSize.
require.Equal(t, 1, numChunks)
}
func TestInsertExecChunk(t *testing.T) {
skipIfWithRealTiKV(t)
store, clean := createMockStoreForSchemaTest(t)
defer clean()
tk := testkit.NewTestKit(t, store)
tk.MustExec("use test")
tk.MustExec("create table test1(a int)")
for i := 0; i < 100; i++ {
tk.MustExec(fmt.Sprintf("insert test1 values (%d)", i))
}
tk.MustExec("create table test2(a int)")
tk.Session().GetSessionVars().SetDistSQLScanConcurrency(1)
tk.MustExec("insert into test2(a) select a from test1;")
rs, err := tk.Exec("select * from test2")
require.NoError(t, err)
defer func() { require.NoError(t, rs.Close()) }()
var idx int
for {
req := rs.NewChunk(nil)
err = rs.Next(context.TODO(), req)
require.NoError(t, err)
if req.NumRows() == 0 {
break
}
for rowIdx := 0; rowIdx < req.NumRows(); rowIdx++ {
row := req.GetRow(rowIdx)
require.Equal(t, int64(idx), row.GetInt64(0))
idx++
}
}
require.Equal(t, 100, idx)
}
func TestUpdateExecChunk(t *testing.T) {
skipIfWithRealTiKV(t)
store, clean := createMockStoreForSchemaTest(t)
defer clean()
tk := testkit.NewTestKit(t, store)
tk.MustExec("use test")
tk.MustExec("create table chk(a int)")
for i := 0; i < 100; i++ {
tk.MustExec(fmt.Sprintf("insert chk values (%d)", i))
}
tk.Session().GetSessionVars().SetDistSQLScanConcurrency(1)
for i := 0; i < 100; i++ {
tk.MustExec(fmt.Sprintf("update chk set a = a + 100 where a = %d", i))
}
rs, err := tk.Exec("select * from chk")
require.NoError(t, err)
defer func() { require.NoError(t, rs.Close()) }()
var idx int
for {
req := rs.NewChunk(nil)
err = rs.Next(context.TODO(), req)
require.NoError(t, err)
if req.NumRows() == 0 {
break
}
for rowIdx := 0; rowIdx < req.NumRows(); rowIdx++ {
row := req.GetRow(rowIdx)
require.Equal(t, int64(idx+100), row.GetInt64(0))
idx++
}
}
require.Equal(t, 100, idx)
}
func TestDeleteExecChunk(t *testing.T) {
skipIfWithRealTiKV(t)
store, clean := createMockStoreForSchemaTest(t)
defer clean()
tk := testkit.NewTestKit(t, store)
tk.MustExec("use test")
tk.MustExec("create table chk(a int)")
for i := 0; i < 100; i++ {
tk.MustExec(fmt.Sprintf("insert chk values (%d)", i))
}
tk.Session().GetSessionVars().SetDistSQLScanConcurrency(1)
for i := 0; i < 99; i++ {
tk.MustExec(fmt.Sprintf("delete from chk where a = %d", i))
}
rs, err := tk.Exec("select * from chk")
require.NoError(t, err)
defer func() { require.NoError(t, rs.Close()) }()
req := rs.NewChunk(nil)
err = rs.Next(context.TODO(), req)
require.NoError(t, err)
require.Equal(t, 1, req.NumRows())
row := req.GetRow(0)
require.Equal(t, int64(99), row.GetInt64(0))
}
func TestDeleteMultiTableExecChunk(t *testing.T) {
skipIfWithRealTiKV(t)
store, clean := createMockStoreForSchemaTest(t)
defer clean()
tk := testkit.NewTestKit(t, store)
tk.MustExec("use test")
tk.MustExec("create table chk1(a int)")
tk.MustExec("create table chk2(a int)")
for i := 0; i < 100; i++ {
tk.MustExec(fmt.Sprintf("insert chk1 values (%d)", i))
}
for i := 0; i < 50; i++ {
tk.MustExec(fmt.Sprintf("insert chk2 values (%d)", i))
}
tk.Session().GetSessionVars().SetDistSQLScanConcurrency(1)
tk.MustExec("delete chk1, chk2 from chk1 inner join chk2 where chk1.a = chk2.a")
rs, err := tk.Exec("select * from chk1")
require.NoError(t, err)
var idx int
for {
req := rs.NewChunk(nil)
err = rs.Next(context.TODO(), req)
require.NoError(t, err)
if req.NumRows() == 0 {
break
}
for i := 0; i < req.NumRows(); i++ {
row := req.GetRow(i)
require.Equal(t, int64(idx+50), row.GetInt64(0))
idx++
}
}
require.Equal(t, 50, idx)
require.NoError(t, rs.Close())
rs, err = tk.Exec("select * from chk2")
require.NoError(t, err)
req := rs.NewChunk(nil)
err = rs.Next(context.TODO(), req)
require.NoError(t, err)
require.Equal(t, 0, req.NumRows())
require.NoError(t, rs.Close())
}
func TestIndexLookUpReaderChunk(t *testing.T) {
skipIfWithRealTiKV(t)
// Since normally a single region mock tikv only returns one partial result we need to manually split the
// table to test multiple chunks.
var cluster testutils.Cluster
store, clean := testkit.CreateMockStore(t, mockstore.WithClusterInspector(func(c testutils.Cluster) {
mockstore.BootstrapWithSingleStore(c)
cluster = c
}))
defer clean()
tk := testkit.NewTestKit(t, store)
tk.MustExec("use test")
tk.MustExec("drop table if exists chk")
tk.MustExec("create table chk (k int unique, c int)")
for i := 0; i < 100; i++ {
tk.MustExec(fmt.Sprintf("insert chk values (%d, %d)", i, i))
}
tbl, err := domain.GetDomain(tk.Session()).InfoSchema().TableByName(model.NewCIStr("test"), model.NewCIStr("chk"))
require.NoError(t, err)
indexStart := tablecodec.EncodeTableIndexPrefix(tbl.Meta().ID, tbl.Indices()[0].Meta().ID)
cluster.SplitKeys(indexStart, indexStart.PrefixNext(), 10)
tk.Session().GetSessionVars().IndexLookupSize = 10
rs, err := tk.Exec("select * from chk order by k")
require.NoError(t, err)
req := rs.NewChunk(nil)
var count int
for {
err = rs.Next(context.TODO(), req)
require.NoError(t, err)
numRows := req.NumRows()
if numRows == 0 {
break
}
for i := 0; i < numRows; i++ {
require.Equal(t, int64(count), req.GetRow(i).GetInt64(0))
require.Equal(t, int64(count), req.GetRow(i).GetInt64(1))
count++
}
}
require.Equal(t, 100, count)
require.NoError(t, rs.Close())
rs, err = tk.Exec("select k from chk where c < 90 order by k")
require.NoError(t, err)
req = rs.NewChunk(nil)
count = 0
for {
err = rs.Next(context.TODO(), req)
require.NoError(t, err)
numRows := req.NumRows()
if numRows == 0 {
break
}
for i := 0; i < numRows; i++ {
require.Equal(t, int64(count), req.GetRow(i).GetInt64(0))
count++
}
}
require.Equal(t, 90, count)
require.NoError(t, rs.Close())
}
func TestDisableTxnAutoRetry(t *testing.T) {
skipIfWithRealTiKV(t)
store, clean := createMockStoreForSchemaTest(t)
defer clean()
tk1 := testkit.NewTestKit(t, store)
tk2 := testkit.NewTestKit(t, store)
tk1.MustExec("use test")
tk2.MustExec("use test")
tk1.MustExec("create table no_retry (id int)")
tk1.MustExec("insert into no_retry values (1)")
tk1.MustExec("set @@tidb_disable_txn_auto_retry = 1")
tk1.MustExec("begin")
tk1.MustExec("update no_retry set id = 2")
tk2.MustExec("begin")
tk2.MustExec("update no_retry set id = 3")
tk2.MustExec("commit")
// No auto retry because tidb_disable_txn_auto_retry is set to 1.
_, err := tk1.Session().Execute(context.Background(), "commit")
require.Error(t, err)
// session 1 starts a transaction early.
// execute a select statement to clear retry history.
tk1.MustExec("select 1")
err = tk1.Session().PrepareTxnCtx(context.Background())
require.NoError(t, err)
// session 2 update the value.
tk2.MustExec("update no_retry set id = 4")
// AutoCommit update will retry, so it would not fail.
tk1.MustExec("update no_retry set id = 5")
// RestrictedSQL should retry.
tk1.Session().GetSessionVars().InRestrictedSQL = true
tk1.MustExec("begin")
tk2.MustExec("update no_retry set id = 6")
tk1.MustExec("update no_retry set id = 7")
tk1.MustExec("commit")
// test for disable transaction local latch
tk1.Session().GetSessionVars().InRestrictedSQL = false
defer config.RestoreFunc()()
config.UpdateGlobal(func(conf *config.Config) {
conf.TxnLocalLatches.Enabled = false
})
tk1.MustExec("begin")
tk1.MustExec("update no_retry set id = 9")
tk2.MustExec("update no_retry set id = 8")
_, err = tk1.Session().Execute(context.Background(), "commit")
require.Error(t, err)
require.True(t, kv.ErrWriteConflict.Equal(err), fmt.Sprintf("err %v", err))
require.Contains(t, err.Error(), kv.TxnRetryableMark)
tk1.MustExec("rollback")
config.UpdateGlobal(func(conf *config.Config) {
conf.TxnLocalLatches.Enabled = true
})
tk1.MustExec("begin")
tk2.MustExec("alter table no_retry add index idx(id)")
tk2.MustQuery("select * from no_retry").Check(testkit.Rows("8"))
tk1.MustExec("update no_retry set id = 10")
_, err = tk1.Session().Execute(context.Background(), "commit")
require.Error(t, err)
// set autocommit to begin and commit
tk1.MustExec("set autocommit = 0")
tk1.MustQuery("select * from no_retry").Check(testkit.Rows("8"))
tk2.MustExec("update no_retry set id = 11")
tk1.MustExec("update no_retry set id = 12")
_, err = tk1.Session().Execute(context.Background(), "set autocommit = 1")
require.Error(t, err)
require.True(t, kv.ErrWriteConflict.Equal(err), fmt.Sprintf("err %v", err))
require.Contains(t, err.Error(), kv.TxnRetryableMark)
tk1.MustExec("rollback")
tk2.MustQuery("select * from no_retry").Check(testkit.Rows("11"))
tk1.MustExec("set autocommit = 0")
tk1.MustQuery("select * from no_retry").Check(testkit.Rows("11"))
tk2.MustExec("update no_retry set id = 13")
tk1.MustExec("update no_retry set id = 14")
_, err = tk1.Session().Execute(context.Background(), "commit")
require.Error(t, err)
require.True(t, kv.ErrWriteConflict.Equal(err), fmt.Sprintf("err %v", err))
require.Contains(t, err.Error(), kv.TxnRetryableMark)
tk1.MustExec("rollback")
tk2.MustQuery("select * from no_retry").Check(testkit.Rows("13"))
}
func TestTxnSize(t *testing.T) {
skipIfWithRealTiKV(t)
store, clean := createMockStoreForSchemaTest(t)
defer clean()
tk := testkit.NewTestKit(t, store)
tk.MustExec("use test")
tk.MustExec("drop table if exists txn_size")
tk.MustExec("create table txn_size (k int , v varchar(64))")
tk.MustExec("begin")
tk.MustExec("insert txn_size values (1, 'dfaasdfsdf')")
tk.MustExec("insert txn_size values (2, 'dsdfaasdfsdf')")
tk.MustExec("insert txn_size values (3, 'abcdefghijkl')")
txn, err := tk.Session().Txn(false)
require.NoError(t, err)
require.Greater(t, txn.Size(), 0)
}
func TestLoadSchemaFailed(t *testing.T) {
skipIfWithRealTiKV(t)
originalRetryTime := domain.SchemaOutOfDateRetryTimes.Load()
originalRetryInterval := domain.SchemaOutOfDateRetryInterval.Load()
domain.SchemaOutOfDateRetryTimes.Store(3)
domain.SchemaOutOfDateRetryInterval.Store(20 * time.Millisecond)
defer func() {
domain.SchemaOutOfDateRetryTimes.Store(originalRetryTime)
domain.SchemaOutOfDateRetryInterval.Store(originalRetryInterval)
}()
store, clean := createMockStoreForSchemaTest(t)
defer clean()
tk := testkit.NewTestKit(t, store)
tk1 := testkit.NewTestKit(t, store)
tk2 := testkit.NewTestKit(t, store)
tk.MustExec("use test")
tk1.MustExec("use test")
tk2.MustExec("use test")
tk.MustExec("create table t (a int);")
tk.MustExec("create table t1 (a int);")
tk.MustExec("create table t2 (a int);")
tk1.MustExec("begin")
tk2.MustExec("begin")
// Make sure loading information schema is failed and server is invalid.
require.NoError(t, failpoint.Enable("github.com/pingcap/tidb/domain/ErrorMockReloadFailed", `return(true)`))
defer func() { require.NoError(t, failpoint.Disable("github.com/pingcap/tidb/domain/ErrorMockReloadFailed")) }()
require.Error(t, domain.GetDomain(tk.Session()).Reload())
lease := domain.GetDomain(tk.Session()).DDL().GetLease()
time.Sleep(lease * 2)
// Make sure executing insert statement is failed when server is invalid.
require.Error(t, tk.ExecToErr("insert t values (100);"))
tk1.MustExec("insert t1 values (100);")
tk2.MustExec("insert t2 values (100);")
require.Error(t, tk1.ExecToErr("commit"))
ver, err := store.CurrentVersion(kv.GlobalTxnScope)
require.NoError(t, err)
require.NotNil(t, ver)
require.NoError(t, failpoint.Disable("github.com/pingcap/tidb/domain/ErrorMockReloadFailed"))
time.Sleep(lease * 2)
tk.MustExec("drop table if exists t;")
tk.MustExec("create table t (a int);")
tk.MustExec("insert t values (100);")
// Make sure insert to table t2 transaction executes.
tk2.MustExec("commit")
}
func TestValidationRecursion(t *testing.T) {
skipIfWithRealTiKV(t)
// We have to expect that validation functions will call GlobalVarsAccessor.GetGlobalSysVar().
// This tests for a regression where GetGlobalSysVar() can not safely call the validation
// function because it might cause infinite recursion.
// See: https://github.com/pingcap/tidb/issues/30255
sv := variable.SysVar{Scope: variable.ScopeGlobal, Name: "mynewsysvar", Value: "test", Validation: func(vars *variable.SessionVars, normalizedValue string, originalValue string, scope variable.ScopeFlag) (string, error) {
return vars.GlobalVarsAccessor.GetGlobalSysVar("mynewsysvar")
}}
variable.RegisterSysVar(&sv)
store, clean := createMockStoreForSchemaTest(t)
defer clean()
tk := testkit.NewTestKit(t, store)
tk.MustExec("use test")
val, err := sv.Validate(tk.Session().GetSessionVars(), "test2", variable.ScopeGlobal)
require.NoError(t, err)
require.Equal(t, "test", val)
}
func TestSchemaCheckerSQL(t *testing.T) {
skipIfWithRealTiKV(t)
store, clean := createMockStoreForSchemaTest(t)
defer clean()
tk := testkit.NewTestKit(t, store)
tk1 := testkit.NewTestKit(t, store)
tk.MustExec("use test")
tk1.MustExec("use test")
// create table
tk.MustExec(`create table t (id int, c int);`)
tk.MustExec(`create table t1 (id int, c int);`)
// insert data
tk.MustExec(`insert into t values(1, 1);`)
// The schema version is out of date in the first transaction, but the SQL can be retried.
tk.MustExec("set @@tidb_disable_txn_auto_retry = 0")
tk.MustExec(`begin;`)
tk1.MustExec(`alter table t add index idx(c);`)
tk.MustExec(`insert into t values(2, 2);`)
tk.MustExec(`commit;`)
// The schema version is out of date in the first transaction, and the SQL can't be retried.
atomic.StoreUint32(&session.SchemaChangedWithoutRetry, 1)
defer func() {
atomic.StoreUint32(&session.SchemaChangedWithoutRetry, 0)
}()
tk.MustExec(`begin;`)
tk1.MustExec(`alter table t modify column c bigint;`)
tk.MustExec(`insert into t values(3, 3);`)
err := tk.ExecToErr(`commit;`)
require.True(t, terror.ErrorEqual(err, domain.ErrInfoSchemaChanged), fmt.Sprintf("err %v", err))
// But the transaction related table IDs aren't in the updated table IDs.
tk.MustExec(`begin;`)
tk1.MustExec(`alter table t add index idx2(c);`)
tk.MustExec(`insert into t1 values(4, 4);`)
tk.MustExec(`commit;`)
// Test for "select for update".
tk.MustExec(`begin;`)
tk1.MustExec(`alter table t add index idx3(c);`)
tk.MustQuery(`select * from t for update`)
require.Error(t, tk.ExecToErr(`commit;`))
// Repeated tests for partitioned table
tk.MustExec(`create table pt (id int, c int) partition by hash (id) partitions 3`)
tk.MustExec(`insert into pt values(1, 1);`)
// The schema version is out of date in the first transaction, and the SQL can't be retried.
tk.MustExec(`begin;`)
tk1.MustExec(`alter table pt modify column c bigint;`)
tk.MustExec(`insert into pt values(3, 3);`)
err = tk.ExecToErr(`commit;`)
require.True(t, terror.ErrorEqual(err, domain.ErrInfoSchemaChanged), fmt.Sprintf("err %v", err))
// But the transaction related table IDs aren't in the updated table IDs.
tk.MustExec(`begin;`)
tk1.MustExec(`alter table pt add index idx2(c);`)
tk.MustExec(`insert into t1 values(4, 4);`)
tk.MustExec(`commit;`)
// Test for "select for update".
tk.MustExec(`begin;`)
tk1.MustExec(`alter table pt add index idx3(c);`)
tk.MustQuery(`select * from pt for update`)
require.Error(t, tk.ExecToErr(`commit;`))
// Test for "select for update".
tk.MustExec(`begin;`)
tk1.MustExec(`alter table pt add index idx4(c);`)
tk.MustQuery(`select * from pt partition (p1) for update`)
require.Error(t, tk.ExecToErr(`commit;`))
}
func TestSchemaCheckerTempTable(t *testing.T) {
skipIfWithRealTiKV(t)
store, clean := createMockStoreForSchemaTest(t)
defer clean()
tk1 := testkit.NewTestKit(t, store)
tk2 := testkit.NewTestKit(t, store)
tk1.MustExec("use test")
tk2.MustExec("use test")
// create table
tk1.MustExec(`drop table if exists normal_table`)
tk1.MustExec(`create table normal_table (id int, c int);`)
defer tk1.MustExec(`drop table if exists normal_table`)
tk1.MustExec(`drop table if exists temp_table`)
tk1.MustExec(`create global temporary table temp_table (id int primary key, c int) on commit delete rows;`)
defer tk1.MustExec(`drop table if exists temp_table`)
// The schema version is out of date in the first transaction, and the SQL can't be retried.
atomic.StoreUint32(&session.SchemaChangedWithoutRetry, 1)
defer func() {
atomic.StoreUint32(&session.SchemaChangedWithoutRetry, 0)
}()
// It's fine to change the schema of temporary tables.
tk1.MustExec(`begin;`)
tk2.MustExec(`alter table temp_table modify column c tinyint;`)
tk1.MustExec(`insert into temp_table values(3, 3);`)
tk1.MustExec(`commit;`)
tk1.MustExec("begin pessimistic")
tk2.MustExec(`alter table temp_table modify column c int;`)
tk1.MustQuery(`select * from temp_table for update;`).Check(testkit.Rows())
tk1.MustExec(`commit;`)
tk1.MustExec("begin pessimistic")
tk2.MustExec(`alter table temp_table modify column c smallint;`)
tk1.MustExec(`insert into temp_table values(3, 4);`)
tk1.MustQuery(`select * from temp_table for update;`).Check(testkit.Rows("3 4"))
tk1.MustExec(`commit;`)
tk1.MustExec("begin pessimistic")
tk2.MustExec(`alter table temp_table modify column c bigint;`)
tk1.MustQuery(`select * from temp_table where id=1 for update;`).Check(testkit.Rows())
tk1.MustExec(`commit;`)
tk1.MustExec("begin pessimistic")
tk2.MustExec(`alter table temp_table modify column c smallint;`)
tk1.MustExec("insert into temp_table values (1, 2), (2, 3), (4, 5)")
tk1.MustQuery(`select * from temp_table where id=1 for update;`).Check(testkit.Rows("1 2"))
tk1.MustExec(`commit;`)
tk1.MustExec("begin pessimistic")
tk2.MustExec(`alter table temp_table modify column c int;`)
tk1.MustQuery(`select * from temp_table where id=1 for update;`).Check(testkit.Rows())
tk1.MustExec(`commit;`)
tk1.MustExec("begin pessimistic")
tk2.MustExec(`alter table temp_table modify column c bigint;`)
tk1.MustQuery(`select * from temp_table where id in (1, 2, 3) for update;`).Check(testkit.Rows())
tk1.MustExec(`commit;`)
tk1.MustExec("begin pessimistic")
tk2.MustExec(`alter table temp_table modify column c int;`)
tk1.MustExec("insert into temp_table values (1, 2), (2, 3), (4, 5)")
tk1.MustQuery(`select * from temp_table where id in (1, 2, 3) for update;`).Check(testkit.Rows("1 2", "2 3"))
tk1.MustExec(`commit;`)
tk1.MustExec("insert into normal_table values(1, 2)")
tk1.MustExec("begin pessimistic")
tk2.MustExec(`alter table temp_table modify column c int;`)
tk1.MustExec(`insert into temp_table values(1, 5);`)
tk1.MustQuery(`select * from temp_table, normal_table where temp_table.id = normal_table.id for update;`).Check(testkit.Rows("1 5 1 2"))
tk1.MustExec(`commit;`)
tk1.MustExec("begin pessimistic")
tk2.MustExec(`alter table normal_table modify column c bigint;`)
tk1.MustQuery(`select * from temp_table, normal_table where temp_table.id = normal_table.id for update;`).Check(testkit.Rows())
tk1.MustExec(`commit;`)
// Truncate will modify table ID.
tk1.MustExec(`begin;`)
tk2.MustExec(`truncate table temp_table;`)
tk1.MustExec(`insert into temp_table values(3, 3);`)
tk1.MustExec(`commit;`)
// It reports error when also changing the schema of a normal table.
tk1.MustExec(`begin;`)
tk2.MustExec(`alter table normal_table modify column c bigint;`)
tk1.MustExec(`insert into temp_table values(3, 3);`)
tk1.MustExec(`insert into normal_table values(3, 3);`)
err := tk1.ExecToErr(`commit;`)
require.True(t, terror.ErrorEqual(err, domain.ErrInfoSchemaChanged), fmt.Sprintf("err %v", err))
tk1.MustExec("begin pessimistic")
tk2.MustExec(`alter table normal_table modify column c int;`)
tk1.MustExec(`insert into temp_table values(1, 6);`)
tk1.MustQuery(`select * from temp_table, normal_table where temp_table.id = normal_table.id for update;`).Check(testkit.Rows("1 6 1 2"))
err = tk1.ExecToErr(`commit;`)
require.True(t, terror.ErrorEqual(err, domain.ErrInfoSchemaChanged), fmt.Sprintf("err %v", err))
}