mirror of https://github.com/prometheus/prometheus
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
224 lines
4.5 KiB
224 lines
4.5 KiB
// Copyright 2015 The Prometheus Authors |
|
// Licensed under the Apache License, Version 2.0 (the "License"); |
|
// you may not use this file except in compliance with the License. |
|
// You may obtain a copy of the License at |
|
// |
|
// http://www.apache.org/licenses/LICENSE-2.0 |
|
// |
|
// Unless required by applicable law or agreed to in writing, software |
|
// distributed under the License is distributed on an "AS IS" BASIS, |
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
|
// See the License for the specific language governing permissions and |
|
// limitations under the License. |
|
|
|
package parser |
|
|
|
import ( |
|
"testing" |
|
|
|
"github.com/stretchr/testify/require" |
|
|
|
"github.com/prometheus/prometheus/model/labels" |
|
) |
|
|
|
func TestExprString(t *testing.T) { |
|
// A list of valid expressions that are expected to be |
|
// returned as out when calling String(). If out is empty the output |
|
// is expected to equal the input. |
|
inputs := []struct { |
|
in, out string |
|
}{ |
|
{ |
|
in: `sum by() (task:errors:rate10s{job="s"})`, |
|
out: `sum(task:errors:rate10s{job="s"})`, |
|
}, |
|
{ |
|
in: `sum by(code) (task:errors:rate10s{job="s"})`, |
|
out: `sum by (code) (task:errors:rate10s{job="s"})`, |
|
}, |
|
{ |
|
in: `sum without() (task:errors:rate10s{job="s"})`, |
|
out: `sum without () (task:errors:rate10s{job="s"})`, |
|
}, |
|
{ |
|
in: `sum without(instance) (task:errors:rate10s{job="s"})`, |
|
out: `sum without (instance) (task:errors:rate10s{job="s"})`, |
|
}, |
|
{ |
|
in: `topk(5, task:errors:rate10s{job="s"})`, |
|
}, |
|
{ |
|
in: `count_values("value", task:errors:rate10s{job="s"})`, |
|
}, |
|
{ |
|
in: `a - on() c`, |
|
out: `a - on () c`, |
|
}, |
|
{ |
|
in: `a - on(b) c`, |
|
out: `a - on (b) c`, |
|
}, |
|
{ |
|
in: `a - on(b) group_left(x) c`, |
|
out: `a - on (b) group_left (x) c`, |
|
}, |
|
{ |
|
in: `a - on(b) group_left(x, y) c`, |
|
out: `a - on (b) group_left (x, y) c`, |
|
}, |
|
{ |
|
in: `a - on(b) group_left c`, |
|
out: `a - on (b) group_left () c`, |
|
}, |
|
{ |
|
in: `a - on(b) group_left() (c)`, |
|
out: `a - on (b) group_left () (c)`, |
|
}, |
|
{ |
|
in: `a - ignoring(b) c`, |
|
out: `a - ignoring (b) c`, |
|
}, |
|
{ |
|
in: `a - ignoring() c`, |
|
out: `a - c`, |
|
}, |
|
{ |
|
in: `up > bool 0`, |
|
}, |
|
{ |
|
in: `a offset 1m`, |
|
}, |
|
{ |
|
in: `a offset -7m`, |
|
}, |
|
{ |
|
in: `a{c="d"}[5m] offset 1m`, |
|
}, |
|
{ |
|
in: `a[5m] offset 1m`, |
|
}, |
|
{ |
|
in: `a[12m] offset -3m`, |
|
}, |
|
{ |
|
in: `a[1h:5m] offset 1m`, |
|
}, |
|
{ |
|
in: `{__name__="a"}`, |
|
}, |
|
{ |
|
in: `a{b!="c"}[1m]`, |
|
}, |
|
{ |
|
in: `a{b=~"c"}[1m]`, |
|
}, |
|
{ |
|
in: `a{b!~"c"}[1m]`, |
|
}, |
|
{ |
|
in: `a @ 10`, |
|
out: `a @ 10.000`, |
|
}, |
|
{ |
|
in: `a[1m] @ 10`, |
|
out: `a[1m] @ 10.000`, |
|
}, |
|
{ |
|
in: `a @ start()`, |
|
}, |
|
{ |
|
in: `a @ end()`, |
|
}, |
|
{ |
|
in: `a[1m] @ start()`, |
|
}, |
|
{ |
|
in: `a[1m] @ end()`, |
|
}, |
|
} |
|
|
|
for _, test := range inputs { |
|
expr, err := ParseExpr(test.in) |
|
require.NoError(t, err) |
|
|
|
exp := test.in |
|
if test.out != "" { |
|
exp = test.out |
|
} |
|
|
|
require.Equal(t, exp, expr.String()) |
|
} |
|
} |
|
|
|
func TestVectorSelector_String(t *testing.T) { |
|
for _, tc := range []struct { |
|
name string |
|
vs VectorSelector |
|
expected string |
|
}{ |
|
{ |
|
name: "empty value", |
|
vs: VectorSelector{}, |
|
expected: ``, |
|
}, |
|
{ |
|
name: "no matchers with name", |
|
vs: VectorSelector{Name: "foobar"}, |
|
expected: `foobar`, |
|
}, |
|
{ |
|
name: "one matcher with name", |
|
vs: VectorSelector{ |
|
Name: "foobar", |
|
LabelMatchers: []*labels.Matcher{ |
|
labels.MustNewMatcher(labels.MatchEqual, "a", "x"), |
|
}, |
|
}, |
|
expected: `foobar{a="x"}`, |
|
}, |
|
{ |
|
name: "two matchers with name", |
|
vs: VectorSelector{ |
|
Name: "foobar", |
|
LabelMatchers: []*labels.Matcher{ |
|
labels.MustNewMatcher(labels.MatchEqual, "a", "x"), |
|
labels.MustNewMatcher(labels.MatchEqual, "b", "y"), |
|
}, |
|
}, |
|
expected: `foobar{a="x",b="y"}`, |
|
}, |
|
{ |
|
name: "two matchers without name", |
|
vs: VectorSelector{ |
|
LabelMatchers: []*labels.Matcher{ |
|
labels.MustNewMatcher(labels.MatchEqual, "a", "x"), |
|
labels.MustNewMatcher(labels.MatchEqual, "b", "y"), |
|
}, |
|
}, |
|
expected: `{a="x",b="y"}`, |
|
}, |
|
{ |
|
name: "name matcher and name", |
|
vs: VectorSelector{ |
|
Name: "foobar", |
|
LabelMatchers: []*labels.Matcher{ |
|
labels.MustNewMatcher(labels.MatchEqual, labels.MetricName, "foobar"), |
|
}, |
|
}, |
|
expected: `foobar`, |
|
}, |
|
{ |
|
name: "name matcher only", |
|
vs: VectorSelector{ |
|
LabelMatchers: []*labels.Matcher{ |
|
labels.MustNewMatcher(labels.MatchEqual, labels.MetricName, "foobar"), |
|
}, |
|
}, |
|
expected: `{__name__="foobar"}`, |
|
}, |
|
} { |
|
t.Run(tc.name, func(t *testing.T) { |
|
require.Equal(t, tc.expected, tc.vs.String()) |
|
}) |
|
} |
|
}
|
|
|