1 В избранное 0 Ответвления 0

OSCHINA-MIRROR/hanchuanchuan-goInception

Присоединиться к Gitlife
Откройте для себя и примите участие в публичных проектах с открытым исходным кодом с участием более 10 миллионов разработчиков. Приватные репозитории также полностью бесплатны :)
Присоединиться бесплатно
Это зеркальный репозиторий, синхронизируется ежедневно с исходного репозитория.
Клонировать/Скачать
cbo_test.go 31 КБ
Копировать Редактировать Исходные данные Просмотреть построчно История
hanchuanchuan Отправлено 6 лет назад 27f3c5a
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852
// 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,
// See the License for the specific language governing permissions and
// limitations under the License.
package core_test
import (
"fmt"
"strings"
"testing"
"github.com/hanchuanchuan/goInception/config"
"github.com/hanchuanchuan/goInception/domain"
"github.com/hanchuanchuan/goInception/kv"
"github.com/hanchuanchuan/goInception/planner"
"github.com/hanchuanchuan/goInception/planner/core"
"github.com/hanchuanchuan/goInception/session"
"github.com/hanchuanchuan/goInception/sessionctx"
"github.com/hanchuanchuan/goInception/statistics"
"github.com/hanchuanchuan/goInception/store/mockstore"
"github.com/hanchuanchuan/goInception/util/testkit"
"github.com/hanchuanchuan/goInception/util/testleak"
. "github.com/pingcap/check"
"github.com/pingcap/errors"
)
var _ = Suite(&testAnalyzeSuite{})
type testAnalyzeSuite struct {
}
func (s *testAnalyzeSuite) TestExplainAnalyze(c *C) {
defer testleak.AfterTest(c)()
store, dom, err := newStoreWithBootstrap()
c.Assert(err, IsNil)
tk := testkit.NewTestKit(c, store)
defer func() {
dom.Close()
store.Close()
}()
tk.MustExec("use test")
tk.MustExec("create table t1(a int, b int, c int, key idx(a, b))")
tk.MustExec("create table t2(a int, b int)")
tk.MustExec("insert into t1 values (1, 1, 1), (2, 2, 2), (3, 3, 3), (4, 4, 4)")
tk.MustExec("insert into t2 values (2, 22), (3, 33), (5, 55)")
tk.MustExec("analyze table t1, t2")
rs := tk.MustQuery("explain analyze select t1.a, t1.b, sum(t1.c) from t1 join t2 on t1.a = t2.b where t1.a > 1")
c.Assert(len(rs.Rows()), Equals, 10)
for _, row := range rs.Rows() {
c.Assert(len(row), Equals, 5)
taskType := row[2].(string)
if taskType != "cop" {
execInfo := row[4].(string)
c.Assert(strings.Contains(execInfo, "time"), Equals, true)
c.Assert(strings.Contains(execInfo, "loops"), Equals, true)
c.Assert(strings.Contains(execInfo, "rows"), Equals, true)
}
}
}
// TestCBOWithoutAnalyze tests the plan with stats that only have count info.
func (s *testAnalyzeSuite) TestCBOWithoutAnalyze(c *C) {
defer testleak.AfterTest(c)()
store, dom, err := newStoreWithBootstrap()
c.Assert(err, IsNil)
testKit := testkit.NewTestKit(c, store)
defer func() {
dom.Close()
store.Close()
}()
testKit.MustExec("use test")
testKit.MustExec("create table t1 (a int)")
testKit.MustExec("create table t2 (a int)")
h := dom.StatsHandle()
c.Assert(h.HandleDDLEvent(<-h.DDLEventCh()), IsNil)
c.Assert(h.HandleDDLEvent(<-h.DDLEventCh()), IsNil)
testKit.MustExec("insert into t1 values (1), (2), (3), (4), (5), (6)")
testKit.MustExec("insert into t2 values (1), (2), (3), (4), (5), (6)")
h.DumpStatsDeltaToKV(statistics.DumpAll)
c.Assert(h.Update(dom.InfoSchema()), IsNil)
testKit.MustQuery("explain select * from t1, t2 where t1.a = t2.a").Check(testkit.Rows(
"HashLeftJoin_8 7.50 root inner join, inner:TableReader_13, equal:[eq(test.t1.a, test.t2.a)]",
"├─TableReader_11 6.00 root data:TableScan_10",
"│ └─TableScan_10 6.00 cop table:t1, range:[-inf,+inf], keep order:false, stats:pseudo",
"└─TableReader_13 6.00 root data:TableScan_12",
" └─TableScan_12 6.00 cop table:t2, range:[-inf,+inf], keep order:false, stats:pseudo",
))
}
func (s *testAnalyzeSuite) TestStraightJoin(c *C) {
defer testleak.AfterTest(c)()
store, dom, err := newStoreWithBootstrap()
c.Assert(err, IsNil)
testKit := testkit.NewTestKit(c, store)
defer func() {
dom.Close()
store.Close()
}()
testKit.MustExec("use test")
h := dom.StatsHandle()
for _, tblName := range []string{"t1", "t2", "t3", "t4"} {
testKit.MustExec(fmt.Sprintf("create table %s (a int)", tblName))
c.Assert(h.HandleDDLEvent(<-h.DDLEventCh()), IsNil)
}
testKit.MustQuery("explain select straight_join * from t1, t2, t3, t4").Check(testkit.Rows(
"HashLeftJoin_10 10000000000000000.00 root inner join, inner:TableReader_23",
"├─HashLeftJoin_12 1000000000000.00 root inner join, inner:TableReader_21",
"│ ├─HashLeftJoin_14 100000000.00 root inner join, inner:TableReader_19",
"│ │ ├─TableReader_17 10000.00 root data:TableScan_16",
"│ │ │ └─TableScan_16 10000.00 cop table:t1, range:[-inf,+inf], keep order:false, stats:pseudo",
"│ │ └─TableReader_19 10000.00 root data:TableScan_18",
"│ │ └─TableScan_18 10000.00 cop table:t2, range:[-inf,+inf], keep order:false, stats:pseudo",
"│ └─TableReader_21 10000.00 root data:TableScan_20",
"│ └─TableScan_20 10000.00 cop table:t3, range:[-inf,+inf], keep order:false, stats:pseudo",
"└─TableReader_23 10000.00 root data:TableScan_22",
" └─TableScan_22 10000.00 cop table:t4, range:[-inf,+inf], keep order:false, stats:pseudo",
))
testKit.MustQuery("explain select * from t1 straight_join t2 straight_join t3 straight_join t4").Check(testkit.Rows(
"HashLeftJoin_10 10000000000000000.00 root inner join, inner:TableReader_23",
"├─HashLeftJoin_12 1000000000000.00 root inner join, inner:TableReader_21",
"│ ├─HashLeftJoin_14 100000000.00 root inner join, inner:TableReader_19",
"│ │ ├─TableReader_17 10000.00 root data:TableScan_16",
"│ │ │ └─TableScan_16 10000.00 cop table:t1, range:[-inf,+inf], keep order:false, stats:pseudo",
"│ │ └─TableReader_19 10000.00 root data:TableScan_18",
"│ │ └─TableScan_18 10000.00 cop table:t2, range:[-inf,+inf], keep order:false, stats:pseudo",
"│ └─TableReader_21 10000.00 root data:TableScan_20",
"│ └─TableScan_20 10000.00 cop table:t3, range:[-inf,+inf], keep order:false, stats:pseudo",
"└─TableReader_23 10000.00 root data:TableScan_22",
" └─TableScan_22 10000.00 cop table:t4, range:[-inf,+inf], keep order:false, stats:pseudo",
))
testKit.MustQuery("explain select straight_join * from t1, t2, t3, t4 where t1.a=t4.a;").Check(testkit.Rows(
"HashLeftJoin_11 1250000000000.00 root inner join, inner:TableReader_24, equal:[eq(test.t1.a, test.t4.a)]",
"├─HashLeftJoin_13 1000000000000.00 root inner join, inner:TableReader_22",
"│ ├─HashLeftJoin_15 100000000.00 root inner join, inner:TableReader_20",
"│ │ ├─TableReader_18 10000.00 root data:TableScan_17",
"│ │ │ └─TableScan_17 10000.00 cop table:t1, range:[-inf,+inf], keep order:false, stats:pseudo",
"│ │ └─TableReader_20 10000.00 root data:TableScan_19",
"│ │ └─TableScan_19 10000.00 cop table:t2, range:[-inf,+inf], keep order:false, stats:pseudo",
"│ └─TableReader_22 10000.00 root data:TableScan_21",
"│ └─TableScan_21 10000.00 cop table:t3, range:[-inf,+inf], keep order:false, stats:pseudo",
"└─TableReader_24 10000.00 root data:TableScan_23",
" └─TableScan_23 10000.00 cop table:t4, range:[-inf,+inf], keep order:false, stats:pseudo",
))
}
func (s *testAnalyzeSuite) TestTableDual(c *C) {
defer testleak.AfterTest(c)()
store, dom, err := newStoreWithBootstrap()
c.Assert(err, IsNil)
defer func() {
dom.Close()
store.Close()
}()
testKit := testkit.NewTestKit(c, store)
testKit.MustExec(`use test`)
h := dom.StatsHandle()
testKit.MustExec(`create table t(a int)`)
testKit.MustExec("insert into t values (1), (2), (3), (4), (5), (6), (7), (8), (9), (10)")
c.Assert(h.HandleDDLEvent(<-h.DDLEventCh()), IsNil)
h.DumpStatsDeltaToKV(statistics.DumpAll)
c.Assert(h.Update(dom.InfoSchema()), IsNil)
testKit.MustQuery(`explain select * from t where 1 = 0`).Check(testkit.Rows(
`TableDual_6 0.00 root rows:0`,
))
testKit.MustQuery(`explain select * from t where 1 = 1 limit 0`).Check(testkit.Rows(
`TableDual_5 0.00 root rows:0`,
))
}
func (s *testAnalyzeSuite) TestEstimation(c *C) {
defer testleak.AfterTest(c)()
store, dom, err := newStoreWithBootstrap()
c.Assert(err, IsNil)
testKit := testkit.NewTestKit(c, store)
defer func() {
dom.Close()
store.Close()
statistics.RatioOfPseudoEstimate = 0.7
}()
statistics.RatioOfPseudoEstimate = 10.0
testKit.MustExec("use test")
testKit.MustExec("create table t (a int)")
testKit.MustExec("insert into t values (1), (2), (3), (4), (5), (6), (7), (8), (9), (10)")
testKit.MustExec("insert into t select * from t")
testKit.MustExec("insert into t select * from t")
h := dom.StatsHandle()
h.HandleDDLEvent(<-h.DDLEventCh())
h.DumpStatsDeltaToKV(statistics.DumpAll)
testKit.MustExec("analyze table t")
for i := 1; i <= 8; i++ {
testKit.MustExec("delete from t where a = ?", i)
}
h.DumpStatsDeltaToKV(statistics.DumpAll)
c.Assert(h.Update(dom.InfoSchema()), IsNil)
testKit.MustQuery("explain select count(*) from t group by a").Check(testkit.Rows(
"HashAgg_9 2.00 root group by:col_1, funcs:count(col_0)",
"└─TableReader_10 2.00 root data:HashAgg_5",
" └─HashAgg_5 2.00 cop group by:test.t.a, funcs:count(1)",
" └─TableScan_8 8.00 cop table:t, range:[-inf,+inf], keep order:false",
))
}
func constructInsertSQL(i, n int) string {
sql := "insert into t (a,b,c,e)values "
for j := 0; j < n; j++ {
sql += fmt.Sprintf("(%d, %d, '%d', %d)", i*n+j, i, i+j, i*n+j)
if j != n-1 {
sql += ", "
}
}
return sql
}
func (s *testAnalyzeSuite) TestIndexRead(c *C) {
defer testleak.AfterTest(c)()
store, dom, err := newStoreWithBootstrap()
c.Assert(err, IsNil)
testKit := testkit.NewTestKit(c, store)
defer func() {
dom.Close()
store.Close()
}()
testKit.MustExec("use test")
testKit.MustExec("drop table if exists t, t1")
testKit.MustExec("create table t (a int primary key, b int, c varchar(200), d datetime DEFAULT CURRENT_TIMESTAMP, e int, ts timestamp DEFAULT CURRENT_TIMESTAMP)")
testKit.MustExec("create index b on t (b)")
testKit.MustExec("create index d on t (d)")
testKit.MustExec("create index e on t (e)")
testKit.MustExec("create index b_c on t (b,c)")
testKit.MustExec("create index ts on t (ts)")
for i := 0; i < 100; i++ {
testKit.MustExec(constructInsertSQL(i, 100))
}
testKit.MustExec("create table t1 (a int, b int, index idx(a), index idxx(b))")
for i := 1; i < 16; i++ {
testKit.MustExec(fmt.Sprintf("insert into t1 values(%v, %v)", i, i))
}
testKit.MustExec("analyze table t")
testKit.MustExec("analyze table t1")
tests := []struct {
sql string
best string
}{
{
sql: "select count(*) from t group by e",
best: "IndexReader(Index(t.e)[[NULL,+inf]]->StreamAgg)->StreamAgg",
},
{
sql: "select count(*) from t where e <= 10 group by e",
best: "IndexReader(Index(t.e)[[-inf,10]]->StreamAgg)->StreamAgg",
},
{
sql: "select count(*) from t where e <= 50",
best: "IndexReader(Index(t.e)[[-inf,50]]->StreamAgg)->StreamAgg",
},
{
sql: "select count(*) from t where c > '1' group by b",
best: "IndexReader(Index(t.b_c)[[NULL,+inf]]->Sel([gt(test.t.c, 1)])->StreamAgg)->StreamAgg",
},
{
sql: "select count(*) from t where e = 1 group by b",
best: "IndexLookUp(Index(t.e)[[1,1]], Table(t)->HashAgg)->HashAgg",
},
{
sql: "select count(*) from t where e > 1 group by b",
best: "IndexLookUp(Index(t.b)[[NULL,+inf]], Table(t)->Sel([gt(test.t.e, 1)]))->StreamAgg",
},
{
sql: "select count(e) from t where t.b <= 20",
best: "IndexLookUp(Index(t.b)[[-inf,20]], Table(t)->HashAgg)->HashAgg",
},
{
sql: "select count(e) from t where t.b <= 30",
best: "IndexLookUp(Index(t.b)[[-inf,30]], Table(t)->HashAgg)->HashAgg",
},
{
sql: "select count(e) from t where t.b <= 40",
best: "IndexLookUp(Index(t.b)[[-inf,40]], Table(t)->HashAgg)->HashAgg",
},
{
sql: "select count(e) from t where t.b <= 50",
best: "IndexLookUp(Index(t.b)[[-inf,50]], Table(t)->HashAgg)->HashAgg",
},
{
sql: "select count(e) from t where t.b <= 100000000000",
best: "TableReader(Table(t)->Sel([le(test.t.b, 100000000000)])->StreamAgg)->StreamAgg",
},
{
sql: "select * from t where t.b <= 40",
best: "IndexLookUp(Index(t.b)[[-inf,40]], Table(t))",
},
{
sql: "select * from t where t.b <= 50",
best: "IndexLookUp(Index(t.b)[[-inf,50]], Table(t))",
},
{
sql: "select * from t where t.b <= 10000000000",
best: "TableReader(Table(t)->Sel([le(test.t.b, 10000000000)]))",
},
// test panic
{
sql: "select * from t where 1 and t.b <= 50",
best: "IndexLookUp(Index(t.b)[[-inf,50]], Table(t))",
},
{
sql: "select * from t where t.b <= 100 order by t.a limit 1",
best: "TableReader(Table(t)->Sel([le(test.t.b, 100)])->Limit)->Limit",
},
{
sql: "select * from t where t.b <= 1 order by t.a limit 10",
best: "IndexLookUp(Index(t.b)[[-inf,1]]->TopN([test.t.a],0,10), Table(t))->TopN([test.t.a],0,10)",
},
{
sql: "select * from t use index(b) where b = 1 order by a",
best: "IndexLookUp(Index(t.b)[[1,1]], Table(t))->Sort",
},
// test datetime
{
sql: "select * from t where d < cast('1991-09-05' as datetime)",
best: "IndexLookUp(Index(t.d)[[-inf,1991-09-05 00:00:00)], Table(t))",
},
// test timestamp
{
sql: "select * from t where ts < '1991-09-05'",
best: "IndexLookUp(Index(t.ts)[[-inf,1991-09-05 00:00:00)], Table(t))",
},
{
sql: "select sum(a) from t1 use index(idx) where a = 3 and b = 100000 group by a limit 1",
best: "IndexLookUp(Index(t1.idx)[[3,3]], Table(t1)->Sel([eq(test.t1.b, 100000)]))->StreamAgg->Limit",
},
}
for _, tt := range tests {
ctx := testKit.Se.(sessionctx.Context)
stmts, err := session.Parse(ctx, tt.sql)
c.Assert(err, IsNil)
c.Assert(stmts, HasLen, 1)
stmt := stmts[0]
is := domain.GetDomain(ctx).InfoSchema()
err = core.Preprocess(ctx, stmt, is, false)
c.Assert(err, IsNil)
p, err := planner.Optimize(ctx, stmt, is)
c.Assert(err, IsNil)
c.Assert(core.ToString(p), Equals, tt.best, Commentf("for %s", tt.sql))
}
}
func (s *testAnalyzeSuite) TestEmptyTable(c *C) {
defer testleak.AfterTest(c)()
store, dom, err := newStoreWithBootstrap()
c.Assert(err, IsNil)
testKit := testkit.NewTestKit(c, store)
defer func() {
dom.Close()
store.Close()
}()
testKit.MustExec("use test")
testKit.MustExec("drop table if exists t, t1")
testKit.MustExec("create table t (c1 int)")
testKit.MustExec("create table t1 (c1 int)")
testKit.MustExec("analyze table t, t1")
tests := []struct {
sql string
best string
}{
{
sql: "select * from t where t.c1 <= 50",
best: "TableReader(Table(t)->Sel([le(test.t.c1, 50)]))",
},
{
sql: "select * from t where c1 in (select c1 from t1)",
best: "LeftHashJoin{TableReader(Table(t))->TableReader(Table(t1))}(test.t.c1,test.t1.c1)",
},
{
sql: "select * from t, t1 where t.c1 = t1.c1",
best: "LeftHashJoin{TableReader(Table(t))->TableReader(Table(t1))}(test.t.c1,test.t1.c1)",
},
{
sql: "select * from t limit 0",
best: "Dual",
},
}
for _, tt := range tests {
ctx := testKit.Se.(sessionctx.Context)
stmts, err := session.Parse(ctx, tt.sql)
c.Assert(err, IsNil)
c.Assert(stmts, HasLen, 1)
stmt := stmts[0]
is := domain.GetDomain(ctx).InfoSchema()
err = core.Preprocess(ctx, stmt, is, false)
c.Assert(err, IsNil)
p, err := planner.Optimize(ctx, stmt, is)
c.Assert(err, IsNil)
c.Assert(core.ToString(p), Equals, tt.best, Commentf("for %s", tt.sql))
}
}
func (s *testAnalyzeSuite) TestAnalyze(c *C) {
defer testleak.AfterTest(c)()
store, dom, err := newStoreWithBootstrap()
c.Assert(err, IsNil)
testKit := testkit.NewTestKit(c, store)
defer func() {
dom.Close()
store.Close()
}()
testKit.MustExec("use test")
testKit.MustExec("drop table if exists t, t1, t2, t3")
testKit.MustExec("create table t (a int, b int)")
testKit.MustExec("create index a on t (a)")
testKit.MustExec("create index b on t (b)")
testKit.MustExec("insert into t (a,b) values (1,1),(1,2),(1,3),(1,4),(2,5),(2,6),(2,7),(2,8)")
testKit.MustExec("analyze table t")
testKit.MustExec("create table t1 (a int, b int)")
testKit.MustExec("create index a on t1 (a)")
testKit.MustExec("create index b on t1 (b)")
testKit.MustExec("insert into t1 (a,b) values (1,1),(1,2),(1,3),(1,4),(2,5),(2,6),(2,7),(2,8)")
testKit.MustExec("create table t2 (a int, b int)")
testKit.MustExec("create index a on t2 (a)")
testKit.MustExec("create index b on t2 (b)")
testKit.MustExec("insert into t2 (a,b) values (1,1),(1,2),(1,3),(1,4),(2,5),(2,6),(2,7),(2,8)")
testKit.MustExec("analyze table t2 index a")
testKit.MustExec("create table t3 (a int, b int)")
testKit.MustExec("create index a on t3 (a)")
testKit.MustExec("set @@session.tidb_enable_table_partition=1")
testKit.MustExec("create table t4 (a int, b int) partition by range (a) (partition p1 values less than (2), partition p2 values less than (3))")
testKit.MustExec("create index a on t4 (a)")
testKit.MustExec("create index b on t4 (b)")
testKit.MustExec("insert into t4 (a,b) values (1,1),(1,2),(1,3),(1,4),(2,5),(2,6),(2,7),(2,8)")
testKit.MustExec("analyze table t4")
tests := []struct {
sql string
best string
}{
{
sql: "analyze table t3",
best: "Analyze{Index(a),Table(b)}",
},
// Test analyze full table.
{
sql: "select * from t where t.a <= 2",
best: "TableReader(Table(t)->Sel([le(test.t.a, 2)]))",
},
{
sql: "select * from t where t.b < 2",
best: "IndexLookUp(Index(t.b)[[-inf,2)], Table(t))",
},
{
sql: "select * from t where t.a = 1 and t.b <= 2",
best: "IndexLookUp(Index(t.b)[[-inf,2]], Table(t)->Sel([eq(test.t.a, 1)]))",
},
// Test not analyzed table.
{
sql: "select * from t1 where t1.a <= 2",
best: "IndexLookUp(Index(t1.a)[[-inf,2]], Table(t1))",
},
{
sql: "select * from t1 where t1.a = 1 and t1.b <= 2",
best: "IndexLookUp(Index(t1.a)[[1,1]], Table(t1)->Sel([le(test.t1.b, 2)]))",
},
// Test analyze single index.
{
sql: "select * from t2 where t2.a <= 2",
best: "TableReader(Table(t2)->Sel([le(test.t2.a, 2)]))",
},
// Test analyze all index.
{
sql: "analyze table t2 index",
best: "Analyze{Index(a),Index(b)}",
},
// Test partitioned table.
{
sql: "select * from t4 where t4.a <= 2",
best: "UnionAll{TableReader(Table(t4)->Sel([le(test.t4.a, 2)]))->TableReader(Table(t4)->Sel([le(test.t4.a, 2)]))}",
},
{
sql: "select * from t4 where t4.b < 2",
best: "UnionAll{IndexLookUp(Index(t4.b)[[-inf,2)], Table(t4))->IndexLookUp(Index(t4.b)[[-inf,2)], Table(t4))}",
},
{
sql: "select * from t4 where t4.a = 1 and t4.b <= 2",
best: "TableReader(Table(t4)->Sel([eq(test.t4.a, 1) le(test.t4.b, 2)]))",
},
// TODO: Refine these tests in the future.
//{
// sql: "select * from t2 where t2.a = 1 and t2.b <= 2",
// best: "IndexLookUp(Index(t2.b)[[-inf,2]], Table(t2)->Sel([eq(test.t2.a, 1)]))",
//},
}
for _, tt := range tests {
ctx := testKit.Se.(sessionctx.Context)
stmts, err := session.Parse(ctx, tt.sql)
c.Assert(err, IsNil)
c.Assert(stmts, HasLen, 1)
stmt := stmts[0]
is := domain.GetDomain(ctx).InfoSchema()
err = core.Preprocess(ctx, stmt, is, false)
c.Assert(err, IsNil)
p, err := planner.Optimize(ctx, stmt, is)
c.Assert(err, IsNil)
c.Assert(core.ToString(p), Equals, tt.best, Commentf("for %s", tt.sql))
}
}
func (s *testAnalyzeSuite) TestOutdatedAnalyze(c *C) {
defer testleak.AfterTest(c)()
store, dom, err := newStoreWithBootstrap()
c.Assert(err, IsNil)
testKit := testkit.NewTestKit(c, store)
defer func() {
dom.Close()
store.Close()
}()
testKit.MustExec("use test")
testKit.MustExec("create table t (a int, b int, index idx(a))")
for i := 0; i < 10; i++ {
testKit.MustExec(fmt.Sprintf("insert into t values (%d,%d)", i, i))
}
h := dom.StatsHandle()
h.HandleDDLEvent(<-h.DDLEventCh())
h.DumpStatsDeltaToKV(statistics.DumpAll)
testKit.MustExec("analyze table t")
testKit.MustExec("insert into t select * from t")
testKit.MustExec("insert into t select * from t")
testKit.MustExec("insert into t select * from t")
h.DumpStatsDeltaToKV(statistics.DumpAll)
c.Assert(h.Update(dom.InfoSchema()), IsNil)
statistics.RatioOfPseudoEstimate = 10.0
testKit.MustQuery("explain select * from t where a <= 5 and b <= 5").Check(testkit.Rows(
"TableReader_7 35.91 root data:Selection_6",
"└─Selection_6 35.91 cop le(test.t.a, 5), le(test.t.b, 5)",
" └─TableScan_5 80.00 cop table:t, range:[-inf,+inf], keep order:false",
))
statistics.RatioOfPseudoEstimate = 0.7
testKit.MustQuery("explain select * from t where a <= 5 and b <= 5").Check(testkit.Rows(
"IndexLookUp_11 8.84 root ",
"├─IndexScan_8 26.59 cop table:t, index:a, range:[-inf,5], keep order:false, stats:pseudo",
"└─Selection_10 8.84 cop le(test.t.b, 5)",
" └─TableScan_9 26.59 cop table:t, keep order:false, stats:pseudo",
))
}
func (s *testAnalyzeSuite) TestPreparedNullParam(c *C) {
defer testleak.AfterTest(c)()
store, dom, err := newStoreWithBootstrap()
c.Assert(err, IsNil)
defer func() {
dom.Close()
store.Close()
}()
cfg := config.GetGlobalConfig()
orgEnable := cfg.PreparedPlanCache.Enabled
orgCapacity := cfg.PreparedPlanCache.Capacity
flags := []bool{false, true}
for _, flag := range flags {
cfg.PreparedPlanCache.Enabled = flag
cfg.PreparedPlanCache.Capacity = 100
testKit := testkit.NewTestKit(c, store)
testKit.MustExec("use test")
testKit.MustExec("drop table if exists t")
testKit.MustExec("create table t (id int, KEY id (id))")
testKit.MustExec("insert into t values (1), (2), (3)")
sql := "select * from t where id = ?"
best := "Dual"
ctx := testKit.Se.(sessionctx.Context)
stmts, err := session.Parse(ctx, sql)
stmt := stmts[0]
is := domain.GetDomain(ctx).InfoSchema()
err = core.Preprocess(ctx, stmt, is, true)
c.Assert(err, IsNil)
p, err := planner.Optimize(ctx, stmt, is)
c.Assert(err, IsNil)
c.Assert(core.ToString(p), Equals, best, Commentf("for %s", sql))
}
cfg.PreparedPlanCache.Enabled = orgEnable
cfg.PreparedPlanCache.Capacity = orgCapacity
}
func (s *testAnalyzeSuite) TestNullCount(c *C) {
defer testleak.AfterTest(c)()
store, dom, err := newStoreWithBootstrap()
c.Assert(err, IsNil)
testKit := testkit.NewTestKit(c, store)
defer func() {
dom.Close()
store.Close()
}()
testKit.MustExec("use test")
testKit.MustExec("drop table if exists t")
testKit.MustExec("create table t (a int, b int, index idx(a))")
testKit.MustExec("insert into t values (null, null), (null, null)")
testKit.MustExec("analyze table t")
testKit.MustQuery("explain select * from t where a is null").Check(testkit.Rows(
"TableReader_7 2.00 root data:Selection_6",
"└─Selection_6 2.00 cop isnull(test.t.a)",
" └─TableScan_5 2.00 cop table:t, range:[-inf,+inf], keep order:false",
))
testKit.MustQuery("explain select * from t use index(idx) where a is null").Check(testkit.Rows(
"IndexLookUp_7 2.00 root ",
"├─IndexScan_5 2.00 cop table:t, index:a, range:[NULL,NULL], keep order:false",
"└─TableScan_6 2.00 cop table:t, keep order:false",
))
h := dom.StatsHandle()
h.Clear()
h.Lease = 1
defer func() { h.Lease = 0 }()
c.Assert(h.Update(dom.InfoSchema()), IsNil)
testKit.MustQuery("explain select * from t where b = 1").Check(testkit.Rows(
"TableReader_7 0.00 root data:Selection_6",
"└─Selection_6 0.00 cop eq(test.t.b, 1)",
" └─TableScan_5 2.00 cop table:t, range:[-inf,+inf], keep order:false",
))
testKit.MustQuery("explain select * from t where b < 1").Check(testkit.Rows(
"TableReader_7 0.00 root data:Selection_6",
"└─Selection_6 0.00 cop lt(test.t.b, 1)",
" └─TableScan_5 2.00 cop table:t, range:[-inf,+inf], keep order:false",
))
}
func (s *testAnalyzeSuite) TestCorrelatedEstimation(c *C) {
defer testleak.AfterTest(c)()
store, dom, err := newStoreWithBootstrap()
c.Assert(err, IsNil)
tk := testkit.NewTestKit(c, store)
defer func() {
dom.Close()
store.Close()
}()
tk.MustExec("use test")
tk.MustExec("create table t(a int, b int, c int, index idx(c))")
tk.MustExec("insert into t values(1,1,1), (2,2,2), (3,3,3), (4,4,4), (5,5,5), (6,6,6), (7,7,7), (8,8,8), (9,9,9),(10,10,10)")
tk.MustExec("analyze table t")
tk.MustQuery("explain select t.c in (select count(*) from t s , t t1 where s.a = t.a and s.a = t1.a) from t;").
Check(testkit.Rows(
"Projection_11 10.00 root 9_aux_0",
"└─Apply_13 10.00 root left outer semi join, inner:StreamAgg_20, equal:[eq(test.t.c, 7_col_0)]",
" ├─TableReader_15 10.00 root data:TableScan_14",
" │ └─TableScan_14 10.00 cop table:t, range:[-inf,+inf], keep order:false",
" └─StreamAgg_20 1.00 root funcs:count(1)",
" └─HashLeftJoin_21 1.00 root inner join, inner:TableReader_28, equal:[eq(s.a, t1.a)]",
" ├─TableReader_25 1.00 root data:Selection_24",
" │ └─Selection_24 1.00 cop eq(s.a, test.t.a)",
" │ └─TableScan_23 10.00 cop table:s, range:[-inf,+inf], keep order:false",
" └─TableReader_28 1.00 root data:Selection_27",
" └─Selection_27 1.00 cop eq(t1.a, test.t.a)",
" └─TableScan_26 10.00 cop table:t1, range:[-inf,+inf], keep order:false",
))
tk.MustQuery("explain select (select concat(t1.a, \",\", t1.b) from t t1 where t1.a=t.a and t1.c=t.c) from t").
Check(testkit.Rows(
"Projection_8 10.00 root concat(t1.a, \",\", t1.b)",
"└─Apply_10 10.00 root left outer join, inner:MaxOneRow_13",
" ├─TableReader_12 10.00 root data:TableScan_11",
" │ └─TableScan_11 10.00 cop table:t, range:[-inf,+inf], keep order:false",
" └─MaxOneRow_13 1.00 root ",
" └─Projection_14 0.80 root concat(cast(t1.a), \",\", cast(t1.b))",
" └─IndexLookUp_21 0.80 root ",
" ├─IndexScan_18 1.00 cop table:t1, index:c, range: decided by [eq(t1.c, test.t.c)], keep order:false",
" └─Selection_20 0.80 cop eq(t1.a, test.t.a)",
" └─TableScan_19 1.00 cop table:t, keep order:false",
))
}
func (s *testAnalyzeSuite) TestInconsistentEstimation(c *C) {
defer testleak.AfterTest(c)()
store, dom, err := newStoreWithBootstrap()
c.Assert(err, IsNil)
tk := testkit.NewTestKit(c, store)
defer func() {
dom.Close()
store.Close()
}()
tk.MustExec("use test")
tk.MustExec("create table t(a int, b int, c int, index ab(a,b), index ac(a,c))")
tk.MustExec("insert into t values (1,1,1), (1000,1000,1000)")
for i := 0; i < 10; i++ {
tk.MustExec("insert into t values (5,5,5), (10,10,10)")
}
tk.MustExec("analyze table t with 2 buckets")
// Force using the histogram to estimate.
tk.MustExec("update mysql.stats_histograms set stats_ver = 0")
dom.StatsHandle().Clear()
dom.StatsHandle().Update(dom.InfoSchema())
// Using the histogram (a, b) to estimate `a = 5` will get 1.22, while using the CM Sketch to estimate
// the `a = 5 and c = 5` will get 10, it is not consistent.
tk.MustQuery("explain select * from t use index(ab) where a = 5 and c = 5").
Check(testkit.Rows(
"IndexLookUp_8 10.00 root ",
"├─IndexScan_5 12.50 cop table:t, index:a, b, range:[5,5], keep order:false",
"└─Selection_7 10.00 cop eq(test.t.c, 5)",
" └─TableScan_6 12.50 cop table:t, keep order:false",
))
}
func newStoreWithBootstrap() (kv.Storage, *domain.Domain, error) {
store, err := mockstore.NewMockTikvStore()
if err != nil {
return nil, nil, errors.Trace(err)
}
session.SetSchemaLease(0)
session.SetStatsLease(0)
dom, err := session.BootstrapSession(store)
dom.SetStatsUpdating(true)
return store, dom, errors.Trace(err)
}
func BenchmarkOptimize(b *testing.B) {
c := &C{}
store, dom, err := newStoreWithBootstrap()
c.Assert(err, IsNil)
defer func() {
dom.Close()
store.Close()
}()
testKit := testkit.NewTestKit(c, store)
testKit.MustExec("use test")
testKit.MustExec("drop table if exists t")
testKit.MustExec("create table t (a int primary key, b int, c varchar(200), d datetime DEFAULT CURRENT_TIMESTAMP, e int, ts timestamp DEFAULT CURRENT_TIMESTAMP)")
testKit.MustExec("create index b on t (b)")
testKit.MustExec("create index d on t (d)")
testKit.MustExec("create index e on t (e)")
testKit.MustExec("create index b_c on t (b,c)")
testKit.MustExec("create index ts on t (ts)")
for i := 0; i < 100; i++ {
testKit.MustExec(constructInsertSQL(i, 100))
}
testKit.MustExec("analyze table t")
tests := []struct {
sql string
best string
}{
{
sql: "select count(*) from t group by e",
best: "IndexReader(Index(t.e)[[NULL,+inf]])->StreamAgg",
},
{
sql: "select count(*) from t where e <= 10 group by e",
best: "IndexReader(Index(t.e)[[-inf,10]])->StreamAgg",
},
{
sql: "select count(*) from t where e <= 50",
best: "IndexReader(Index(t.e)[[-inf,50]]->HashAgg)->HashAgg",
},
{
sql: "select count(*) from t where c > '1' group by b",
best: "IndexReader(Index(t.b_c)[[NULL,+inf]]->Sel([gt(test.t.c, 1)]))->StreamAgg",
},
{
sql: "select count(*) from t where e = 1 group by b",
best: "IndexLookUp(Index(t.e)[[1,1]], Table(t)->HashAgg)->HashAgg",
},
{
sql: "select count(*) from t where e > 1 group by b",
best: "TableReader(Table(t)->Sel([gt(test.t.e, 1)])->HashAgg)->HashAgg",
},
{
sql: "select count(e) from t where t.b <= 20",
best: "IndexLookUp(Index(t.b)[[-inf,20]], Table(t)->HashAgg)->HashAgg",
},
{
sql: "select count(e) from t where t.b <= 30",
best: "IndexLookUp(Index(t.b)[[-inf,30]], Table(t)->HashAgg)->HashAgg",
},
{
sql: "select count(e) from t where t.b <= 40",
best: "IndexLookUp(Index(t.b)[[-inf,40]], Table(t)->HashAgg)->HashAgg",
},
{
sql: "select count(e) from t where t.b <= 50",
best: "TableReader(Table(t)->Sel([le(test.t.b, 50)])->HashAgg)->HashAgg",
},
{
sql: "select * from t where t.b <= 40",
best: "IndexLookUp(Index(t.b)[[-inf,40]], Table(t))",
},
{
sql: "select * from t where t.b <= 50",
best: "TableReader(Table(t)->Sel([le(test.t.b, 50)]))",
},
// test panic
{
sql: "select * from t where 1 and t.b <= 50",
best: "TableReader(Table(t)->Sel([le(test.t.b, 50)]))",
},
{
sql: "select * from t where t.b <= 100 order by t.a limit 1",
best: "TableReader(Table(t)->Sel([le(test.t.b, 100)])->Limit)->Limit",
},
{
sql: "select * from t where t.b <= 1 order by t.a limit 10",
best: "IndexLookUp(Index(t.b)[[-inf,1]]->TopN([test.t.a],0,10), Table(t))->TopN([test.t.a],0,10)",
},
{
sql: "select * from t use index(b) where b = 1 order by a",
best: "IndexLookUp(Index(t.b)[[1,1]], Table(t))->Sort",
},
// test datetime
{
sql: "select * from t where d < cast('1991-09-05' as datetime)",
best: "IndexLookUp(Index(t.d)[[-inf,1991-09-05 00:00:00)], Table(t))",
},
// test timestamp
{
sql: "select * from t where ts < '1991-09-05'",
best: "IndexLookUp(Index(t.ts)[[-inf,1991-09-05 00:00:00)], Table(t))",
},
}
for _, tt := range tests {
ctx := testKit.Se.(sessionctx.Context)
stmts, err := session.Parse(ctx, tt.sql)
c.Assert(err, IsNil)
c.Assert(stmts, HasLen, 1)
stmt := stmts[0]
is := domain.GetDomain(ctx).InfoSchema()
err = core.Preprocess(ctx, stmt, is, false)
c.Assert(err, IsNil)
b.Run(tt.sql, func(b *testing.B) {
b.ResetTimer()
for i := 0; i < b.N; i++ {
_, err := planner.Optimize(ctx, stmt, is)
c.Assert(err, IsNil)
}
b.ReportAllocs()
})
}
}

Комментарий ( 0 )

Вы можете оставить комментарий после Вход в систему

1
https://gitlife.ru/oschina-mirror/hanchuanchuan-goInception.git
git@gitlife.ru:oschina-mirror/hanchuanchuan-goInception.git
oschina-mirror
hanchuanchuan-goInception
hanchuanchuan-goInception
v1.2.4