mirror of
https://github.com/tailscale/tailscale.git
synced 2024-11-30 05:25:35 +00:00
8546ff98fb
This splits Prometheus metric handlers exposed by tsweb into two modules: - `varz.Handler` exposes Prometheus metrics generated by our expvar converter; - `promvarz.Handler` combines our expvar-converted metrics and native Prometheus metrics. By default, tsweb will use the promvarz handler, however users can keep using only the expvar converter. Specifically, `tailscaled` now uses `varz.Handler` explicitly, which avoids a dependency on the (heavyweight) Prometheus client. Updates https://github.com/tailscale/corp/issues/10205 Signed-off-by: Anton Tolchanov <anton@tailscale.com>
49 lines
1.3 KiB
Go
49 lines
1.3 KiB
Go
// Copyright (c) Tailscale Inc & AUTHORS
|
|
// SPDX-License-Identifier: BSD-3-Clause
|
|
|
|
// Package promvarz combines Prometheus metrics exported by our expvar converter
|
|
// (tsweb/varz) with metrics exported by the official Prometheus client.
|
|
package promvarz
|
|
|
|
import (
|
|
"fmt"
|
|
"net/http"
|
|
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
"github.com/prometheus/common/expfmt"
|
|
"tailscale.com/tsweb/varz"
|
|
)
|
|
|
|
// Handler returns Prometheus metrics exported by our expvar converter
|
|
// and the official Prometheus client.
|
|
func Handler(w http.ResponseWriter, r *http.Request) {
|
|
if err := gatherNativePrometheusMetrics(w); err != nil {
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
w.Write([]byte(err.Error()))
|
|
return
|
|
}
|
|
varz.Handler(w, r)
|
|
}
|
|
|
|
// gatherNativePrometheusMetrics writes metrics from the default
|
|
// metric registry in text format.
|
|
func gatherNativePrometheusMetrics(w http.ResponseWriter) error {
|
|
enc := expfmt.NewEncoder(w, expfmt.FmtText)
|
|
mfs, err := prometheus.DefaultGatherer.Gather()
|
|
if err != nil {
|
|
return fmt.Errorf("could not gather metrics from DefaultGatherer: %w", err)
|
|
}
|
|
|
|
for _, mf := range mfs {
|
|
if err := enc.Encode(mf); err != nil {
|
|
return fmt.Errorf("could not encode metric %v: %w", mf, err)
|
|
}
|
|
}
|
|
if closer, ok := enc.(expfmt.Closer); ok {
|
|
if err := closer.Close(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|