mirror of
				https://github.com/prometheus/prometheus.git
				synced 2025-11-04 10:21:02 +01:00 
			
		
		
		
	This creates a new `model` directory and moves all data-model related packages over there: exemplar labels relabel rulefmt textparse timestamp value All the others are more or less utilities and have been moved to `util`: gate logging modetimevfs pool runtime Signed-off-by: beorn7 <beorn@grafana.com>
		
			
				
	
	
		
			216 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			216 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
// 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"})`,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			in: `sum without() (task:errors:rate10s{job="s"})`,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			in: `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`,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			in: `a - on(b) c`,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			in: `a - on(b) group_left(x) c`,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			in: `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)`,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			in: `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())
 | 
						|
		})
 | 
						|
	}
 | 
						|
}
 |