mirror of
https://github.com/grafana/grafana.git
synced 2024-11-25 18:30:41 -06:00
cbbbe2e6f6
* introduce a function checkIfSeriesNeedToBeFixed to scan all value fields in the response and provide a function that updates Series so they can be uniquely identifiable. Only Graphite and TestData are checked. * update `convertDataFramesToResults` to run this function and provide it to WideToMany * update WideToMany to run the fix function if it is not nil
95 lines
2.4 KiB
Go
95 lines
2.4 KiB
Go
package backtesting
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"time"
|
|
|
|
"github.com/grafana/grafana-plugin-sdk-go/data"
|
|
|
|
"github.com/grafana/grafana/pkg/expr"
|
|
"github.com/grafana/grafana/pkg/expr/mathexp"
|
|
"github.com/grafana/grafana/pkg/services/ngalert/eval"
|
|
)
|
|
|
|
// DataEvaluator is evaluator that evaluates data
|
|
type dataEvaluator struct {
|
|
refID string
|
|
data []mathexp.Series
|
|
downsampleFunction string
|
|
upsampleFunction string
|
|
}
|
|
|
|
func newDataEvaluator(refID string, frame *data.Frame) (*dataEvaluator, error) {
|
|
series, err := expr.WideToMany(frame, nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
for _, s := range series {
|
|
s.SortByTime(false)
|
|
}
|
|
|
|
return &dataEvaluator{
|
|
refID: refID,
|
|
data: series,
|
|
downsampleFunction: "last",
|
|
upsampleFunction: "pad",
|
|
}, nil
|
|
}
|
|
|
|
func (d *dataEvaluator) Eval(_ context.Context, from time.Time, interval time.Duration, evaluations int, callback callbackFunc) error {
|
|
var resampled = make([]mathexp.Series, 0, len(d.data))
|
|
to := from.Add(time.Duration(evaluations) * interval)
|
|
for _, s := range d.data {
|
|
// making sure the input data frame is aligned with the interval
|
|
r, err := s.Resample(d.refID, interval, d.downsampleFunction, d.upsampleFunction, from, to.Add(-interval)) // we want to query [from,to)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
resampled = append(resampled, r)
|
|
}
|
|
|
|
for i := 0; i < evaluations; i++ {
|
|
result := make([]eval.Result, 0, len(resampled))
|
|
var now time.Time
|
|
for _, series := range resampled {
|
|
snow := series.GetTime(i)
|
|
if !now.IsZero() && now != snow { // this should not happen because all series' belong to a single data frame
|
|
return errors.New("failed to resample input data. timestamps are not aligned")
|
|
}
|
|
now = snow
|
|
value := series.GetValue(i)
|
|
var state = eval.Normal
|
|
if value == nil {
|
|
continue
|
|
} else if *value != 0 {
|
|
state = eval.Alerting
|
|
}
|
|
result = append(result, eval.Result{
|
|
Instance: series.GetLabels(),
|
|
State: state,
|
|
Results: nil,
|
|
Values: map[string]eval.NumberValueCapture{
|
|
d.refID: {
|
|
Var: d.refID,
|
|
Labels: series.GetLabels(),
|
|
Value: value,
|
|
},
|
|
},
|
|
EvaluatedAt: now,
|
|
})
|
|
}
|
|
if len(result) == 0 {
|
|
result = append(result, eval.Result{
|
|
State: eval.NoData,
|
|
EvaluatedAt: now,
|
|
})
|
|
}
|
|
err := callback(i, now, result)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|