0
0
mirror of https://github.com/go-gitea/gitea.git synced 2025-07-19 06:20:48 +02:00

feat(metrics): add http metrics

This commit is contained in:
TheFox0x7 2025-01-16 21:02:50 +01:00
parent 1dad933ec5
commit 3e06c39d2f
2 changed files with 51 additions and 13 deletions

View File

@ -11,10 +11,15 @@ import (
"code.gitea.io/gitea/modules/setting" "code.gitea.io/gitea/modules/setting"
"github.com/prometheus/client_golang/prometheus" "github.com/prometheus/client_golang/prometheus"
"github.com/prometheus/common/model"
) )
const namespace = "gitea_" const namespace = "gitea_"
func init() {
model.NameValidationScheme = model.UTF8Validation
}
// Collector implements the prometheus.Collector interface and // Collector implements the prometheus.Collector interface and
// exposes gitea metrics for prometheus // exposes gitea metrics for prometheus
type Collector struct { type Collector struct {

View File

@ -6,8 +6,8 @@ package common
import ( import (
"fmt" "fmt"
"net/http" "net/http"
"strconv"
"strings" "strings"
"time"
"code.gitea.io/gitea/modules/cache" "code.gitea.io/gitea/modules/cache"
"code.gitea.io/gitea/modules/gtprof" "code.gitea.io/gitea/modules/gtprof"
@ -19,17 +19,42 @@ import (
"gitea.com/go-chi/session" "gitea.com/go-chi/session"
"github.com/chi-middleware/proxy" "github.com/chi-middleware/proxy"
"github.com/felixge/httpsnoop"
"github.com/go-chi/chi/v5" "github.com/go-chi/chi/v5"
"github.com/prometheus/client_golang/prometheus" "github.com/prometheus/client_golang/prometheus"
"github.com/prometheus/client_golang/prometheus/promauto" "github.com/prometheus/client_golang/prometheus/promauto"
) )
var responseLatency = promauto.NewHistogramVec(prometheus.HistogramOpts{ var (
// reqInflightGauge tracks the amount of currently handled requests
reqInflightGauge = promauto.NewGaugeVec(prometheus.GaugeOpts{
Namespace: "http", Namespace: "http",
Subsystem: "request", Subsystem: "server",
Name: "duration_seconds", Name: "active_requests",
Help: "Gitea response time", Help: "Number of active HTTP server requests.",
}, []string{"route"}) }, []string{"http.request.method"})
// reqDurationHistogram tracks the time taken by http request
reqDurationHistogram = promauto.NewHistogramVec(prometheus.HistogramOpts{
Namespace: "http",
Subsystem: "server",
Name: "request_duration",
Help: "Measures the latency of HTTP requests processed by the server",
}, []string{"http.request.method", "http.response.status_code", "http.route"})
// reqSizeHistogram tracks the size of request
reqSizeHistogram = promauto.NewHistogramVec(prometheus.HistogramOpts{
Namespace: "http",
Subsystem: "server_request",
Name: "body_size",
Help: "Size of HTTP server request bodies.",
}, []string{"http.request.method", "http.response.status_code", "http.route"})
// respSizeHistogram tracks the size of the response
respSizeHistogram = promauto.NewHistogramVec(prometheus.HistogramOpts{
Namespace: "http",
Subsystem: "server_response",
Name: "body_size",
Help: "Size of HTTP server response bodies.",
}, []string{"http.request.method", "http.response.status_code", "http.route"})
)
// ProtocolMiddlewares returns HTTP protocol related middlewares, and it provides a global panic recovery // ProtocolMiddlewares returns HTTP protocol related middlewares, and it provides a global panic recovery
func ProtocolMiddlewares() (handlers []any) { func ProtocolMiddlewares() (handlers []any) {
@ -120,17 +145,25 @@ func ForwardedHeadersHandler(limit int, trustedProxies []string) func(h http.Han
return proxy.ForwardedHeaders(opt) return proxy.ForwardedHeaders(opt)
} }
// RouteMetrics instruments http requests and responses
func RouteMetrics() func(h http.Handler) http.Handler { func RouteMetrics() func(h http.Handler) http.Handler {
return func(next http.Handler) http.Handler { return func(next http.Handler) http.Handler {
return http.HandlerFunc(func(resp http.ResponseWriter, req *http.Request) { return http.HandlerFunc(func(resp http.ResponseWriter, req *http.Request) {
start := time.Now() inflight := reqInflightGauge.WithLabelValues(req.Method)
inflight.Inc()
defer inflight.Dec()
m := httpsnoop.CaptureMetrics(next, resp, req)
next.ServeHTTP(resp, req) next.ServeHTTP(resp, req)
elapsed := time.Since(start).Seconds() route := chi.RouteContext(req.Context()).RoutePattern()
route := "ui" code := strconv.Itoa(m.Code)
if strings.HasPrefix(req.URL.Path, "/api") { reqDurationHistogram.WithLabelValues(req.Method, code, route).Observe(m.Duration.Seconds())
route = "api" respSizeHistogram.WithLabelValues(req.Method, code, route).Observe(float64(m.Written))
size := req.ContentLength
if size < 0 {
size = 0
} }
responseLatency.WithLabelValues(route).Observe(elapsed) reqSizeHistogram.WithLabelValues(req.Method, code, route).Observe(float64(size))
}) })
} }
} }