Skip to content

Commit

Permalink
This is an automated cherry-pick of pingcap#57964
Browse files Browse the repository at this point in the history
Signed-off-by: ti-chi-bot <[email protected]>
  • Loading branch information
wjhuang2016 authored and ti-chi-bot committed Dec 9, 2024
1 parent 1e2bb37 commit be0ea47
Show file tree
Hide file tree
Showing 3 changed files with 129 additions and 9 deletions.
3 changes: 2 additions & 1 deletion pkg/ddl/tests/metadatalock/BUILD.bazel
Original file line number Diff line number Diff line change
Expand Up @@ -8,12 +8,13 @@ go_test(
"mdl_test.go",
],
flaky = True,
shard_count = 36,
shard_count = 37,
deps = [
"//pkg/config",
"//pkg/ddl",
"//pkg/ddl/ingest/testutil",
"//pkg/errno",
"//pkg/meta/model",
"//pkg/server",
"//pkg/testkit",
"//pkg/testkit/testsetup",
Expand Down
103 changes: 103 additions & 0 deletions pkg/ddl/tests/metadatalock/mdl_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -15,14 +15,20 @@
package metadatalocktest

import (
"context"
"fmt"
"sync"
"testing"
"time"

"github.com/pingcap/failpoint"
<<<<<<< HEAD
=======
"github.com/pingcap/tidb/pkg/ddl"
>>>>>>> 6b17068d75f (planner: fix incorrectly using the schema for plan cache (#57964))
ingesttestutil "github.com/pingcap/tidb/pkg/ddl/ingest/testutil"
mysql "github.com/pingcap/tidb/pkg/errno"
"github.com/pingcap/tidb/pkg/meta/model"
"github.com/pingcap/tidb/pkg/server"
"github.com/pingcap/tidb/pkg/testkit"
"github.com/stretchr/testify/require"
Expand Down Expand Up @@ -997,6 +1003,103 @@ func TestMDLPreparePlanCacheExecute2(t *testing.T) {
tk.MustExec("admin check table t")
}

// TestMDLPreparePlanCacheExecuteInsert makes sure the insert statement handle the schema correctly in plan cache.
func TestMDLPreparePlanCacheExecuteInsert(t *testing.T) {
store, dom := testkit.CreateMockStoreAndDomain(t)
defer ingesttestutil.InjectMockBackendMgr(t, store)()

sv := server.CreateMockServer(t, store)

sv.SetDomain(dom)
dom.InfoSyncer().SetSessionManager(sv)
defer sv.Close()

conn1 := server.CreateMockConn(t, sv)
tk := testkit.NewTestKitWithSession(t, store, conn1.Context().Session)
conn2 := server.CreateMockConn(t, sv)
tkDDL := testkit.NewTestKitWithSession(t, store, conn2.Context().Session)
conn3 := server.CreateMockConn(t, sv)
tk3 := testkit.NewTestKitWithSession(t, store, conn3.Context().Session)
tk.MustExec("use test")
tk.MustExec("set global tidb_enable_metadata_lock=1")
tk.MustExec("create table t(a int primary key, b int);")
tk.MustExec("create table t2(a int);")
tk.MustExec("insert into t values(1, 1), (2, 2), (3, 3), (4, 4);")

tk.MustExec(`begin`)
tk.MustExec(`prepare delete_stmt from 'delete from t where a = ?'`)
tk.MustExec(`prepare insert_stmt from 'insert into t values (?, ?)'`)
tk.MustExec(`commit`)

tk.MustExec(`begin`)
tk.MustExec(`set @a = 4, @b= 4;`)
tk.MustExec(`execute delete_stmt using @a;`)
tk.MustExec(`execute insert_stmt using @a, @b;`)
tk.MustExec(`commit`)

tk.MustExec("begin")

ch := make(chan struct{})

first := true
testfailpoint.EnableCall(t, "github.com/pingcap/tidb/pkg/ddl/onJobUpdated", func(job *model.Job) {
switch job.SchemaState {
case model.StateWriteReorganization:
tbl, _ := dom.InfoSchema().TableByID(context.Background(), job.TableID)
idx := tbl.Meta().FindIndexByName("idx")
switch idx.BackfillState {
case model.BackfillStateRunning:
if first {
tk.MustExec(`begin`)
tk.MustExec(`set @a=9;`)
tk.MustExec(`execute delete_stmt using @a;`)
tk.MustQuery("select @@last_plan_from_cache;").Check(testkit.Rows("0"))
tk.MustExec(`set @a=6, @b=4;`)
tk.MustExec(`execute insert_stmt using @a, @b;`)
tk.MustQuery("select @@last_plan_from_cache;").Check(testkit.Rows("0"))
tk.MustExec(`commit`)
tk.MustExec(`begin`)
tk.MustExec(`set @a=4;`)
tk.MustExec(`execute delete_stmt using @a;`)
tk.MustQuery("select @@last_plan_from_cache;").Check(testkit.Rows("1"))
tk.MustExec(`set @a=4, @b=4;`)
tk.MustExec(`execute insert_stmt using @a, @b;`)
tk.MustQuery("select @@last_plan_from_cache;").Check(testkit.Rows("0"))
tk.MustExec(`commit`)

tk.MustExec("begin")
// Activate txn.
tk.MustExec("select * from t2")
first = false
tk3.MustExec("insert into test.t values(10000, 1000)")
return
}
}
}
})

ddl.MockDMLExecutionMerging = func() {
tk.MustExec(`execute delete_stmt using @a;`)
tk.MustQuery("select @@last_plan_from_cache;").Check(testkit.Rows("0"))
tk.MustExec(`execute insert_stmt using @a, @b;`)
tk.MustExec("commit")
}
require.NoError(t, failpoint.Enable("github.com/pingcap/tidb/pkg/ddl/mockDMLExecutionMerging", "1*return(true)->return(false)"))

var wg sync.WaitGroup
wg.Add(1)
go func() {
<-ch
tkDDL.MustExec("alter table test.t add index idx(a);")
wg.Done()
}()

ch <- struct{}{}
wg.Wait()

tk.MustExec("admin check table t")
}

func TestMDLDisable2Enable(t *testing.T) {
store, dom := testkit.CreateMockStoreAndDomain(t)
sv := server.CreateMockServer(t, store)
Expand Down
32 changes: 24 additions & 8 deletions pkg/planner/core/plan_cache.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,11 @@ package core
import (
"bytes"
"context"
<<<<<<< HEAD
=======
"math"
"time"
>>>>>>> 6b17068d75f (planner: fix incorrectly using the schema for plan cache (#57964))

"github.com/pingcap/errors"
"github.com/pingcap/tidb/pkg/bindinfo"
Expand All @@ -41,9 +46,16 @@ import (
"github.com/pingcap/tidb/pkg/util/chunk"
"github.com/pingcap/tidb/pkg/util/collate"
"github.com/pingcap/tidb/pkg/util/dbterror/plannererrors"
<<<<<<< HEAD
"github.com/pingcap/tidb/pkg/util/kvcache"
utilpc "github.com/pingcap/tidb/pkg/util/plancache"
"github.com/pingcap/tidb/pkg/util/ranger"
=======
"github.com/pingcap/tidb/pkg/util/hint"
"github.com/pingcap/tidb/pkg/util/intest"
"github.com/pingcap/tidb/pkg/util/logutil"
"go.uber.org/zap"
>>>>>>> 6b17068d75f (planner: fix incorrectly using the schema for plan cache (#57964))
)

// PlanCacheKeyTestIssue43667 is only for test.
Expand Down Expand Up @@ -114,23 +126,27 @@ func planCachePreprocess(ctx context.Context, sctx sessionctx.Context, isNonPrep
if err != nil {
return plannererrors.ErrSchemaChanged.GenWithStack("Schema change caused error: %s", err.Error())
}
// Table ID is changed, for example, drop & create table, truncate table.
delete(stmt.RelateVersion, stmt.tbls[i].Meta().ID)
stmt.tbls[i] = tblByName
stmt.RelateVersion[tblByName.Meta().ID] = tblByName.Meta().Revision
tbl = tblByName
}
<<<<<<< HEAD
newTbl, err := tryLockMDLAndUpdateSchemaIfNecessary(sctx.GetPlanCtx(), stmt.dbName[i], stmt.tbls[i], is)
=======
// newTbl is the 'should be used' table info for this execution.
newTbl, err := tryLockMDLAndUpdateSchemaIfNecessary(ctx, sctx.GetPlanCtx(), stmt.dbName[i], tbl, is)
>>>>>>> 6b17068d75f (planner: fix incorrectly using the schema for plan cache (#57964))
if err != nil {
logutil.BgLogger().Warn("meet error during tryLockMDLAndUpdateSchemaIfNecessary", zap.String("table name", tbl.Meta().Name.String()), zap.Error(err))
// Invalid the cache key related fields to avoid using plan cache.
stmt.RelateVersion[tbl.Meta().ID] = math.MaxUint64
schemaNotMatch = true
continue
}
// The revision of tbl and newTbl may not be the same.
// Example:
// The version of stmt.tbls[i] is taken from the prepare statement and is revision v1.
// When stmt.tbls[i] is locked in MDL, the revision of newTbl is also v1.
// The revision of tbl is v2. The reason may have other statements trigger "tryLockMDLAndUpdateSchemaIfNecessary" before, leading to tbl revision update.
if stmt.tbls[i].Meta().Revision != newTbl.Meta().Revision || (tbl != nil && tbl.Meta().Revision != newTbl.Meta().Revision) {
if stmt.tbls[i].Meta().Revision != newTbl.Meta().Revision {
schemaNotMatch = true
}
// Update the cache key related fields.
stmt.tbls[i] = newTbl
stmt.RelateVersion[newTbl.Meta().ID] = newTbl.Meta().Revision
}
Expand Down

0 comments on commit be0ea47

Please sign in to comment.