2020-10-18 18:16:02 +02:00
|
|
|
package metric
|
|
|
|
|
|
|
|
import (
|
2021-01-22 11:30:49 +01:00
|
|
|
"context"
|
2021-04-13 11:36:36 +02:00
|
|
|
"github.com/goharbor/harbor/src/lib/config"
|
2020-10-18 18:16:02 +02:00
|
|
|
"net/http"
|
|
|
|
"strconv"
|
2021-01-22 11:30:49 +01:00
|
|
|
"strings"
|
2020-10-18 18:16:02 +02:00
|
|
|
"time"
|
|
|
|
|
2020-11-05 08:14:53 +01:00
|
|
|
"github.com/goharbor/harbor/src/lib"
|
2020-10-18 18:16:02 +02:00
|
|
|
"github.com/goharbor/harbor/src/lib/metric"
|
|
|
|
)
|
|
|
|
|
2021-01-22 11:30:49 +01:00
|
|
|
// ContextOpIDKey ...
|
|
|
|
type contextOpIDKey struct{}
|
|
|
|
|
|
|
|
const (
|
|
|
|
// CatalogOperationID ...
|
|
|
|
CatalogOperationID = "v2_catalog"
|
|
|
|
// ListTagOperationID ...
|
|
|
|
ListTagOperationID = "v2_tags"
|
|
|
|
// ManifestOperationID ...
|
|
|
|
ManifestOperationID = "v2_manifest"
|
|
|
|
// BlobsOperationID ...
|
|
|
|
BlobsOperationID = "v2_blob"
|
|
|
|
// BlobsUploadOperationID ...
|
|
|
|
BlobsUploadOperationID = "v2_blob_upload"
|
|
|
|
// OthersOperationID ...
|
|
|
|
OthersOperationID = "v2_others"
|
|
|
|
)
|
|
|
|
|
|
|
|
// SetMetricOpID used to set operation ID for metrics
|
|
|
|
func SetMetricOpID(ctx context.Context, value string) {
|
|
|
|
if config.Metric().Enabled {
|
|
|
|
v := ctx.Value(contextOpIDKey{}).(*string)
|
|
|
|
*v = value
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func isChartMuseumURL(url string) bool {
|
|
|
|
return strings.HasPrefix(url, "/chartrepo/") || strings.HasPrefix(url, "/api/chartrepo/")
|
|
|
|
}
|
|
|
|
|
2020-10-18 18:16:02 +02:00
|
|
|
func instrumentHandler(next http.Handler) http.Handler {
|
|
|
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
2021-01-22 11:30:49 +01:00
|
|
|
metric.TotalInFlightGauge.Inc()
|
|
|
|
defer metric.TotalInFlightGauge.Dec()
|
|
|
|
now, rc, op := time.Now(), lib.NewResponseRecorder(w), ""
|
|
|
|
ctx := context.WithValue(r.Context(), contextOpIDKey{}, &op)
|
|
|
|
next.ServeHTTP(rc, r.WithContext(ctx))
|
2021-01-29 11:07:06 +01:00
|
|
|
if len(op) == 0 {
|
|
|
|
if isChartMuseumURL(r.URL.Path) {
|
|
|
|
op = "chartmuseum"
|
|
|
|
} else {
|
|
|
|
// From swagger's perspective the operation of this legacy URL is unknown
|
|
|
|
op = "unknown"
|
|
|
|
}
|
2021-01-22 11:30:49 +01:00
|
|
|
}
|
|
|
|
metric.TotalReqDurSummary.WithLabelValues(r.Method, op).Observe(time.Since(now).Seconds())
|
|
|
|
metric.TotalReqCnt.WithLabelValues(r.Method, strconv.Itoa(rc.StatusCode), op).Inc()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func transparentHandler(next http.Handler) http.Handler {
|
|
|
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
next.ServeHTTP(w, r)
|
2020-10-18 18:16:02 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// Middleware returns a middleware for handling requests
|
|
|
|
func Middleware() func(http.Handler) http.Handler {
|
|
|
|
if config.Metric().Enabled {
|
2020-12-01 10:12:19 +01:00
|
|
|
return instrumentHandler
|
2020-10-18 18:16:02 +02:00
|
|
|
}
|
2021-01-22 11:30:49 +01:00
|
|
|
return transparentHandler
|
|
|
|
}
|
|
|
|
|
|
|
|
// InjectOpIDMiddleware returns a middleware used for injecting operations ID
|
|
|
|
func InjectOpIDMiddleware(opID string) func(next http.Handler) http.Handler {
|
2020-10-18 18:16:02 +02:00
|
|
|
return func(next http.Handler) http.Handler {
|
2021-01-22 11:30:49 +01:00
|
|
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
SetMetricOpID(r.Context(), opID)
|
|
|
|
next.ServeHTTP(w, r)
|
2020-10-18 18:16:02 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|