// Copyright 2017 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 core_test import ( "testing" "github.com/pingcap/tidb/expression" "github.com/pingcap/tidb/infoschema" "github.com/pingcap/tidb/parser" "github.com/pingcap/tidb/parser/ast" "github.com/pingcap/tidb/parser/model" "github.com/pingcap/tidb/parser/mysql" "github.com/pingcap/tidb/planner/core" "github.com/pingcap/tidb/testkit" driver "github.com/pingcap/tidb/types/parser_driver" "github.com/stretchr/testify/require" ) func TestCacheable(t *testing.T) { store := testkit.CreateMockStore(t) tk := testkit.NewTestKit(t, store) tk.MustExec("use test") tk.MustExec("create table t1(a int, b int) partition by range(a) ( partition p0 values less than (6), partition p1 values less than (11) )") tk.MustExec("create table t2(a int, b int) partition by hash(a) partitions 11") tk.MustExec("create table t3(a int, b int)") tbl := &ast.TableName{Schema: model.NewCIStr("test"), Name: model.NewCIStr("t3")} is := tk.Session().GetInfoSchema().(infoschema.InfoSchema) // test non-SelectStmt/-InsertStmt/-DeleteStmt/-UpdateStmt/-SetOprStmt var stmt ast.Node = &ast.ShowStmt{} require.False(t, core.Cacheable(stmt, is)) stmt = &ast.LoadDataStmt{} require.False(t, core.Cacheable(stmt, is)) // test SetOprStmt stmt = &ast.SetOprStmt{} require.True(t, core.Cacheable(stmt, is)) tableRefsClause := &ast.TableRefsClause{TableRefs: &ast.Join{Left: &ast.TableSource{Source: tbl}}} // test InsertStmt stmt = &ast.InsertStmt{Table: tableRefsClause} // insert-values-stmt require.False(t, core.Cacheable(stmt, is)) stmt = &ast.InsertStmt{Table: tableRefsClause, Select: &ast.SelectStmt{}} // insert-select-stmt require.True(t, core.Cacheable(stmt, is)) // test DeleteStmt whereExpr := &ast.FuncCallExpr{} stmt = &ast.DeleteStmt{ TableRefs: tableRefsClause, Where: whereExpr, } require.True(t, core.Cacheable(stmt, is)) for funcName := range expression.UnCacheableFunctions { whereExpr.FnName = model.NewCIStr(funcName) require.False(t, core.Cacheable(stmt, is)) } whereExpr.FnName = model.NewCIStr(ast.Rand) require.True(t, core.Cacheable(stmt, is)) stmt = &ast.DeleteStmt{ TableRefs: tableRefsClause, Where: &ast.ExistsSubqueryExpr{}, } require.False(t, core.Cacheable(stmt, is)) limitStmt := &ast.Limit{ Count: &driver.ParamMarkerExpr{}, } stmt = &ast.DeleteStmt{ TableRefs: tableRefsClause, Limit: limitStmt, } require.False(t, core.Cacheable(stmt, is)) limitStmt = &ast.Limit{ Offset: &driver.ParamMarkerExpr{}, } stmt = &ast.DeleteStmt{ TableRefs: tableRefsClause, Limit: limitStmt, } require.False(t, core.Cacheable(stmt, is)) limitStmt = &ast.Limit{} stmt = &ast.DeleteStmt{ TableRefs: tableRefsClause, Limit: limitStmt, } require.True(t, core.Cacheable(stmt, is)) stmt.(*ast.DeleteStmt).TableHints = append(stmt.(*ast.DeleteStmt).TableHints, &ast.TableOptimizerHint{ HintName: model.NewCIStr(core.HintIgnorePlanCache), }) require.False(t, core.Cacheable(stmt, is)) // test UpdateStmt whereExpr = &ast.FuncCallExpr{} stmt = &ast.UpdateStmt{ TableRefs: tableRefsClause, Where: whereExpr, } require.True(t, core.Cacheable(stmt, is)) for funcName := range expression.UnCacheableFunctions { whereExpr.FnName = model.NewCIStr(funcName) require.False(t, core.Cacheable(stmt, is)) } whereExpr.FnName = model.NewCIStr(ast.Rand) require.True(t, core.Cacheable(stmt, is)) stmt = &ast.UpdateStmt{ TableRefs: tableRefsClause, Where: &ast.ExistsSubqueryExpr{}, } require.False(t, core.Cacheable(stmt, is)) limitStmt = &ast.Limit{ Count: &driver.ParamMarkerExpr{}, } stmt = &ast.UpdateStmt{ TableRefs: tableRefsClause, Limit: limitStmt, } require.False(t, core.Cacheable(stmt, is)) limitStmt = &ast.Limit{ Offset: &driver.ParamMarkerExpr{}, } stmt = &ast.UpdateStmt{ TableRefs: tableRefsClause, Limit: limitStmt, } require.False(t, core.Cacheable(stmt, is)) limitStmt = &ast.Limit{} stmt = &ast.UpdateStmt{ TableRefs: tableRefsClause, Limit: limitStmt, } require.True(t, core.Cacheable(stmt, is)) stmt.(*ast.UpdateStmt).TableHints = append(stmt.(*ast.UpdateStmt).TableHints, &ast.TableOptimizerHint{ HintName: model.NewCIStr(core.HintIgnorePlanCache), }) require.False(t, core.Cacheable(stmt, is)) // test SelectStmt whereExpr = &ast.FuncCallExpr{} stmt = &ast.SelectStmt{ Where: whereExpr, } require.True(t, core.Cacheable(stmt, is)) for funcName := range expression.UnCacheableFunctions { whereExpr.FnName = model.NewCIStr(funcName) require.False(t, core.Cacheable(stmt, is)) } whereExpr.FnName = model.NewCIStr(ast.Rand) require.True(t, core.Cacheable(stmt, is)) stmt = &ast.SelectStmt{ Where: &ast.ExistsSubqueryExpr{}, } require.False(t, core.Cacheable(stmt, is)) limitStmt = &ast.Limit{ Count: &driver.ParamMarkerExpr{}, } stmt = &ast.SelectStmt{ Limit: limitStmt, } require.False(t, core.Cacheable(stmt, is)) limitStmt = &ast.Limit{ Offset: &driver.ParamMarkerExpr{}, } stmt = &ast.SelectStmt{ Limit: limitStmt, } require.False(t, core.Cacheable(stmt, is)) limitStmt = &ast.Limit{} stmt = &ast.SelectStmt{ Limit: limitStmt, } require.True(t, core.Cacheable(stmt, is)) paramExpr := &driver.ParamMarkerExpr{} orderByClause := &ast.OrderByClause{Items: []*ast.ByItem{{Expr: paramExpr}}} stmt = &ast.SelectStmt{ OrderBy: orderByClause, } require.False(t, core.Cacheable(stmt, is)) valExpr := &driver.ValueExpr{} orderByClause = &ast.OrderByClause{Items: []*ast.ByItem{{Expr: valExpr}}} stmt = &ast.SelectStmt{ OrderBy: orderByClause, } require.True(t, core.Cacheable(stmt, is)) stmt.(*ast.SelectStmt).TableHints = append(stmt.(*ast.SelectStmt).TableHints, &ast.TableOptimizerHint{ HintName: model.NewCIStr(core.HintIgnorePlanCache), }) require.False(t, core.Cacheable(stmt, is)) boundExpr := &ast.FrameBound{Expr: &driver.ParamMarkerExpr{}} require.False(t, core.Cacheable(boundExpr, is)) // Partition table can not be cached. join := &ast.Join{ Left: &ast.TableName{Schema: model.NewCIStr("test"), Name: model.NewCIStr("t1")}, Right: &ast.TableName{Schema: model.NewCIStr("test"), Name: model.NewCIStr("t2")}, } stmt = &ast.SelectStmt{ From: &ast.TableRefsClause{ TableRefs: join, }, } require.False(t, core.Cacheable(stmt, is)) join = &ast.Join{ Left: &ast.TableName{Schema: model.NewCIStr("test"), Name: model.NewCIStr("t3")}, } stmt = &ast.SelectStmt{ From: &ast.TableRefsClause{ TableRefs: join, }, } require.True(t, core.Cacheable(stmt, is)) } func TestNonPreparedPlanCacheable(t *testing.T) { store := testkit.CreateMockStore(t) tk := testkit.NewTestKit(t, store) tk.MustExec("use test") tk.MustExec("create table t1(a int, b int, index idx_b(b)) partition by range(a) ( partition p0 values less than (6), partition p1 values less than (11) )") tk.MustExec("create table t2(a int, b int) partition by hash(a) partitions 11") tk.MustExec("create table t3(a int, b int)") is := tk.Session().GetInfoSchema().(infoschema.InfoSchema) p := parser.New() charset := mysql.DefaultCharset collation := mysql.DefaultCollationName supported := []string{ "select * from t where a<10", "select * from t where a<13 and b<15", "select * from t where b=13", "select * from t where c<8", "select * from t where d>8", "select * from t where c=8 and d>10", "select * from t where a<12 and b<13 and c<12 and d>2", "select * from t where a in (1, 2, 3)", "select * from t where a<13 or b<15", "select * from t where a<13 or b<15 and c=13", } unsupported := []string{ "select /*+ use_index(t1, idx_b) */ * from t1 where a > 1 and b < 2", // hint "select distinct a from t1 where a > 1 and b < 2", // distinct "select count(*) from t1 where a > 1 and b < 2 group by a", // group by "select a, sum(b) as c from t1 where a > 1 and b < 2 group by a having sum(b) > 1", // having "select * from t1 limit 1", // limit "select * from t1 order by a", // order by "select * from t1, t2", // join "select * from (select * from t1) t", // sub-query "insert into t1 values(1, 1)", // insert "insert into t1(a, b) select a, b from t1", // insert into select "update t1 set a = 1 where b = 2", // update "delete from t1 where b = 1", // delete "select * from t where a+b=13", // '+' "select * from t where mod(a, 3)=1", // mod "select * from t where d>now()", // now } for _, q := range unsupported { stmt, err := p.ParseOneStmt(q, charset, collation) require.NoError(t, err) require.False(t, core.NonPreparedPlanCacheable(stmt, is)) } for _, q := range supported { stmt, err := p.ParseOneStmt(q, charset, collation) require.NoError(t, err) require.True(t, core.NonPreparedPlanCacheable(stmt, is)) } }