2023-10-13 13:54:33 +02:00
|
|
|
package rule
|
2020-05-10 18:58:17 +02:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2021-09-13 14:48:18 +02:00
|
|
|
"fmt"
|
2023-09-06 16:29:59 +02:00
|
|
|
"math"
|
2023-10-13 13:54:33 +02:00
|
|
|
"os"
|
2022-03-29 15:09:07 +02:00
|
|
|
"reflect"
|
2020-05-10 18:58:17 +02:00
|
|
|
"sort"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
2023-10-13 13:54:33 +02:00
|
|
|
"gopkg.in/yaml.v2"
|
|
|
|
|
2020-06-01 12:46:37 +02:00
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmalert/config"
|
2023-10-13 13:54:33 +02:00
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmalert/datasource"
|
2020-05-10 18:58:17 +02:00
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmalert/notifier"
|
2022-12-06 15:36:46 +01:00
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmalert/remotewrite"
|
2023-10-13 13:54:33 +02:00
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/app/vmalert/templates"
|
2022-12-06 15:36:46 +01:00
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/decimal"
|
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/prompbmarshal"
|
2022-02-11 15:17:00 +01:00
|
|
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/promutils"
|
2020-05-10 18:58:17 +02:00
|
|
|
)
|
|
|
|
|
2020-09-03 00:00:55 +02:00
|
|
|
func init() {
|
|
|
|
// Disable rand sleep on group start during tests in order to speed up test execution.
|
|
|
|
// Rand sleep is needed only in prod code.
|
2023-10-13 13:54:33 +02:00
|
|
|
SkipRandSleepOnGroupStart = true
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestMain(m *testing.M) {
|
|
|
|
if err := templates.Load([]string{}, true); err != nil {
|
|
|
|
fmt.Println("failed to load template for test")
|
|
|
|
os.Exit(1)
|
|
|
|
}
|
|
|
|
os.Exit(m.Run())
|
2020-09-03 00:00:55 +02:00
|
|
|
}
|
|
|
|
|
2020-05-10 18:58:17 +02:00
|
|
|
func TestUpdateWith(t *testing.T) {
|
|
|
|
testCases := []struct {
|
|
|
|
name string
|
2020-06-15 21:15:47 +02:00
|
|
|
currentRules []config.Rule
|
|
|
|
newRules []config.Rule
|
2020-05-10 18:58:17 +02:00
|
|
|
}{
|
|
|
|
{
|
|
|
|
"new rule",
|
2020-06-15 21:15:47 +02:00
|
|
|
nil,
|
|
|
|
[]config.Rule{{Alert: "bar"}},
|
2020-05-10 18:58:17 +02:00
|
|
|
},
|
|
|
|
{
|
2020-06-01 12:46:37 +02:00
|
|
|
"update alerting rule",
|
2023-09-06 16:29:59 +02:00
|
|
|
[]config.Rule{
|
|
|
|
{
|
|
|
|
Alert: "foo",
|
|
|
|
Expr: "up > 0",
|
|
|
|
For: promutils.NewDuration(time.Second),
|
|
|
|
Labels: map[string]string{
|
|
|
|
"bar": "baz",
|
|
|
|
},
|
|
|
|
Annotations: map[string]string{
|
|
|
|
"summary": "{{ $value|humanize }}",
|
|
|
|
"description": "{{$labels}}",
|
|
|
|
},
|
2020-05-10 18:58:17 +02:00
|
|
|
},
|
2023-07-27 15:13:13 +02:00
|
|
|
{
|
|
|
|
Alert: "bar",
|
|
|
|
Expr: "up > 0",
|
|
|
|
For: promutils.NewDuration(time.Second),
|
|
|
|
Labels: map[string]string{
|
|
|
|
"bar": "baz",
|
|
|
|
},
|
2023-09-06 16:29:59 +02:00
|
|
|
},
|
|
|
|
},
|
2023-07-27 15:13:13 +02:00
|
|
|
[]config.Rule{
|
|
|
|
{
|
|
|
|
Alert: "foo",
|
|
|
|
Expr: "up > 10",
|
|
|
|
For: promutils.NewDuration(time.Second),
|
|
|
|
Labels: map[string]string{
|
|
|
|
"baz": "bar",
|
|
|
|
},
|
|
|
|
Annotations: map[string]string{
|
|
|
|
"summary": "none",
|
|
|
|
},
|
2020-05-10 18:58:17 +02:00
|
|
|
},
|
2023-07-27 15:13:13 +02:00
|
|
|
{
|
|
|
|
Alert: "bar",
|
|
|
|
Expr: "up > 0",
|
|
|
|
For: promutils.NewDuration(2 * time.Second),
|
|
|
|
KeepFiringFor: promutils.NewDuration(time.Minute),
|
|
|
|
Labels: map[string]string{
|
|
|
|
"bar": "baz",
|
|
|
|
},
|
2023-09-06 16:29:59 +02:00
|
|
|
},
|
|
|
|
},
|
2020-05-10 18:58:17 +02:00
|
|
|
},
|
2020-06-01 12:46:37 +02:00
|
|
|
{
|
|
|
|
"update recording rule",
|
2020-06-15 21:15:47 +02:00
|
|
|
[]config.Rule{{
|
|
|
|
Record: "foo",
|
|
|
|
Expr: "max(up)",
|
2020-06-01 12:46:37 +02:00
|
|
|
Labels: map[string]string{
|
|
|
|
"bar": "baz",
|
|
|
|
},
|
|
|
|
}},
|
2020-06-15 21:15:47 +02:00
|
|
|
[]config.Rule{{
|
|
|
|
Record: "foo",
|
|
|
|
Expr: "min(up)",
|
2020-06-01 12:46:37 +02:00
|
|
|
Labels: map[string]string{
|
|
|
|
"baz": "bar",
|
|
|
|
},
|
|
|
|
}},
|
|
|
|
},
|
2020-05-10 18:58:17 +02:00
|
|
|
{
|
|
|
|
"empty rule",
|
2020-06-15 21:15:47 +02:00
|
|
|
[]config.Rule{{Alert: "foo"}, {Record: "bar"}},
|
|
|
|
nil,
|
2020-05-10 18:58:17 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
"multiple rules",
|
2020-06-15 21:15:47 +02:00
|
|
|
[]config.Rule{
|
|
|
|
{Alert: "bar"},
|
|
|
|
{Alert: "baz"},
|
|
|
|
{Alert: "foo"},
|
2020-06-01 12:46:37 +02:00
|
|
|
},
|
2020-06-15 21:15:47 +02:00
|
|
|
[]config.Rule{
|
|
|
|
{Alert: "baz"},
|
|
|
|
{Record: "foo"},
|
2020-06-01 12:46:37 +02:00
|
|
|
},
|
2020-05-15 08:55:22 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
"replace rule",
|
2020-06-15 21:15:47 +02:00
|
|
|
[]config.Rule{{Alert: "foo1"}},
|
|
|
|
[]config.Rule{{Alert: "foo2"}},
|
2020-05-15 08:55:22 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
"replace multiple rules",
|
2020-06-15 21:15:47 +02:00
|
|
|
[]config.Rule{
|
|
|
|
{Alert: "foo1"},
|
|
|
|
{Record: "foo2"},
|
|
|
|
{Alert: "foo3"},
|
2020-06-01 12:46:37 +02:00
|
|
|
},
|
2020-06-15 21:15:47 +02:00
|
|
|
[]config.Rule{
|
|
|
|
{Alert: "foo3"},
|
|
|
|
{Alert: "foo4"},
|
|
|
|
{Record: "foo5"},
|
2020-06-01 12:46:37 +02:00
|
|
|
},
|
2020-05-10 18:58:17 +02:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tc := range testCases {
|
|
|
|
t.Run(tc.name, func(t *testing.T) {
|
2020-06-15 21:15:47 +02:00
|
|
|
g := &Group{Name: "test"}
|
2023-10-13 13:54:33 +02:00
|
|
|
qb := &datasource.FakeQuerier{}
|
2020-06-15 21:15:47 +02:00
|
|
|
for _, r := range tc.currentRules {
|
|
|
|
r.ID = config.HashRule(r)
|
2021-04-28 22:41:15 +02:00
|
|
|
g.Rules = append(g.Rules, g.newRule(qb, r))
|
2020-06-15 21:15:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ng := &Group{Name: "test"}
|
|
|
|
for _, r := range tc.newRules {
|
|
|
|
r.ID = config.HashRule(r)
|
2021-04-28 22:41:15 +02:00
|
|
|
ng.Rules = append(ng.Rules, ng.newRule(qb, r))
|
2020-06-15 21:15:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
err := g.updateWith(ng)
|
2020-06-01 12:46:37 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2020-05-10 18:58:17 +02:00
|
|
|
|
|
|
|
if len(g.Rules) != len(tc.newRules) {
|
|
|
|
t.Fatalf("expected to have %d rules; got: %d",
|
|
|
|
len(g.Rules), len(tc.newRules))
|
|
|
|
}
|
2020-05-15 08:55:22 +02:00
|
|
|
sort.Slice(g.Rules, func(i, j int) bool {
|
2020-06-01 12:46:37 +02:00
|
|
|
return g.Rules[i].ID() < g.Rules[j].ID()
|
2020-05-15 08:55:22 +02:00
|
|
|
})
|
2020-06-15 21:15:47 +02:00
|
|
|
sort.Slice(ng.Rules, func(i, j int) bool {
|
|
|
|
return ng.Rules[i].ID() < ng.Rules[j].ID()
|
|
|
|
})
|
2020-05-10 18:58:17 +02:00
|
|
|
for i, r := range g.Rules {
|
2020-06-15 21:15:47 +02:00
|
|
|
got, want := r, ng.Rules[i]
|
2020-06-01 12:46:37 +02:00
|
|
|
if got.ID() != want.ID() {
|
|
|
|
t.Fatalf("expected to have rule %q; got %q", want, got)
|
2020-05-10 18:58:17 +02:00
|
|
|
}
|
2023-10-13 13:54:33 +02:00
|
|
|
if err := CompareRules(t, got, want); err != nil {
|
2022-07-18 11:02:51 +02:00
|
|
|
t.Fatalf("comparison error: %s", err)
|
2020-05-10 18:58:17 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestGroupStart(t *testing.T) {
|
2023-10-13 13:54:33 +02:00
|
|
|
const (
|
|
|
|
rules = `
|
|
|
|
- name: groupTest
|
|
|
|
rules:
|
|
|
|
- alert: VMRows
|
|
|
|
for: 1ms
|
|
|
|
expr: vm_rows > 0
|
|
|
|
labels:
|
|
|
|
label: bar
|
|
|
|
host: "{{ $labels.instance }}"
|
|
|
|
annotations:
|
|
|
|
summary: "{{ $value }}"
|
|
|
|
`
|
|
|
|
)
|
|
|
|
var groups []config.Group
|
|
|
|
err := yaml.Unmarshal([]byte(rules), &groups)
|
2020-05-10 18:58:17 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("failed to parse rules: %s", err)
|
|
|
|
}
|
2022-03-16 16:26:33 +01:00
|
|
|
|
2023-10-13 13:54:33 +02:00
|
|
|
fs := &datasource.FakeQuerier{}
|
|
|
|
fn := ¬ifier.FakeNotifier{}
|
2021-04-28 22:41:15 +02:00
|
|
|
|
2022-03-16 16:26:33 +01:00
|
|
|
const evalInterval = time.Millisecond
|
2023-10-13 13:54:33 +02:00
|
|
|
g := NewGroup(groups[0], fs, evalInterval, map[string]string{"cluster": "east-1"})
|
2021-04-28 22:41:15 +02:00
|
|
|
g.Concurrency = 2
|
2020-05-10 18:58:17 +02:00
|
|
|
|
|
|
|
const inst1, inst2, job = "foo", "bar", "baz"
|
|
|
|
m1 := metricWithLabels(t, "instance", inst1, "job", job)
|
|
|
|
m2 := metricWithLabels(t, "instance", inst2, "job", job)
|
|
|
|
|
2020-06-01 12:46:37 +02:00
|
|
|
r := g.Rules[0].(*AlertingRule)
|
vmalert: fix labels and annotations processing for alerts (#2403)
To improve compatibility with Prometheus alerting the order of
templates processing has changed.
Before, vmalert did all labels processing beforehand. It meant
all extra labels (such as `alertname`, `alertgroup` or rule labels)
were available in templating. All collisions were resolved in favour
of extra labels.
In Prometheus, only labels from the received metric are available in
templating, so no collisions are possible.
This change makes vmalert's behaviour similar to Prometheus.
For example, consider alerting rule which is triggered by time series
with `alertname` label. In vmalert, this label would be overriden
by alerting rule's name everywhere: for alert labels, for annotations, etc.
In Prometheus, it would be overriden for alert's labels only, but in annotations
the original label value would be available.
See more details here https://github.com/prometheus/compliance/issues/80
Signed-off-by: hagen1778 <roman@victoriametrics.com>
2022-04-06 20:24:45 +02:00
|
|
|
alert1, err := r.newAlert(m1, nil, time.Now(), nil)
|
2020-05-10 18:58:17 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("faield to create alert: %s", err)
|
|
|
|
}
|
|
|
|
alert1.State = notifier.StateFiring
|
2020-11-09 23:27:32 +01:00
|
|
|
// add external label
|
|
|
|
alert1.Labels["cluster"] = "east-1"
|
2023-10-13 13:54:33 +02:00
|
|
|
// add rule labels
|
2020-11-09 23:27:32 +01:00
|
|
|
alert1.Labels["label"] = "bar"
|
|
|
|
alert1.Labels["host"] = inst1
|
2021-10-22 11:30:38 +02:00
|
|
|
// add service labels
|
|
|
|
alert1.Labels[alertNameLabel] = alert1.Name
|
|
|
|
alert1.Labels[alertGroupNameLabel] = g.Name
|
vmalert: fix labels and annotations processing for alerts (#2403)
To improve compatibility with Prometheus alerting the order of
templates processing has changed.
Before, vmalert did all labels processing beforehand. It meant
all extra labels (such as `alertname`, `alertgroup` or rule labels)
were available in templating. All collisions were resolved in favour
of extra labels.
In Prometheus, only labels from the received metric are available in
templating, so no collisions are possible.
This change makes vmalert's behaviour similar to Prometheus.
For example, consider alerting rule which is triggered by time series
with `alertname` label. In vmalert, this label would be overriden
by alerting rule's name everywhere: for alert labels, for annotations, etc.
In Prometheus, it would be overriden for alert's labels only, but in annotations
the original label value would be available.
See more details here https://github.com/prometheus/compliance/issues/80
Signed-off-by: hagen1778 <roman@victoriametrics.com>
2022-04-06 20:24:45 +02:00
|
|
|
alert1.ID = hash(alert1.Labels)
|
2020-05-10 18:58:17 +02:00
|
|
|
|
vmalert: fix labels and annotations processing for alerts (#2403)
To improve compatibility with Prometheus alerting the order of
templates processing has changed.
Before, vmalert did all labels processing beforehand. It meant
all extra labels (such as `alertname`, `alertgroup` or rule labels)
were available in templating. All collisions were resolved in favour
of extra labels.
In Prometheus, only labels from the received metric are available in
templating, so no collisions are possible.
This change makes vmalert's behaviour similar to Prometheus.
For example, consider alerting rule which is triggered by time series
with `alertname` label. In vmalert, this label would be overriden
by alerting rule's name everywhere: for alert labels, for annotations, etc.
In Prometheus, it would be overriden for alert's labels only, but in annotations
the original label value would be available.
See more details here https://github.com/prometheus/compliance/issues/80
Signed-off-by: hagen1778 <roman@victoriametrics.com>
2022-04-06 20:24:45 +02:00
|
|
|
alert2, err := r.newAlert(m2, nil, time.Now(), nil)
|
2020-05-10 18:58:17 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("faield to create alert: %s", err)
|
|
|
|
}
|
|
|
|
alert2.State = notifier.StateFiring
|
2020-11-09 23:27:32 +01:00
|
|
|
// add external label
|
|
|
|
alert2.Labels["cluster"] = "east-1"
|
2023-10-13 13:54:33 +02:00
|
|
|
// add rule labels
|
2020-11-09 23:27:32 +01:00
|
|
|
alert2.Labels["label"] = "bar"
|
|
|
|
alert2.Labels["host"] = inst2
|
2021-10-22 11:30:38 +02:00
|
|
|
// add service labels
|
|
|
|
alert2.Labels[alertNameLabel] = alert2.Name
|
|
|
|
alert2.Labels[alertGroupNameLabel] = g.Name
|
vmalert: fix labels and annotations processing for alerts (#2403)
To improve compatibility with Prometheus alerting the order of
templates processing has changed.
Before, vmalert did all labels processing beforehand. It meant
all extra labels (such as `alertname`, `alertgroup` or rule labels)
were available in templating. All collisions were resolved in favour
of extra labels.
In Prometheus, only labels from the received metric are available in
templating, so no collisions are possible.
This change makes vmalert's behaviour similar to Prometheus.
For example, consider alerting rule which is triggered by time series
with `alertname` label. In vmalert, this label would be overriden
by alerting rule's name everywhere: for alert labels, for annotations, etc.
In Prometheus, it would be overriden for alert's labels only, but in annotations
the original label value would be available.
See more details here https://github.com/prometheus/compliance/issues/80
Signed-off-by: hagen1778 <roman@victoriametrics.com>
2022-04-06 20:24:45 +02:00
|
|
|
alert2.ID = hash(alert2.Labels)
|
2020-05-10 18:58:17 +02:00
|
|
|
|
|
|
|
finished := make(chan struct{})
|
2023-10-13 13:54:33 +02:00
|
|
|
fs.Add(m1)
|
|
|
|
fs.Add(m2)
|
2020-05-10 18:58:17 +02:00
|
|
|
go func() {
|
2023-10-13 13:54:33 +02:00
|
|
|
g.Start(context.Background(), func() []notifier.Notifier { return []notifier.Notifier{fn} }, nil, fs)
|
2020-05-10 18:58:17 +02:00
|
|
|
close(finished)
|
|
|
|
}()
|
|
|
|
|
|
|
|
// wait for multiple evals
|
|
|
|
time.Sleep(20 * evalInterval)
|
|
|
|
|
2023-10-13 13:54:33 +02:00
|
|
|
gotAlerts := fn.GetAlerts()
|
2020-05-10 18:58:17 +02:00
|
|
|
expectedAlerts := []notifier.Alert{*alert1, *alert2}
|
|
|
|
compareAlerts(t, expectedAlerts, gotAlerts)
|
|
|
|
|
2023-10-13 13:54:33 +02:00
|
|
|
gotAlertsNum := fn.GetCounter()
|
2022-03-16 16:26:33 +01:00
|
|
|
if gotAlertsNum < len(expectedAlerts)*2 {
|
|
|
|
t.Fatalf("expected to receive at least %d alerts; got %d instead",
|
|
|
|
len(expectedAlerts)*2, gotAlertsNum)
|
|
|
|
}
|
|
|
|
|
2020-05-10 18:58:17 +02:00
|
|
|
// reset previous data
|
2023-10-13 13:54:33 +02:00
|
|
|
fs.Reset()
|
2020-05-10 18:58:17 +02:00
|
|
|
// and set only one datapoint for response
|
2023-10-13 13:54:33 +02:00
|
|
|
fs.Add(m1)
|
2020-05-10 18:58:17 +02:00
|
|
|
|
|
|
|
// wait for multiple evals
|
|
|
|
time.Sleep(20 * evalInterval)
|
|
|
|
|
2023-10-13 13:54:33 +02:00
|
|
|
gotAlerts = fn.GetAlerts()
|
2022-03-29 15:09:07 +02:00
|
|
|
alert2.State = notifier.StateInactive
|
|
|
|
expectedAlerts = []notifier.Alert{*alert1, *alert2}
|
2020-05-10 18:58:17 +02:00
|
|
|
compareAlerts(t, expectedAlerts, gotAlerts)
|
|
|
|
|
2023-10-13 13:54:33 +02:00
|
|
|
g.Close()
|
2020-05-10 18:58:17 +02:00
|
|
|
<-finished
|
|
|
|
}
|
2021-09-13 14:48:18 +02:00
|
|
|
|
|
|
|
func TestResolveDuration(t *testing.T) {
|
|
|
|
testCases := []struct {
|
|
|
|
groupInterval time.Duration
|
|
|
|
maxDuration time.Duration
|
2022-03-16 16:26:33 +01:00
|
|
|
resendDelay time.Duration
|
2021-09-13 14:48:18 +02:00
|
|
|
expected time.Duration
|
|
|
|
}{
|
2022-03-16 16:26:33 +01:00
|
|
|
{time.Minute, 0, 0, 4 * time.Minute},
|
|
|
|
{time.Minute, 0, 2 * time.Minute, 8 * time.Minute},
|
|
|
|
{time.Minute, 4 * time.Minute, 4 * time.Minute, 4 * time.Minute},
|
|
|
|
{2 * time.Minute, time.Minute, 2 * time.Minute, time.Minute},
|
|
|
|
{time.Minute, 2 * time.Minute, 1 * time.Minute, 2 * time.Minute},
|
|
|
|
{2 * time.Minute, 0, 1 * time.Minute, 8 * time.Minute},
|
|
|
|
{0, 0, 0, 0},
|
2021-09-13 14:48:18 +02:00
|
|
|
}
|
2022-03-16 16:26:33 +01:00
|
|
|
|
2021-09-13 14:48:18 +02:00
|
|
|
for _, tc := range testCases {
|
|
|
|
t.Run(fmt.Sprintf("%v-%v-%v", tc.groupInterval, tc.expected, tc.maxDuration), func(t *testing.T) {
|
2022-03-29 15:09:07 +02:00
|
|
|
got := getResolveDuration(tc.groupInterval, tc.resendDelay, tc.maxDuration)
|
2021-09-13 14:48:18 +02:00
|
|
|
if got != tc.expected {
|
|
|
|
t.Errorf("expected to have %v; got %v", tc.expected, got)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2022-03-29 15:09:07 +02:00
|
|
|
|
|
|
|
func TestGetStaleSeries(t *testing.T) {
|
|
|
|
ts := time.Now()
|
|
|
|
e := &executor{
|
2023-10-13 13:54:33 +02:00
|
|
|
PreviouslySentSeriesToRW: make(map[uint64]map[string][]prompbmarshal.Label),
|
2022-03-29 15:09:07 +02:00
|
|
|
}
|
2023-10-13 13:54:33 +02:00
|
|
|
f := func(r Rule, labels, expLabels [][]prompbmarshal.Label) {
|
2022-03-29 15:09:07 +02:00
|
|
|
t.Helper()
|
|
|
|
var tss []prompbmarshal.TimeSeries
|
|
|
|
for _, l := range labels {
|
|
|
|
tss = append(tss, newTimeSeriesPB([]float64{1}, []int64{ts.Unix()}, l))
|
|
|
|
}
|
2023-10-13 13:54:33 +02:00
|
|
|
staleS := e.getStaleSeries(r, tss, ts)
|
2022-03-29 15:09:07 +02:00
|
|
|
if staleS == nil && expLabels == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if len(staleS) != len(expLabels) {
|
|
|
|
t.Fatalf("expected to get %d stale series, got %d",
|
|
|
|
len(expLabels), len(staleS))
|
|
|
|
}
|
|
|
|
for i, exp := range expLabels {
|
|
|
|
got := staleS[i]
|
|
|
|
if !reflect.DeepEqual(exp, got.Labels) {
|
|
|
|
t.Fatalf("expected to get labels: \n%v;\ngot instead: \n%v",
|
|
|
|
exp, got.Labels)
|
|
|
|
}
|
|
|
|
if len(got.Samples) != 1 {
|
|
|
|
t.Fatalf("expected to have 1 sample; got %d", len(got.Samples))
|
|
|
|
}
|
|
|
|
if !decimal.IsStaleNaN(got.Samples[0].Value) {
|
|
|
|
t.Fatalf("expected sample value to be %v; got %v", decimal.StaleNaN, got.Samples[0].Value)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// warn: keep in mind, that executor holds the state, so sequence of f calls matters
|
|
|
|
|
|
|
|
// single series
|
|
|
|
f(&AlertingRule{RuleID: 1},
|
|
|
|
[][]prompbmarshal.Label{toPromLabels(t, "__name__", "job:foo", "job", "foo")},
|
|
|
|
nil)
|
|
|
|
f(&AlertingRule{RuleID: 1},
|
|
|
|
[][]prompbmarshal.Label{toPromLabels(t, "__name__", "job:foo", "job", "foo")},
|
|
|
|
nil)
|
|
|
|
f(&AlertingRule{RuleID: 1},
|
|
|
|
nil,
|
|
|
|
[][]prompbmarshal.Label{toPromLabels(t, "__name__", "job:foo", "job", "foo")})
|
|
|
|
f(&AlertingRule{RuleID: 1},
|
|
|
|
nil,
|
|
|
|
nil)
|
|
|
|
|
|
|
|
// multiple series
|
|
|
|
f(&AlertingRule{RuleID: 1},
|
|
|
|
[][]prompbmarshal.Label{
|
|
|
|
toPromLabels(t, "__name__", "job:foo", "job", "foo"),
|
|
|
|
toPromLabels(t, "__name__", "job:foo", "job", "bar"),
|
|
|
|
},
|
|
|
|
nil)
|
|
|
|
f(&AlertingRule{RuleID: 1},
|
|
|
|
[][]prompbmarshal.Label{toPromLabels(t, "__name__", "job:foo", "job", "bar")},
|
|
|
|
[][]prompbmarshal.Label{toPromLabels(t, "__name__", "job:foo", "job", "foo")})
|
|
|
|
f(&AlertingRule{RuleID: 1},
|
|
|
|
[][]prompbmarshal.Label{toPromLabels(t, "__name__", "job:foo", "job", "bar")},
|
|
|
|
nil)
|
|
|
|
f(&AlertingRule{RuleID: 1},
|
|
|
|
nil,
|
|
|
|
[][]prompbmarshal.Label{toPromLabels(t, "__name__", "job:foo", "job", "bar")})
|
|
|
|
|
|
|
|
// multiple rules and series
|
|
|
|
f(&AlertingRule{RuleID: 1},
|
|
|
|
[][]prompbmarshal.Label{
|
|
|
|
toPromLabels(t, "__name__", "job:foo", "job", "foo"),
|
|
|
|
toPromLabels(t, "__name__", "job:foo", "job", "bar"),
|
|
|
|
},
|
|
|
|
nil)
|
|
|
|
f(&AlertingRule{RuleID: 2},
|
|
|
|
[][]prompbmarshal.Label{
|
|
|
|
toPromLabels(t, "__name__", "job:foo", "job", "foo"),
|
|
|
|
toPromLabels(t, "__name__", "job:foo", "job", "bar"),
|
|
|
|
},
|
|
|
|
nil)
|
|
|
|
f(&AlertingRule{RuleID: 1},
|
|
|
|
[][]prompbmarshal.Label{toPromLabels(t, "__name__", "job:foo", "job", "bar")},
|
|
|
|
[][]prompbmarshal.Label{toPromLabels(t, "__name__", "job:foo", "job", "foo")})
|
|
|
|
f(&AlertingRule{RuleID: 1},
|
|
|
|
[][]prompbmarshal.Label{toPromLabels(t, "__name__", "job:foo", "job", "bar")},
|
|
|
|
nil)
|
|
|
|
}
|
2022-05-13 10:04:49 +02:00
|
|
|
|
|
|
|
func TestPurgeStaleSeries(t *testing.T) {
|
|
|
|
ts := time.Now()
|
|
|
|
labels := toPromLabels(t, "__name__", "job:foo", "job", "foo")
|
|
|
|
tss := []prompbmarshal.TimeSeries{newTimeSeriesPB([]float64{1}, []int64{ts.Unix()}, labels)}
|
|
|
|
|
|
|
|
f := func(curRules, newRules, expStaleRules []Rule) {
|
|
|
|
t.Helper()
|
|
|
|
e := &executor{
|
2023-10-13 13:54:33 +02:00
|
|
|
PreviouslySentSeriesToRW: make(map[uint64]map[string][]prompbmarshal.Label),
|
2022-05-13 10:04:49 +02:00
|
|
|
}
|
|
|
|
// seed executor with series for
|
|
|
|
// current rules
|
|
|
|
for _, rule := range curRules {
|
|
|
|
e.getStaleSeries(rule, tss, ts)
|
|
|
|
}
|
|
|
|
|
|
|
|
e.purgeStaleSeries(newRules)
|
|
|
|
|
2023-10-13 13:54:33 +02:00
|
|
|
if len(e.PreviouslySentSeriesToRW) != len(expStaleRules) {
|
2022-05-13 10:04:49 +02:00
|
|
|
t.Fatalf("expected to get %d stale series, got %d",
|
2023-10-13 13:54:33 +02:00
|
|
|
len(expStaleRules), len(e.PreviouslySentSeriesToRW))
|
2022-05-13 10:04:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, exp := range expStaleRules {
|
2023-10-13 13:54:33 +02:00
|
|
|
if _, ok := e.PreviouslySentSeriesToRW[exp.ID()]; !ok {
|
2022-05-13 10:04:49 +02:00
|
|
|
t.Fatalf("expected to have rule %d; got nil instead", exp.ID())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
f(nil, nil, nil)
|
|
|
|
f(
|
|
|
|
nil,
|
|
|
|
[]Rule{&AlertingRule{RuleID: 1}},
|
|
|
|
nil,
|
|
|
|
)
|
|
|
|
f(
|
|
|
|
[]Rule{&AlertingRule{RuleID: 1}},
|
|
|
|
nil,
|
|
|
|
nil,
|
|
|
|
)
|
|
|
|
f(
|
|
|
|
[]Rule{&AlertingRule{RuleID: 1}},
|
|
|
|
[]Rule{&AlertingRule{RuleID: 2}},
|
|
|
|
nil,
|
|
|
|
)
|
|
|
|
f(
|
|
|
|
[]Rule{&AlertingRule{RuleID: 1}, &AlertingRule{RuleID: 2}},
|
|
|
|
[]Rule{&AlertingRule{RuleID: 2}},
|
|
|
|
[]Rule{&AlertingRule{RuleID: 2}},
|
|
|
|
)
|
|
|
|
f(
|
|
|
|
[]Rule{&AlertingRule{RuleID: 1}, &AlertingRule{RuleID: 2}},
|
|
|
|
[]Rule{&AlertingRule{RuleID: 1}, &AlertingRule{RuleID: 2}},
|
|
|
|
[]Rule{&AlertingRule{RuleID: 1}, &AlertingRule{RuleID: 2}},
|
|
|
|
)
|
|
|
|
}
|
2022-06-18 09:11:37 +02:00
|
|
|
|
|
|
|
func TestFaultyNotifier(t *testing.T) {
|
2023-10-13 13:54:33 +02:00
|
|
|
fq := &datasource.FakeQuerier{}
|
|
|
|
fq.Add(metricWithValueAndLabels(t, 1, "__name__", "foo", "job", "bar"))
|
2022-06-18 09:11:37 +02:00
|
|
|
|
|
|
|
r := newTestAlertingRule("instant", 0)
|
|
|
|
r.q = fq
|
|
|
|
|
2023-10-13 13:54:33 +02:00
|
|
|
fn := ¬ifier.FakeNotifier{}
|
2022-06-18 09:11:37 +02:00
|
|
|
e := &executor{
|
2023-10-13 13:54:33 +02:00
|
|
|
Notifiers: func() []notifier.Notifier {
|
2022-06-18 09:11:37 +02:00
|
|
|
return []notifier.Notifier{
|
2023-10-13 13:54:33 +02:00
|
|
|
¬ifier.FaultyNotifier{},
|
2022-06-18 09:11:37 +02:00
|
|
|
fn,
|
|
|
|
}
|
|
|
|
},
|
|
|
|
}
|
|
|
|
delay := 5 * time.Second
|
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), delay)
|
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
_ = e.exec(ctx, r, time.Now(), 0, 10)
|
|
|
|
}()
|
|
|
|
|
|
|
|
tn := time.Now()
|
|
|
|
deadline := tn.Add(delay / 2)
|
|
|
|
for {
|
2023-10-13 13:54:33 +02:00
|
|
|
if fn.GetCounter() > 0 {
|
2022-06-18 09:11:37 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
if tn.After(deadline) {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
tn = time.Now()
|
|
|
|
time.Sleep(time.Millisecond * 100)
|
|
|
|
}
|
|
|
|
t.Fatalf("alive notifier didn't receive notification by %v", deadline)
|
|
|
|
}
|
2022-12-06 15:36:46 +01:00
|
|
|
|
|
|
|
func TestFaultyRW(t *testing.T) {
|
2023-10-13 13:54:33 +02:00
|
|
|
fq := &datasource.FakeQuerier{}
|
|
|
|
fq.Add(metricWithValueAndLabels(t, 1, "__name__", "foo", "job", "bar"))
|
2022-12-06 15:36:46 +01:00
|
|
|
|
|
|
|
r := &RecordingRule{
|
|
|
|
Name: "test",
|
|
|
|
q: fq,
|
2023-10-13 13:54:33 +02:00
|
|
|
state: &ruleState{entries: make([]StateEntry, 10)},
|
2022-12-06 15:36:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
e := &executor{
|
2023-10-13 13:54:33 +02:00
|
|
|
Rw: &remotewrite.Client{},
|
|
|
|
PreviouslySentSeriesToRW: make(map[uint64]map[string][]prompbmarshal.Label),
|
2022-12-06 15:36:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
err := e.exec(context.Background(), r, time.Now(), 0, 10)
|
|
|
|
if err == nil {
|
|
|
|
t.Fatalf("expected to get an error from faulty RW client, got nil instead")
|
|
|
|
}
|
|
|
|
}
|
2023-03-01 15:48:20 +01:00
|
|
|
|
|
|
|
func TestCloseWithEvalInterruption(t *testing.T) {
|
2023-10-13 13:54:33 +02:00
|
|
|
const (
|
|
|
|
rules = `
|
|
|
|
- name: groupTest
|
|
|
|
rules:
|
|
|
|
- alert: VMRows
|
|
|
|
for: 1ms
|
|
|
|
expr: vm_rows > 0
|
|
|
|
labels:
|
|
|
|
label: bar
|
|
|
|
host: "{{ $labels.instance }}"
|
|
|
|
annotations:
|
|
|
|
summary: "{{ $value }}"
|
|
|
|
`
|
|
|
|
)
|
|
|
|
var groups []config.Group
|
|
|
|
err := yaml.Unmarshal([]byte(rules), &groups)
|
2023-03-01 15:48:20 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("failed to parse rules: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
const delay = time.Second * 2
|
2023-10-13 13:54:33 +02:00
|
|
|
fq := &datasource.FakeQuerierWithDelay{Delay: delay}
|
2023-03-01 15:48:20 +01:00
|
|
|
|
|
|
|
const evalInterval = time.Millisecond
|
2023-10-13 13:54:33 +02:00
|
|
|
g := NewGroup(groups[0], fq, evalInterval, nil)
|
2023-03-01 15:48:20 +01:00
|
|
|
|
2023-10-13 13:54:33 +02:00
|
|
|
go g.Start(context.Background(), nil, nil, nil)
|
2023-03-01 15:48:20 +01:00
|
|
|
|
|
|
|
time.Sleep(evalInterval * 20)
|
|
|
|
|
|
|
|
go func() {
|
2023-10-13 13:54:33 +02:00
|
|
|
g.Close()
|
2023-03-01 15:48:20 +01:00
|
|
|
}()
|
|
|
|
|
|
|
|
deadline := time.Tick(delay / 2)
|
|
|
|
select {
|
|
|
|
case <-deadline:
|
|
|
|
t.Fatalf("deadline for close exceeded")
|
|
|
|
case <-g.finishedCh:
|
|
|
|
}
|
|
|
|
}
|
2023-09-06 16:29:59 +02:00
|
|
|
|
|
|
|
func TestGroupStartDelay(t *testing.T) {
|
|
|
|
g := &Group{}
|
|
|
|
// interval of 5min and key generate a static delay of 30s
|
|
|
|
g.Interval = time.Minute * 5
|
|
|
|
key := uint64(math.MaxUint64 / 10)
|
|
|
|
|
|
|
|
f := func(atS, expS string) {
|
|
|
|
t.Helper()
|
2023-10-10 12:41:19 +02:00
|
|
|
at, err := time.Parse(time.RFC3339Nano, atS)
|
2023-09-06 16:29:59 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2023-10-10 12:41:19 +02:00
|
|
|
expTS, err := time.Parse(time.RFC3339Nano, expS)
|
2023-09-06 16:29:59 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
delay := delayBeforeStart(at, key, g.Interval, g.EvalOffset)
|
|
|
|
gotStart := at.Add(delay)
|
|
|
|
if expTS != gotStart {
|
|
|
|
t.Errorf("expected to get %v; got %v instead", expTS, gotStart)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// test group without offset
|
2023-10-10 12:41:19 +02:00
|
|
|
f("2023-01-01T00:00:00.000+00:00", "2023-01-01T00:00:30.000+00:00")
|
|
|
|
f("2023-01-01T00:00:00.999+00:00", "2023-01-01T00:00:30.000+00:00")
|
|
|
|
f("2023-01-01T00:00:29.000+00:00", "2023-01-01T00:00:30.000+00:00")
|
|
|
|
f("2023-01-01T00:00:31.000+00:00", "2023-01-01T00:05:30.000+00:00")
|
2023-09-06 16:29:59 +02:00
|
|
|
|
|
|
|
// test group with offset smaller than above fixed randSleep,
|
|
|
|
// this way randSleep will always be enough
|
|
|
|
offset := 20 * time.Second
|
|
|
|
g.EvalOffset = &offset
|
|
|
|
|
2023-10-10 12:41:19 +02:00
|
|
|
f("2023-01-01T00:00:00.000+00:00", "2023-01-01T00:00:30.000+00:00")
|
|
|
|
f("2023-01-01T00:00:29.000+00:00", "2023-01-01T00:00:30.000+00:00")
|
|
|
|
f("2023-01-01T00:00:31.000+00:00", "2023-01-01T00:05:30.000+00:00")
|
2023-09-06 16:29:59 +02:00
|
|
|
|
|
|
|
// test group with offset bigger than above fixed randSleep,
|
|
|
|
// this way offset will be added to delay
|
|
|
|
offset = 3 * time.Minute
|
|
|
|
g.EvalOffset = &offset
|
|
|
|
|
2023-10-10 12:41:19 +02:00
|
|
|
f("2023-01-01T00:00:00.000+00:00", "2023-01-01T00:03:30.000+00:00")
|
|
|
|
f("2023-01-01T00:00:29.000+00:00", "2023-01-01T00:03:30.000+00:00")
|
|
|
|
f("2023-01-01T00:01:00.000+00:00", "2023-01-01T00:08:30.000+00:00")
|
|
|
|
f("2023-01-01T00:03:30.000+00:00", "2023-01-01T00:08:30.000+00:00")
|
|
|
|
f("2023-01-01T00:07:30.000+00:00", "2023-01-01T00:13:30.000+00:00")
|
2023-09-06 16:29:59 +02:00
|
|
|
|
|
|
|
offset = 10 * time.Minute
|
|
|
|
g.EvalOffset = &offset
|
|
|
|
// interval of 1h and key generate a static delay of 6m
|
|
|
|
g.Interval = time.Hour
|
|
|
|
|
2023-10-10 12:41:19 +02:00
|
|
|
f("2023-01-01T00:00:00.000+00:00", "2023-01-01T00:16:00.000+00:00")
|
|
|
|
f("2023-01-01T00:05:00.000+00:00", "2023-01-01T00:16:00.000+00:00")
|
|
|
|
f("2023-01-01T00:30:00.000+00:00", "2023-01-01T01:16:00.000+00:00")
|
|
|
|
}
|
2023-09-06 16:29:59 +02:00
|
|
|
|
2023-10-10 12:41:19 +02:00
|
|
|
func TestGetPrometheusReqTimestamp(t *testing.T) {
|
|
|
|
offset := 30 * time.Minute
|
|
|
|
disableAlign := false
|
|
|
|
testCases := []struct {
|
|
|
|
name string
|
|
|
|
g *Group
|
|
|
|
originTS, expTS string
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
"with query align",
|
|
|
|
&Group{
|
|
|
|
Interval: time.Hour,
|
|
|
|
},
|
|
|
|
"2023-08-28T11:11:00+00:00",
|
|
|
|
"2023-08-28T11:00:00+00:00",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"without query align",
|
|
|
|
&Group{
|
|
|
|
Interval: time.Hour,
|
|
|
|
evalAlignment: &disableAlign,
|
|
|
|
},
|
|
|
|
"2023-08-28T11:11:00+00:00",
|
|
|
|
"2023-08-28T11:11:00+00:00",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"with eval_offset, find previous offset point",
|
|
|
|
&Group{
|
|
|
|
EvalOffset: &offset,
|
|
|
|
Interval: time.Hour,
|
|
|
|
},
|
|
|
|
"2023-08-28T11:11:00+00:00",
|
|
|
|
"2023-08-28T10:30:00+00:00",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"with eval_offset",
|
|
|
|
&Group{
|
|
|
|
EvalOffset: &offset,
|
|
|
|
Interval: time.Hour,
|
|
|
|
},
|
|
|
|
"2023-08-28T11:41:00+00:00",
|
|
|
|
"2023-08-28T11:30:00+00:00",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
|
|
originT, _ := time.Parse(time.RFC3339, tc.originTS)
|
|
|
|
expT, _ := time.Parse(time.RFC3339, tc.expTS)
|
|
|
|
gotTS := tc.g.adjustReqTimestamp(originT)
|
|
|
|
if !gotTS.Equal(expT) {
|
|
|
|
t.Fatalf("get wrong prometheus request timestamp, expect %s, got %s", expT, gotTS)
|
|
|
|
}
|
|
|
|
}
|
2023-09-06 16:29:59 +02:00
|
|
|
}
|
2023-10-13 13:54:33 +02:00
|
|
|
|
|
|
|
func TestRangeIterator(t *testing.T) {
|
|
|
|
testCases := []struct {
|
|
|
|
ri rangeIterator
|
|
|
|
result [][2]time.Time
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
ri: rangeIterator{
|
|
|
|
start: parseTime(t, "2021-01-01T12:00:00.000Z"),
|
|
|
|
end: parseTime(t, "2021-01-01T12:30:00.000Z"),
|
|
|
|
step: 5 * time.Minute,
|
|
|
|
},
|
|
|
|
result: [][2]time.Time{
|
|
|
|
{parseTime(t, "2021-01-01T12:00:00.000Z"), parseTime(t, "2021-01-01T12:05:00.000Z")},
|
|
|
|
{parseTime(t, "2021-01-01T12:05:00.000Z"), parseTime(t, "2021-01-01T12:10:00.000Z")},
|
|
|
|
{parseTime(t, "2021-01-01T12:10:00.000Z"), parseTime(t, "2021-01-01T12:15:00.000Z")},
|
|
|
|
{parseTime(t, "2021-01-01T12:15:00.000Z"), parseTime(t, "2021-01-01T12:20:00.000Z")},
|
|
|
|
{parseTime(t, "2021-01-01T12:20:00.000Z"), parseTime(t, "2021-01-01T12:25:00.000Z")},
|
|
|
|
{parseTime(t, "2021-01-01T12:25:00.000Z"), parseTime(t, "2021-01-01T12:30:00.000Z")},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ri: rangeIterator{
|
|
|
|
start: parseTime(t, "2021-01-01T12:00:00.000Z"),
|
|
|
|
end: parseTime(t, "2021-01-01T12:30:00.000Z"),
|
|
|
|
step: 45 * time.Minute,
|
|
|
|
},
|
|
|
|
result: [][2]time.Time{
|
|
|
|
{parseTime(t, "2021-01-01T12:00:00.000Z"), parseTime(t, "2021-01-01T12:30:00.000Z")},
|
|
|
|
{parseTime(t, "2021-01-01T12:30:00.000Z"), parseTime(t, "2021-01-01T12:30:00.000Z")},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
ri: rangeIterator{
|
|
|
|
start: parseTime(t, "2021-01-01T12:00:12.000Z"),
|
|
|
|
end: parseTime(t, "2021-01-01T12:00:17.000Z"),
|
|
|
|
step: time.Second,
|
|
|
|
},
|
|
|
|
result: [][2]time.Time{
|
|
|
|
{parseTime(t, "2021-01-01T12:00:12.000Z"), parseTime(t, "2021-01-01T12:00:13.000Z")},
|
|
|
|
{parseTime(t, "2021-01-01T12:00:13.000Z"), parseTime(t, "2021-01-01T12:00:14.000Z")},
|
|
|
|
{parseTime(t, "2021-01-01T12:00:14.000Z"), parseTime(t, "2021-01-01T12:00:15.000Z")},
|
|
|
|
{parseTime(t, "2021-01-01T12:00:15.000Z"), parseTime(t, "2021-01-01T12:00:16.000Z")},
|
|
|
|
{parseTime(t, "2021-01-01T12:00:16.000Z"), parseTime(t, "2021-01-01T12:00:17.000Z")},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, tc := range testCases {
|
|
|
|
t.Run(fmt.Sprintf("case %d", i), func(t *testing.T) {
|
|
|
|
var j int
|
|
|
|
for tc.ri.next() {
|
|
|
|
if len(tc.result) < j+1 {
|
|
|
|
t.Fatalf("unexpected result for iterator on step %d: %v - %v",
|
|
|
|
j, tc.ri.s, tc.ri.e)
|
|
|
|
}
|
|
|
|
s, e := tc.ri.s, tc.ri.e
|
|
|
|
expS, expE := tc.result[j][0], tc.result[j][1]
|
|
|
|
if s != expS {
|
|
|
|
t.Fatalf("expected to get start=%v; got %v", expS, s)
|
|
|
|
}
|
|
|
|
if e != expE {
|
|
|
|
t.Fatalf("expected to get end=%v; got %v", expE, e)
|
|
|
|
}
|
|
|
|
j++
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseTime(t *testing.T, s string) time.Time {
|
|
|
|
t.Helper()
|
|
|
|
tt, err := time.Parse("2006-01-02T15:04:05.000Z", s)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
return tt
|
|
|
|
}
|