Skip to content

Commit

Permalink
Aggregate gin routes for Prometheus metrics
Browse files Browse the repository at this point in the history
  • Loading branch information
haoming29 committed May 13, 2024
1 parent ee3ea0c commit bb887e1
Show file tree
Hide file tree
Showing 2 changed files with 113 additions and 7 deletions.
54 changes: 47 additions & 7 deletions web_ui/ui.go
Original file line number Diff line number Diff line change
Expand Up @@ -36,21 +36,21 @@ import (
"syscall"
"time"

"github.com/pelicanplatform/pelican/config"
"github.com/pelicanplatform/pelican/server_structs"
"github.com/pelicanplatform/pelican/server_utils"
"github.com/spf13/viper"

"github.com/gin-gonic/gin"
"github.com/gin-gonic/gin/binding"
"github.com/pelicanplatform/pelican/metrics"
"github.com/pelicanplatform/pelican/param"
"github.com/pkg/errors"
log "github.com/sirupsen/logrus"
"github.com/spf13/viper"
ginprometheus "github.com/zsais/go-gin-prometheus"
"go.uber.org/atomic"
"golang.org/x/sync/errgroup"
"golang.org/x/term"

"github.com/pelicanplatform/pelican/config"
"github.com/pelicanplatform/pelican/metrics"
"github.com/pelicanplatform/pelican/param"
"github.com/pelicanplatform/pelican/server_structs"
"github.com/pelicanplatform/pelican/server_utils"
)

var (
Expand Down Expand Up @@ -371,12 +371,52 @@ func configureCommonEndpoints(engine *gin.Engine) error {
return nil
}

// Map gin routes for Prometheus metrics to reduce metric cardinality
func mapPrometheusPath(c *gin.Context) string {
url := c.Request.URL.Path
// Frontend static resources
if strings.HasPrefix(url, "/view/_next/") {
url = "/view/_next/:resource"
return url
}
if strings.HasPrefix(url, "/api/v1.0/director/healthTest/") {
url = "/api/v1.0/director/healthTest/:testfile"
return url
}
// Only keeps two level depth for object access
if strings.HasPrefix(url, "/api/v1.0/director/object/") {
objectPath := strings.TrimPrefix(path.Clean(url), "/api/v1.0/director/object/")
twoLevels := strings.Split(objectPath, "/")
if len(twoLevels) <= 2 {
return url
} else {
aggPrefix := fmt.Sprintf("/%s/%s", twoLevels[0], twoLevels[1])
url := "/api/v1.0/director/object" + aggPrefix + "/:path"
return url
}
}
// Only keeps two level depth for object access
if strings.HasPrefix(url, "/api/v1.0/director/origin/") {
objectPath := strings.TrimPrefix(path.Clean(url), "/api/v1.0/director/origin/")
twoLevels := strings.Split(objectPath, "/")
if len(twoLevels) <= 2 {
return url
} else {
aggPrefix := fmt.Sprintf("/%s/%s", twoLevels[0], twoLevels[1])
url := "/api/v1.0/director/origin" + aggPrefix + "/:path"
return url
}
}
return url
}

// Configure metrics related endpoints, including Prometheus and /health API
func configureMetrics(engine *gin.Engine) error {
// Add authorization to /metric endpoint
engine.Use(promMetricAuthHandler)

prometheusMonitor := ginprometheus.NewPrometheus("gin")
prometheusMonitor.ReqCntURLLabelMappingFn = mapPrometheusPath
prometheusMonitor.Use(engine)

engine.GET("/api/v1.0/metrics/health", AuthHandler, AdminAuthHandler, func(ctx *gin.Context) {
Expand Down
66 changes: 66 additions & 0 deletions web_ui/ui_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -330,3 +330,69 @@ func TestHandleWebUIAuth(t *testing.T) {
assert.Equal(t, "", r.Result().Header.Get("Location"))
})
}

func TestMapPrometheusPath(t *testing.T) {
t.Run("aggregate-frontend-path", func(t *testing.T) {
c, _ := gin.CreateTestContext(httptest.NewRecorder())
req := httptest.NewRequest("GET", "/view/_next/static/123.js", nil)
c.Request = req

get := mapPrometheusPath(c)
assert.Equal(t, "/view/_next/:resource", get)
})

t.Run("aggregate-healthtest-path", func(t *testing.T) {
c, _ := gin.CreateTestContext(httptest.NewRecorder())
req := httptest.NewRequest("GET", "/api/v1.0/director/healthTest/pelican/self-test-monitoring-123-456.txt", nil)
c.Request = req

get := mapPrometheusPath(c)
assert.Equal(t, "/api/v1.0/director/healthTest/:testfile", get)
})

t.Run("aggregate-two-level-origin-redirect-path", func(t *testing.T) {
c, _ := gin.CreateTestContext(httptest.NewRecorder())
req := httptest.NewRequest("GET", "/api/v1.0/director/object/foo/bar/barz", nil)
c.Request = req

get := mapPrometheusPath(c)
assert.Equal(t, "/api/v1.0/director/object/foo/bar/:path", get)

c, _ = gin.CreateTestContext(httptest.NewRecorder())
req = httptest.NewRequest("GET", "/api/v1.0/director/object/foo/bar.txt", nil)
c.Request = req

get = mapPrometheusPath(c)
assert.Equal(t, "/api/v1.0/director/object/foo/bar.txt", get)

c, _ = gin.CreateTestContext(httptest.NewRecorder())
req = httptest.NewRequest("GET", "/api/v1.0/director/object/foo/bar/level3/level4/file.txt", nil)
c.Request = req

get = mapPrometheusPath(c)
assert.Equal(t, "/api/v1.0/director/object/foo/bar/:path", get)
})

t.Run("aggregate-two-level-object-redirect-path", func(t *testing.T) {
c, _ := gin.CreateTestContext(httptest.NewRecorder())
req := httptest.NewRequest("GET", "/api/v1.0/director/origin/foo/bar/barz", nil)
c.Request = req

get := mapPrometheusPath(c)
assert.Equal(t, "/api/v1.0/director/origin/foo/bar/:path", get)

c, _ = gin.CreateTestContext(httptest.NewRecorder())
req = httptest.NewRequest("GET", "/api/v1.0/director/origin/foo/bar.txt", nil)
c.Request = req

get = mapPrometheusPath(c)
assert.Equal(t, "/api/v1.0/director/origin/foo/bar.txt", get)

c, _ = gin.CreateTestContext(httptest.NewRecorder())
req = httptest.NewRequest("GET", "/api/v1.0/director/origin/foo/bar/level3/level4/file.txt", nil)
c.Request = req

get = mapPrometheusPath(c)
assert.Equal(t, "/api/v1.0/director/origin/foo/bar/:path", get)
})
}

0 comments on commit bb887e1

Please sign in to comment.