mirror of
https://github.com/bitechdev/ResolveSpec.git
synced 2026-01-01 01:34:25 +00:00
Some checks failed
Build , Vet Test, and Lint / Run Vet Tests (1.24.x) (push) Successful in -25m44s
Build , Vet Test, and Lint / Run Vet Tests (1.23.x) (push) Successful in -25m19s
Build , Vet Test, and Lint / Lint Code (push) Successful in -25m18s
Build , Vet Test, and Lint / Build (push) Successful in -25m30s
Tests / Unit Tests (push) Failing after -25m51s
Tests / Integration Tests (push) Failing after -25m53s
253 lines
7.4 KiB
Go
253 lines
7.4 KiB
Go
package metrics
|
|
|
|
import (
|
|
"net/http"
|
|
"strconv"
|
|
"time"
|
|
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
"github.com/prometheus/client_golang/prometheus/promauto"
|
|
"github.com/prometheus/client_golang/prometheus/promhttp"
|
|
)
|
|
|
|
// PrometheusProvider implements the Provider interface using Prometheus
|
|
type PrometheusProvider struct {
|
|
requestDuration *prometheus.HistogramVec
|
|
requestTotal *prometheus.CounterVec
|
|
requestsInFlight prometheus.Gauge
|
|
dbQueryDuration *prometheus.HistogramVec
|
|
dbQueryTotal *prometheus.CounterVec
|
|
cacheHits *prometheus.CounterVec
|
|
cacheMisses *prometheus.CounterVec
|
|
cacheSize *prometheus.GaugeVec
|
|
eventPublished *prometheus.CounterVec
|
|
eventProcessed *prometheus.CounterVec
|
|
eventDuration *prometheus.HistogramVec
|
|
eventQueueSize prometheus.Gauge
|
|
panicsTotal *prometheus.CounterVec
|
|
}
|
|
|
|
// NewPrometheusProvider creates a new Prometheus metrics provider
|
|
// If cfg is nil, default configuration will be used
|
|
func NewPrometheusProvider(cfg *Config) *PrometheusProvider {
|
|
// Use default config if none provided
|
|
if cfg == nil {
|
|
cfg = DefaultConfig()
|
|
} else {
|
|
// Apply defaults for any missing values
|
|
cfg.ApplyDefaults()
|
|
}
|
|
|
|
// Helper to add namespace prefix if configured
|
|
metricName := func(name string) string {
|
|
if cfg.Namespace != "" {
|
|
return cfg.Namespace + "_" + name
|
|
}
|
|
return name
|
|
}
|
|
|
|
return &PrometheusProvider{
|
|
requestDuration: promauto.NewHistogramVec(
|
|
prometheus.HistogramOpts{
|
|
Name: metricName("http_request_duration_seconds"),
|
|
Help: "HTTP request duration in seconds",
|
|
Buckets: cfg.HTTPRequestBuckets,
|
|
},
|
|
[]string{"method", "path", "status"},
|
|
),
|
|
requestTotal: promauto.NewCounterVec(
|
|
prometheus.CounterOpts{
|
|
Name: metricName("http_requests_total"),
|
|
Help: "Total number of HTTP requests",
|
|
},
|
|
[]string{"method", "path", "status"},
|
|
),
|
|
|
|
requestsInFlight: promauto.NewGauge(
|
|
prometheus.GaugeOpts{
|
|
Name: metricName("http_requests_in_flight"),
|
|
Help: "Current number of HTTP requests being processed",
|
|
},
|
|
),
|
|
dbQueryDuration: promauto.NewHistogramVec(
|
|
prometheus.HistogramOpts{
|
|
Name: metricName("db_query_duration_seconds"),
|
|
Help: "Database query duration in seconds",
|
|
Buckets: cfg.DBQueryBuckets,
|
|
},
|
|
[]string{"operation", "table"},
|
|
),
|
|
dbQueryTotal: promauto.NewCounterVec(
|
|
prometheus.CounterOpts{
|
|
Name: metricName("db_queries_total"),
|
|
Help: "Total number of database queries",
|
|
},
|
|
[]string{"operation", "table", "status"},
|
|
),
|
|
cacheHits: promauto.NewCounterVec(
|
|
prometheus.CounterOpts{
|
|
Name: metricName("cache_hits_total"),
|
|
Help: "Total number of cache hits",
|
|
},
|
|
[]string{"provider"},
|
|
),
|
|
cacheMisses: promauto.NewCounterVec(
|
|
prometheus.CounterOpts{
|
|
Name: metricName("cache_misses_total"),
|
|
Help: "Total number of cache misses",
|
|
},
|
|
[]string{"provider"},
|
|
),
|
|
cacheSize: promauto.NewGaugeVec(
|
|
prometheus.GaugeOpts{
|
|
Name: metricName("cache_size_items"),
|
|
Help: "Number of items in cache",
|
|
},
|
|
[]string{"provider"},
|
|
),
|
|
eventPublished: promauto.NewCounterVec(
|
|
prometheus.CounterOpts{
|
|
Name: metricName("events_published_total"),
|
|
Help: "Total number of events published",
|
|
},
|
|
[]string{"source", "event_type"},
|
|
),
|
|
eventProcessed: promauto.NewCounterVec(
|
|
prometheus.CounterOpts{
|
|
Name: metricName("events_processed_total"),
|
|
Help: "Total number of events processed",
|
|
},
|
|
[]string{"source", "event_type", "status"},
|
|
),
|
|
eventDuration: promauto.NewHistogramVec(
|
|
prometheus.HistogramOpts{
|
|
Name: metricName("event_processing_duration_seconds"),
|
|
Help: "Event processing duration in seconds",
|
|
Buckets: cfg.DBQueryBuckets, // Events are typically fast like DB queries
|
|
},
|
|
[]string{"source", "event_type"},
|
|
),
|
|
eventQueueSize: promauto.NewGauge(
|
|
prometheus.GaugeOpts{
|
|
Name: metricName("event_queue_size"),
|
|
Help: "Current number of events in queue",
|
|
},
|
|
),
|
|
panicsTotal: promauto.NewCounterVec(
|
|
prometheus.CounterOpts{
|
|
Name: metricName("panics_total"),
|
|
Help: "Total number of panics",
|
|
},
|
|
[]string{"method"},
|
|
),
|
|
}
|
|
}
|
|
|
|
// ResponseWriter wraps http.ResponseWriter to capture status code
|
|
type ResponseWriter struct {
|
|
http.ResponseWriter
|
|
statusCode int
|
|
}
|
|
|
|
func NewResponseWriter(w http.ResponseWriter) *ResponseWriter {
|
|
return &ResponseWriter{
|
|
ResponseWriter: w,
|
|
statusCode: http.StatusOK,
|
|
}
|
|
}
|
|
|
|
func (rw *ResponseWriter) WriteHeader(code int) {
|
|
rw.statusCode = code
|
|
rw.ResponseWriter.WriteHeader(code)
|
|
}
|
|
|
|
// RecordHTTPRequest implements Provider interface
|
|
func (p *PrometheusProvider) RecordHTTPRequest(method, path, status string, duration time.Duration) {
|
|
p.requestDuration.WithLabelValues(method, path, status).Observe(duration.Seconds())
|
|
p.requestTotal.WithLabelValues(method, path, status).Inc()
|
|
}
|
|
|
|
// IncRequestsInFlight implements Provider interface
|
|
func (p *PrometheusProvider) IncRequestsInFlight() {
|
|
p.requestsInFlight.Inc()
|
|
}
|
|
|
|
// DecRequestsInFlight implements Provider interface
|
|
func (p *PrometheusProvider) DecRequestsInFlight() {
|
|
p.requestsInFlight.Dec()
|
|
}
|
|
|
|
// RecordDBQuery implements Provider interface
|
|
func (p *PrometheusProvider) RecordDBQuery(operation, table string, duration time.Duration, err error) {
|
|
status := "success"
|
|
if err != nil {
|
|
status = "error"
|
|
}
|
|
p.dbQueryDuration.WithLabelValues(operation, table).Observe(duration.Seconds())
|
|
p.dbQueryTotal.WithLabelValues(operation, table, status).Inc()
|
|
}
|
|
|
|
// RecordCacheHit implements Provider interface
|
|
func (p *PrometheusProvider) RecordCacheHit(provider string) {
|
|
p.cacheHits.WithLabelValues(provider).Inc()
|
|
}
|
|
|
|
// RecordCacheMiss implements Provider interface
|
|
func (p *PrometheusProvider) RecordCacheMiss(provider string) {
|
|
p.cacheMisses.WithLabelValues(provider).Inc()
|
|
}
|
|
|
|
// UpdateCacheSize implements Provider interface
|
|
func (p *PrometheusProvider) UpdateCacheSize(provider string, size int64) {
|
|
p.cacheSize.WithLabelValues(provider).Set(float64(size))
|
|
}
|
|
|
|
// RecordEventPublished implements Provider interface
|
|
func (p *PrometheusProvider) RecordEventPublished(source, eventType string) {
|
|
p.eventPublished.WithLabelValues(source, eventType).Inc()
|
|
}
|
|
|
|
// RecordEventProcessed implements Provider interface
|
|
func (p *PrometheusProvider) RecordEventProcessed(source, eventType, status string, duration time.Duration) {
|
|
p.eventProcessed.WithLabelValues(source, eventType, status).Inc()
|
|
p.eventDuration.WithLabelValues(source, eventType).Observe(duration.Seconds())
|
|
}
|
|
|
|
// UpdateEventQueueSize implements Provider interface
|
|
func (p *PrometheusProvider) UpdateEventQueueSize(size int64) {
|
|
p.eventQueueSize.Set(float64(size))
|
|
}
|
|
|
|
// RecordPanic implements the Provider interface
|
|
func (p *PrometheusProvider) RecordPanic(methodName string) {
|
|
p.panicsTotal.WithLabelValues(methodName).Inc()
|
|
}
|
|
|
|
// Handler implements Provider interface
|
|
func (p *PrometheusProvider) Handler() http.Handler {
|
|
return promhttp.Handler()
|
|
}
|
|
|
|
// Middleware returns an HTTP middleware that collects metrics
|
|
func (p *PrometheusProvider) Middleware(next http.Handler) http.Handler {
|
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
start := time.Now()
|
|
|
|
// Increment in-flight requests
|
|
p.IncRequestsInFlight()
|
|
defer p.DecRequestsInFlight()
|
|
|
|
// Wrap response writer to capture status code
|
|
rw := NewResponseWriter(w)
|
|
|
|
// Call next handler
|
|
next.ServeHTTP(rw, r)
|
|
|
|
// Record metrics
|
|
duration := time.Since(start)
|
|
status := strconv.Itoa(rw.statusCode)
|
|
|
|
p.RecordHTTPRequest(r.Method, r.URL.Path, status, duration)
|
|
})
|
|
}
|