Files
tidb/executor/show_stats.go
Ewan Chou 62433663e5 util/types: change types path (#5007)
* util/types: change types path
2017-11-04 10:37:14 -05:00

164 lines
4.4 KiB
Go

// 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 executor
import (
"time"
"github.com/juju/errors"
"github.com/pingcap/tidb/mysql"
"github.com/pingcap/tidb/sessionctx"
"github.com/pingcap/tidb/statistics"
"github.com/pingcap/tidb/store/tikv/oracle"
"github.com/pingcap/tidb/types"
"github.com/pingcap/tidb/util/codec"
)
func (e *ShowExec) fetchShowStatsMeta() error {
do := sessionctx.GetDomain(e.ctx)
h := do.StatsHandle()
dbs := do.InfoSchema().AllSchemas()
for _, db := range dbs {
for _, tbl := range db.Tables {
statsTbl := h.GetTableStats(tbl.ID)
if !statsTbl.Pseudo {
row := types.MakeDatums(
db.Name.O,
tbl.Name.O,
e.versionToTime(statsTbl.Version),
statsTbl.ModifyCount,
statsTbl.Count,
)
e.rows = append(e.rows, row)
}
}
}
return nil
}
func (e *ShowExec) fetchShowStatsHistogram() error {
do := sessionctx.GetDomain(e.ctx)
h := do.StatsHandle()
dbs := do.InfoSchema().AllSchemas()
for _, db := range dbs {
for _, tbl := range db.Tables {
statsTbl := h.GetTableStats(tbl.ID)
if !statsTbl.Pseudo {
for _, col := range statsTbl.Columns {
e.rows = append(e.rows, e.histogramToRow(db.Name.O, tbl.Name.O, col.Info.Name.O, 0, col.Histogram))
}
for _, idx := range statsTbl.Indices {
e.rows = append(e.rows, e.histogramToRow(db.Name.O, tbl.Name.O, idx.Info.Name.O, 1, idx.Histogram))
}
}
}
}
return nil
}
func (e *ShowExec) histogramToRow(dbName string, tblName string, colName string, isIndex int, hist statistics.Histogram) Row {
return types.MakeDatums(
dbName,
tblName,
colName,
isIndex,
e.versionToTime(hist.LastUpdateVersion),
hist.NDV,
hist.NullCount,
)
}
func (e *ShowExec) versionToTime(version uint64) types.Time {
t := time.Unix(0, oracle.ExtractPhysical(version)*int64(time.Millisecond))
return types.Time{Time: types.FromGoTime(t), Type: mysql.TypeDatetime}
}
func (e *ShowExec) fetchShowStatsBuckets() error {
do := sessionctx.GetDomain(e.ctx)
h := do.StatsHandle()
dbs := do.InfoSchema().AllSchemas()
for _, db := range dbs {
for _, tbl := range db.Tables {
statsTbl := h.GetTableStats(tbl.ID)
if !statsTbl.Pseudo {
for _, col := range statsTbl.Columns {
rows, err := e.bucketsToRows(db.Name.O, tbl.Name.O, col.Info.Name.O, 0, col.Histogram)
if err != nil {
return errors.Trace(err)
}
e.rows = append(e.rows, rows...)
}
for _, idx := range statsTbl.Indices {
rows, err := e.bucketsToRows(db.Name.O, tbl.Name.O, idx.Info.Name.O, len(idx.Info.Columns), idx.Histogram)
if err != nil {
return errors.Trace(err)
}
e.rows = append(e.rows, rows...)
}
}
}
}
return nil
}
// bucketsToRows converts histogram buckets to rows. If the histogram is built from index, then numOfCols equals to number
// of index columns, else numOfCols is 0.
func (e *ShowExec) bucketsToRows(dbName, tblName, colName string, numOfCols int, hist statistics.Histogram) ([]Row, error) {
isIndex := 0
if numOfCols > 0 {
isIndex = 1
}
var rows []Row
for i, bkt := range hist.Buckets {
lowerBoundStr, err := e.valueToString(bkt.LowerBound, numOfCols)
if err != nil {
return nil, errors.Trace(err)
}
upperBoundStr, err := e.valueToString(bkt.UpperBound, numOfCols)
if err != nil {
return nil, errors.Trace(err)
}
row := types.MakeDatums(
dbName,
tblName,
colName,
isIndex,
i,
bkt.Count,
bkt.Repeats,
lowerBoundStr,
upperBoundStr,
)
rows = append(rows, row)
}
return rows, nil
}
// valueToString converts a possible encoded value to a formatted string. If the value is encoded, then
// size equals to number of origin values, else size is 0.
func (e *ShowExec) valueToString(value types.Datum, size int) (string, error) {
if size == 0 {
return value.ToString()
}
decodedVals, err := codec.Decode(value.GetBytes(), size)
if err != nil {
return "", errors.Trace(err)
}
str, err := types.DatumsToString(decodedVals)
if err != nil {
return "", errors.Trace(err)
}
return str, nil
}