2018-08-23 01:21:38 +03:00
|
|
|
// Copyright 2018 The Prometheus Authors
|
2018-08-23 00:59:40 +03:00
|
|
|
// 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.
|
|
|
|
|
2018-08-23 00:57:08 +03:00
|
|
|
package testutil
|
2018-07-03 14:25:00 +03:00
|
|
|
|
|
|
|
import (
|
2022-08-05 17:27:47 +03:00
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
2018-08-23 01:21:38 +03:00
|
|
|
"strings"
|
2018-07-03 14:25:00 +03:00
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
|
|
)
|
|
|
|
|
2018-09-04 13:38:29 +03:00
|
|
|
type untypedCollector struct{}
|
|
|
|
|
|
|
|
func (u untypedCollector) Describe(c chan<- *prometheus.Desc) {
|
|
|
|
c <- prometheus.NewDesc("name", "help", nil, nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (u untypedCollector) Collect(c chan<- prometheus.Metric) {
|
|
|
|
c <- prometheus.MustNewConstMetric(
|
|
|
|
prometheus.NewDesc("name", "help", nil, nil),
|
|
|
|
prometheus.UntypedValue,
|
|
|
|
2001,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestToFloat64(t *testing.T) {
|
|
|
|
gaugeWithAValueSet := prometheus.NewGauge(prometheus.GaugeOpts{})
|
|
|
|
gaugeWithAValueSet.Set(3.14)
|
|
|
|
|
|
|
|
counterVecWithOneElement := prometheus.NewCounterVec(prometheus.CounterOpts{}, []string{"foo"})
|
|
|
|
counterVecWithOneElement.WithLabelValues("bar").Inc()
|
|
|
|
|
|
|
|
counterVecWithTwoElements := prometheus.NewCounterVec(prometheus.CounterOpts{}, []string{"foo"})
|
|
|
|
counterVecWithTwoElements.WithLabelValues("bar").Add(42)
|
|
|
|
counterVecWithTwoElements.WithLabelValues("baz").Inc()
|
|
|
|
|
|
|
|
histogramVecWithOneElement := prometheus.NewHistogramVec(prometheus.HistogramOpts{}, []string{"foo"})
|
|
|
|
histogramVecWithOneElement.WithLabelValues("bar").Observe(2.7)
|
|
|
|
|
|
|
|
scenarios := map[string]struct {
|
|
|
|
collector prometheus.Collector
|
|
|
|
panics bool
|
|
|
|
want float64
|
|
|
|
}{
|
|
|
|
"simple counter": {
|
|
|
|
collector: prometheus.NewCounter(prometheus.CounterOpts{}),
|
|
|
|
panics: false,
|
|
|
|
want: 0,
|
|
|
|
},
|
|
|
|
"simple gauge": {
|
|
|
|
collector: prometheus.NewGauge(prometheus.GaugeOpts{}),
|
|
|
|
panics: false,
|
|
|
|
want: 0,
|
|
|
|
},
|
|
|
|
"simple untyped": {
|
|
|
|
collector: untypedCollector{},
|
|
|
|
panics: false,
|
|
|
|
want: 2001,
|
|
|
|
},
|
|
|
|
"simple histogram": {
|
|
|
|
collector: prometheus.NewHistogram(prometheus.HistogramOpts{}),
|
|
|
|
panics: true,
|
|
|
|
},
|
|
|
|
"simple summary": {
|
|
|
|
collector: prometheus.NewSummary(prometheus.SummaryOpts{}),
|
|
|
|
panics: true,
|
|
|
|
},
|
|
|
|
"simple gauge with an actual value set": {
|
|
|
|
collector: gaugeWithAValueSet,
|
|
|
|
panics: false,
|
|
|
|
want: 3.14,
|
|
|
|
},
|
|
|
|
"counter vec with zero elements": {
|
|
|
|
collector: prometheus.NewCounterVec(prometheus.CounterOpts{}, nil),
|
|
|
|
panics: true,
|
|
|
|
},
|
|
|
|
"counter vec with one element": {
|
|
|
|
collector: counterVecWithOneElement,
|
|
|
|
panics: false,
|
|
|
|
want: 1,
|
|
|
|
},
|
|
|
|
"counter vec with two elements": {
|
|
|
|
collector: counterVecWithTwoElements,
|
|
|
|
panics: true,
|
|
|
|
},
|
|
|
|
"histogram vec with one element": {
|
|
|
|
collector: histogramVecWithOneElement,
|
|
|
|
panics: true,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for n, s := range scenarios {
|
|
|
|
t.Run(n, func(t *testing.T) {
|
|
|
|
defer func() {
|
|
|
|
r := recover()
|
|
|
|
if r == nil && s.panics {
|
|
|
|
t.Error("expected panic")
|
|
|
|
} else if r != nil && !s.panics {
|
|
|
|
t.Error("unexpected panic: ", r)
|
|
|
|
}
|
|
|
|
// Any other combination is the expected outcome.
|
|
|
|
}()
|
|
|
|
if got := ToFloat64(s.collector); got != s.want {
|
|
|
|
t.Errorf("want %f, got %f", s.want, got)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-23 01:21:38 +03:00
|
|
|
func TestCollectAndCompare(t *testing.T) {
|
2018-07-03 14:25:00 +03:00
|
|
|
const metadata = `
|
|
|
|
# HELP some_total A value that represents a counter.
|
|
|
|
# TYPE some_total counter
|
|
|
|
`
|
|
|
|
|
|
|
|
c := prometheus.NewCounter(prometheus.CounterOpts{
|
|
|
|
Name: "some_total",
|
|
|
|
Help: "A value that represents a counter.",
|
|
|
|
ConstLabels: prometheus.Labels{
|
|
|
|
"label1": "value1",
|
|
|
|
},
|
|
|
|
})
|
|
|
|
c.Inc()
|
|
|
|
|
|
|
|
expected := `
|
2018-08-23 00:54:26 +03:00
|
|
|
|
2019-01-28 01:18:44 +03:00
|
|
|
some_total{ label1 = "value1" } 1
|
2018-07-03 14:25:00 +03:00
|
|
|
`
|
|
|
|
|
2018-08-23 01:21:38 +03:00
|
|
|
if err := CollectAndCompare(c, strings.NewReader(metadata+expected), "some_total"); err != nil {
|
2018-07-03 14:25:00 +03:00
|
|
|
t.Errorf("unexpected collecting result:\n%s", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-13 17:48:48 +03:00
|
|
|
func TestCollectAndCompareNoLabel(t *testing.T) {
|
|
|
|
const metadata = `
|
|
|
|
# HELP some_total A value that represents a counter.
|
|
|
|
# TYPE some_total counter
|
|
|
|
`
|
|
|
|
|
|
|
|
c := prometheus.NewCounter(prometheus.CounterOpts{
|
|
|
|
Name: "some_total",
|
|
|
|
Help: "A value that represents a counter.",
|
|
|
|
})
|
|
|
|
c.Inc()
|
|
|
|
|
|
|
|
expected := `
|
|
|
|
|
2019-01-28 01:18:44 +03:00
|
|
|
some_total 1
|
2018-11-13 17:48:48 +03:00
|
|
|
`
|
|
|
|
|
|
|
|
if err := CollectAndCompare(c, strings.NewReader(metadata+expected), "some_total"); err != nil {
|
|
|
|
t.Errorf("unexpected collecting result:\n%s", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-20 12:57:30 +03:00
|
|
|
func TestCollectAndCompareNoHelp(t *testing.T) {
|
|
|
|
const metadata = `
|
|
|
|
# TYPE some_total counter
|
|
|
|
`
|
|
|
|
|
|
|
|
c := prometheus.NewCounter(prometheus.CounterOpts{
|
|
|
|
Name: "some_total",
|
|
|
|
})
|
|
|
|
c.Inc()
|
|
|
|
|
|
|
|
expected := `
|
|
|
|
|
|
|
|
some_total 1
|
|
|
|
`
|
|
|
|
|
|
|
|
if err := CollectAndCompare(c, strings.NewReader(metadata+expected), "some_total"); err != nil {
|
|
|
|
t.Errorf("unexpected collecting result:\n%s", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-14 14:33:19 +03:00
|
|
|
func TestCollectAndCompareHistogram(t *testing.T) {
|
|
|
|
inputs := []struct {
|
|
|
|
name string
|
|
|
|
c prometheus.Collector
|
|
|
|
metadata string
|
|
|
|
expect string
|
|
|
|
observation float64
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "Testing Histogram Collector",
|
|
|
|
c: prometheus.NewHistogram(prometheus.HistogramOpts{
|
|
|
|
Name: "some_histogram",
|
|
|
|
Help: "An example of a histogram",
|
|
|
|
Buckets: []float64{1, 2, 3},
|
|
|
|
}),
|
|
|
|
metadata: `
|
|
|
|
# HELP some_histogram An example of a histogram
|
|
|
|
# TYPE some_histogram histogram
|
|
|
|
`,
|
|
|
|
expect: `
|
2019-01-28 01:18:44 +03:00
|
|
|
some_histogram{le="1"} 0
|
|
|
|
some_histogram{le="2"} 0
|
|
|
|
some_histogram{le="3"} 1
|
|
|
|
some_histogram_bucket{le="+Inf"} 1
|
2018-11-14 14:33:19 +03:00
|
|
|
some_histogram_sum 2.5
|
|
|
|
some_histogram_count 1
|
|
|
|
|
|
|
|
`,
|
|
|
|
observation: 2.5,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "Testing HistogramVec Collector",
|
|
|
|
c: prometheus.NewHistogramVec(prometheus.HistogramOpts{
|
|
|
|
Name: "some_histogram",
|
|
|
|
Help: "An example of a histogram",
|
|
|
|
Buckets: []float64{1, 2, 3},
|
|
|
|
}, []string{"test"}),
|
|
|
|
|
|
|
|
metadata: `
|
|
|
|
# HELP some_histogram An example of a histogram
|
|
|
|
# TYPE some_histogram histogram
|
|
|
|
`,
|
|
|
|
expect: `
|
2019-01-28 01:18:44 +03:00
|
|
|
some_histogram_bucket{test="test",le="1"} 0
|
|
|
|
some_histogram_bucket{test="test",le="2"} 0
|
|
|
|
some_histogram_bucket{test="test",le="3"} 1
|
|
|
|
some_histogram_bucket{test="test",le="+Inf"} 1
|
2018-11-14 14:33:19 +03:00
|
|
|
some_histogram_sum{test="test"} 2.5
|
2019-01-28 01:18:44 +03:00
|
|
|
some_histogram_count{test="test"} 1
|
2018-11-14 14:33:19 +03:00
|
|
|
|
|
|
|
`,
|
|
|
|
observation: 2.5,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, input := range inputs {
|
|
|
|
switch collector := input.c.(type) {
|
|
|
|
case prometheus.Histogram:
|
|
|
|
collector.Observe(input.observation)
|
|
|
|
case *prometheus.HistogramVec:
|
|
|
|
collector.WithLabelValues("test").Observe(input.observation)
|
|
|
|
default:
|
2023-09-21 14:31:08 +03:00
|
|
|
t.Fatalf("unsupported collector tested")
|
2018-11-14 14:33:19 +03:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
t.Run(input.name, func(t *testing.T) {
|
|
|
|
if err := CollectAndCompare(input.c, strings.NewReader(input.metadata+input.expect)); err != nil {
|
|
|
|
t.Errorf("unexpected collecting result:\n%s", err)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-03 14:25:00 +03:00
|
|
|
func TestNoMetricFilter(t *testing.T) {
|
|
|
|
const metadata = `
|
|
|
|
# HELP some_total A value that represents a counter.
|
|
|
|
# TYPE some_total counter
|
|
|
|
`
|
|
|
|
|
|
|
|
c := prometheus.NewCounter(prometheus.CounterOpts{
|
|
|
|
Name: "some_total",
|
|
|
|
Help: "A value that represents a counter.",
|
|
|
|
ConstLabels: prometheus.Labels{
|
|
|
|
"label1": "value1",
|
|
|
|
},
|
|
|
|
})
|
|
|
|
c.Inc()
|
|
|
|
|
|
|
|
expected := `
|
2019-01-28 01:18:44 +03:00
|
|
|
some_total{label1="value1"} 1
|
2018-07-03 14:25:00 +03:00
|
|
|
`
|
|
|
|
|
2018-08-23 01:21:38 +03:00
|
|
|
if err := CollectAndCompare(c, strings.NewReader(metadata+expected)); err != nil {
|
2018-07-03 14:25:00 +03:00
|
|
|
t.Errorf("unexpected collecting result:\n%s", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestMetricNotFound(t *testing.T) {
|
|
|
|
const metadata = `
|
|
|
|
# HELP some_other_metric A value that represents a counter.
|
|
|
|
# TYPE some_other_metric counter
|
|
|
|
`
|
|
|
|
|
|
|
|
c := prometheus.NewCounter(prometheus.CounterOpts{
|
|
|
|
Name: "some_total",
|
|
|
|
Help: "A value that represents a counter.",
|
|
|
|
ConstLabels: prometheus.Labels{
|
|
|
|
"label1": "value1",
|
|
|
|
},
|
|
|
|
})
|
|
|
|
c.Inc()
|
|
|
|
|
|
|
|
expected := `
|
2019-01-28 01:18:44 +03:00
|
|
|
some_other_metric{label1="value1"} 1
|
2018-07-03 14:25:00 +03:00
|
|
|
`
|
2024-04-12 12:28:01 +03:00
|
|
|
/*
|
|
|
|
expectedError := `
|
|
|
|
|
|
|
|
Diff:
|
|
|
|
--- metric output does not match expectation; want
|
|
|
|
+++ got:
|
|
|
|
@@ -1,4 +1,4 @@
|
|
|
|
-(bytes.Buffer) # HELP some_other_metric A value that represents a counter.
|
|
|
|
-# TYPE some_other_metric counter
|
|
|
|
-some_other_metric{label1="value1"} 1
|
|
|
|
+(bytes.Buffer) # HELP some_total A value that represents a counter.
|
|
|
|
+# TYPE some_total counter
|
|
|
|
+some_total{label1="value1"} 1
|
|
|
|
|
|
|
|
`
|
|
|
|
*/
|
2018-07-03 14:25:00 +03:00
|
|
|
|
|
|
|
expectedError := `
|
|
|
|
|
2022-04-13 20:38:05 +03:00
|
|
|
Diff:
|
2024-04-12 12:28:01 +03:00
|
|
|
(
|
|
|
|
"""
|
|
|
|
- # HELP some_total A value that represents a counter.
|
|
|
|
- # TYPE some_total counter
|
|
|
|
- some_total{label1="value1"} 1
|
|
|
|
+ # HELP some_other_metric A value that represents a counter.
|
|
|
|
+ # TYPE some_other_metric counter
|
|
|
|
+ some_other_metric{label1="value1"} 1
|
|
|
|
"""
|
|
|
|
)
|
2018-07-03 14:25:00 +03:00
|
|
|
`
|
|
|
|
|
2018-08-23 01:21:38 +03:00
|
|
|
err := CollectAndCompare(c, strings.NewReader(metadata+expected))
|
2018-07-03 14:25:00 +03:00
|
|
|
if err == nil {
|
|
|
|
t.Error("Expected error, got no error.")
|
|
|
|
}
|
|
|
|
|
|
|
|
if err.Error() != expectedError {
|
2019-01-06 02:22:03 +03:00
|
|
|
t.Errorf("Expected\n%#+v\nGot:\n%#+v", expectedError, err.Error())
|
2018-07-03 14:25:00 +03:00
|
|
|
}
|
|
|
|
}
|
2020-05-14 01:17:45 +03:00
|
|
|
|
2022-08-05 17:27:47 +03:00
|
|
|
func TestScrapeAndCompare(t *testing.T) {
|
|
|
|
const expected = `
|
|
|
|
# HELP some_total A value that represents a counter.
|
|
|
|
# TYPE some_total counter
|
|
|
|
|
|
|
|
some_total{ label1 = "value1" } 1
|
|
|
|
`
|
|
|
|
|
|
|
|
expectedReader := strings.NewReader(expected)
|
|
|
|
|
|
|
|
ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
fmt.Fprintln(w, expected)
|
|
|
|
}))
|
|
|
|
defer ts.Close()
|
|
|
|
|
|
|
|
if err := ScrapeAndCompare(ts.URL, expectedReader, "some_total"); err != nil {
|
|
|
|
t.Errorf("unexpected scraping result:\n%s", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-22 13:45:45 +03:00
|
|
|
func TestScrapeAndCompareWithMultipleExpected(t *testing.T) {
|
|
|
|
const expected = `
|
|
|
|
# HELP some_total A value that represents a counter.
|
|
|
|
# TYPE some_total counter
|
|
|
|
|
|
|
|
some_total{ label1 = "value1" } 1
|
|
|
|
|
|
|
|
# HELP some_total2 A value that represents a counter.
|
|
|
|
# TYPE some_total2 counter
|
|
|
|
|
|
|
|
some_total2{ label2 = "value2" } 1
|
|
|
|
`
|
|
|
|
|
|
|
|
expectedReader := strings.NewReader(expected)
|
|
|
|
|
|
|
|
ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
fmt.Fprintln(w, expected)
|
|
|
|
}))
|
|
|
|
defer ts.Close()
|
|
|
|
|
|
|
|
if err := ScrapeAndCompare(ts.URL, expectedReader, "some_total2"); err != nil {
|
|
|
|
t.Errorf("unexpected scraping result:\n%s", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-05 17:27:47 +03:00
|
|
|
func TestScrapeAndCompareFetchingFail(t *testing.T) {
|
|
|
|
err := ScrapeAndCompare("some_url", strings.NewReader("some expectation"), "some_total")
|
|
|
|
if err == nil {
|
|
|
|
t.Errorf("expected an error but got nil")
|
|
|
|
}
|
|
|
|
if !strings.HasPrefix(err.Error(), "scraping metrics failed") {
|
|
|
|
t.Errorf("unexpected error happened: %s", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestScrapeAndCompareBadStatusCode(t *testing.T) {
|
|
|
|
const expected = `
|
|
|
|
# HELP some_total A value that represents a counter.
|
|
|
|
# TYPE some_total counter
|
|
|
|
|
|
|
|
some_total{ label1 = "value1" } 1
|
|
|
|
`
|
|
|
|
|
|
|
|
expectedReader := strings.NewReader(expected)
|
|
|
|
|
|
|
|
ts := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
w.WriteHeader(http.StatusBadGateway)
|
|
|
|
fmt.Fprintln(w, expected)
|
|
|
|
}))
|
|
|
|
defer ts.Close()
|
|
|
|
|
|
|
|
err := ScrapeAndCompare(ts.URL, expectedReader, "some_total")
|
|
|
|
if err == nil {
|
|
|
|
t.Errorf("expected an error but got nil")
|
|
|
|
}
|
|
|
|
if !strings.HasPrefix(err.Error(), "the scraping target returned a status code other than 200") {
|
|
|
|
t.Errorf("unexpected error happened: %s", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-14 01:17:45 +03:00
|
|
|
func TestCollectAndCount(t *testing.T) {
|
|
|
|
c := prometheus.NewCounterVec(
|
|
|
|
prometheus.CounterOpts{
|
|
|
|
Name: "some_total",
|
|
|
|
Help: "A value that represents a counter.",
|
|
|
|
},
|
|
|
|
[]string{"foo"},
|
|
|
|
)
|
|
|
|
if got, want := CollectAndCount(c), 0; got != want {
|
|
|
|
t.Errorf("unexpected metric count, got %d, want %d", got, want)
|
|
|
|
}
|
|
|
|
c.WithLabelValues("bar")
|
|
|
|
if got, want := CollectAndCount(c), 1; got != want {
|
|
|
|
t.Errorf("unexpected metric count, got %d, want %d", got, want)
|
|
|
|
}
|
|
|
|
c.WithLabelValues("baz")
|
|
|
|
if got, want := CollectAndCount(c), 2; got != want {
|
|
|
|
t.Errorf("unexpected metric count, got %d, want %d", got, want)
|
|
|
|
}
|
|
|
|
if got, want := CollectAndCount(c, "some_total"), 2; got != want {
|
|
|
|
t.Errorf("unexpected metric count, got %d, want %d", got, want)
|
|
|
|
}
|
|
|
|
if got, want := CollectAndCount(c, "some_other_total"), 0; got != want {
|
|
|
|
t.Errorf("unexpected metric count, got %d, want %d", got, want)
|
|
|
|
}
|
|
|
|
}
|