Files
tidb/executor/seqtest/prepared_test.go
王超 529ce88c1c *: add TxnManager to manage txn in session (#30574)
* *: add TxnManager to manage txn in session

* modify

* add tests

* move failpoint content to a single file
2021-12-22 13:53:43 +08:00

911 lines
32 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,
// 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 executor_test
import (
"context"
"crypto/tls"
"fmt"
"math"
"testing"
"time"
"github.com/pingcap/tidb/executor"
"github.com/pingcap/tidb/infoschema"
"github.com/pingcap/tidb/metrics"
"github.com/pingcap/tidb/parser/mysql"
plannercore "github.com/pingcap/tidb/planner/core"
"github.com/pingcap/tidb/session"
"github.com/pingcap/tidb/session/txninfo"
"github.com/pingcap/tidb/testkit"
"github.com/pingcap/tidb/types"
"github.com/pingcap/tidb/util"
"github.com/pingcap/tidb/util/kvcache"
dto "github.com/prometheus/client_model/go"
"github.com/stretchr/testify/require"
)
func TestPrepared(t *testing.T) {
store, clean := testkit.CreateMockStore(t)
defer clean()
orgEnable := plannercore.PreparedPlanCacheEnabled()
defer func() {
plannercore.SetPreparedPlanCache(orgEnable)
}()
flags := []bool{false, true}
ctx := context.Background()
for _, flag := range flags {
plannercore.SetPreparedPlanCache(flag)
tk := testkit.NewTestKit(t, store)
se, err := session.CreateSession4TestWithOpt(store, &session.Opt{
PreparedPlanCache: kvcache.NewSimpleLRUCache(100, 0.1, math.MaxUint64),
})
require.NoError(t, err)
tk.SetSession(se)
tk.MustExec("use test")
tk.MustExec("drop table if exists prepare_test")
tk.MustExec("create table prepare_test (id int PRIMARY KEY AUTO_INCREMENT, c1 int, c2 int, c3 int default 1)")
tk.MustExec("insert prepare_test (c1) values (1),(2),(NULL)")
tk.MustExec(`prepare stmt_test_1 from 'select id from prepare_test where id > ?';`)
tk.MustExec(`set @a = 1;`)
tk.MustExec(`execute stmt_test_1 using @a;`)
tk.MustExec(`prepare stmt_test_2 from 'select 1'`)
// Prepare multiple statement is not allowed.
_, err = tk.Exec(`prepare stmt_test_3 from 'select id from prepare_test where id > ?;select id from prepare_test where id > ?;'`)
require.True(t, executor.ErrPrepareMulti.Equal(err))
// The variable count does not match.
tk.MustExec(`prepare stmt_test_4 from 'select id from prepare_test where id > ? and id < ?';`)
tk.MustExec(`set @a = 1;`)
_, err = tk.Exec(`execute stmt_test_4 using @a;`)
require.True(t, plannercore.ErrWrongParamCount.Equal(err))
// Prepare and deallocate prepared statement immediately.
tk.MustExec(`prepare stmt_test_5 from 'select id from prepare_test where id > ?';`)
tk.MustExec(`deallocate prepare stmt_test_5;`)
// Statement not found.
_, err = tk.Exec("deallocate prepare stmt_test_5")
require.True(t, plannercore.ErrStmtNotFound.Equal(err))
// incorrect SQLs in prepare. issue #3738, SQL in prepare stmt is parsed in DoPrepare.
_, err = tk.Exec(`prepare p from "delete from t where a = 7 or 1=1/*' and b = 'p'";`)
require.EqualError(t, err, `[parser:1064]You have an error in your SQL syntax; check the manual that corresponds to your TiDB version for the right syntax to use near '/*' and b = 'p'' at line 1`)
// The `stmt_test5` should not be found.
_, err = tk.Exec(`set @a = 1; execute stmt_test_5 using @a;`)
require.True(t, plannercore.ErrStmtNotFound.Equal(err))
// Use parameter marker with argument will run prepared statement.
result := tk.MustQuery("select distinct c1, c2 from prepare_test where c1 = ?", 1)
result.Check(testkit.Rows("1 <nil>"))
// Call Session PrepareStmt directly to get stmtID.
query := "select c1, c2 from prepare_test where c1 = ?"
stmtID, _, _, err := tk.Session().PrepareStmt(query)
require.NoError(t, err)
rs, err := tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(1)})
require.NoError(t, err)
tk.ResultSetToResult(rs, fmt.Sprintf("%v", rs)).Check(testkit.Rows("1 <nil>"))
tk.MustExec("delete from prepare_test")
query = "select c1 from prepare_test where c1 = (select c1 from prepare_test where c1 = ?)"
stmtID, _, _, err = tk.Session().PrepareStmt(query)
require.NoError(t, err)
tk1 := testkit.NewTestKit(t, store)
se, err = session.CreateSession4TestWithOpt(store, &session.Opt{
PreparedPlanCache: kvcache.NewSimpleLRUCache(100, 0.1, math.MaxUint64),
})
require.NoError(t, err)
tk1.SetSession(se)
tk1.MustExec("use test")
tk1.MustExec("insert prepare_test (c1) values (3)")
rs, err = tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(3)})
require.NoError(t, err)
tk.ResultSetToResult(rs, fmt.Sprintf("%v", rs)).Check(testkit.Rows("3"))
tk.MustExec("delete from prepare_test")
query = "select c1 from prepare_test where c1 = (select c1 from prepare_test where c1 = ?)"
stmtID, _, _, err = tk.Session().PrepareStmt(query)
require.NoError(t, err)
rs, err = tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(3)})
require.NoError(t, err)
require.NoError(t, rs.Close())
tk1.MustExec("insert prepare_test (c1) values (3)")
rs, err = tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(3)})
require.NoError(t, err)
tk.ResultSetToResult(rs, fmt.Sprintf("%v", rs)).Check(testkit.Rows("3"))
tk.MustExec("delete from prepare_test")
query = "select c1 from prepare_test where c1 in (select c1 from prepare_test where c1 = ?)"
stmtID, _, _, err = tk.Session().PrepareStmt(query)
require.NoError(t, err)
rs, err = tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(3)})
require.NoError(t, err)
require.NoError(t, rs.Close())
tk1.MustExec("insert prepare_test (c1) values (3)")
rs, err = tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(3)})
require.NoError(t, err)
tk.ResultSetToResult(rs, fmt.Sprintf("%v", rs)).Check(testkit.Rows("3"))
tk.MustExec("begin")
tk.MustExec("insert prepare_test (c1) values (4)")
query = "select c1, c2 from prepare_test where c1 = ?"
stmtID, _, _, err = tk.Session().PrepareStmt(query)
require.NoError(t, err)
tk.MustExec("rollback")
rs, err = tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(4)})
require.NoError(t, err)
tk.ResultSetToResult(rs, fmt.Sprintf("%v", rs)).Check(testkit.Rows())
// Check that ast.Statement created by executor.CompileExecutePreparedStmt has query text.
stmt, _, _, err := executor.CompileExecutePreparedStmt(context.TODO(), tk.Session(), stmtID,
tk.Session().GetInfoSchema().(infoschema.InfoSchema), 0, []types.Datum{types.NewDatum(1)})
require.NoError(t, err)
require.Equal(t, query, stmt.OriginText())
// Check that rebuild plan works.
err = tk.Session().PrepareTxnCtx(ctx)
require.NoError(t, err)
_, err = stmt.RebuildPlan(ctx)
require.NoError(t, err)
rs, err = stmt.Exec(ctx)
require.NoError(t, err)
req := rs.NewChunk(nil)
err = rs.Next(ctx, req)
require.NoError(t, err)
require.NoError(t, rs.Close())
// Make schema change.
tk.MustExec("drop table if exists prepare2")
_, err = tk.Exec("create table prepare2 (a int)")
require.NoError(t, err)
// Should success as the changed schema do not affect the prepared statement.
rs, err = tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(1)})
require.NoError(t, err)
if rs != nil {
require.NoError(t, rs.Close())
}
// Drop a column so the prepared statement become invalid.
query = "select c1, c2 from prepare_test where c1 = ?"
stmtID, _, _, err = tk.Session().PrepareStmt(query)
require.NoError(t, err)
tk.MustExec("alter table prepare_test drop column c2")
_, err = tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(1)})
require.True(t, plannercore.ErrUnknownColumn.Equal(err))
tk.MustExec("drop table prepare_test")
_, err = tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(1)})
require.True(t, plannercore.ErrSchemaChanged.Equal(err))
// issue 3381
tk.MustExec("drop table if exists prepare3")
tk.MustExec("create table prepare3 (a decimal(1))")
tk.MustExec("prepare stmt from 'insert into prepare3 value(123)'")
_, err = tk.Exec("execute stmt")
require.Error(t, err)
_, _, fields, err := tk.Session().PrepareStmt("select a from prepare3")
require.NoError(t, err)
require.Equal(t, "test", fields[0].DBName.L)
require.Equal(t, "prepare3", fields[0].TableAsName.L)
require.Equal(t, "a", fields[0].ColumnAsName.L)
_, _, fields, err = tk.Session().PrepareStmt("select a from prepare3 where ?")
require.NoError(t, err)
require.Equal(t, "test", fields[0].DBName.L)
require.Equal(t, "prepare3", fields[0].TableAsName.L)
require.Equal(t, "a", fields[0].ColumnAsName.L)
_, _, fields, err = tk.Session().PrepareStmt("select (1,1) in (select 1,1)")
require.NoError(t, err)
require.Equal(t, "", fields[0].DBName.L)
require.Equal(t, "", fields[0].TableAsName.L)
require.Equal(t, "(1,1) in (select 1,1)", fields[0].ColumnAsName.L)
_, _, fields, err = tk.Session().PrepareStmt("select a from prepare3 where a = (" +
"select a from prepare2 where a = ?)")
require.NoError(t, err)
require.Equal(t, "test", fields[0].DBName.L)
require.Equal(t, "prepare3", fields[0].TableAsName.L)
require.Equal(t, "a", fields[0].ColumnAsName.L)
_, _, fields, err = tk.Session().PrepareStmt("select * from prepare3 as t1 join prepare3 as t2")
require.NoError(t, err)
require.Equal(t, "test", fields[0].DBName.L)
require.Equal(t, "t1", fields[0].TableAsName.L)
require.Equal(t, "a", fields[0].ColumnAsName.L)
require.Equal(t, "test", fields[1].DBName.L)
require.Equal(t, "t2", fields[1].TableAsName.L)
require.Equal(t, "a", fields[1].ColumnAsName.L)
_, _, fields, err = tk.Session().PrepareStmt("update prepare3 set a = ?")
require.NoError(t, err)
require.Len(t, fields, 0)
// issue 8074
tk.MustExec("drop table if exists prepare1;")
tk.MustExec("create table prepare1 (a decimal(1))")
tk.MustExec("insert into prepare1 values(1);")
_, err = tk.Exec("prepare stmt FROM @sql1")
require.EqualError(t, err, "[parser:1064]You have an error in your SQL syntax; check the manual that corresponds to your TiDB version for the right syntax to use line 1 column 4 near \"NULL\" ")
tk.MustExec("SET @sql = 'update prepare1 set a=5 where a=?';")
_, err = tk.Exec("prepare stmt FROM @sql")
require.NoError(t, err)
tk.MustExec("set @var=1;")
_, err = tk.Exec("execute stmt using @var")
require.NoError(t, err)
tk.MustQuery("select a from prepare1;").Check(testkit.Rows("5"))
// issue 19371
tk.MustExec("SET @sql = 'update prepare1 set a=a+1';")
_, err = tk.Exec("prepare stmt FROM @SQL")
require.NoError(t, err)
_, err = tk.Exec("execute stmt")
require.NoError(t, err)
tk.MustQuery("select a from prepare1;").Check(testkit.Rows("6"))
_, err = tk.Exec("prepare stmt FROM @Sql")
require.NoError(t, err)
_, err = tk.Exec("execute stmt")
require.NoError(t, err)
tk.MustQuery("select a from prepare1;").Check(testkit.Rows("7"))
// Coverage.
exec := &executor.ExecuteExec{}
err = exec.Next(ctx, nil)
require.NoError(t, err)
err = exec.Close()
require.NoError(t, err)
// issue 8065
stmtID, _, _, err = tk.Session().PrepareStmt("select ? from dual")
require.NoError(t, err)
_, err = tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(1)})
require.NoError(t, err)
stmtID, _, _, err = tk.Session().PrepareStmt("update prepare1 set a = ? where a = ?")
require.NoError(t, err)
_, err = tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(1), types.NewDatum(1)})
require.NoError(t, err)
}
}
func TestPreparedLimitOffset(t *testing.T) {
store, clean := testkit.CreateMockStore(t)
defer clean()
orgEnable := plannercore.PreparedPlanCacheEnabled()
defer func() {
plannercore.SetPreparedPlanCache(orgEnable)
}()
flags := []bool{false, true}
ctx := context.Background()
for _, flag := range flags {
plannercore.SetPreparedPlanCache(flag)
tk := testkit.NewTestKit(t, store)
se, err := session.CreateSession4TestWithOpt(store, &session.Opt{
PreparedPlanCache: kvcache.NewSimpleLRUCache(100, 0.1, math.MaxUint64),
})
require.NoError(t, err)
tk.SetSession(se)
tk.MustExec("use test")
tk.MustExec("drop table if exists prepare_test")
tk.MustExec("create table prepare_test (id int PRIMARY KEY AUTO_INCREMENT, c1 int, c2 int, c3 int default 1)")
tk.MustExec("insert prepare_test (c1) values (1),(2),(NULL)")
tk.MustExec(`prepare stmt_test_1 from 'select id from prepare_test limit ? offset ?'; set @a = 1, @b=1;`)
r := tk.MustQuery(`execute stmt_test_1 using @a, @b;`)
r.Check(testkit.Rows("2"))
tk.MustExec(`set @a=1.1`)
r = tk.MustQuery(`execute stmt_test_1 using @a, @b;`)
r.Check(testkit.Rows("2"))
tk.MustExec(`set @c="-1"`)
_, err = tk.Exec("execute stmt_test_1 using @c, @c")
require.True(t, plannercore.ErrWrongArguments.Equal(err))
stmtID, _, _, err := tk.Session().PrepareStmt("select id from prepare_test limit ?")
require.NoError(t, err)
_, err = tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(1)})
require.NoError(t, err)
}
}
func TestPreparedNullParam(t *testing.T) {
store, clean := testkit.CreateMockStore(t)
defer clean()
orgEnable := plannercore.PreparedPlanCacheEnabled()
defer func() {
plannercore.SetPreparedPlanCache(orgEnable)
}()
flags := []bool{false, true}
for _, flag := range flags {
plannercore.SetPreparedPlanCache(flag)
tk := testkit.NewTestKit(t, store)
se, err := session.CreateSession4TestWithOpt(store, &session.Opt{
PreparedPlanCache: kvcache.NewSimpleLRUCache(100, 0.1, math.MaxUint64),
})
require.NoError(t, err)
tk.SetSession(se)
tk.MustExec("use test")
tk.MustExec("drop table if exists t")
tk.MustExec("create table t (id int, KEY id (id))")
tk.MustExec("insert into t values (1), (2), (3)")
tk.MustExec(`prepare stmt from 'select * from t use index(id) where id = ?'`)
r := tk.MustQuery(`execute stmt using @id;`)
r.Check(nil)
r = tk.MustQuery(`execute stmt using @id;`)
r.Check(nil)
tk.MustExec(`set @id="1"`)
r = tk.MustQuery(`execute stmt using @id;`)
r.Check(testkit.Rows("1"))
r = tk.MustQuery(`execute stmt using @id2;`)
r.Check(nil)
r = tk.MustQuery(`execute stmt using @id;`)
r.Check(testkit.Rows("1"))
}
}
func TestPrepareWithAggregation(t *testing.T) {
store, clean := testkit.CreateMockStore(t)
defer clean()
orgEnable := plannercore.PreparedPlanCacheEnabled()
defer func() {
plannercore.SetPreparedPlanCache(orgEnable)
}()
flags := []bool{false, true}
for _, flag := range flags {
plannercore.SetPreparedPlanCache(flag)
tk := testkit.NewTestKit(t, store)
se, err := session.CreateSession4TestWithOpt(store, &session.Opt{
PreparedPlanCache: kvcache.NewSimpleLRUCache(100, 0.1, math.MaxUint64),
})
require.NoError(t, err)
tk.SetSession(se)
tk.MustExec("use test")
tk.MustExec("drop table if exists t")
tk.MustExec("create table t (id int primary key)")
tk.MustExec("insert into t values (1), (2), (3)")
tk.MustExec(`prepare stmt from 'select sum(id) from t where id = ?'`)
tk.MustExec(`set @id="1"`)
r := tk.MustQuery(`execute stmt using @id;`)
r.Check(testkit.Rows("1"))
r = tk.MustQuery(`execute stmt using @id;`)
r.Check(testkit.Rows("1"))
}
}
func TestPreparedIssue7579(t *testing.T) {
store, clean := testkit.CreateMockStore(t)
defer clean()
orgEnable := plannercore.PreparedPlanCacheEnabled()
defer func() {
plannercore.SetPreparedPlanCache(orgEnable)
}()
flags := []bool{false, true}
for _, flag := range flags {
plannercore.SetPreparedPlanCache(flag)
tk := testkit.NewTestKit(t, store)
se, err := session.CreateSession4TestWithOpt(store, &session.Opt{
PreparedPlanCache: kvcache.NewSimpleLRUCache(100, 0.1, math.MaxUint64),
})
require.NoError(t, err)
tk.SetSession(se)
tk.MustExec("use test")
tk.MustExec("drop table if exists t")
tk.MustExec("create table t (a int, b int, index a_idx(a))")
tk.MustExec("insert into t values (1,1), (2,2), (null,3)")
r := tk.MustQuery("select a, b from t order by b asc;")
r.Check(testkit.Rows("1 1", "2 2", "<nil> 3"))
tk.MustExec(`prepare stmt from 'select a, b from t where ? order by b asc'`)
r = tk.MustQuery(`execute stmt using @param;`)
r.Check(nil)
tk.MustExec(`set @param = true`)
r = tk.MustQuery(`execute stmt using @param;`)
r.Check(testkit.Rows("1 1", "2 2", "<nil> 3"))
tk.MustExec(`set @param = false`)
r = tk.MustQuery(`execute stmt using @param;`)
r.Check(nil)
tk.MustExec(`set @param = 1`)
r = tk.MustQuery(`execute stmt using @param;`)
r.Check(testkit.Rows("1 1", "2 2", "<nil> 3"))
tk.MustExec(`set @param = 0`)
r = tk.MustQuery(`execute stmt using @param;`)
r.Check(nil)
}
}
func TestPreparedInsert(t *testing.T) {
store, clean := testkit.CreateMockStore(t)
defer clean()
orgEnable := plannercore.PreparedPlanCacheEnabled()
defer func() {
plannercore.SetPreparedPlanCache(orgEnable)
}()
metrics.ResettablePlanCacheCounterFortTest = true
metrics.PlanCacheCounter.Reset()
counter := metrics.PlanCacheCounter.WithLabelValues("prepare")
pb := &dto.Metric{}
flags := []bool{false, true}
for _, flag := range flags {
plannercore.SetPreparedPlanCache(flag)
tk := testkit.NewTestKit(t, store)
se, err := session.CreateSession4TestWithOpt(store, &session.Opt{
PreparedPlanCache: kvcache.NewSimpleLRUCache(100, 0.1, math.MaxUint64),
})
require.NoError(t, err)
tk.SetSession(se)
tk.MustExec("use test")
tk.MustExec("drop table if exists prepare_test")
tk.MustExec("create table prepare_test (id int PRIMARY KEY, c1 int)")
tk.MustExec(`prepare stmt_insert from 'insert into prepare_test values (?, ?)'`)
tk.MustExec(`set @a=1,@b=1; execute stmt_insert using @a, @b;`)
if flag {
err = counter.Write(pb)
require.NoError(t, err)
hit := pb.GetCounter().GetValue()
require.Equal(t, float64(0), hit)
}
tk.MustExec(`set @a=2,@b=2; execute stmt_insert using @a, @b;`)
if flag {
err = counter.Write(pb)
require.NoError(t, err)
hit := pb.GetCounter().GetValue()
require.Equal(t, float64(1), hit)
}
tk.MustExec(`set @a=3,@b=3; execute stmt_insert using @a, @b;`)
if flag {
err = counter.Write(pb)
require.NoError(t, err)
hit := pb.GetCounter().GetValue()
require.Equal(t, float64(2), hit)
}
result := tk.MustQuery("select id, c1 from prepare_test where id = ?", 1)
result.Check(testkit.Rows("1 1"))
result = tk.MustQuery("select id, c1 from prepare_test where id = ?", 2)
result.Check(testkit.Rows("2 2"))
result = tk.MustQuery("select id, c1 from prepare_test where id = ?", 3)
result.Check(testkit.Rows("3 3"))
tk.MustExec(`prepare stmt_insert_select from 'insert into prepare_test (id, c1) select id + 100, c1 + 100 from prepare_test where id = ?'`)
tk.MustExec(`set @a=1; execute stmt_insert_select using @a;`)
if flag {
err = counter.Write(pb)
require.NoError(t, err)
hit := pb.GetCounter().GetValue()
require.Equal(t, float64(2), hit)
}
tk.MustExec(`set @a=2; execute stmt_insert_select using @a;`)
if flag {
err = counter.Write(pb)
require.NoError(t, err)
hit := pb.GetCounter().GetValue()
require.Equal(t, float64(3), hit)
}
tk.MustExec(`set @a=3; execute stmt_insert_select using @a;`)
if flag {
err = counter.Write(pb)
require.NoError(t, err)
hit := pb.GetCounter().GetValue()
require.Equal(t, float64(4), hit)
}
result = tk.MustQuery("select id, c1 from prepare_test where id = ?", 101)
result.Check(testkit.Rows("101 101"))
result = tk.MustQuery("select id, c1 from prepare_test where id = ?", 102)
result.Check(testkit.Rows("102 102"))
result = tk.MustQuery("select id, c1 from prepare_test where id = ?", 103)
result.Check(testkit.Rows("103 103"))
}
}
func TestPreparedUpdate(t *testing.T) {
store, clean := testkit.CreateMockStore(t)
defer clean()
orgEnable := plannercore.PreparedPlanCacheEnabled()
defer func() {
plannercore.SetPreparedPlanCache(orgEnable)
}()
metrics.ResettablePlanCacheCounterFortTest = true
metrics.PlanCacheCounter.Reset()
counter := metrics.PlanCacheCounter.WithLabelValues("prepare")
pb := &dto.Metric{}
flags := []bool{false, true}
for _, flag := range flags {
plannercore.SetPreparedPlanCache(flag)
tk := testkit.NewTestKit(t, store)
se, err := session.CreateSession4TestWithOpt(store, &session.Opt{
PreparedPlanCache: kvcache.NewSimpleLRUCache(100, 0.1, math.MaxUint64),
})
require.NoError(t, err)
tk.SetSession(se)
tk.MustExec("use test")
tk.MustExec("drop table if exists prepare_test")
tk.MustExec("create table prepare_test (id int PRIMARY KEY, c1 int)")
tk.MustExec(`insert into prepare_test values (1, 1)`)
tk.MustExec(`insert into prepare_test values (2, 2)`)
tk.MustExec(`insert into prepare_test values (3, 3)`)
tk.MustExec(`prepare stmt_update from 'update prepare_test set c1 = c1 + ? where id = ?'`)
tk.MustExec(`set @a=1,@b=100; execute stmt_update using @b,@a;`)
if flag {
err = counter.Write(pb)
require.NoError(t, err)
hit := pb.GetCounter().GetValue()
require.Equal(t, float64(0), hit)
}
tk.MustExec(`set @a=2,@b=200; execute stmt_update using @b,@a;`)
if flag {
err = counter.Write(pb)
require.NoError(t, err)
hit := pb.GetCounter().GetValue()
require.Equal(t, float64(1), hit)
}
tk.MustExec(`set @a=3,@b=300; execute stmt_update using @b,@a;`)
if flag {
err = counter.Write(pb)
require.NoError(t, err)
hit := pb.GetCounter().GetValue()
require.Equal(t, float64(2), hit)
}
result := tk.MustQuery("select id, c1 from prepare_test where id = ?", 1)
result.Check(testkit.Rows("1 101"))
result = tk.MustQuery("select id, c1 from prepare_test where id = ?", 2)
result.Check(testkit.Rows("2 202"))
result = tk.MustQuery("select id, c1 from prepare_test where id = ?", 3)
result.Check(testkit.Rows("3 303"))
}
}
func TestIssue21884(t *testing.T) {
store, clean := testkit.CreateMockStore(t)
defer clean()
orgEnable := plannercore.PreparedPlanCacheEnabled()
defer func() {
plannercore.SetPreparedPlanCache(orgEnable)
}()
plannercore.SetPreparedPlanCache(false)
tk := testkit.NewTestKit(t, store)
tk.MustExec("use test")
tk.MustExec("drop table if exists prepare_test")
tk.MustExec("create table prepare_test(a bigint primary key, status bigint, last_update_time datetime)")
tk.MustExec("insert into prepare_test values (100, 0, '2020-12-18 20:00:00')")
tk.MustExec("prepare stmt from 'update prepare_test set status = ?, last_update_time = now() where a = 100'")
tk.MustExec("set @status = 1")
tk.MustExec("execute stmt using @status")
updateTime := tk.MustQuery("select last_update_time from prepare_test").Rows()[0][0]
// Sleep 1 second to make sure `last_update_time` is updated.
time.Sleep(1 * time.Second)
tk.MustExec("execute stmt using @status")
newUpdateTime := tk.MustQuery("select last_update_time from prepare_test").Rows()[0][0]
require.NotEqual(t, newUpdateTime, updateTime)
}
func TestPreparedDelete(t *testing.T) {
store, clean := testkit.CreateMockStore(t)
defer clean()
orgEnable := plannercore.PreparedPlanCacheEnabled()
defer func() {
plannercore.SetPreparedPlanCache(orgEnable)
}()
metrics.ResettablePlanCacheCounterFortTest = true
metrics.PlanCacheCounter.Reset()
counter := metrics.PlanCacheCounter.WithLabelValues("prepare")
pb := &dto.Metric{}
flags := []bool{false, true}
for _, flag := range flags {
plannercore.SetPreparedPlanCache(flag)
tk := testkit.NewTestKit(t, store)
se, err := session.CreateSession4TestWithOpt(store, &session.Opt{
PreparedPlanCache: kvcache.NewSimpleLRUCache(100, 0.1, math.MaxUint64),
})
require.NoError(t, err)
tk.SetSession(se)
tk.MustExec("use test")
tk.MustExec("drop table if exists prepare_test")
tk.MustExec("create table prepare_test (id int PRIMARY KEY, c1 int)")
tk.MustExec(`insert into prepare_test values (1, 1)`)
tk.MustExec(`insert into prepare_test values (2, 2)`)
tk.MustExec(`insert into prepare_test values (3, 3)`)
tk.MustExec(`prepare stmt_delete from 'delete from prepare_test where id = ?'`)
tk.MustExec(`set @a=1; execute stmt_delete using @a;`)
if flag {
err = counter.Write(pb)
require.NoError(t, err)
hit := pb.GetCounter().GetValue()
require.Equal(t, float64(0), hit)
}
tk.MustExec(`set @a=2; execute stmt_delete using @a;`)
if flag {
err = counter.Write(pb)
require.NoError(t, err)
hit := pb.GetCounter().GetValue()
require.Equal(t, float64(1), hit)
}
tk.MustExec(`set @a=3; execute stmt_delete using @a;`)
if flag {
err = counter.Write(pb)
require.NoError(t, err)
hit := pb.GetCounter().GetValue()
require.Equal(t, float64(2), hit)
}
result := tk.MustQuery("select id, c1 from prepare_test where id = ?", 1)
result.Check(nil)
result = tk.MustQuery("select id, c1 from prepare_test where id = ?", 2)
result.Check(nil)
result = tk.MustQuery("select id, c1 from prepare_test where id = ?", 3)
result.Check(nil)
}
}
func TestPrepareDealloc(t *testing.T) {
store, clean := testkit.CreateMockStore(t)
defer clean()
orgEnable := plannercore.PreparedPlanCacheEnabled()
defer func() {
plannercore.SetPreparedPlanCache(orgEnable)
}()
plannercore.SetPreparedPlanCache(true)
tk := testkit.NewTestKit(t, store)
se, err := session.CreateSession4TestWithOpt(store, &session.Opt{
PreparedPlanCache: kvcache.NewSimpleLRUCache(3, 0.1, math.MaxUint64),
})
require.NoError(t, err)
tk.SetSession(se)
tk.MustExec("use test")
tk.MustExec("drop table if exists prepare_test")
tk.MustExec("create table prepare_test (id int PRIMARY KEY, c1 int)")
require.Equal(t, 0, tk.Session().PreparedPlanCache().Size())
tk.MustExec(`prepare stmt1 from 'select * from prepare_test'`)
tk.MustExec("execute stmt1")
tk.MustExec(`prepare stmt2 from 'select * from prepare_test'`)
tk.MustExec("execute stmt2")
tk.MustExec(`prepare stmt3 from 'select * from prepare_test'`)
tk.MustExec("execute stmt3")
tk.MustExec(`prepare stmt4 from 'select * from prepare_test'`)
tk.MustExec("execute stmt4")
require.Equal(t, 3, tk.Session().PreparedPlanCache().Size())
tk.MustExec("deallocate prepare stmt1")
require.Equal(t, 3, tk.Session().PreparedPlanCache().Size())
tk.MustExec("deallocate prepare stmt2")
tk.MustExec("deallocate prepare stmt3")
tk.MustExec("deallocate prepare stmt4")
require.Equal(t, 0, tk.Session().PreparedPlanCache().Size())
}
func TestPreparedIssue8153(t *testing.T) {
store, clean := testkit.CreateMockStore(t)
defer clean()
orgEnable := plannercore.PreparedPlanCacheEnabled()
defer func() {
plannercore.SetPreparedPlanCache(orgEnable)
}()
flags := []bool{false, true}
for _, flag := range flags {
plannercore.SetPreparedPlanCache(flag)
tk := testkit.NewTestKit(t, store)
se, err := session.CreateSession4TestWithOpt(store, &session.Opt{
PreparedPlanCache: kvcache.NewSimpleLRUCache(100, 0.1, math.MaxUint64),
})
require.NoError(t, err)
tk.SetSession(se)
tk.MustExec("use test")
tk.MustExec("drop table if exists t")
tk.MustExec("create table t (a int, b int)")
tk.MustExec("insert into t (a, b) values (1,3), (2,2), (3,1)")
tk.MustExec(`prepare stmt from 'select * from t order by ? asc'`)
r := tk.MustQuery(`execute stmt using @param;`)
r.Check(testkit.Rows("1 3", "2 2", "3 1"))
tk.MustExec(`set @param = 1`)
r = tk.MustQuery(`execute stmt using @param;`)
r.Check(testkit.Rows("1 3", "2 2", "3 1"))
tk.MustExec(`set @param = 2`)
r = tk.MustQuery(`execute stmt using @param;`)
r.Check(testkit.Rows("3 1", "2 2", "1 3"))
tk.MustExec(`set @param = 3`)
_, err = tk.Exec(`execute stmt using @param;`)
require.EqualError(t, err, "[planner:1054]Unknown column '?' in 'order clause'")
tk.MustExec(`set @param = '##'`)
r = tk.MustQuery(`execute stmt using @param;`)
r.Check(testkit.Rows("1 3", "2 2", "3 1"))
tk.MustExec("insert into t (a, b) values (1,1), (1,2), (2,1), (2,3), (3,2), (3,3)")
tk.MustExec(`prepare stmt from 'select ?, sum(a) from t group by ?'`)
tk.MustExec(`set @a=1,@b=1`)
r = tk.MustQuery(`execute stmt using @a,@b;`)
r.Check(testkit.Rows("1 18"))
tk.MustExec(`set @a=1,@b=2`)
_, err = tk.Exec(`execute stmt using @a,@b;`)
require.EqualError(t, err, "[planner:1056]Can't group on 'sum(a)'")
}
}
func TestPreparedIssue8644(t *testing.T) {
store, clean := testkit.CreateMockStore(t)
defer clean()
orgEnable := plannercore.PreparedPlanCacheEnabled()
defer func() {
plannercore.SetPreparedPlanCache(orgEnable)
}()
flags := []bool{false, true}
for _, flag := range flags {
plannercore.SetPreparedPlanCache(flag)
tk := testkit.NewTestKit(t, store)
se, err := session.CreateSession4TestWithOpt(store, &session.Opt{
PreparedPlanCache: kvcache.NewSimpleLRUCache(100, 0.1, math.MaxUint64),
})
require.NoError(t, err)
tk.SetSession(se)
tk.MustExec("use test")
tk.MustExec("drop table if exists t")
tk.MustExec("create table t(data mediumblob)")
tk.MustExec(`prepare stmt from 'insert t (data) values (?)'`)
tk.MustExec(`set @a = 'a'`)
tk.MustExec(`execute stmt using @a;`)
tk.MustExec(`set @b = 'aaaaaaaaaaaaaaaaaa'`)
tk.MustExec(`execute stmt using @b;`)
r := tk.MustQuery(`select * from t`)
r.Check(testkit.Rows("a", "aaaaaaaaaaaaaaaaaa"))
tk.MustExec("drop table if exists t")
tk.MustExec("create table t(data decimal)")
tk.MustExec(`prepare stmt from 'insert t (data) values (?)'`)
tk.MustExec(`set @a = '1'`)
tk.MustExec(`execute stmt using @a;`)
tk.MustExec(`set @b = '11111.11111'`) // '.11111' will be truncated.
tk.MustExec(`execute stmt using @b;`)
r = tk.MustQuery(`select * from t`)
r.Check(testkit.Rows("1", "11111"))
tk.MustExec("drop table if exists t")
tk.MustExec("create table t(data decimal(10,3));")
tk.MustExec("prepare stmt from 'insert t (data) values (?)';")
tk.MustExec("set @a = 1.1;")
tk.MustExec("execute stmt using @a;")
tk.MustExec("set @b = 11.11;")
tk.MustExec("execute stmt using @b;")
r = tk.MustQuery(`select * from t`)
r.Check(testkit.Rows("1.100", "11.110"))
}
}
// mockSessionManager is a mocked session manager which is used for test.
type mockSessionManager1 struct {
Se session.Session
}
func (msm *mockSessionManager1) ShowTxnList() []*txninfo.TxnInfo {
panic("unimplemented!")
}
// ShowProcessList implements the SessionManager.ShowProcessList interface.
func (msm *mockSessionManager1) ShowProcessList() map[uint64]*util.ProcessInfo {
ret := make(map[uint64]*util.ProcessInfo)
return ret
}
func (msm *mockSessionManager1) GetProcessInfo(_ uint64) (*util.ProcessInfo, bool) {
pi := msm.Se.ShowProcess()
return pi, true
}
// Kill implements the SessionManager.Kill interface.
func (msm *mockSessionManager1) Kill(_ uint64, _ bool) {}
func (msm *mockSessionManager1) KillAllConnections() {}
func (msm *mockSessionManager1) ServerID() uint64 {
return 1
}
func (msm *mockSessionManager1) UpdateTLSConfig(_ *tls.Config) {}
func TestPreparedIssue17419(t *testing.T) {
store, dom, clean := testkit.CreateMockStoreAndDomain(t)
defer clean()
ctx := context.Background()
tk := testkit.NewTestKit(t, store)
tk.MustExec("use test")
tk.MustExec("drop table if exists t")
tk.MustExec("create table t (a int)")
tk.MustExec("insert into t (a) values (1), (2), (3)")
tk1 := testkit.NewTestKit(t, store)
query := "select * from test.t"
stmtID, _, _, err := tk1.Session().PrepareStmt(query)
require.NoError(t, err)
sm := &mockSessionManager1{
Se: tk1.Session(),
}
tk1.Session().SetSessionManager(sm)
dom.ExpensiveQueryHandle().SetSessionManager(sm)
rs, err := tk1.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{})
require.NoError(t, err)
tk1.ResultSetToResult(rs, fmt.Sprintf("%v", rs)).Check(testkit.Rows("1", "2", "3"))
tk1.Session().SetProcessInfo("", time.Now(), mysql.ComStmtExecute, 0)
dom.ExpensiveQueryHandle().LogOnQueryExceedMemQuota(tk.Session().GetSessionVars().ConnectionID)
// After entirely fixing https://github.com/pingcap/tidb/issues/17419
// require.NotNil(t, tk1.Session().ShowProcess().Plan)
// _, ok := tk1.Session().ShowProcess().Plan.(*plannercore.Execute)
// require.True(t, ok)
}