Files
tidb/expression/builtin_control.go

702 lines
22 KiB
Go

// Copyright 2015 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 expression
import (
"github.com/cznic/mathutil"
"github.com/juju/errors"
"github.com/pingcap/tidb/context"
"github.com/pingcap/tidb/mysql"
"github.com/pingcap/tidb/util/charset"
"github.com/pingcap/tidb/util/types"
"github.com/pingcap/tidb/util/types/json"
"github.com/pingcap/tipb/go-tipb"
)
var (
_ functionClass = &caseWhenFunctionClass{}
_ functionClass = &ifFunctionClass{}
_ functionClass = &ifNullFunctionClass{}
)
var (
_ builtinFunc = &builtinCaseWhenIntSig{}
_ builtinFunc = &builtinCaseWhenRealSig{}
_ builtinFunc = &builtinCaseWhenDecimalSig{}
_ builtinFunc = &builtinCaseWhenStringSig{}
_ builtinFunc = &builtinCaseWhenTimeSig{}
_ builtinFunc = &builtinCaseWhenDurationSig{}
_ builtinFunc = &builtinIfNullIntSig{}
_ builtinFunc = &builtinIfNullRealSig{}
_ builtinFunc = &builtinIfNullDecimalSig{}
_ builtinFunc = &builtinIfNullStringSig{}
_ builtinFunc = &builtinIfNullTimeSig{}
_ builtinFunc = &builtinIfNullDurationSig{}
_ builtinFunc = &builtinIfNullJSONSig{}
_ builtinFunc = &builtinIfIntSig{}
_ builtinFunc = &builtinIfRealSig{}
_ builtinFunc = &builtinIfDecimalSig{}
_ builtinFunc = &builtinIfStringSig{}
_ builtinFunc = &builtinIfTimeSig{}
_ builtinFunc = &builtinIfDurationSig{}
_ builtinFunc = &builtinIfJSONSig{}
)
type caseWhenFunctionClass struct {
baseFunctionClass
}
// Infer result type for builtin IF, IFNULL && NULLIF.
func inferType4ControlFuncs(tp1, tp2 *types.FieldType) *types.FieldType {
retTp, typeClass := &types.FieldType{}, types.ClassString
if tp1.Tp == mysql.TypeNull {
*retTp, typeClass = *tp2, tp2.ToClass()
// If both arguments are NULL, make resulting type BINARY(0).
if tp2.Tp == mysql.TypeNull {
retTp.Tp, typeClass = mysql.TypeString, types.ClassString
retTp.Flen, retTp.Decimal = 0, 0
types.SetBinChsClnFlag(retTp)
}
} else if tp2.Tp == mysql.TypeNull {
*retTp, typeClass = *tp1, tp1.ToClass()
} else {
var unsignedFlag uint
typeClass = types.AggTypeClass([]*types.FieldType{tp1, tp2}, &unsignedFlag)
retTp = types.AggFieldType([]*types.FieldType{tp1, tp2})
if typeClass == types.ClassInt {
retTp.Decimal = 0
} else {
if tp1.Decimal == types.UnspecifiedLength || tp2.Decimal == types.UnspecifiedLength {
retTp.Decimal = types.UnspecifiedLength
} else {
retTp.Decimal = mathutil.Max(tp1.Decimal, tp2.Decimal)
}
}
if types.IsNonBinaryStr(tp1) && !types.IsBinaryStr(tp2) {
retTp.Charset, retTp.Collate, retTp.Flag = charset.CharsetUTF8, charset.CollationUTF8, 0
if mysql.HasBinaryFlag(tp1.Flag) {
retTp.Flag |= mysql.BinaryFlag
}
} else if types.IsNonBinaryStr(tp2) && !types.IsBinaryStr(tp1) {
retTp.Charset, retTp.Collate, retTp.Flag = charset.CharsetUTF8, charset.CollationUTF8, 0
if mysql.HasBinaryFlag(tp2.Flag) {
retTp.Flag |= mysql.BinaryFlag
}
} else if types.IsBinaryStr(tp1) || types.IsBinaryStr(tp2) || typeClass != types.ClassString {
types.SetBinChsClnFlag(retTp)
} else {
retTp.Charset, retTp.Collate, retTp.Flag = charset.CharsetUTF8, charset.CollationUTF8, 0
}
if typeClass == types.ClassDecimal || typeClass == types.ClassInt {
unsignedFlag1, unsignedFlag2 := mysql.HasUnsignedFlag(tp1.Flag), mysql.HasUnsignedFlag(tp2.Flag)
flagLen1, flagLen2 := 0, 0
if !unsignedFlag1 {
flagLen1 = 1
}
if !unsignedFlag2 {
flagLen2 = 1
}
len1 := tp1.Flen - flagLen1
len2 := tp2.Flen - flagLen2
if tp1.Decimal != types.UnspecifiedLength {
len1 -= tp1.Decimal
}
if tp1.Decimal != types.UnspecifiedLength {
len2 -= tp2.Decimal
}
retTp.Flen = mathutil.Max(len1, len2) + retTp.Decimal + 1
} else {
retTp.Flen = mathutil.Max(tp1.Flen, tp2.Flen)
}
}
// Fix decimal for int and string.
fieldTp := fieldTp2EvalTp(retTp)
if fieldTp == tpInt {
retTp.Decimal = 0
} else if fieldTp == tpString {
if tp1.Tp != mysql.TypeNull || tp2.Tp != mysql.TypeNull {
retTp.Decimal = types.UnspecifiedLength
}
}
return retTp
}
func (c *caseWhenFunctionClass) getFunction(ctx context.Context, args []Expression) (sig builtinFunc, err error) {
if err = c.verifyArgs(args); err != nil {
return nil, errors.Trace(err)
}
l := len(args)
// Fill in each 'THEN' clause parameter type.
fieldTps := make([]*types.FieldType, 0, (l+1)/2)
decimal, flen, isBinaryStr := args[1].GetType().Decimal, 0, false
for i := 1; i < l; i += 2 {
fieldTps = append(fieldTps, args[i].GetType())
decimal = mathutil.Max(decimal, args[i].GetType().Decimal)
flen = mathutil.Max(flen, args[i].GetType().Flen)
isBinaryStr = isBinaryStr || types.IsBinaryStr(args[i].GetType())
}
if l%2 == 1 {
fieldTps = append(fieldTps, args[l-1].GetType())
decimal = mathutil.Max(decimal, args[l-1].GetType().Decimal)
flen = mathutil.Max(flen, args[l-1].GetType().Flen)
isBinaryStr = isBinaryStr || types.IsBinaryStr(args[l-1].GetType())
}
fieldTp := types.AggFieldType(fieldTps)
tp := fieldTp2EvalTp(fieldTp)
if tp == tpInt {
decimal = 0
}
fieldTp.Decimal, fieldTp.Flen = decimal, flen
if fieldTp.ToClass() == types.ClassString && !isBinaryStr {
fieldTp.Charset, fieldTp.Collate = mysql.DefaultCharset, mysql.DefaultCollationName
}
// Set retType to BINARY(0) if all arguments are of type NULL.
if fieldTp.Tp == mysql.TypeNull {
fieldTp.Flen, fieldTp.Decimal = 0, -1
types.SetBinChsClnFlag(fieldTp)
}
argTps := make([]evalTp, 0, l)
for i := 0; i < l-1; i += 2 {
argTps = append(argTps, tpInt, tp)
}
if l%2 == 1 {
argTps = append(argTps, tp)
}
bf := newBaseBuiltinFuncWithTp(args, ctx, tp, argTps...)
bf.tp = fieldTp
switch tp {
case tpInt:
bf.tp.Decimal = 0
sig = &builtinCaseWhenIntSig{baseIntBuiltinFunc{bf}}
case tpReal:
sig = &builtinCaseWhenRealSig{baseRealBuiltinFunc{bf}}
case tpDecimal:
sig = &builtinCaseWhenDecimalSig{baseDecimalBuiltinFunc{bf}}
case tpString:
bf.tp.Decimal = types.UnspecifiedLength
sig = &builtinCaseWhenStringSig{baseStringBuiltinFunc{bf}}
case tpDatetime, tpTimestamp:
sig = &builtinCaseWhenTimeSig{baseTimeBuiltinFunc{bf}}
case tpDuration:
sig = &builtinCaseWhenDurationSig{baseDurationBuiltinFunc{bf}}
}
return sig.setSelf(sig), nil
}
type builtinCaseWhenIntSig struct {
baseIntBuiltinFunc
}
// evalInt evals a builtinCaseWhenIntSig.
// See https://dev.mysql.com/doc/refman/5.7/en/case.html
func (b *builtinCaseWhenIntSig) evalInt(row []types.Datum) (ret int64, isNull bool, err error) {
sc := b.ctx.GetSessionVars().StmtCtx
var condition int64
args, l := b.getArgs(), len(b.getArgs())
for i := 0; i < l-1; i += 2 {
condition, isNull, err = args[i].EvalInt(row, sc)
if err != nil {
return 0, isNull, errors.Trace(err)
}
if isNull || condition == 0 {
continue
}
ret, isNull, err = args[i+1].EvalInt(row, sc)
return ret, isNull, errors.Trace(err)
}
// when clause(condition, result) -> args[i], args[i+1]; (i >= 0 && i+1 < l-1)
// else clause -> args[l-1]
// If case clause has else clause, l%2 == 1.
if l%2 == 1 {
ret, isNull, err = args[l-1].EvalInt(row, sc)
return ret, isNull, errors.Trace(err)
}
return ret, true, nil
}
type builtinCaseWhenRealSig struct {
baseRealBuiltinFunc
}
// evalReal evals a builtinCaseWhenRealSig.
// See https://dev.mysql.com/doc/refman/5.7/en/case.html
func (b *builtinCaseWhenRealSig) evalReal(row []types.Datum) (ret float64, isNull bool, err error) {
sc := b.ctx.GetSessionVars().StmtCtx
var condition int64
args, l := b.getArgs(), len(b.getArgs())
for i := 0; i < l-1; i += 2 {
condition, isNull, err = args[i].EvalInt(row, sc)
if err != nil {
return 0, isNull, errors.Trace(err)
}
if isNull || condition == 0 {
continue
}
ret, isNull, err = args[i+1].EvalReal(row, sc)
return ret, isNull, errors.Trace(err)
}
// when clause(condition, result) -> args[i], args[i+1]; (i >= 0 && i+1 < l-1)
// else clause -> args[l-1]
// If case clause has else clause, l%2 == 1.
if l%2 == 1 {
ret, isNull, err = args[l-1].EvalReal(row, sc)
return ret, isNull, errors.Trace(err)
}
return ret, true, nil
}
type builtinCaseWhenDecimalSig struct {
baseDecimalBuiltinFunc
}
// evalDecimal evals a builtinCaseWhenDecimalSig.
// See https://dev.mysql.com/doc/refman/5.7/en/case.html
func (b *builtinCaseWhenDecimalSig) evalDecimal(row []types.Datum) (ret *types.MyDecimal, isNull bool, err error) {
sc := b.ctx.GetSessionVars().StmtCtx
var condition int64
args, l := b.getArgs(), len(b.getArgs())
for i := 0; i < l-1; i += 2 {
condition, isNull, err = args[i].EvalInt(row, sc)
if err != nil {
return nil, isNull, errors.Trace(err)
}
if isNull || condition == 0 {
continue
}
ret, isNull, err = args[i+1].EvalDecimal(row, sc)
return ret, isNull, errors.Trace(err)
}
// when clause(condition, result) -> args[i], args[i+1]; (i >= 0 && i+1 < l-1)
// else clause -> args[l-1]
// If case clause has else clause, l%2 == 1.
if l%2 == 1 {
ret, isNull, err = args[l-1].EvalDecimal(row, sc)
return ret, isNull, errors.Trace(err)
}
return ret, true, nil
}
type builtinCaseWhenStringSig struct {
baseStringBuiltinFunc
}
// evalString evals a builtinCaseWhenStringSig.
// See https://dev.mysql.com/doc/refman/5.7/en/case.html
func (b *builtinCaseWhenStringSig) evalString(row []types.Datum) (ret string, isNull bool, err error) {
sc := b.ctx.GetSessionVars().StmtCtx
var condition int64
args, l := b.getArgs(), len(b.getArgs())
for i := 0; i < l-1; i += 2 {
condition, isNull, err = args[i].EvalInt(row, sc)
if err != nil {
return "", isNull, errors.Trace(err)
}
if isNull || condition == 0 {
continue
}
ret, isNull, err = args[i+1].EvalString(row, sc)
return ret, isNull, errors.Trace(err)
}
// when clause(condition, result) -> args[i], args[i+1]; (i >= 0 && i+1 < l-1)
// else clause -> args[l-1]
// If case clause has else clause, l%2 == 1.
if l%2 == 1 {
ret, isNull, err = args[l-1].EvalString(row, sc)
return ret, isNull, errors.Trace(err)
}
return ret, true, nil
}
type builtinCaseWhenTimeSig struct {
baseTimeBuiltinFunc
}
// evalTime evals a builtinCaseWhenTimeSig.
// See https://dev.mysql.com/doc/refman/5.7/en/case.html
func (b *builtinCaseWhenTimeSig) evalTime(row []types.Datum) (ret types.Time, isNull bool, err error) {
sc := b.ctx.GetSessionVars().StmtCtx
var condition int64
args, l := b.getArgs(), len(b.getArgs())
for i := 0; i < l-1; i += 2 {
condition, isNull, err = args[i].EvalInt(row, sc)
if err != nil {
return ret, isNull, errors.Trace(err)
}
if isNull || condition == 0 {
continue
}
ret, isNull, err = args[i+1].EvalTime(row, sc)
return ret, isNull, errors.Trace(err)
}
// when clause(condition, result) -> args[i], args[i+1]; (i >= 0 && i+1 < l-1)
// else clause -> args[l-1]
// If case clause has else clause, l%2 == 1.
if l%2 == 1 {
ret, isNull, err = args[l-1].EvalTime(row, sc)
return ret, isNull, errors.Trace(err)
}
return ret, true, nil
}
type builtinCaseWhenDurationSig struct {
baseDurationBuiltinFunc
}
// evalDuration evals a builtinCaseWhenDurationSig.
// See https://dev.mysql.com/doc/refman/5.7/en/case.html
func (b *builtinCaseWhenDurationSig) evalDuration(row []types.Datum) (ret types.Duration, isNull bool, err error) {
sc := b.ctx.GetSessionVars().StmtCtx
var condition int64
args, l := b.getArgs(), len(b.getArgs())
for i := 0; i < l-1; i += 2 {
condition, isNull, err = args[i].EvalInt(row, sc)
if err != nil {
return ret, false, errors.Trace(err)
}
if isNull || condition == 0 {
continue
}
ret, isNull, err = args[i+1].EvalDuration(row, sc)
return ret, isNull, errors.Trace(err)
}
// when clause(condition, result) -> args[i], args[i+1]; (i >= 0 && i+1 < l-1)
// else clause -> args[l-1]
// If case clause has else clause, l%2 == 1.
if l%2 == 1 {
ret, isNull, err = args[l-1].EvalDuration(row, sc)
return ret, isNull, errors.Trace(err)
}
return ret, true, nil
}
type ifFunctionClass struct {
baseFunctionClass
}
// See https://dev.mysql.com/doc/refman/5.7/en/control-flow-functions.html#function_if
func (c *ifFunctionClass) getFunction(ctx context.Context, args []Expression) (sig builtinFunc, err error) {
if err = c.verifyArgs(args); err != nil {
return nil, errors.Trace(err)
}
retTp := inferType4ControlFuncs(args[1].GetType(), args[2].GetType())
evalTps := fieldTp2EvalTp(retTp)
bf := newBaseBuiltinFuncWithTp(args, ctx, evalTps, tpInt, evalTps, evalTps)
bf.tp = retTp
switch evalTps {
case tpInt:
sig = &builtinIfIntSig{baseIntBuiltinFunc{bf}}
sig.setPbCode(tipb.ScalarFuncSig_IfInt)
case tpReal:
sig = &builtinIfRealSig{baseRealBuiltinFunc{bf}}
sig.setPbCode(tipb.ScalarFuncSig_IfReal)
case tpDecimal:
sig = &builtinIfDecimalSig{baseDecimalBuiltinFunc{bf}}
sig.setPbCode(tipb.ScalarFuncSig_IfDecimal)
case tpString:
sig = &builtinIfStringSig{baseStringBuiltinFunc{bf}}
sig.setPbCode(tipb.ScalarFuncSig_IfString)
case tpDatetime, tpTimestamp:
sig = &builtinIfTimeSig{baseTimeBuiltinFunc{bf}}
sig.setPbCode(tipb.ScalarFuncSig_IfTime)
case tpDuration:
sig = &builtinIfDurationSig{baseDurationBuiltinFunc{bf}}
sig.setPbCode(tipb.ScalarFuncSig_IfDuration)
case tpJSON:
sig = &builtinIfJSONSig{baseJSONBuiltinFunc{bf}}
sig.setPbCode(tipb.ScalarFuncSig_IfJson)
}
return sig.setSelf(sig), nil
}
type builtinIfIntSig struct {
baseIntBuiltinFunc
}
func (b *builtinIfIntSig) evalInt(row []types.Datum) (ret int64, isNull bool, err error) {
sc := b.ctx.GetSessionVars().StmtCtx
arg0, isNull0, err := b.args[0].EvalInt(row, sc)
if err != nil {
return 0, false, errors.Trace(err)
}
arg1, isNull1, err := b.args[1].EvalInt(row, sc)
if (!isNull0 && arg0 != 0) || err != nil {
return arg1, isNull1, errors.Trace(err)
}
arg2, isNull2, err := b.args[2].EvalInt(row, sc)
return arg2, isNull2, errors.Trace(err)
}
type builtinIfRealSig struct {
baseRealBuiltinFunc
}
func (b *builtinIfRealSig) evalReal(row []types.Datum) (ret float64, isNull bool, err error) {
sc := b.ctx.GetSessionVars().StmtCtx
arg0, isNull0, err := b.args[0].EvalInt(row, sc)
if err != nil {
return 0, false, errors.Trace(err)
}
arg1, isNull1, err := b.args[1].EvalReal(row, sc)
if (!isNull0 && arg0 != 0) || err != nil {
return arg1, isNull1, errors.Trace(err)
}
arg2, isNull2, err := b.args[2].EvalReal(row, sc)
return arg2, isNull2, errors.Trace(err)
}
type builtinIfDecimalSig struct {
baseDecimalBuiltinFunc
}
func (b *builtinIfDecimalSig) evalDecimal(row []types.Datum) (ret *types.MyDecimal, isNull bool, err error) {
sc := b.ctx.GetSessionVars().StmtCtx
arg0, isNull0, err := b.args[0].EvalInt(row, sc)
if err != nil {
return nil, false, errors.Trace(err)
}
arg1, isNull1, err := b.args[1].EvalDecimal(row, sc)
if (!isNull0 && arg0 != 0) || err != nil {
return arg1, isNull1, errors.Trace(err)
}
arg2, isNull2, err := b.args[2].EvalDecimal(row, sc)
return arg2, isNull2, errors.Trace(err)
}
type builtinIfStringSig struct {
baseStringBuiltinFunc
}
func (b *builtinIfStringSig) evalString(row []types.Datum) (ret string, isNull bool, err error) {
sc := b.ctx.GetSessionVars().StmtCtx
arg0, isNull0, err := b.args[0].EvalInt(row, sc)
if err != nil {
return "", false, errors.Trace(err)
}
arg1, isNull1, err := b.args[1].EvalString(row, sc)
if (!isNull0 && arg0 != 0) || err != nil {
return arg1, isNull1, errors.Trace(err)
}
arg2, isNull2, err := b.args[2].EvalString(row, sc)
return arg2, isNull2, errors.Trace(err)
}
type builtinIfTimeSig struct {
baseTimeBuiltinFunc
}
func (b *builtinIfTimeSig) evalTime(row []types.Datum) (ret types.Time, isNull bool, err error) {
sc := b.ctx.GetSessionVars().StmtCtx
arg0, isNull0, err := b.args[0].EvalInt(row, sc)
if err != nil {
return ret, false, errors.Trace(err)
}
arg1, isNull1, err := b.args[1].EvalTime(row, sc)
if (!isNull0 && arg0 != 0) || err != nil {
return arg1, isNull1, errors.Trace(err)
}
arg2, isNull2, err := b.args[2].EvalTime(row, sc)
return arg2, isNull2, errors.Trace(err)
}
type builtinIfDurationSig struct {
baseDurationBuiltinFunc
}
func (b *builtinIfDurationSig) evalDuration(row []types.Datum) (ret types.Duration, isNull bool, err error) {
sc := b.ctx.GetSessionVars().StmtCtx
arg0, isNull0, err := b.args[0].EvalInt(row, sc)
if err != nil {
return ret, false, errors.Trace(err)
}
arg1, isNull1, err := b.args[1].EvalDuration(row, sc)
if (!isNull0 && arg0 != 0) || err != nil {
return arg1, isNull1, errors.Trace(err)
}
arg2, isNull2, err := b.args[2].EvalDuration(row, sc)
return arg2, isNull2, errors.Trace(err)
}
type builtinIfJSONSig struct {
baseJSONBuiltinFunc
}
func (b *builtinIfJSONSig) evalJSON(row []types.Datum) (ret json.JSON, isNull bool, err error) {
sc := b.ctx.GetSessionVars().StmtCtx
arg0, isNull0, err := b.args[0].EvalInt(row, sc)
if err != nil {
return ret, false, errors.Trace(err)
}
arg1, isNull1, err := b.args[1].EvalJSON(row, sc)
if err != nil {
return ret, false, errors.Trace(err)
}
arg2, isNull2, err := b.args[2].EvalJSON(row, sc)
if err != nil {
return ret, false, errors.Trace(err)
}
switch {
case isNull0 || arg0 == 0:
ret, isNull = arg2, isNull2
case arg0 != 0:
ret, isNull = arg1, isNull1
}
return
}
type ifNullFunctionClass struct {
baseFunctionClass
}
func (c *ifNullFunctionClass) getFunction(ctx context.Context, args []Expression) (sig builtinFunc, err error) {
if err = errors.Trace(c.verifyArgs(args)); err != nil {
return nil, errors.Trace(err)
}
tp0, tp1 := args[0].GetType(), args[1].GetType()
retTp := inferType4ControlFuncs(tp0, tp1)
retTp.Flag |= (tp0.Flag & mysql.NotNullFlag) | (tp1.Flag & mysql.NotNullFlag)
if tp0.Tp == mysql.TypeNull && tp1.Tp == mysql.TypeNull {
retTp.Tp = mysql.TypeNull
retTp.Flen, retTp.Decimal = 0, -1
types.SetBinChsClnFlag(retTp)
}
evalTps := fieldTp2EvalTp(retTp)
bf := newBaseBuiltinFuncWithTp(args, ctx, evalTps, evalTps, evalTps)
bf.tp = retTp
switch evalTps {
case tpInt:
sig = &builtinIfNullIntSig{baseIntBuiltinFunc{bf}}
sig.setPbCode(tipb.ScalarFuncSig_IfNullInt)
case tpReal:
sig = &builtinIfNullRealSig{baseRealBuiltinFunc{bf}}
sig.setPbCode(tipb.ScalarFuncSig_IfNullReal)
case tpDecimal:
sig = &builtinIfNullDecimalSig{baseDecimalBuiltinFunc{bf}}
sig.setPbCode(tipb.ScalarFuncSig_IfNullDecimal)
case tpString:
sig = &builtinIfNullStringSig{baseStringBuiltinFunc{bf}}
sig.setPbCode(tipb.ScalarFuncSig_IfNullString)
case tpDatetime, tpTimestamp:
sig = &builtinIfNullTimeSig{baseTimeBuiltinFunc{bf}}
sig.setPbCode(tipb.ScalarFuncSig_IfNullTime)
case tpDuration:
sig = &builtinIfNullDurationSig{baseDurationBuiltinFunc{bf}}
sig.setPbCode(tipb.ScalarFuncSig_IfNullDuration)
case tpJSON:
sig = &builtinIfNullJSONSig{baseJSONBuiltinFunc{bf}}
sig.setPbCode(tipb.ScalarFuncSig_IfNullJson)
}
return sig.setSelf(sig), nil
}
type builtinIfNullIntSig struct {
baseIntBuiltinFunc
}
func (b *builtinIfNullIntSig) evalInt(row []types.Datum) (int64, bool, error) {
sc := b.ctx.GetSessionVars().StmtCtx
arg0, isNull, err := b.args[0].EvalInt(row, sc)
if !isNull || err != nil {
return arg0, false, errors.Trace(err)
}
arg1, isNull, err := b.args[1].EvalInt(row, sc)
return arg1, isNull, errors.Trace(err)
}
type builtinIfNullRealSig struct {
baseRealBuiltinFunc
}
func (b *builtinIfNullRealSig) evalReal(row []types.Datum) (float64, bool, error) {
sc := b.ctx.GetSessionVars().StmtCtx
arg0, isNull, err := b.args[0].EvalReal(row, sc)
if !isNull || err != nil {
return arg0, false, errors.Trace(err)
}
arg1, isNull, err := b.args[1].EvalReal(row, sc)
return arg1, isNull, errors.Trace(err)
}
type builtinIfNullDecimalSig struct {
baseDecimalBuiltinFunc
}
func (b *builtinIfNullDecimalSig) evalDecimal(row []types.Datum) (*types.MyDecimal, bool, error) {
sc := b.ctx.GetSessionVars().StmtCtx
arg0, isNull, err := b.args[0].EvalDecimal(row, sc)
if !isNull || err != nil {
return arg0, false, errors.Trace(err)
}
arg1, isNull, err := b.args[1].EvalDecimal(row, sc)
return arg1, isNull, errors.Trace(err)
}
type builtinIfNullStringSig struct {
baseStringBuiltinFunc
}
func (b *builtinIfNullStringSig) evalString(row []types.Datum) (string, bool, error) {
sc := b.ctx.GetSessionVars().StmtCtx
arg0, isNull, err := b.args[0].EvalString(row, sc)
if !isNull || err != nil {
return arg0, false, errors.Trace(err)
}
arg1, isNull, err := b.args[1].EvalString(row, sc)
return arg1, isNull, errors.Trace(err)
}
type builtinIfNullTimeSig struct {
baseTimeBuiltinFunc
}
func (b *builtinIfNullTimeSig) evalTime(row []types.Datum) (types.Time, bool, error) {
sc := b.ctx.GetSessionVars().StmtCtx
arg0, isNull, err := b.args[0].EvalTime(row, sc)
if !isNull || err != nil {
return arg0, false, errors.Trace(err)
}
arg1, isNull, err := b.args[1].EvalTime(row, sc)
return arg1, isNull, errors.Trace(err)
}
type builtinIfNullDurationSig struct {
baseDurationBuiltinFunc
}
func (b *builtinIfNullDurationSig) evalDuration(row []types.Datum) (types.Duration, bool, error) {
sc := b.ctx.GetSessionVars().StmtCtx
arg0, isNull, err := b.args[0].EvalDuration(row, sc)
if !isNull || err != nil {
return arg0, false, errors.Trace(err)
}
arg1, isNull, err := b.args[1].EvalDuration(row, sc)
return arg1, isNull, errors.Trace(err)
}
type builtinIfNullJSONSig struct {
baseJSONBuiltinFunc
}
func (b *builtinIfNullJSONSig) evalJSON(row []types.Datum) (json.JSON, bool, error) {
sc := b.ctx.GetSessionVars().StmtCtx
arg0, isNull, err := b.args[0].EvalJSON(row, sc)
if !isNull {
return arg0, false, errors.Trace(err)
}
arg1, isNull, err := b.args[1].EvalJSON(row, sc)
return arg1, isNull, errors.Trace(err)
}