mirror of
https://github.com/grafana/grafana.git
synced 2025-01-16 19:52:33 -06:00
040ce40113
Co-authored-by: Yuriy Tseretyan <yuriy.tseretyan@grafana.com> Co-authored-by: gillesdemey <gilles.de.mey@gmail.com>
523 lines
12 KiB
Go
523 lines
12 KiB
Go
package mathexp
|
|
|
|
import (
|
|
"fmt"
|
|
"math"
|
|
"reflect"
|
|
"runtime"
|
|
|
|
"github.com/grafana/grafana-plugin-sdk-go/data"
|
|
"github.com/grafana/grafana/pkg/expr/mathexp/parse"
|
|
)
|
|
|
|
// Expr holds a parsed math command expression.
|
|
type Expr struct {
|
|
*parse.Tree
|
|
}
|
|
|
|
// State embeds a parsed Expr with variables and their results
|
|
// so the expression can act on them.
|
|
type State struct {
|
|
*Expr
|
|
Vars Vars
|
|
// Could hold more properties that change behavior around:
|
|
// - Unions (How many result A and many Result B in case A + B are joined)
|
|
// - NaN/Null behavior
|
|
RefID string
|
|
}
|
|
|
|
// Vars holds the results of datasource queries or other expression commands.
|
|
type Vars map[string]Results
|
|
|
|
// New creates a new expression tree
|
|
func New(expr string, funcs ...map[string]parse.Func) (*Expr, error) {
|
|
funcs = append(funcs, builtins)
|
|
t, err := parse.Parse(expr, funcs...)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
e := &Expr{
|
|
Tree: t,
|
|
}
|
|
return e, nil
|
|
}
|
|
|
|
// Execute applies a parse expression to the context and executes it
|
|
func (e *Expr) Execute(refID string, vars Vars) (r Results, err error) {
|
|
s := &State{
|
|
Expr: e,
|
|
Vars: vars,
|
|
RefID: refID,
|
|
}
|
|
return e.executeState(s)
|
|
}
|
|
|
|
func (e *Expr) executeState(s *State) (r Results, err error) {
|
|
defer errRecover(&err, s)
|
|
r, err = s.walk(e.Tree.Root)
|
|
return
|
|
}
|
|
|
|
// errRecover is the handler that turns panics into returns from the top
|
|
// level of Parse.
|
|
func errRecover(errp *error, s *State) {
|
|
e := recover()
|
|
if e != nil {
|
|
switch err := e.(type) {
|
|
case runtime.Error:
|
|
panic(e)
|
|
case error:
|
|
*errp = err
|
|
default:
|
|
panic(e)
|
|
}
|
|
}
|
|
}
|
|
|
|
// walk is the top level function to walk a parsed expression
|
|
// with its associate variables
|
|
func (e *State) walk(node parse.Node) (res Results, err error) {
|
|
switch node := node.(type) {
|
|
case *parse.ScalarNode:
|
|
res = NewScalarResults(e.RefID, &node.Float64)
|
|
case *parse.VarNode:
|
|
res = e.Vars[node.Name]
|
|
case *parse.BinaryNode:
|
|
res, err = e.walkBinary(node)
|
|
case *parse.UnaryNode:
|
|
res, err = e.walkUnary(node)
|
|
case *parse.FuncNode:
|
|
res, err = e.walkFunc(node)
|
|
default:
|
|
return res, fmt.Errorf("expr: can not walk node type: %s", node.Type())
|
|
}
|
|
return
|
|
}
|
|
|
|
func (e *State) walkUnary(node *parse.UnaryNode) (Results, error) {
|
|
a, err := e.walk(node.Arg)
|
|
if err != nil {
|
|
return Results{}, err
|
|
}
|
|
newResults := Results{}
|
|
for _, val := range a.Values {
|
|
var newVal Value
|
|
switch rt := val.(type) {
|
|
case Scalar:
|
|
newVal = NewScalar(e.RefID, nil)
|
|
f := rt.GetFloat64Value()
|
|
if f != nil {
|
|
newF, err := unaryOp(node.OpStr, *f)
|
|
if err != nil {
|
|
return newResults, err
|
|
}
|
|
newVal = NewScalar(e.RefID, &newF)
|
|
}
|
|
case Number:
|
|
newVal, err = e.unaryNumber(rt, node.OpStr)
|
|
case Series:
|
|
newVal, err = e.unarySeries(rt, node.OpStr)
|
|
default:
|
|
return newResults, fmt.Errorf("can not perform a unary operation on type %v", rt.Type())
|
|
}
|
|
if err != nil {
|
|
return newResults, err
|
|
}
|
|
newResults.Values = append(newResults.Values, newVal)
|
|
}
|
|
return newResults, nil
|
|
}
|
|
|
|
func (e *State) unarySeries(s Series, op string) (Series, error) {
|
|
newSeries := NewSeries(e.RefID, s.GetLabels(), s.Len())
|
|
for i := 0; i < s.Len(); i++ {
|
|
t, f := s.GetPoint(i)
|
|
if f == nil {
|
|
newSeries.SetPoint(i, t, nil)
|
|
continue
|
|
}
|
|
newF, err := unaryOp(op, *f)
|
|
if err != nil {
|
|
return newSeries, err
|
|
}
|
|
newSeries.SetPoint(i, t, &newF)
|
|
}
|
|
return newSeries, nil
|
|
}
|
|
|
|
func (e *State) unaryNumber(n Number, op string) (Number, error) {
|
|
newNumber := NewNumber(e.RefID, n.GetLabels())
|
|
|
|
f := n.GetFloat64Value()
|
|
if f != nil {
|
|
newF, err := unaryOp(op, *f)
|
|
if err != nil {
|
|
return newNumber, err
|
|
}
|
|
newNumber.SetValue(&newF)
|
|
}
|
|
return newNumber, nil
|
|
}
|
|
|
|
// unaryOp performs a unary operation on a float.
|
|
func unaryOp(op string, a float64) (r float64, err error) {
|
|
if math.IsNaN(a) {
|
|
return math.NaN(), nil
|
|
}
|
|
switch op {
|
|
case "!":
|
|
if a == 0 {
|
|
r = 1
|
|
} else {
|
|
r = 0
|
|
}
|
|
case "-":
|
|
r = -a
|
|
default:
|
|
return r, fmt.Errorf("expr: unknown unary operator %s", op)
|
|
}
|
|
return
|
|
}
|
|
|
|
// Union holds to Values from Two sets where their labels are compatible (TODO: define compatible).
|
|
// This is a intermediate container for Binary operations such (e.g. A + B).
|
|
type Union struct {
|
|
Labels data.Labels
|
|
A, B Value
|
|
}
|
|
|
|
// union creates Union objects based on the Labels attached to each Series or Number
|
|
// within a collection of Series or Numbers. The Unions are used with binary
|
|
// operations. The labels of the Union will the taken from result with a greater
|
|
// number of tags.
|
|
func union(aResults, bResults Results) []*Union {
|
|
unions := []*Union{}
|
|
if len(aResults.Values) == 0 || len(bResults.Values) == 0 {
|
|
return unions
|
|
}
|
|
for _, a := range aResults.Values {
|
|
for _, b := range bResults.Values {
|
|
var labels data.Labels
|
|
aLabels := a.GetLabels()
|
|
bLabels := b.GetLabels()
|
|
switch {
|
|
case aLabels.Equals(bLabels) || len(aLabels) == 0 || len(bLabels) == 0:
|
|
l := aLabels
|
|
if len(aLabels) == 0 {
|
|
l = bLabels
|
|
}
|
|
labels = l
|
|
case len(aLabels) == len(bLabels):
|
|
continue // invalid union, drop for now
|
|
case aLabels.Contains(bLabels):
|
|
labels = aLabels
|
|
case bLabels.Contains(aLabels):
|
|
labels = bLabels
|
|
default:
|
|
continue
|
|
}
|
|
u := &Union{
|
|
Labels: labels,
|
|
A: a,
|
|
B: b,
|
|
}
|
|
unions = append(unions, u)
|
|
}
|
|
}
|
|
if len(unions) == 0 && len(aResults.Values) == 1 && len(bResults.Values) == 1 {
|
|
// In the case of only 1 thing on each side of the operator, we combine them
|
|
// and strip the tags.
|
|
// This isn't ideal for understanding behavior, but will make more stuff work when
|
|
// combining different datasources without munging.
|
|
// This choice is highly questionable in the long term.
|
|
unions = append(unions, &Union{
|
|
A: aResults.Values[0],
|
|
B: bResults.Values[0],
|
|
})
|
|
}
|
|
return unions
|
|
}
|
|
|
|
func (e *State) walkBinary(node *parse.BinaryNode) (Results, error) {
|
|
res := Results{Values{}}
|
|
ar, err := e.walk(node.Args[0])
|
|
if err != nil {
|
|
return res, err
|
|
}
|
|
br, err := e.walk(node.Args[1])
|
|
if err != nil {
|
|
return res, err
|
|
}
|
|
unions := union(ar, br)
|
|
for _, uni := range unions {
|
|
var value Value
|
|
switch at := uni.A.(type) {
|
|
case Scalar:
|
|
aFloat := at.GetFloat64Value()
|
|
switch bt := uni.B.(type) {
|
|
// Scalar op Scalar
|
|
case Scalar:
|
|
bFloat := bt.GetFloat64Value()
|
|
if aFloat == nil || bFloat == nil {
|
|
value = NewScalar(e.RefID, nil)
|
|
break
|
|
}
|
|
f := math.NaN()
|
|
if aFloat != nil && bFloat != nil {
|
|
f, err = binaryOp(node.OpStr, *aFloat, *bFloat)
|
|
if err != nil {
|
|
return res, err
|
|
}
|
|
}
|
|
value = NewScalar(e.RefID, &f)
|
|
// Scalar op Scalar
|
|
case Number:
|
|
value, err = e.biScalarNumber(uni.Labels, node.OpStr, bt, aFloat, false)
|
|
// Scalar op Series
|
|
case Series:
|
|
value, err = e.biSeriesNumber(uni.Labels, node.OpStr, bt, aFloat, false)
|
|
default:
|
|
return res, fmt.Errorf("not implemented: binary %v on %T and %T", node.OpStr, uni.A, uni.B)
|
|
}
|
|
case Series:
|
|
switch bt := uni.B.(type) {
|
|
// Series Op Scalar
|
|
case Scalar:
|
|
bFloat := bt.GetFloat64Value()
|
|
value, err = e.biSeriesNumber(uni.Labels, node.OpStr, at, bFloat, true)
|
|
// case Series Op Number
|
|
case Number:
|
|
bFloat := bt.GetFloat64Value()
|
|
value, err = e.biSeriesNumber(uni.Labels, node.OpStr, at, bFloat, true)
|
|
// case Series op Series
|
|
case Series:
|
|
value, err = e.biSeriesSeries(uni.Labels, node.OpStr, at, bt)
|
|
default:
|
|
return res, fmt.Errorf("not implemented: binary %v on %T and %T", node.OpStr, uni.A, uni.B)
|
|
}
|
|
case Number:
|
|
aFloat := at.GetFloat64Value()
|
|
switch bt := uni.B.(type) {
|
|
case Scalar:
|
|
bFloat := bt.GetFloat64Value()
|
|
value, err = e.biScalarNumber(uni.Labels, node.OpStr, at, bFloat, true)
|
|
case Number:
|
|
bFloat := bt.GetFloat64Value()
|
|
value, err = e.biScalarNumber(uni.Labels, node.OpStr, at, bFloat, true)
|
|
case Series:
|
|
value, err = e.biSeriesNumber(uni.Labels, node.OpStr, bt, aFloat, false)
|
|
default:
|
|
return res, fmt.Errorf("not implemented: binary %v on %T and %T", node.OpStr, uni.A, uni.B)
|
|
}
|
|
default:
|
|
return res, fmt.Errorf("not implemented: binary %v on %T and %T", node.OpStr, uni.A, uni.B)
|
|
}
|
|
if err != nil {
|
|
return res, err
|
|
}
|
|
res.Values = append(res.Values, value)
|
|
}
|
|
return res, nil
|
|
}
|
|
|
|
// binaryOp performs a binary operations (e.g. A+B or A>B) on two
|
|
// float values
|
|
// nolint:gocyclo
|
|
func binaryOp(op string, a, b float64) (r float64, err error) {
|
|
// Test short circuit before NaN.
|
|
switch op {
|
|
case "||":
|
|
if a != 0 {
|
|
return 1, nil
|
|
}
|
|
case "&&":
|
|
if a == 0 {
|
|
return 0, nil
|
|
}
|
|
}
|
|
if math.IsNaN(a) || math.IsNaN(b) {
|
|
return math.NaN(), nil
|
|
}
|
|
switch op {
|
|
case "+":
|
|
r = a + b
|
|
case "*":
|
|
r = a * b
|
|
case "-":
|
|
r = a - b
|
|
case "/":
|
|
r = a / b
|
|
case "**":
|
|
r = math.Pow(a, b)
|
|
case "%":
|
|
r = math.Mod(a, b)
|
|
case "==":
|
|
if a == b {
|
|
r = 1
|
|
} else {
|
|
r = 0
|
|
}
|
|
case ">":
|
|
if a > b {
|
|
r = 1
|
|
} else {
|
|
r = 0
|
|
}
|
|
case "!=":
|
|
if a != b {
|
|
r = 1
|
|
} else {
|
|
r = 0
|
|
}
|
|
case "<":
|
|
if a < b {
|
|
r = 1
|
|
} else {
|
|
r = 0
|
|
}
|
|
case ">=":
|
|
if a >= b {
|
|
r = 1
|
|
} else {
|
|
r = 0
|
|
}
|
|
case "<=":
|
|
if a <= b {
|
|
r = 1
|
|
} else {
|
|
r = 0
|
|
}
|
|
case "||":
|
|
if a != 0 || b != 0 {
|
|
r = 1
|
|
} else {
|
|
r = 0
|
|
}
|
|
case "&&":
|
|
if a != 0 && b != 0 {
|
|
r = 1
|
|
} else {
|
|
r = 0
|
|
}
|
|
default:
|
|
return r, fmt.Errorf("expr: unknown operator %s", op)
|
|
}
|
|
return r, nil
|
|
}
|
|
|
|
func (e *State) biScalarNumber(labels data.Labels, op string, number Number, scalarVal *float64, numberFirst bool) (Number, error) {
|
|
newNumber := NewNumber(e.RefID, labels)
|
|
f := number.GetFloat64Value()
|
|
if f == nil || scalarVal == nil {
|
|
newNumber.SetValue(nil)
|
|
return newNumber, nil
|
|
}
|
|
nF := math.NaN()
|
|
var err error
|
|
if numberFirst {
|
|
nF, err = binaryOp(op, *f, *scalarVal)
|
|
} else {
|
|
nF, err = binaryOp(op, *scalarVal, *f)
|
|
}
|
|
if err != nil {
|
|
return newNumber, err
|
|
}
|
|
newNumber.SetValue(&nF)
|
|
return newNumber, nil
|
|
}
|
|
|
|
func (e *State) biSeriesNumber(labels data.Labels, op string, s Series, scalarVal *float64, seriesFirst bool) (Series, error) {
|
|
newSeries := NewSeries(e.RefID, labels, s.Len())
|
|
var err error
|
|
for i := 0; i < s.Len(); i++ {
|
|
nF := math.NaN()
|
|
t, f := s.GetPoint(i)
|
|
if f == nil || scalarVal == nil {
|
|
newSeries.SetPoint(i, t, nil)
|
|
continue
|
|
}
|
|
if seriesFirst {
|
|
nF, err = binaryOp(op, *f, *scalarVal)
|
|
} else {
|
|
nF, err = binaryOp(op, *scalarVal, *f)
|
|
}
|
|
if err != nil {
|
|
return newSeries, err
|
|
}
|
|
newSeries.SetPoint(i, t, &nF)
|
|
}
|
|
return newSeries, nil
|
|
}
|
|
|
|
// ... if would you like some series with your series and then get some series, or is that enough series?
|
|
// biSeriesSeries performs a the binary operation for each value in the two series where the times
|
|
// are equal. If there are datapoints in A or B that do not share a time, they will be dropped.
|
|
func (e *State) biSeriesSeries(labels data.Labels, op string, aSeries, bSeries Series) (Series, error) {
|
|
bPoints := make(map[string]*float64)
|
|
for i := 0; i < bSeries.Len(); i++ {
|
|
t, f := bSeries.GetPoint(i)
|
|
bPoints[t.UTC().String()] = f
|
|
}
|
|
|
|
newSeries := NewSeries(e.RefID, labels, 0)
|
|
for aIdx := 0; aIdx < aSeries.Len(); aIdx++ {
|
|
aTime, aF := aSeries.GetPoint(aIdx)
|
|
bF, ok := bPoints[aTime.UTC().String()]
|
|
if !ok {
|
|
continue
|
|
}
|
|
if aF == nil || bF == nil {
|
|
newSeries.AppendPoint(aTime, nil)
|
|
continue
|
|
}
|
|
nF, err := binaryOp(op, *aF, *bF)
|
|
if err != nil {
|
|
return newSeries, err
|
|
}
|
|
newSeries.AppendPoint(aTime, &nF)
|
|
}
|
|
return newSeries, nil
|
|
}
|
|
|
|
func (e *State) walkFunc(node *parse.FuncNode) (Results, error) {
|
|
var res Results
|
|
var err error
|
|
var in []reflect.Value
|
|
for _, a := range node.Args {
|
|
var v interface{}
|
|
switch t := a.(type) {
|
|
case *parse.StringNode:
|
|
v = t.Text
|
|
case *parse.VarNode:
|
|
v = e.Vars[t.Name]
|
|
case *parse.ScalarNode:
|
|
v = NewScalarResults(e.RefID, &t.Float64)
|
|
case *parse.FuncNode:
|
|
v, err = e.walkFunc(t)
|
|
case *parse.UnaryNode:
|
|
v, err = e.walkUnary(t)
|
|
case *parse.BinaryNode:
|
|
v, err = e.walkBinary(t)
|
|
default:
|
|
return res, fmt.Errorf("expr: unknown func arg type: %T", t)
|
|
}
|
|
if err != nil {
|
|
return res, err
|
|
}
|
|
in = append(in, reflect.ValueOf(v))
|
|
}
|
|
|
|
f := reflect.ValueOf(node.F.F)
|
|
|
|
fr := f.Call(append([]reflect.Value{reflect.ValueOf(e)}, in...))
|
|
|
|
res = fr[0].Interface().(Results)
|
|
if len(fr) > 1 && !fr[1].IsNil() {
|
|
err := fr[1].Interface().(error)
|
|
if err != nil {
|
|
return res, err
|
|
}
|
|
}
|
|
return res, nil
|
|
}
|