tidb index_advise 源码

  • 2022-09-19
  • 浏览 (303)

tidb index_advise 代码

文件路径:/executor/index_advise.go

// Copyright 2019 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

import (
	"context"
	"strings"

	"github.com/pingcap/errors"
	"github.com/pingcap/tidb/parser"
	"github.com/pingcap/tidb/parser/ast"
	"github.com/pingcap/tidb/sessionctx"
	"github.com/pingcap/tidb/util"
	"github.com/pingcap/tidb/util/chunk"
)

// IndexAdviseExec represents a index advise executor.
type IndexAdviseExec struct {
	baseExecutor

	IsLocal         bool
	indexAdviseInfo *IndexAdviseInfo
}

// Next implements the Executor Next interface.
func (e *IndexAdviseExec) Next(ctx context.Context, req *chunk.Chunk) error {
	if !e.IsLocal {
		return errors.New("Index Advise: don't support load file without local field")
	}
	if e.indexAdviseInfo.Path == "" {
		return errors.New("Index Advise: infile path is empty")
	}
	if len(e.indexAdviseInfo.LinesInfo.Terminated) == 0 {
		return errors.New("Index Advise: don't support advise index for SQL terminated by nil")
	}

	if val := e.ctx.Value(IndexAdviseVarKey); val != nil {
		e.ctx.SetValue(IndexAdviseVarKey, nil)
		return errors.New("Index Advise: previous index advise option isn't closed normally")
	}
	e.ctx.SetValue(IndexAdviseVarKey, e.indexAdviseInfo)
	return nil
}

// Close implements the Executor Close interface.
func (e *IndexAdviseExec) Close() error {
	return nil
}

// Open implements the Executor Open interface.
func (e *IndexAdviseExec) Open(ctx context.Context) error {
	return nil
}

// IndexAdviseInfo saves the information of index advise operation.
type IndexAdviseInfo struct {
	Path        string
	MaxMinutes  uint64
	MaxIndexNum *ast.MaxIndexNumClause
	LinesInfo   *ast.LinesClause
	Ctx         sessionctx.Context
	StmtNodes   [][]ast.StmtNode
	Result      *IndexAdvice
}

func (e *IndexAdviseInfo) getStmtNodes(data []byte) error {
	str := string(data)
	sqls := strings.Split(str, e.LinesInfo.Terminated)

	j := 0
	for i, sql := range sqls {
		if sql != "\n" && sql != "" && strings.HasPrefix(sql, e.LinesInfo.Starting) {
			sqls[j] = sqls[i]
			j++
		}
	}
	sqls = sqls[:j]

	sv := e.Ctx.GetSessionVars()
	e.StmtNodes = make([][]ast.StmtNode, len(sqls))
	sqlParser := parser.New()
	for i, sql := range sqls {
		stmtNodes, warns, err := sqlParser.ParseSQL(sql)
		if err != nil {
			return err
		}
		for _, warn := range warns {
			sv.StmtCtx.AppendWarning(util.SyntaxWarn(warn))
		}
		curStmtNodes := make([]ast.StmtNode, len(stmtNodes))
		copy(curStmtNodes, stmtNodes)
		e.StmtNodes[i] = curStmtNodes
	}
	return nil
}

func (e *IndexAdviseInfo) prepareInfo(data []byte) error {
	if e.MaxMinutes == 0 {
		return errors.New("Index Advise: the maximum execution time limit should be greater than 0")
	}
	if e.MaxIndexNum != nil {
		if e.MaxIndexNum.PerTable == 0 || e.MaxIndexNum.PerDB == 0 {
			return errors.New("Index Advise: the maximum number of indexes should be greater than 0")
		}
	}
	return e.getStmtNodes(data)
}

// GetIndexAdvice gets the index advice by workload file.
func (e *IndexAdviseInfo) GetIndexAdvice(ctx context.Context, data []byte) error {
	if err := e.prepareInfo(data); err != nil {
		return err
	}
	// TODO: Finish the index advise process. It will be done in another PR.
	return nil
}

// IndexAdvice represents the index advice.
type IndexAdvice struct {
	// TODO: Define index advice data structure and implements the ResultSet interface. It will be done in another PR
}

// IndexAdviseVarKeyType is a dummy type to avoid naming collision in context.
type IndexAdviseVarKeyType int

// String defines a Stringer function for debugging and pretty printing.
func (k IndexAdviseVarKeyType) String() string {
	return "index_advise_var"
}

// IndexAdviseVarKey is a variable key for index advise.
const IndexAdviseVarKey IndexAdviseVarKeyType = 0

相关信息

tidb 源码目录

相关文章

tidb adapter 源码

tidb admin 源码

tidb admin_plugins 源码

tidb admin_telemetry 源码

tidb aggregate 源码

tidb analyze 源码

tidb analyze_col 源码

tidb analyze_col_v2 源码

tidb analyze_fast 源码

tidb analyze_global_stats 源码

0  赞