mirror of
https://github.com/grafana/grafana.git
synced 2025-01-09 15:43:23 -06:00
05fd7eb047
When there is a single frame with no fields (e.g. splunk datasource) SSE errors when trying to figure out the data type. This frame needs to exist since this is where the executedQueryString metadata exists. This adds a new return type to SSE to represent no data, so the original frame with its metadata can still be maintained.
532 lines
12 KiB
Go
532 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)
|
|
case NoData:
|
|
newVal = NoData{}.New()
|
|
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{}
|
|
aValueLen := len(aResults.Values)
|
|
bValueLen := len(bResults.Values)
|
|
if aValueLen == 0 || bValueLen == 0 {
|
|
return unions
|
|
}
|
|
if aValueLen == 1 || bValueLen == 1 {
|
|
if aResults.Values[0].Type() == parse.TypeNoData || bResults.Values[0].Type() == parse.TypeNoData {
|
|
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
|
|
}
|