2020-01-14 10:38:55 +00:00
|
|
|
/*
|
|
|
|
Copyright 2019 The Kubernetes 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 testutil
|
|
|
|
|
|
|
|
import (
|
2020-04-14 07:04:33 +00:00
|
|
|
"fmt"
|
2020-01-14 10:38:55 +00:00
|
|
|
"io"
|
|
|
|
|
|
|
|
"github.com/prometheus/client_golang/prometheus/testutil"
|
|
|
|
|
2020-04-14 07:04:33 +00:00
|
|
|
apimachineryversion "k8s.io/apimachinery/pkg/version"
|
2020-01-14 10:38:55 +00:00
|
|
|
"k8s.io/component-base/metrics"
|
2023-08-17 05:15:28 +00:00
|
|
|
"k8s.io/component-base/metrics/legacyregistry"
|
2020-01-14 10:38:55 +00:00
|
|
|
)
|
|
|
|
|
2024-05-15 06:54:18 +00:00
|
|
|
type TB interface {
|
|
|
|
Logf(format string, args ...any)
|
|
|
|
Errorf(format string, args ...any)
|
|
|
|
Fatalf(format string, args ...any)
|
|
|
|
}
|
|
|
|
|
2020-01-14 10:38:55 +00:00
|
|
|
// CollectAndCompare registers the provided Collector with a newly created
|
|
|
|
// pedantic Registry. It then does the same as GatherAndCompare, gathering the
|
|
|
|
// metrics from the pedantic Registry.
|
|
|
|
func CollectAndCompare(c metrics.Collector, expected io.Reader, metricNames ...string) error {
|
2020-12-17 12:28:29 +00:00
|
|
|
lintProblems, err := testutil.CollectAndLint(c, metricNames...)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := getLintError(lintProblems); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-01-14 10:38:55 +00:00
|
|
|
return testutil.CollectAndCompare(c, expected, metricNames...)
|
|
|
|
}
|
|
|
|
|
|
|
|
// GatherAndCompare gathers all metrics from the provided Gatherer and compares
|
|
|
|
// it to an expected output read from the provided Reader in the Prometheus text
|
|
|
|
// exposition format. If any metricNames are provided, only metrics with those
|
|
|
|
// names are compared.
|
|
|
|
func GatherAndCompare(g metrics.Gatherer, expected io.Reader, metricNames ...string) error {
|
2020-12-17 12:28:29 +00:00
|
|
|
lintProblems, err := testutil.GatherAndLint(g, metricNames...)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := getLintError(lintProblems); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-01-14 10:38:55 +00:00
|
|
|
return testutil.GatherAndCompare(g, expected, metricNames...)
|
|
|
|
}
|
|
|
|
|
|
|
|
// CustomCollectAndCompare registers the provided StableCollector with a newly created
|
|
|
|
// registry. It then does the same as GatherAndCompare, gathering the
|
|
|
|
// metrics from the pedantic Registry.
|
|
|
|
func CustomCollectAndCompare(c metrics.StableCollector, expected io.Reader, metricNames ...string) error {
|
|
|
|
registry := metrics.NewKubeRegistry()
|
|
|
|
registry.CustomMustRegister(c)
|
|
|
|
|
|
|
|
return GatherAndCompare(registry, expected, metricNames...)
|
|
|
|
}
|
2020-04-14 07:04:33 +00:00
|
|
|
|
2023-06-01 16:58:10 +00:00
|
|
|
// ScrapeAndCompare calls a remote exporter's endpoint which is expected to return some metrics in
|
|
|
|
// plain text format. Then it compares it with the results that the `expected` would return.
|
|
|
|
// If the `metricNames` is not empty it would filter the comparison only to the given metric names.
|
|
|
|
func ScrapeAndCompare(url string, expected io.Reader, metricNames ...string) error {
|
|
|
|
return testutil.ScrapeAndCompare(url, expected, metricNames...)
|
|
|
|
}
|
|
|
|
|
2020-04-14 07:04:33 +00:00
|
|
|
// NewFakeKubeRegistry creates a fake `KubeRegistry` that takes the input version as `build in version`.
|
|
|
|
// It should only be used in testing scenario especially for the deprecated metrics.
|
|
|
|
// The input version format should be `major.minor.patch`, e.g. '1.18.0'.
|
|
|
|
func NewFakeKubeRegistry(ver string) metrics.KubeRegistry {
|
|
|
|
backup := metrics.BuildVersion
|
|
|
|
defer func() {
|
|
|
|
metrics.BuildVersion = backup
|
|
|
|
}()
|
|
|
|
|
|
|
|
metrics.BuildVersion = func() apimachineryversion.Info {
|
|
|
|
return apimachineryversion.Info{
|
|
|
|
GitVersion: fmt.Sprintf("v%s-alpha+1.12345", ver),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return metrics.NewKubeRegistry()
|
|
|
|
}
|
2023-08-17 05:15:28 +00:00
|
|
|
|
2024-05-15 06:54:18 +00:00
|
|
|
func AssertVectorCount(t TB, name string, labelFilter map[string]string, wantCount int) {
|
2023-08-17 05:15:28 +00:00
|
|
|
metrics, err := legacyregistry.DefaultGatherer.Gather()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Failed to gather metrics: %s", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
counterSum := 0
|
|
|
|
for _, mf := range metrics {
|
|
|
|
if mf.GetName() != name {
|
|
|
|
continue // Ignore other metrics.
|
|
|
|
}
|
|
|
|
for _, metric := range mf.GetMetric() {
|
|
|
|
if !LabelsMatch(metric, labelFilter) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
counterSum += int(metric.GetCounter().GetValue())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if wantCount != counterSum {
|
|
|
|
t.Errorf("Wanted count %d, got %d for metric %s with labels %#+v", wantCount, counterSum, name, labelFilter)
|
|
|
|
for _, mf := range metrics {
|
|
|
|
if mf.GetName() == name {
|
|
|
|
for _, metric := range mf.GetMetric() {
|
|
|
|
t.Logf("\tnear match: %s", metric.String())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-05-15 06:54:18 +00:00
|
|
|
func AssertHistogramTotalCount(t TB, name string, labelFilter map[string]string, wantCount int) {
|
2023-08-17 05:15:28 +00:00
|
|
|
metrics, err := legacyregistry.DefaultGatherer.Gather()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Failed to gather metrics: %s", err)
|
|
|
|
}
|
|
|
|
counterSum := 0
|
|
|
|
for _, mf := range metrics {
|
|
|
|
if mf.GetName() != name {
|
|
|
|
continue // Ignore other metrics.
|
|
|
|
}
|
|
|
|
for _, metric := range mf.GetMetric() {
|
|
|
|
if !LabelsMatch(metric, labelFilter) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
counterSum += int(metric.GetHistogram().GetSampleCount())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if wantCount != counterSum {
|
|
|
|
t.Errorf("Wanted count %d, got %d for metric %s with labels %#+v", wantCount, counterSum, name, labelFilter)
|
|
|
|
for _, mf := range metrics {
|
|
|
|
if mf.GetName() == name {
|
|
|
|
for _, metric := range mf.GetMetric() {
|
|
|
|
t.Logf("\tnear match: %s\n", metric.String())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|