90 Star 494 Fork 152

平凯星辰(北京)科技有限公司/tidb

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
local_aggregate.go 7.18 KB
一键复制 编辑 原始数据 按行查看 历史
Shen Li 提交于 2017-10-09 01:31 . localstore: Fix make check shadow (#4718)
// Copyright 2016 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 localstore
import (
"bytes"
"github.com/juju/errors"
"github.com/pingcap/tidb/distsql/xeval"
"github.com/pingcap/tidb/util/codec"
"github.com/pingcap/tidb/util/types"
"github.com/pingcap/tipb/go-tipb"
)
var singleGroup = []byte("SingleGroup")
func (rs *localRegion) getGroupKey(ctx *selectContext) ([]byte, error) {
items := ctx.sel.GetGroupBy()
if len(items) == 0 {
return singleGroup, nil
}
vals := make([]types.Datum, 0, len(items))
for _, item := range items {
v, err := ctx.eval.Eval(item.Expr)
if err != nil {
return nil, errors.Trace(err)
}
vals = append(vals, v)
}
bs, err := codec.EncodeValue(nil, vals...)
if err != nil {
return nil, errors.Trace(err)
}
return bs, nil
}
// Update aggregate functions with rows.
func (rs *localRegion) aggregate(ctx *selectContext, h int64, row map[int64][]byte) error {
// Put row data into evaluate context for later evaluation.
err := rs.setColumnValueToCtx(ctx, h, row, ctx.aggColumns)
if err != nil {
return errors.Trace(err)
}
// Get group key.
gk, err := rs.getGroupKey(ctx)
if err != nil {
return errors.Trace(err)
}
if _, ok := ctx.groups[string(gk)]; !ok {
ctx.groups[string(gk)] = true
ctx.groupKeys = append(ctx.groupKeys, gk)
}
// Update aggregate funcs.
for _, agg := range ctx.aggregates {
agg.currentGroup = gk
args := make([]types.Datum, 0, len(agg.expr.Children))
// Evaluate arguments.
for _, x := range agg.expr.Children {
cv, err1 := ctx.eval.Eval(x)
if err1 != nil {
return errors.Trace(err1)
}
args = append(args, cv)
}
err = agg.update(ctx, args)
if err != nil {
return errors.Trace(err)
}
}
return nil
}
// Partial result for a single aggregate group.
type aggItem struct {
// Number of rows, this could be used in cout/avg
count uint64
// This could be used to store sum/max/min/first
value types.Datum
// TODO: support group_concat
buffer *bytes.Buffer // Buffer is used for group_concat.
// It will check if the agg has met the first row key.
gotFirstRow bool
}
// This is similar to ast.AggregateFuncExpr but use tipb.Expr.
type aggregateFuncExpr struct {
expr *tipb.Expr
currentGroup []byte
// contextPerGroupMap is used to store aggregate evaluation context.
// Each entry for a group.
contextPerGroupMap map[string](*aggItem)
}
// Clear clears aggregate computing context.
func (n *aggregateFuncExpr) clear() {
n.currentGroup = []byte{}
n.contextPerGroupMap = nil
}
// Update is used for update aggregate context.
func (n *aggregateFuncExpr) update(ctx *selectContext, args []types.Datum) error {
switch n.expr.GetTp() {
case tipb.ExprType_Count:
return n.updateCount(ctx, args)
case tipb.ExprType_First:
return n.updateFirst(ctx, args)
case tipb.ExprType_Sum, tipb.ExprType_Avg:
return n.updateSum(ctx, args)
case tipb.ExprType_Max:
return n.updateMaxMin(ctx, args, true)
case tipb.ExprType_Min:
return n.updateMaxMin(ctx, args, false)
}
return errors.Errorf("Unknown AggExpr: %v", n.expr.GetTp())
}
func (n *aggregateFuncExpr) toDatums(ctx *selectContext) (ds []types.Datum, err error) {
switch n.expr.GetTp() {
case tipb.ExprType_Count:
ds = n.getCountDatum()
case tipb.ExprType_First, tipb.ExprType_Max, tipb.ExprType_Min:
ds = n.getValueDatum()
case tipb.ExprType_Sum:
d, err := getSumValue(ctx, n.getAggItem())
if err != nil {
return nil, errors.Trace(err)
}
ds = []types.Datum{d}
case tipb.ExprType_Avg:
item := n.getAggItem()
sum, err := getSumValue(ctx, item)
if err != nil {
return nil, errors.Trace(err)
}
cnt := types.NewUintDatum(item.count)
ds = []types.Datum{cnt, sum}
}
return
}
func getSumValue(ctx *selectContext, item *aggItem) (types.Datum, error) {
v := item.value
var d types.Datum
if !v.IsNull() {
// For sum result, we should convert it to decimal.
de, err1 := v.ToDecimal(ctx.sc)
if err1 != nil {
return d, errors.Trace(err1)
}
d = types.NewDecimalDatum(de)
}
return d, nil
}
// Convert count to datum list.
func (n *aggregateFuncExpr) getCountDatum() []types.Datum {
item := n.getAggItem()
return []types.Datum{types.NewUintDatum(item.count)}
}
// Convert value to datum list.
func (n *aggregateFuncExpr) getValueDatum() []types.Datum {
item := n.getAggItem()
return []types.Datum{item.value}
}
var singleGroupKey = []byte("SingleGroup")
// getAggItem gets aggregate evaluation context for the current group.
// If it is nil, add a new context into contextPerGroupMap.
func (n *aggregateFuncExpr) getAggItem() *aggItem {
if n.currentGroup == nil {
n.currentGroup = singleGroupKey
}
if n.contextPerGroupMap == nil {
n.contextPerGroupMap = make(map[string](*aggItem))
}
if _, ok := n.contextPerGroupMap[string(n.currentGroup)]; !ok {
n.contextPerGroupMap[string(n.currentGroup)] = &aggItem{}
}
return n.contextPerGroupMap[string(n.currentGroup)]
}
func (n *aggregateFuncExpr) updateCount(ctx *selectContext, args []types.Datum) error {
for _, a := range args {
if a.IsNull() {
return nil
}
}
aggItem := n.getAggItem()
aggItem.count++
return nil
}
func (n *aggregateFuncExpr) updateFirst(ctx *selectContext, args []types.Datum) error {
aggItem := n.getAggItem()
if aggItem.gotFirstRow {
return nil
}
if len(args) != 1 {
return errors.New("Wrong number of args for AggFuncFirstRow")
}
aggItem.value = args[0]
aggItem.gotFirstRow = true
return nil
}
func (n *aggregateFuncExpr) updateSum(ctx *selectContext, args []types.Datum) error {
if len(args) != 1 {
// This should not happen. The length of argument list is already checked in the early stage.
// This is just in case of error.
return errors.Errorf("Wrong number of argument for sum, need 1 but get %d", len(args))
}
arg := args[0]
if arg.IsNull() {
return nil
}
aggItem := n.getAggItem()
if aggItem.value.IsNull() {
aggItem.value = arg
aggItem.count = 1
return nil
}
var err error
aggItem.value, err = xeval.ComputeArithmetic(ctx.sc, tipb.ExprType_Plus, arg, aggItem.value)
if err != nil {
return errors.Trace(err)
}
aggItem.count++
return nil
}
func (n *aggregateFuncExpr) updateMaxMin(ctx *selectContext, args []types.Datum, max bool) error {
if len(args) != 1 {
// This should not happen. The length of argument list is already checked in the early stage.
// This is just in case of error.
name := "max"
if !max {
name = "min"
}
return errors.Errorf("Wrong number of argument for %s, need 1 but get %d", name, len(args))
}
arg := args[0]
if arg.IsNull() {
return nil
}
aggItem := n.getAggItem()
if aggItem.value.IsNull() {
aggItem.value = arg
return nil
}
c, err := aggItem.value.CompareDatum(ctx.sc, &arg)
if err != nil {
return errors.Trace(err)
}
if max {
if c == -1 {
aggItem.value = arg
}
} else if c == 1 {
aggItem.value = arg
}
return nil
}
Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
Go
1
https://gitee.com/pingcap/tidb.git
git@gitee.com:pingcap/tidb.git
pingcap
tidb
tidb
v1.0.5

搜索帮助