tidb analyze_global_stats 源码

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

tidb analyze_global_stats 代码

文件路径:/executor/analyze_global_stats.go

// Copyright 2022 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"

	"github.com/pingcap/tidb/domain"
	"github.com/pingcap/tidb/infoschema"
	"github.com/pingcap/tidb/types"
	"github.com/pingcap/tidb/util/logutil"
	"go.uber.org/zap"
)

type globalStatsKey struct {
	tableID int64
	indexID int64
}

type globalStatsInfo struct {
	isIndex int
	// When the `isIndex == 0`, histIDs will be the column IDs.
	// Otherwise, histIDs will only contain the index ID.
	histIDs      []int64
	statsVersion int
}

// globalStatsMap is a map used to store which partition tables and the corresponding indexes need global-level stats.
// The meaning of key in map is the structure that used to store the tableID and indexID.
// The meaning of value in map is some additional information needed to build global-level stats.
type globalStatsMap map[globalStatsKey]globalStatsInfo

func (e *AnalyzeExec) handleGlobalStats(ctx context.Context, needGlobalStats bool, globalStatsMap globalStatsMap) error {
	if !needGlobalStats {
		return nil
	}
	statsHandle := domain.GetDomain(e.ctx).StatsHandle()
	for globalStatsID, info := range globalStatsMap {
		globalOpts := e.opts
		if e.OptionsMap != nil {
			if v2Options, ok := e.OptionsMap[globalStatsID.tableID]; ok {
				globalOpts = v2Options.FilledOpts
			}
		}
		globalStats, err := statsHandle.MergePartitionStats2GlobalStatsByTableID(e.ctx, globalOpts, e.ctx.GetInfoSchema().(infoschema.InfoSchema), globalStatsID.tableID, info.isIndex, info.histIDs)
		if err != nil {
			if types.ErrPartitionStatsMissing.Equal(err) || types.ErrPartitionColumnStatsMissing.Equal(err) {
				// When we find some partition-level stats are missing, we need to report warning.
				e.ctx.GetSessionVars().StmtCtx.AppendWarning(err)
				continue
			}
			return err
		}
		for i := 0; i < globalStats.Num; i++ {
			hg, cms, topN := globalStats.Hg[i], globalStats.Cms[i], globalStats.TopN[i]
			// fms for global stats doesn't need to dump to kv.
			err = statsHandle.SaveStatsToStorage(globalStatsID.tableID, globalStats.Count, info.isIndex, hg, cms, topN, info.statsVersion, 1, true)
			if err != nil {
				logutil.Logger(ctx).Error("save global-level stats to storage failed", zap.Error(err))
			}
			// Dump stats to historical storage.
			if err := e.recordHistoricalStats(globalStatsID.tableID); err != nil {
				logutil.BgLogger().Error("record historical stats failed", zap.Error(err))
			}
		}
	}
	return nil
}

相关信息

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_idx 源码

0  赞