mirror of
https://github.com/zitadel/zitadel.git
synced 2025-01-12 00:43:40 +00:00
129 lines
3.1 KiB
Go
129 lines
3.1 KiB
Go
|
package metrics
|
||
|
|
||
|
import (
|
||
|
"net/http"
|
||
|
"strings"
|
||
|
)
|
||
|
|
||
|
const (
|
||
|
RequestCounter = "http.server.request_count"
|
||
|
RequestCountDescription = "Request counter"
|
||
|
TotalRequestCounter = "http.server.total_request_count"
|
||
|
TotalRequestDescription = "Total return code counter"
|
||
|
ReturnCodeCounter = "http.server.return_code_counter"
|
||
|
ReturnCodeCounterDescription = "Return code counter"
|
||
|
Method = "method"
|
||
|
URI = "uri"
|
||
|
ReturnCode = "return_code"
|
||
|
)
|
||
|
|
||
|
type Handler struct {
|
||
|
handler http.Handler
|
||
|
methods []MetricType
|
||
|
filters []Filter
|
||
|
}
|
||
|
|
||
|
type MetricType int32
|
||
|
|
||
|
const (
|
||
|
MetricTypeTotalCount MetricType = iota
|
||
|
MetricTypeStatusCode
|
||
|
MetricTypeRequestCount
|
||
|
)
|
||
|
|
||
|
type StatusRecorder struct {
|
||
|
http.ResponseWriter
|
||
|
Status int
|
||
|
}
|
||
|
|
||
|
func (r *StatusRecorder) WriteHeader(status int) {
|
||
|
r.Status = status
|
||
|
r.ResponseWriter.WriteHeader(status)
|
||
|
}
|
||
|
|
||
|
type Filter func(*http.Request) bool
|
||
|
|
||
|
func NewMetricsHandler(handler http.Handler, metricMethods []MetricType, ignoredEndpoints ...string) http.Handler {
|
||
|
h := Handler{
|
||
|
handler: handler,
|
||
|
methods: metricMethods,
|
||
|
}
|
||
|
if len(ignoredEndpoints) > 0 {
|
||
|
h.filters = append(h.filters, shouldNotIgnore(ignoredEndpoints...))
|
||
|
}
|
||
|
return &h
|
||
|
}
|
||
|
|
||
|
// ServeHTTP serves HTTP requests (http.Handler)
|
||
|
func (h *Handler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
||
|
if len(h.methods) == 0 {
|
||
|
h.handler.ServeHTTP(w, r)
|
||
|
return
|
||
|
}
|
||
|
for _, f := range h.filters {
|
||
|
if !f(r) {
|
||
|
// Simply pass through to the handler if a filter rejects the request
|
||
|
h.handler.ServeHTTP(w, r)
|
||
|
return
|
||
|
}
|
||
|
}
|
||
|
recorder := &StatusRecorder{
|
||
|
ResponseWriter: w,
|
||
|
Status: 200,
|
||
|
}
|
||
|
h.handler.ServeHTTP(recorder, r)
|
||
|
if h.containsMetricsMethod(MetricTypeRequestCount) {
|
||
|
RegisterRequestCounter(r)
|
||
|
}
|
||
|
if h.containsMetricsMethod(MetricTypeTotalCount) {
|
||
|
RegisterTotalRequestCounter(r)
|
||
|
}
|
||
|
if h.containsMetricsMethod(MetricTypeStatusCode) {
|
||
|
RegisterRequestCodeCounter(recorder, r)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (h *Handler) containsMetricsMethod(method MetricType) bool {
|
||
|
for _, m := range h.methods {
|
||
|
if m == method {
|
||
|
return true
|
||
|
}
|
||
|
}
|
||
|
return false
|
||
|
}
|
||
|
|
||
|
func RegisterRequestCounter(r *http.Request) {
|
||
|
var labels = map[string]interface{}{
|
||
|
URI: strings.Split(r.RequestURI, "?")[0],
|
||
|
Method: r.Method,
|
||
|
}
|
||
|
RegisterCounter(RequestCounter, RequestCountDescription)
|
||
|
AddCount(r.Context(), RequestCounter, 1, labels)
|
||
|
}
|
||
|
|
||
|
func RegisterTotalRequestCounter(r *http.Request) {
|
||
|
RegisterCounter(TotalRequestCounter, TotalRequestDescription)
|
||
|
AddCount(r.Context(), TotalRequestCounter, 1, nil)
|
||
|
}
|
||
|
|
||
|
func RegisterRequestCodeCounter(recorder *StatusRecorder, r *http.Request) {
|
||
|
var labels = map[string]interface{}{
|
||
|
URI: strings.Split(r.RequestURI, "?")[0],
|
||
|
Method: r.Method,
|
||
|
ReturnCode: recorder.Status,
|
||
|
}
|
||
|
RegisterCounter(ReturnCodeCounter, ReturnCodeCounterDescription)
|
||
|
AddCount(r.Context(), ReturnCodeCounter, 1, labels)
|
||
|
}
|
||
|
|
||
|
func shouldNotIgnore(endpoints ...string) func(r *http.Request) bool {
|
||
|
return func(r *http.Request) bool {
|
||
|
for _, endpoint := range endpoints {
|
||
|
if strings.HasPrefix(r.URL.RequestURI(), endpoint) {
|
||
|
return false
|
||
|
}
|
||
|
}
|
||
|
return true
|
||
|
}
|
||
|
}
|