mirror of
https://github.com/grafana/grafana.git
synced 2025-02-25 18:55:37 -06:00
* Chore: Enable more go-ruleguard rules Signed-off-by: Arve Knudsen <arve.knudsen@gmail.com>
84 lines
2.1 KiB
Go
84 lines
2.1 KiB
Go
package mathexp
|
|
|
|
import (
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/grafana/grafana-plugin-sdk-go/backend"
|
|
"github.com/grafana/grafana-plugin-sdk-go/data"
|
|
)
|
|
|
|
// Resample turns the Series into a Number based on the given reduction function
|
|
func (s Series) Resample(refID string, interval time.Duration, downsampler string, upsampler string, tr backend.TimeRange) (Series, error) {
|
|
newSeriesLength := int(float64(tr.To.Sub(tr.From).Nanoseconds()) / float64(interval.Nanoseconds()))
|
|
if newSeriesLength <= 0 {
|
|
return s, fmt.Errorf("the series cannot be sampled further; the time range is shorter than the interval")
|
|
}
|
|
resampled := NewSeries(refID, s.GetLabels(), s.TimeIdx, s.TimeIsNullable, s.ValueIdx, s.ValueIsNullable, newSeriesLength+1)
|
|
bookmark := 0
|
|
var lastSeen *float64
|
|
idx := 0
|
|
t := tr.From
|
|
for !t.After(tr.To) && idx <= newSeriesLength {
|
|
vals := make([]*float64, 0)
|
|
sIdx := bookmark
|
|
for {
|
|
if sIdx == s.Len() {
|
|
break
|
|
}
|
|
st, v := s.GetPoint(sIdx)
|
|
if st.After(t) {
|
|
break
|
|
}
|
|
bookmark++
|
|
sIdx++
|
|
lastSeen = v
|
|
vals = append(vals, v)
|
|
}
|
|
var value *float64
|
|
if len(vals) == 0 { // upsampling
|
|
switch upsampler {
|
|
case "pad":
|
|
if lastSeen != nil {
|
|
value = lastSeen
|
|
} else {
|
|
value = nil
|
|
}
|
|
case "backfilling":
|
|
if sIdx == s.Len() { // no vals left
|
|
value = nil
|
|
} else {
|
|
_, value = s.GetPoint(sIdx)
|
|
}
|
|
case "fillna":
|
|
value = nil
|
|
default:
|
|
return s, fmt.Errorf("upsampling %v not implemented", upsampler)
|
|
}
|
|
} else { // downsampling
|
|
fVec := data.NewField("", s.GetLabels(), vals)
|
|
var tmp *float64
|
|
switch downsampler {
|
|
case "sum":
|
|
tmp = Sum(fVec)
|
|
case "mean":
|
|
tmp = Avg(fVec)
|
|
case "min":
|
|
tmp = Min(fVec)
|
|
case "max":
|
|
tmp = Max(fVec)
|
|
default:
|
|
return s, fmt.Errorf("downsampling %v not implemented", downsampler)
|
|
}
|
|
value = tmp
|
|
}
|
|
tv := t // his is required otherwise all points keep the latest timestamp; anything better?
|
|
if err := resampled.SetPoint(idx, &tv, value); err != nil {
|
|
return resampled, err
|
|
}
|
|
t = t.Add(interval)
|
|
idx++
|
|
}
|
|
return resampled, nil
|
|
}
|