|
|
|
@ -15,8 +15,10 @@ package rules
|
|
|
|
|
|
|
|
|
|
import ( |
|
|
|
|
"testing" |
|
|
|
|
"time" |
|
|
|
|
|
|
|
|
|
"github.com/prometheus/prometheus/pkg/labels" |
|
|
|
|
"github.com/prometheus/prometheus/pkg/timestamp" |
|
|
|
|
"github.com/prometheus/prometheus/promql" |
|
|
|
|
"github.com/prometheus/prometheus/util/testutil" |
|
|
|
|
) |
|
|
|
@ -37,3 +39,106 @@ annotations:
|
|
|
|
|
got := rule.HTMLSnippet("/test/prefix") |
|
|
|
|
testutil.Assert(t, want == got, "incorrect HTML snippet; want:\n\n|%v|\n\ngot:\n\n|%v|", want, got) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
func TestAlertingRuleLabelsUpdate(t *testing.T) { |
|
|
|
|
suite, err := promql.NewTest(t, ` |
|
|
|
|
load 1m |
|
|
|
|
http_requests{job="app-server", instance="0"} 75 85 70 70 |
|
|
|
|
`) |
|
|
|
|
testutil.Ok(t, err) |
|
|
|
|
defer suite.Close() |
|
|
|
|
|
|
|
|
|
err = suite.Run() |
|
|
|
|
testutil.Ok(t, err) |
|
|
|
|
|
|
|
|
|
expr, err := promql.ParseExpr(`http_requests < 100`) |
|
|
|
|
testutil.Ok(t, err) |
|
|
|
|
|
|
|
|
|
rule := NewAlertingRule( |
|
|
|
|
"HTTPRequestRateLow", |
|
|
|
|
expr, |
|
|
|
|
time.Minute, |
|
|
|
|
// Basing alerting rule labels off of a value that can change is a very bad idea.
|
|
|
|
|
// If an alert is going back and forth between two label values it will never fire.
|
|
|
|
|
// Instead, you should write two alerts with constant labels.
|
|
|
|
|
labels.FromStrings("severity", "{{ if lt $value 80.0 }}critical{{ else }}warning{{ end }}"), |
|
|
|
|
nil, true, nil, |
|
|
|
|
) |
|
|
|
|
|
|
|
|
|
results := []promql.Vector{ |
|
|
|
|
promql.Vector{ |
|
|
|
|
{ |
|
|
|
|
Metric: labels.FromStrings( |
|
|
|
|
"__name__", "ALERTS", |
|
|
|
|
"alertname", "HTTPRequestRateLow", |
|
|
|
|
"alertstate", "pending", |
|
|
|
|
"instance", "0", |
|
|
|
|
"job", "app-server", |
|
|
|
|
"severity", "critical", |
|
|
|
|
), |
|
|
|
|
Point: promql.Point{V: 1}, |
|
|
|
|
}, |
|
|
|
|
}, |
|
|
|
|
promql.Vector{ |
|
|
|
|
{ |
|
|
|
|
Metric: labels.FromStrings( |
|
|
|
|
"__name__", "ALERTS", |
|
|
|
|
"alertname", "HTTPRequestRateLow", |
|
|
|
|
"alertstate", "pending", |
|
|
|
|
"instance", "0", |
|
|
|
|
"job", "app-server", |
|
|
|
|
"severity", "warning", |
|
|
|
|
), |
|
|
|
|
Point: promql.Point{V: 1}, |
|
|
|
|
}, |
|
|
|
|
}, |
|
|
|
|
promql.Vector{ |
|
|
|
|
{ |
|
|
|
|
Metric: labels.FromStrings( |
|
|
|
|
"__name__", "ALERTS", |
|
|
|
|
"alertname", "HTTPRequestRateLow", |
|
|
|
|
"alertstate", "pending", |
|
|
|
|
"instance", "0", |
|
|
|
|
"job", "app-server", |
|
|
|
|
"severity", "critical", |
|
|
|
|
), |
|
|
|
|
Point: promql.Point{V: 1}, |
|
|
|
|
}, |
|
|
|
|
}, |
|
|
|
|
promql.Vector{ |
|
|
|
|
{ |
|
|
|
|
Metric: labels.FromStrings( |
|
|
|
|
"__name__", "ALERTS", |
|
|
|
|
"alertname", "HTTPRequestRateLow", |
|
|
|
|
"alertstate", "firing", |
|
|
|
|
"instance", "0", |
|
|
|
|
"job", "app-server", |
|
|
|
|
"severity", "critical", |
|
|
|
|
), |
|
|
|
|
Point: promql.Point{V: 1}, |
|
|
|
|
}, |
|
|
|
|
}, |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
baseTime := time.Unix(0, 0) |
|
|
|
|
for i, result := range results { |
|
|
|
|
t.Logf("case %d", i) |
|
|
|
|
evalTime := baseTime.Add(time.Duration(i) * time.Minute) |
|
|
|
|
result[0].Point.T = timestamp.FromTime(evalTime) |
|
|
|
|
res, err := rule.Eval(suite.Context(), evalTime, EngineQueryFunc(suite.QueryEngine(), suite.Storage()), nil) |
|
|
|
|
testutil.Ok(t, err) |
|
|
|
|
|
|
|
|
|
var filteredRes promql.Vector // After removing 'ALERTS_FOR_STATE' samples.
|
|
|
|
|
for _, smpl := range res { |
|
|
|
|
smplName := smpl.Metric.Get("__name__") |
|
|
|
|
if smplName == "ALERTS" { |
|
|
|
|
filteredRes = append(filteredRes, smpl) |
|
|
|
|
} else { |
|
|
|
|
// If not 'ALERTS', it has to be 'ALERTS_FOR_STATE'.
|
|
|
|
|
testutil.Equals(t, smplName, "ALERTS_FOR_STATE") |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
testutil.Equals(t, result, filteredRes) |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|