// clear the current test storage of all inserted samples. func (t *Test) clear() { if t.closeStorage != nil { t.closeStorage() } if t.queryEngine != nil { t.queryEngine.Stop() } var closer testutil.Closer t.storage, closer = local.NewTestStorage(t, 1) t.closeStorage = closer.Close t.queryEngine = NewEngine(t.storage, nil) }
// clear the current test storage of all inserted samples. func (t *Test) clear() { if t.closeStorage != nil { t.closeStorage() } if t.cancelCtx != nil { t.cancelCtx() } var closer testutil.Closer t.storage, closer = local.NewTestStorage(t, 2) t.closeStorage = closer.Close t.queryEngine = NewEngine(t.storage, nil) t.context, t.cancelCtx = context.WithCancel(context.Background()) }
func TestRuleEval(t *testing.T) { storage, closer := local.NewTestStorage(t, 2) defer closer.Close() engine := promql.NewEngine(storage, nil) ctx, cancelCtx := context.WithCancel(context.Background()) defer cancelCtx() now := model.Now() suite := []struct { name string expr promql.Expr labels model.LabelSet result model.Vector }{ { name: "nolabels", expr: &promql.NumberLiteral{Val: 1}, labels: model.LabelSet{}, result: model.Vector{&model.Sample{ Value: 1, Timestamp: now, Metric: model.Metric{"__name__": "nolabels"}, }}, }, { name: "labels", expr: &promql.NumberLiteral{Val: 1}, labels: model.LabelSet{"foo": "bar"}, result: model.Vector{&model.Sample{ Value: 1, Timestamp: now, Metric: model.Metric{"__name__": "labels", "foo": "bar"}, }}, }, } for _, test := range suite { rule := NewRecordingRule(test.name, test.expr, test.labels) result, err := rule.eval(ctx, now, engine, "") if err != nil { t.Fatalf("Error evaluating %s", test.name) } if !reflect.DeepEqual(result, test.result) { t.Fatalf("Error: expected %q, got %q", test.result, result) } } }
func TestTemplateExpansion(t *testing.T) { scenarios := []testTemplatesScenario{ { // No template. text: "plain text", output: "plain text", }, { // Simple value. text: "{{ 1 }}", output: "1", }, { // HTML escaping. text: "{{ \"<b>\" }}", output: "<b>", html: true, }, { // Disabling HTML escaping. text: "{{ \"<b>\" | safeHtml }}", output: "<b>", html: true, }, { // HTML escaping doesn't apply to non-html. text: "{{ \"<b>\" }}", output: "<b>", }, { // Pass multiple arguments to templates. text: "{{define \"x\"}}{{.arg0}} {{.arg1}}{{end}}{{template \"x\" (args 1 \"2\")}}", output: "1 2", }, { text: "{{ query \"1.5\" | first | value }}", output: "1.5", }, { // Get value from scalar query. text: "{{ query \"scalar(count(metric))\" | first | value }}", output: "2", }, { // Get value from query. text: "{{ query \"metric{instance='a'}\" | first | value }}", output: "11", }, { // Get label from query. text: "{{ query \"metric{instance='a'}\" | first | label \"instance\" }}", output: "a", }, { // Range over query and sort by label. text: "{{ range query \"metric\" | sortByLabel \"instance\" }}{{.Labels.instance}}:{{.Value}}: {{end}}", output: "a:11: b:21: ", }, { // Unparsable template. text: "{{", shouldFail: true, }, { // Error in function. text: "{{ query \"missing\" | first }}", shouldFail: true, }, { // Panic. text: "{{ (query \"missing\").banana }}", shouldFail: true, }, { // Regex replacement. text: "{{ reReplaceAll \"(a)b\" \"x$1\" \"ab\" }}", output: "xa", }, { // Humanize. text: "{{ range . }}{{ humanize . }}:{{ end }}", input: []float64{0.0, 1.0, 1234567.0, .12}, output: "0:1:1.235M:120m:", }, { // Humanize1024. text: "{{ range . }}{{ humanize1024 . }}:{{ end }}", input: []float64{0.0, 1.0, 1048576.0, .12}, output: "0:1:1Mi:0.12:", }, { // HumanizeDuration - seconds. text: "{{ range . }}{{ humanizeDuration . }}:{{ end }}", input: []float64{0, 1, 60, 3600, 86400, 86400 + 3600, -(86400*2 + 3600*3 + 60*4 + 5), 899.99}, output: "0s:1s:1m 0s:1h 0m 0s:1d 0h 0m 0s:1d 1h 0m 0s:-2d 3h 4m 5s:14m 59s:", }, { // HumanizeDuration - subsecond and fractional seconds. text: "{{ range . }}{{ humanizeDuration . }}:{{ end }}", input: []float64{.1, .0001, .12345, 60.1, 60.5, 1.2345, 12.345}, output: "100ms:100us:123.5ms:1m 0s:1m 0s:1.234s:12.35s:", }, { // Humanize* Inf and NaN. text: "{{ range . }}{{ humanize . }}:{{ humanize1024 . }}:{{ humanizeDuration . }}:{{humanizeTimestamp .}}:{{ end }}", input: []float64{math.Inf(1), math.Inf(-1), math.NaN()}, output: "+Inf:+Inf:+Inf:+Inf:-Inf:-Inf:-Inf:-Inf:NaN:NaN:NaN:NaN:", }, { // HumanizeTimestamp - clientmodel.SampleValue input. text: "{{ 1435065584.128 | humanizeTimestamp }}", output: "2015-06-23 13:19:44.128 +0000 UTC", }, { // Title. text: "{{ \"aa bb CC\" | title }}", output: "Aa Bb CC", }, { // Match. text: "{{ match \"a+\" \"aa\" }} {{ match \"a+\" \"b\" }}", output: "true false", }, { // graphLink. text: "{{ graphLink \"up\" }}", output: "/graph#%5B%7B%22expr%22%3A%22up%22%2C%22tab%22%3A0%7D%5D", }, { // tableLink. text: "{{ tableLink \"up\" }}", output: "/graph#%5B%7B%22expr%22%3A%22up%22%2C%22tab%22%3A1%7D%5D", }, { // tmpl. text: "{{ define \"a\" }}x{{ end }}{{ $name := \"a\"}}{{ tmpl $name . }}", output: "x", html: true, }, } time := clientmodel.Timestamp(0) storage, closer := local.NewTestStorage(t, 1) defer closer.Close() storage.Append(&clientmodel.Sample{ Metric: clientmodel.Metric{ clientmodel.MetricNameLabel: "metric", "instance": "a"}, Value: 11, }) storage.Append(&clientmodel.Sample{ Metric: clientmodel.Metric{ clientmodel.MetricNameLabel: "metric", "instance": "b"}, Value: 21, }) storage.WaitForIndexing() engine := promql.NewEngine(storage, nil) for i, s := range scenarios { var result string var err error expander := NewTemplateExpander(s.text, "test", s.input, time, engine, "") if s.html { result, err = expander.ExpandHTML(nil) } else { result, err = expander.Expand() } if s.shouldFail { if err == nil { t.Fatalf("%d. Error not returned from %v", i, s.text) } continue } if err != nil { t.Fatalf("%d. Error returned from %v: %v", i, s.text, err) continue } if result != s.output { t.Fatalf("%d. Error in result from %v: Expected '%v' Got '%v'", i, s.text, s.output, result) continue } } }
func TestAlertingRule(t *testing.T) { // Labels in expected output need to be alphabetically sorted. var evalOutputs = [][]string{ { `ALERTS{alertname="HttpRequestRateLow", alertstate="pending", group="canary", instance="0", job="app-server", severity="critical"} => 1 @[%v]`, `ALERTS{alertname="HttpRequestRateLow", alertstate="pending", group="canary", instance="1", job="app-server", severity="critical"} => 1 @[%v]`, }, { `ALERTS{alertname="HttpRequestRateLow", alertstate="pending", group="canary", instance="0", job="app-server", severity="critical"} => 0 @[%v]`, `ALERTS{alertname="HttpRequestRateLow", alertstate="firing", group="canary", instance="0", job="app-server", severity="critical"} => 1 @[%v]`, `ALERTS{alertname="HttpRequestRateLow", alertstate="pending", group="canary", instance="1", job="app-server", severity="critical"} => 0 @[%v]`, `ALERTS{alertname="HttpRequestRateLow", alertstate="firing", group="canary", instance="1", job="app-server", severity="critical"} => 1 @[%v]`, }, { `ALERTS{alertname="HttpRequestRateLow", alertstate="firing", group="canary", instance="1", job="app-server", severity="critical"} => 0 @[%v]`, `ALERTS{alertname="HttpRequestRateLow", alertstate="firing", group="canary", instance="0", job="app-server", severity="critical"} => 0 @[%v]`, }, { /* empty */ }, { /* empty */ }, } storage, closer := local.NewTestStorage(t, 1) defer closer.Close() storeMatrix(storage, testMatrix) engine := promql.NewEngine(storage, nil) defer engine.Stop() expr, err := promql.ParseExpr(`http_requests{group="canary", job="app-server"} < 100`) if err != nil { t.Fatalf("Unable to parse alert expression: %s", err) } alertLabels := clientmodel.LabelSet{ "severity": "critical", } rule := NewAlertingRule("HttpRequestRateLow", expr, time.Minute, alertLabels, "summary", "description") for i, expectedLines := range evalOutputs { evalTime := testStartTime.Add(testSampleInterval * time.Duration(i)) res, err := rule.eval(evalTime, engine) if err != nil { t.Fatalf("Error during alerting rule evaluation: %s", err) } actualLines := strings.Split(res.String(), "\n") expectedLines := annotateWithTime(expectedLines, evalTime) if actualLines[0] == "" { actualLines = []string{} } failed := false if len(actualLines) != len(expectedLines) { t.Errorf("%d. Number of samples in expected and actual output don't match (%d vs. %d)", i, len(expectedLines), len(actualLines)) failed = true } for j, expectedSample := range expectedLines { found := false for _, actualSample := range actualLines { if actualSample == expectedSample { found = true } } if !found { t.Errorf("%d.%d. Couldn't find expected sample in output: '%v'", i, j, expectedSample) failed = true } } if failed { t.Fatalf("%d. Expected and actual outputs don't match:\n%v", i, vectorComparisonString(expectedLines, actualLines)) } } }
func TestQuery(t *testing.T) { scenarios := []struct { // URL query string. queryStr string // Expected HTTP response status code. status int // Regex to match against response body. bodyRe string }{ { queryStr: "", status: http.StatusOK, bodyRe: `{"type":"error","value":"Parse error at char 1: no expression found in input","version":1}`, }, { queryStr: "expr=testmetric", status: http.StatusOK, bodyRe: `{"type":"vector","value":\[\{"metric":{"__name__":"testmetric"},"value":"0","timestamp":\d+\.\d+}\],"version":1\}`, }, { queryStr: "expr=testmetric×tamp=" + testTimestamp.String(), status: http.StatusOK, bodyRe: `{"type":"vector","value":\[\{"metric":{"__name__":"testmetric"},"value":"0","timestamp":` + testTimestamp.String() + `}\],"version":1\}`, }, { queryStr: "expr=testmetric×tamp=" + testTimestamp.Add(-time.Hour).String(), status: http.StatusOK, bodyRe: `{"type":"vector","value":\[\],"version":1\}`, }, { queryStr: "timestamp=invalid", status: http.StatusBadRequest, bodyRe: "invalid query timestamp", }, { queryStr: "expr=(badexpression", status: http.StatusOK, bodyRe: `{"type":"error","value":"Parse error at char 15: unclosed left parenthesis","version":1}`, }, } storage, closer := local.NewTestStorage(t, 1) defer closer.Close() storage.Append(&clientmodel.Sample{ Metric: clientmodel.Metric{ clientmodel.MetricNameLabel: "testmetric", }, Timestamp: testTimestamp, Value: 0, }) storage.WaitForIndexing() api := &API{ Now: testNow, Storage: storage, QueryEngine: promql.NewEngine(storage, nil), } rtr := route.New() api.Register(rtr.WithPrefix("/api")) server := httptest.NewServer(rtr) defer server.Close() for i, s := range scenarios { // Do query. resp, err := http.Get(server.URL + "/api/query?" + s.queryStr) if err != nil { t.Fatalf("%d. Error querying API: %s", i, err) } // Check status code. if resp.StatusCode != s.status { t.Fatalf("%d. Unexpected status code; got %d, want %d", i, resp.StatusCode, s.status) } // Check response headers. ct := resp.Header["Content-Type"] if len(ct) != 1 { t.Fatalf("%d. Unexpected number of 'Content-Type' headers; got %d, want 1", i, len(ct)) } if ct[0] != "application/json" { t.Fatalf("%d. Unexpected 'Content-Type' header; got %s; want %s", i, ct[0], "application/json") } // Check body. b, err := ioutil.ReadAll(resp.Body) if err != nil { t.Fatalf("%d. Error reading response body: %s", i, err) } re := regexp.MustCompile(s.bodyRe) if !re.Match(b) { t.Fatalf("%d. Body didn't match '%s'. Body: %s", i, s.bodyRe, string(b)) } } }
func TestRangedEvaluationRegressions(t *testing.T) { scenarios := []struct { in ast.Matrix out ast.Matrix expr string }{ { // Testing COWMetric behavior in drop_common_labels. in: ast.Matrix{ { Metric: clientmodel.COWMetric{ Metric: clientmodel.Metric{ clientmodel.MetricNameLabel: "testmetric", "testlabel": "1", }, }, Values: metric.Values{ { Timestamp: testStartTime, Value: 1, }, { Timestamp: testStartTime.Add(time.Hour), Value: 1, }, }, }, { Metric: clientmodel.COWMetric{ Metric: clientmodel.Metric{ clientmodel.MetricNameLabel: "testmetric", "testlabel": "2", }, }, Values: metric.Values{ { Timestamp: testStartTime.Add(time.Hour), Value: 2, }, }, }, }, out: ast.Matrix{ { Metric: clientmodel.COWMetric{ Metric: clientmodel.Metric{ clientmodel.MetricNameLabel: "testmetric", }, }, Values: metric.Values{ { Timestamp: testStartTime, Value: 1, }, }, }, { Metric: clientmodel.COWMetric{ Metric: clientmodel.Metric{ clientmodel.MetricNameLabel: "testmetric", "testlabel": "1", }, }, Values: metric.Values{ { Timestamp: testStartTime.Add(time.Hour), Value: 1, }, }, }, { Metric: clientmodel.COWMetric{ Metric: clientmodel.Metric{ clientmodel.MetricNameLabel: "testmetric", "testlabel": "2", }, }, Values: metric.Values{ { Timestamp: testStartTime.Add(time.Hour), Value: 2, }, }, }, }, expr: "drop_common_labels(testmetric)", }, { // Testing COWMetric behavior in vector aggregation. in: ast.Matrix{ { Metric: clientmodel.COWMetric{ Metric: clientmodel.Metric{ clientmodel.MetricNameLabel: "testmetric", "testlabel": "1", }, }, Values: metric.Values{ { Timestamp: testStartTime, Value: 1, }, { Timestamp: testStartTime.Add(time.Hour), Value: 1, }, }, }, { Metric: clientmodel.COWMetric{ Metric: clientmodel.Metric{ clientmodel.MetricNameLabel: "testmetric", "testlabel": "2", }, }, Values: metric.Values{ { Timestamp: testStartTime, Value: 2, }, }, }, }, out: ast.Matrix{ { Metric: clientmodel.COWMetric{ Metric: clientmodel.Metric{}, }, Values: metric.Values{ { Timestamp: testStartTime, Value: 3, }, }, }, { Metric: clientmodel.COWMetric{ Metric: clientmodel.Metric{ "testlabel": "1", }, }, Values: metric.Values{ { Timestamp: testStartTime.Add(time.Hour), Value: 1, }, }, }, }, expr: "sum(testmetric) keeping_extra", }, } for i, s := range scenarios { storage, closer := local.NewTestStorage(t) storeMatrix(storage, s.in) expr, err := LoadExprFromString(s.expr) if err != nil { t.Fatalf("%d. Error parsing expression: %v", i, err) } got, err := ast.EvalVectorRange( expr.(ast.VectorNode), testStartTime, testStartTime.Add(time.Hour), time.Hour, storage, stats.NewTimerGroup(), ) if err != nil { t.Fatalf("%d. Error evaluating expression: %v", i, err) } if got.String() != s.out.String() { t.Fatalf("%d. Expression: %s\n\ngot:\n=====\n%v\n====\n\nwant:\n=====\n%v\n=====\n", i, s.expr, got.String(), s.out.String()) } closer.Close() } }
func newTestStorage(t testing.TB) (storage local.Storage, closer test.Closer) { storage, closer = local.NewTestStorage(t) storeMatrix(storage, testMatrix) return storage, closer }