Skip to content

Commit

Permalink
evalengine: Implement PERIOD_DIFF (vitessio#16557)
Browse files Browse the repository at this point in the history
Signed-off-by: Noble Mittal <[email protected]>
  • Loading branch information
beingnoble03 authored Aug 12, 2024
1 parent 3c2e8f9 commit 7f09809
Show file tree
Hide file tree
Showing 5 changed files with 116 additions and 0 deletions.
12 changes: 12 additions & 0 deletions go/vt/vtgate/evalengine/cached_size.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

20 changes: 20 additions & 0 deletions go/vt/vtgate/evalengine/compiler_asm.go
Original file line number Diff line number Diff line change
Expand Up @@ -4319,6 +4319,26 @@ func (asm *assembler) Fn_PERIOD_ADD() {
}, "FN PERIOD_ADD INT64(SP-2) INT64(SP-1)")
}

func (asm *assembler) Fn_PERIOD_DIFF() {
asm.adjustStack(-1)
asm.emit(func(env *ExpressionEnv) int {
if env.vm.stack[env.vm.sp-2] == nil {
env.vm.sp--
return 1
}
period1 := env.vm.stack[env.vm.sp-2].(*evalInt64).i
period2 := env.vm.stack[env.vm.sp-1].(*evalInt64).i
res, err := periodDiff(period1, period2)
if err != nil {
env.vm.err = err
return 0
}
env.vm.stack[env.vm.sp-2] = res
env.vm.sp--
return 1
}, "FN PERIOD_DIFF INT64(SP-2) INT64(SP-1)")
}

func (asm *assembler) Interval(l int) {
asm.adjustStack(-l)
asm.emit(func(env *ExpressionEnv) int {
Expand Down
55 changes: 55 additions & 0 deletions go/vt/vtgate/evalengine/fn_time.go
Original file line number Diff line number Diff line change
Expand Up @@ -181,6 +181,10 @@ type (
CallExpr
}

builtinPeriodDiff struct {
CallExpr
}

builtinDateMath struct {
CallExpr
sub bool
Expand Down Expand Up @@ -222,6 +226,7 @@ var _ IR = (*builtinWeekOfYear)(nil)
var _ IR = (*builtinYear)(nil)
var _ IR = (*builtinYearWeek)(nil)
var _ IR = (*builtinPeriodAdd)(nil)
var _ IR = (*builtinPeriodDiff)(nil)

func (call *builtinNow) eval(env *ExpressionEnv) (eval, error) {
now := env.time(call.utc)
Expand Down Expand Up @@ -2021,6 +2026,56 @@ func (call *builtinPeriodAdd) compile(c *compiler) (ctype, error) {
return ctype{Type: sqltypes.Int64, Flag: period.Flag | months.Flag | flagNullable}, nil
}

func periodDiff(period1, period2 int64) (*evalInt64, error) {
if !datetime.ValidatePeriod(period1) || !datetime.ValidatePeriod(period2) {
return nil, vterrors.NewErrorf(vtrpcpb.Code_INVALID_ARGUMENT, vterrors.WrongArguments, "Incorrect arguments to period_diff")
}
res := datetime.PeriodToMonths(period1) - datetime.PeriodToMonths(period2)
return newEvalInt64(res), nil
}

func (b *builtinPeriodDiff) eval(env *ExpressionEnv) (eval, error) {
p1, p2, err := b.arg2(env)
if err != nil {
return nil, err
}
if p1 == nil || p2 == nil {
return nil, nil
}
period1 := evalToInt64(p1)
period2 := evalToInt64(p2)
return periodDiff(period1.i, period2.i)
}

func (call *builtinPeriodDiff) compile(c *compiler) (ctype, error) {
period1, err := call.Arguments[0].compile(c)
if err != nil {
return ctype{}, err
}
period2, err := call.Arguments[1].compile(c)
if err != nil {
return ctype{}, err
}

skip := c.compileNullCheck2(period1, period2)

switch period1.Type {
case sqltypes.Int64:
default:
c.asm.Convert_xi(2)
}

switch period2.Type {
case sqltypes.Int64:
default:
c.asm.Convert_xi(1)
}

c.asm.Fn_PERIOD_DIFF()
c.asm.jumpDestination(skip)
return ctype{Type: sqltypes.Int64, Flag: period1.Flag | period2.Flag | flagNullable}, nil
}

func evalToInterval(itv eval, unit datetime.IntervalType, negate bool) *datetime.Interval {
switch itv := itv.(type) {
case *evalBytes:
Expand Down
22 changes: 22 additions & 0 deletions go/vt/vtgate/evalengine/testcases/cases.go
Original file line number Diff line number Diff line change
Expand Up @@ -156,6 +156,7 @@ var Cases = []TestCase{
{Run: FnYear},
{Run: FnYearWeek},
{Run: FnPeriodAdd},
{Run: FnPeriodDiff},
{Run: FnInetAton},
{Run: FnInetNtoa},
{Run: FnInet6Aton},
Expand Down Expand Up @@ -2245,6 +2246,27 @@ func FnPeriodAdd(yield Query) {
}
}

func FnPeriodDiff(yield Query) {
for _, p1 := range inputBitwise {
for _, p2 := range inputBitwise {
yield(fmt.Sprintf("PERIOD_DIFF(%s, %s)", p1, p2), nil)
}
}
for _, p1 := range inputPeriods {
for _, p2 := range inputPeriods {
yield(fmt.Sprintf("PERIOD_DIFF(%s, %s)", p1, p2), nil)
}
}

mysqlDocSamples := []string{
`PERIOD_DIFF(200802,200703)`,
}

for _, q := range mysqlDocSamples {
yield(q, nil)
}
}

func FnInetAton(yield Query) {
for _, d := range ipInputs {
yield(fmt.Sprintf("INET_ATON(%s)", d), nil)
Expand Down
7 changes: 7 additions & 0 deletions go/vt/vtgate/evalengine/translate_builtin.go
Original file line number Diff line number Diff line change
Expand Up @@ -535,6 +535,13 @@ func (ast *astCompiler) translateFuncExpr(fn *sqlparser.FuncExpr) (IR, error) {
default:
return nil, argError(method)
}
case "period_diff":
switch len(args) {
case 2:
return &builtinPeriodDiff{CallExpr: call}, nil
default:
return nil, argError(method)
}
case "inet_aton":
if len(args) != 1 {
return nil, argError(method)
Expand Down

0 comments on commit 7f09809

Please sign in to comment.