2021-09-07 02:35:37 -05:00
|
|
|
package intervalv2
|
2021-07-15 09:45:59 -05:00
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"regexp"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/grafana/grafana-plugin-sdk-go/backend"
|
2021-10-19 07:56:57 -05:00
|
|
|
"github.com/grafana/grafana-plugin-sdk-go/backend/gtime"
|
2021-07-15 09:45:59 -05:00
|
|
|
"github.com/grafana/grafana/pkg/tsdb/interval"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
defaultRes int64 = 1500
|
|
|
|
defaultMinInterval = time.Millisecond * 1
|
|
|
|
year = time.Hour * 24 * 365
|
|
|
|
day = time.Hour * 24
|
|
|
|
)
|
|
|
|
|
|
|
|
type Interval struct {
|
|
|
|
Text string
|
|
|
|
Value time.Duration
|
|
|
|
}
|
|
|
|
|
|
|
|
type intervalCalculator struct {
|
|
|
|
minInterval time.Duration
|
|
|
|
}
|
|
|
|
|
|
|
|
type Calculator interface {
|
2021-09-13 07:58:32 -05:00
|
|
|
Calculate(timerange backend.TimeRange, minInterval time.Duration, maxDataPoints int64) Interval
|
2021-08-05 02:09:49 -05:00
|
|
|
CalculateSafeInterval(timerange backend.TimeRange, resolution int64) Interval
|
2021-07-15 09:45:59 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
type CalculatorOptions struct {
|
|
|
|
MinInterval time.Duration
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewCalculator(opts ...CalculatorOptions) *intervalCalculator {
|
|
|
|
calc := &intervalCalculator{}
|
|
|
|
|
|
|
|
for _, o := range opts {
|
|
|
|
if o.MinInterval == 0 {
|
|
|
|
calc.minInterval = defaultMinInterval
|
|
|
|
} else {
|
|
|
|
calc.minInterval = o.MinInterval
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return calc
|
|
|
|
}
|
|
|
|
|
|
|
|
func (i *Interval) Milliseconds() int64 {
|
|
|
|
return i.Value.Nanoseconds() / int64(time.Millisecond)
|
|
|
|
}
|
|
|
|
|
2021-09-13 07:58:32 -05:00
|
|
|
func (ic *intervalCalculator) Calculate(timerange backend.TimeRange, minInterval time.Duration, maxDataPoints int64) Interval {
|
2021-07-15 09:45:59 -05:00
|
|
|
to := timerange.To.UnixNano()
|
|
|
|
from := timerange.From.UnixNano()
|
2021-09-13 07:58:32 -05:00
|
|
|
resolution := maxDataPoints
|
|
|
|
if resolution == 0 {
|
|
|
|
resolution = defaultRes
|
|
|
|
}
|
|
|
|
|
|
|
|
calculatedInterval := time.Duration((to - from) / resolution)
|
2021-08-05 02:09:49 -05:00
|
|
|
|
2021-09-13 07:58:32 -05:00
|
|
|
if calculatedInterval < minInterval {
|
2021-09-09 07:05:08 -05:00
|
|
|
return Interval{Text: interval.FormatDuration(minInterval), Value: minInterval}
|
2021-07-15 09:45:59 -05:00
|
|
|
}
|
|
|
|
|
2021-09-13 07:58:32 -05:00
|
|
|
rounded := roundInterval(calculatedInterval)
|
2021-09-09 07:05:08 -05:00
|
|
|
|
|
|
|
return Interval{Text: interval.FormatDuration(rounded), Value: rounded}
|
2021-08-05 02:09:49 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
func (ic *intervalCalculator) CalculateSafeInterval(timerange backend.TimeRange, safeRes int64) Interval {
|
|
|
|
to := timerange.To.UnixNano()
|
|
|
|
from := timerange.From.UnixNano()
|
|
|
|
safeInterval := time.Duration((to - from) / safeRes)
|
|
|
|
|
|
|
|
rounded := roundInterval(safeInterval)
|
2021-07-15 09:45:59 -05:00
|
|
|
return Interval{Text: interval.FormatDuration(rounded), Value: rounded}
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetIntervalFrom returns the minimum interval.
|
|
|
|
// dsInterval is the string representation of data source min interval, if configured.
|
|
|
|
// queryInterval is the string representation of query interval (min interval), e.g. "10ms" or "10s".
|
|
|
|
// queryIntervalMS is a pre-calculated numeric representation of the query interval in milliseconds.
|
|
|
|
func GetIntervalFrom(dsInterval, queryInterval string, queryIntervalMS int64, defaultInterval time.Duration) (time.Duration, error) {
|
2021-09-07 02:35:37 -05:00
|
|
|
// Apparently we are setting default value of queryInterval to 0s now
|
2021-09-10 15:56:15 -05:00
|
|
|
interval := queryInterval
|
|
|
|
if interval == "0s" {
|
|
|
|
interval = ""
|
2021-09-07 02:35:37 -05:00
|
|
|
}
|
2021-09-10 15:56:15 -05:00
|
|
|
if interval == "" {
|
2021-07-15 09:45:59 -05:00
|
|
|
if queryIntervalMS != 0 {
|
|
|
|
return time.Duration(queryIntervalMS) * time.Millisecond, nil
|
|
|
|
}
|
|
|
|
}
|
2021-09-10 15:56:15 -05:00
|
|
|
if interval == "" && dsInterval != "" {
|
2021-07-15 09:45:59 -05:00
|
|
|
interval = dsInterval
|
|
|
|
}
|
|
|
|
if interval == "" {
|
|
|
|
return defaultInterval, nil
|
|
|
|
}
|
2021-09-10 15:56:15 -05:00
|
|
|
|
|
|
|
parsedInterval, err := ParseIntervalStringToTimeDuration(interval)
|
|
|
|
if err != nil {
|
|
|
|
return time.Duration(0), err
|
|
|
|
}
|
|
|
|
|
|
|
|
return parsedInterval, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func ParseIntervalStringToTimeDuration(interval string) (time.Duration, error) {
|
|
|
|
formattedInterval := strings.Replace(strings.Replace(interval, "<", "", 1), ">", "", 1)
|
|
|
|
isPureNum, err := regexp.MatchString(`^\d+$`, formattedInterval)
|
2021-07-15 09:45:59 -05:00
|
|
|
if err != nil {
|
|
|
|
return time.Duration(0), err
|
|
|
|
}
|
|
|
|
if isPureNum {
|
2021-09-10 15:56:15 -05:00
|
|
|
formattedInterval += "s"
|
2021-07-15 09:45:59 -05:00
|
|
|
}
|
2021-10-19 07:56:57 -05:00
|
|
|
parsedInterval, err := gtime.ParseDuration(formattedInterval)
|
2021-07-15 09:45:59 -05:00
|
|
|
if err != nil {
|
|
|
|
return time.Duration(0), err
|
|
|
|
}
|
|
|
|
return parsedInterval, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// FormatDuration converts a duration into the kbn format e.g. 1m 2h or 3d
|
|
|
|
func FormatDuration(inter time.Duration) string {
|
|
|
|
if inter >= year {
|
|
|
|
return fmt.Sprintf("%dy", inter/year)
|
|
|
|
}
|
|
|
|
|
|
|
|
if inter >= day {
|
|
|
|
return fmt.Sprintf("%dd", inter/day)
|
|
|
|
}
|
|
|
|
|
|
|
|
if inter >= time.Hour {
|
|
|
|
return fmt.Sprintf("%dh", inter/time.Hour)
|
|
|
|
}
|
|
|
|
|
|
|
|
if inter >= time.Minute {
|
|
|
|
return fmt.Sprintf("%dm", inter/time.Minute)
|
|
|
|
}
|
|
|
|
|
|
|
|
if inter >= time.Second {
|
|
|
|
return fmt.Sprintf("%ds", inter/time.Second)
|
|
|
|
}
|
|
|
|
|
|
|
|
if inter >= time.Millisecond {
|
|
|
|
return fmt.Sprintf("%dms", inter/time.Millisecond)
|
|
|
|
}
|
|
|
|
|
|
|
|
return "1ms"
|
|
|
|
}
|
|
|
|
|
|
|
|
//nolint: gocyclo
|
|
|
|
func roundInterval(interval time.Duration) time.Duration {
|
|
|
|
switch {
|
|
|
|
// 0.015s
|
|
|
|
case interval <= 15*time.Millisecond:
|
|
|
|
return time.Millisecond * 10 // 0.01s
|
|
|
|
// 0.035s
|
|
|
|
case interval <= 35*time.Millisecond:
|
|
|
|
return time.Millisecond * 20 // 0.02s
|
|
|
|
// 0.075s
|
|
|
|
case interval <= 75*time.Millisecond:
|
|
|
|
return time.Millisecond * 50 // 0.05s
|
|
|
|
// 0.15s
|
|
|
|
case interval <= 150*time.Millisecond:
|
|
|
|
return time.Millisecond * 100 // 0.1s
|
|
|
|
// 0.35s
|
|
|
|
case interval <= 350*time.Millisecond:
|
|
|
|
return time.Millisecond * 200 // 0.2s
|
|
|
|
// 0.75s
|
|
|
|
case interval <= 750*time.Millisecond:
|
|
|
|
return time.Millisecond * 500 // 0.5s
|
|
|
|
// 1.5s
|
|
|
|
case interval <= 1500*time.Millisecond:
|
|
|
|
return time.Millisecond * 1000 // 1s
|
|
|
|
// 3.5s
|
|
|
|
case interval <= 3500*time.Millisecond:
|
|
|
|
return time.Millisecond * 2000 // 2s
|
|
|
|
// 7.5s
|
|
|
|
case interval <= 7500*time.Millisecond:
|
|
|
|
return time.Millisecond * 5000 // 5s
|
|
|
|
// 12.5s
|
|
|
|
case interval <= 12500*time.Millisecond:
|
|
|
|
return time.Millisecond * 10000 // 10s
|
|
|
|
// 17.5s
|
|
|
|
case interval <= 17500*time.Millisecond:
|
|
|
|
return time.Millisecond * 15000 // 15s
|
|
|
|
// 25s
|
|
|
|
case interval <= 25000*time.Millisecond:
|
|
|
|
return time.Millisecond * 20000 // 20s
|
|
|
|
// 45s
|
|
|
|
case interval <= 45000*time.Millisecond:
|
|
|
|
return time.Millisecond * 30000 // 30s
|
|
|
|
// 1.5m
|
|
|
|
case interval <= 90000*time.Millisecond:
|
|
|
|
return time.Millisecond * 60000 // 1m
|
|
|
|
// 3.5m
|
|
|
|
case interval <= 210000*time.Millisecond:
|
|
|
|
return time.Millisecond * 120000 // 2m
|
|
|
|
// 7.5m
|
|
|
|
case interval <= 450000*time.Millisecond:
|
|
|
|
return time.Millisecond * 300000 // 5m
|
|
|
|
// 12.5m
|
|
|
|
case interval <= 750000*time.Millisecond:
|
|
|
|
return time.Millisecond * 600000 // 10m
|
|
|
|
// 12.5m
|
|
|
|
case interval <= 1050000*time.Millisecond:
|
|
|
|
return time.Millisecond * 900000 // 15m
|
|
|
|
// 25m
|
|
|
|
case interval <= 1500000*time.Millisecond:
|
|
|
|
return time.Millisecond * 1200000 // 20m
|
|
|
|
// 45m
|
|
|
|
case interval <= 2700000*time.Millisecond:
|
|
|
|
return time.Millisecond * 1800000 // 30m
|
|
|
|
// 1.5h
|
|
|
|
case interval <= 5400000*time.Millisecond:
|
|
|
|
return time.Millisecond * 3600000 // 1h
|
|
|
|
// 2.5h
|
|
|
|
case interval <= 9000000*time.Millisecond:
|
|
|
|
return time.Millisecond * 7200000 // 2h
|
|
|
|
// 4.5h
|
|
|
|
case interval <= 16200000*time.Millisecond:
|
|
|
|
return time.Millisecond * 10800000 // 3h
|
|
|
|
// 9h
|
|
|
|
case interval <= 32400000*time.Millisecond:
|
|
|
|
return time.Millisecond * 21600000 // 6h
|
|
|
|
// 24h
|
|
|
|
case interval <= 86400000*time.Millisecond:
|
|
|
|
return time.Millisecond * 43200000 // 12h
|
|
|
|
// 48h
|
|
|
|
case interval <= 172800000*time.Millisecond:
|
|
|
|
return time.Millisecond * 86400000 // 24h
|
|
|
|
// 1w
|
|
|
|
case interval <= 604800000*time.Millisecond:
|
|
|
|
return time.Millisecond * 86400000 // 24h
|
|
|
|
// 3w
|
|
|
|
case interval <= 1814400000*time.Millisecond:
|
|
|
|
return time.Millisecond * 604800000 // 1w
|
|
|
|
// 2y
|
|
|
|
case interval < 3628800000*time.Millisecond:
|
|
|
|
return time.Millisecond * 2592000000 // 30d
|
|
|
|
default:
|
|
|
|
return time.Millisecond * 31536000000 // 1y
|
|
|
|
}
|
|
|
|
}
|