Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Metrics become available only on second .Set() #1682

Closed
marevers opened this issue Nov 12, 2024 · 4 comments
Closed

Metrics become available only on second .Set() #1682

marevers opened this issue Nov 12, 2024 · 4 comments
Labels

Comments

@marevers
Copy link
Contributor

marevers commented Nov 12, 2024

I am observing some strange behavior using .WithLabelValues and creating GaugeVec metrics using promauto.

Here's a shortened/abbreviated rundown of how I am generating my metrics:

// Define exporter
type exporter struct {
	Config  config
	Reg     *prometheus.Registry
	Metrics struct {
		MetricVec    *prometheus.GaugeVec
	}
}

// Create a registry
func createRegistry() *prometheus.Registry {
	reg := prometheus.NewRegistry()

	reg.MustRegister(
		collectors.NewGoCollector(),
		collectors.NewProcessCollector(collectors.ProcessCollectorOpts{}),
	)

	return reg
}

// Register a metric with the exporter
func (e *exporter) registerMetrics(labels []string) {
	e.Metrics.MetricVec = promauto.With(e.Reg).NewGaugeVec(prometheus.GaugeOpts{
		Name:      "my_metric",
		Namespace: "my_metric_prefix",
		Help:      "Help of the metric",
	}, labels)
}

// Initialise exporter and register metrics
exporter := &exporter{
	Config: cfg,
	Reg:    createRegistry(),
}

exporter.registerMetrics(labels)

On an interval I am calling the following function (again, abbreviated for readability):

func (e *exporter) calculateMetrics() {
	if someCondition {
		e.Metrics.MetricVec.WithLabelValues(labelValues...).Set(1)
	} else {
		e.Metrics.MetricVec.WithLabelValues(labelValues...).Set(0)
	}
}

Now, interestingly enough, only on the second time calculateMetrics is called, my metrics become available/visible (on promhttp.HandlerFor(exporter.Reg, promhttp.HandlerOpts{}). The very first time, the metrics are seemingly correctly set but they are not visible. This causes the exporter to be delayed in providing the metrics, e.g. if the interval is set to 5 minutes, the metrics become available after 10 minutes.

Is this a usage error, or is the behavior somehow bugged?

@dosubot dosubot bot added the bug label Nov 12, 2024
Copy link

dosubot bot commented Nov 12, 2024

Hey @marevers! I'm here to help you with any bugs, questions, or contributions. Let's tackle this together!

This behavior might be related to how metrics are initialized and registered in Prometheus when using GaugeVec with WithLabelValues. A similar issue was discussed in a closed issue on the client_golang repository, where metrics like CounterVec were not included in the Gather results unless WithLabelValues was called at least once [1].

In your case, it seems that the metrics are not being initialized or registered in the registry until the second call to Set. This could be due to the fact that the metric is only being created and registered in the registry when WithLabelValues is called for the first time, but it might not be fully visible until the second call.

To address this, you can try explicitly initializing the metric with a default value (e.g., 0) right after registering it. This ensures that the metric is present in the registry from the start. Here's a possible modification to your code:

// Register a metric with the exporter
func (e *exporter) registerMetrics(labels []string) {
	e.Metrics.MetricVec = promauto.With(e.Reg).NewGaugeVec(prometheus.GaugeOpts{
		Name:      "my_metric",
		Namespace: "my_metric_prefix",
		Help:      "Help of the metric",
	}, labels)

	// Initialize the metric with default label values
	e.Metrics.MetricVec.WithLabelValues(make([]string, len(labels))...).Set(0)
}

This approach initializes the metric with default label values, ensuring it is visible in the registry from the beginning.

To continue talking to Dosu, mention @dosu.

@marevers
Copy link
Contributor Author

This solution unfortunately doesn't work, because e.Metrics.MetricVec.WithLabelValues(make([]string, len(labels))...).Set(0) applies the labels with empty label values. As the correct label values are not yet defined - they are retrieved in the course of calculateMetrics() - it still results in the metrics appearing on the second run of the aforementioned function.

I have applied a provisional fix by just running calculateMetrics() twice before starting the interval goroutine, but I do wonder if there is a proper, idiomatic way of handling this.

@bwplotka
Copy link
Member

Thanks for reporting!

e.Metrics.MetricVec = promauto.With(e.Reg).NewGaugeVec(prometheus.GaugeOpts{
		Name:      "my_metric",
		Namespace: "my_metric_prefix",
		Help:      "Help of the metric",
	}, labels)

So just this line obv will not give you the my_metric{<your dynamic label>, hope that's clear. The first one calculateMetrics (or WithLabelValues(labelValues...) with or without set) should give you those metrics.

I wonder, maybe there's a bug in your calculateMetrics() with the labelValues you are passing? But even then, even empty string should appear.

We can create a quick unit test to verify here (help wanted) and try to repro this in isolation, but you are right, first WithLabelValues(labelValues...) (with or without Set) should immediately create a gauge.

@bwplotka
Copy link
Member

It feels there's some bug on your side e.g. maybe you check HTTP endpoint too quickly when the first Set was not yet done?

The following repro works on main. Closing for now, we can reopen if we have a repro unit test that proves a bug or so (:

package client_golang_test

import (
	"bytes"
	"testing"

	"github.com/prometheus/client_golang/prometheus"
	"github.com/prometheus/client_golang/prometheus/promauto"
	"github.com/prometheus/client_golang/prometheus/testutil"
)

// Define exporter
type exporter struct {
	Reg     *prometheus.Registry
	Metrics struct {
		MetricVec *prometheus.GaugeVec
	}
}

// Create a registry
func createRegistry() *prometheus.Registry {
	reg := prometheus.NewRegistry()
	return reg
}

// Register a metric with the exporter
func (e *exporter) registerMetrics(labels []string) {
	e.Metrics.MetricVec = promauto.With(e.Reg).NewGaugeVec(prometheus.GaugeOpts{
		Name:      "my_metric",
		Namespace: "my_metric_prefix",
		Help:      "Help of the metric",
	}, labels)
}

// Repro: https://github.com/prometheus/client_golang/issues/1682
func TestSet(t *testing.T) {
	e := &exporter{
		Reg: createRegistry(),
	}
	e.registerMetrics([]string{"label1", "label2"})

	// Expect no metric.
	if err := testutil.CollectAndCompare(e.Reg, bytes.NewReader([]byte(``)), "my_metric_prefix_my_metric"); err != nil {
		t.Fatal(err)
	}

	// Expect no metric zero.
	e.Metrics.MetricVec.WithLabelValues("1", "2").Set(0)
	if err := testutil.GatherAndCompare(e.Reg, bytes.NewReader([]byte(`
# HELP my_metric_prefix_my_metric Help of the metric
# TYPE my_metric_prefix_my_metric gauge
my_metric_prefix_my_metric{label1="1", label2="2"} 0
`)), "my_metric_prefix_my_metric"); err != nil {
		t.Fatal(err)
	}
}

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Projects
None yet
Development

No branches or pull requests

2 participants