mirror of
https://github.com/grafana/grafana.git
synced 2025-02-25 18:55:37 -06:00
return /metrics before session middleware
This commit is contained in:
parent
6b1ae1a8a8
commit
6d22a67a30
@ -6,7 +6,6 @@ import (
|
|||||||
"github.com/grafana/grafana/pkg/api/dtos"
|
"github.com/grafana/grafana/pkg/api/dtos"
|
||||||
"github.com/grafana/grafana/pkg/middleware"
|
"github.com/grafana/grafana/pkg/middleware"
|
||||||
m "github.com/grafana/grafana/pkg/models"
|
m "github.com/grafana/grafana/pkg/models"
|
||||||
"github.com/prometheus/client_golang/prometheus/promhttp"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// Register adds http routes
|
// Register adds http routes
|
||||||
@ -98,8 +97,6 @@ func (hs *HttpServer) registerRoutes() {
|
|||||||
// api renew session based on remember cookie
|
// api renew session based on remember cookie
|
||||||
r.Get("/api/login/ping", quota("session"), LoginApiPing)
|
r.Get("/api/login/ping", quota("session"), LoginApiPing)
|
||||||
|
|
||||||
r.Get("/metrics", promhttp.Handler())
|
|
||||||
|
|
||||||
// authed api
|
// authed api
|
||||||
r.Group("/api", func() {
|
r.Group("/api", func() {
|
||||||
|
|
||||||
@ -266,9 +263,6 @@ func (hs *HttpServer) registerRoutes() {
|
|||||||
r.Get("/tsdb/testdata/gensql", reqGrafanaAdmin, wrap(GenerateSqlTestData))
|
r.Get("/tsdb/testdata/gensql", reqGrafanaAdmin, wrap(GenerateSqlTestData))
|
||||||
r.Get("/tsdb/testdata/random-walk", wrap(GetTestDataRandomWalk))
|
r.Get("/tsdb/testdata/random-walk", wrap(GetTestDataRandomWalk))
|
||||||
|
|
||||||
// metrics
|
|
||||||
//r.Get("/metrics", wrap(GetInternalMetrics))
|
|
||||||
|
|
||||||
r.Group("/alerts", func() {
|
r.Group("/alerts", func() {
|
||||||
r.Post("/test", bind(dtos.AlertTestCommand{}), wrap(AlertTest))
|
r.Post("/test", bind(dtos.AlertTestCommand{}), wrap(AlertTest))
|
||||||
r.Post("/:alertId/pause", bind(dtos.PauseAlertCommand{}), wrap(PauseAlert), reqEditorRole)
|
r.Post("/:alertId/pause", bind(dtos.PauseAlertCommand{}), wrap(PauseAlert), reqEditorRole)
|
||||||
|
@ -11,6 +11,8 @@ import (
|
|||||||
"path"
|
"path"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/prometheus/client_golang/prometheus/promhttp"
|
||||||
|
|
||||||
gocache "github.com/patrickmn/go-cache"
|
gocache "github.com/patrickmn/go-cache"
|
||||||
macaron "gopkg.in/macaron.v1"
|
macaron "gopkg.in/macaron.v1"
|
||||||
|
|
||||||
@ -165,6 +167,7 @@ func (hs *HttpServer) newMacaron() *macaron.Macaron {
|
|||||||
}))
|
}))
|
||||||
|
|
||||||
m.Use(hs.healthHandler)
|
m.Use(hs.healthHandler)
|
||||||
|
m.Use(hs.metricsEndpoint)
|
||||||
m.Use(middleware.GetContextHandler())
|
m.Use(middleware.GetContextHandler())
|
||||||
m.Use(middleware.Sessioner(&setting.SessionOptions))
|
m.Use(middleware.Sessioner(&setting.SessionOptions))
|
||||||
m.Use(middleware.RequestMetrics())
|
m.Use(middleware.RequestMetrics())
|
||||||
@ -180,6 +183,14 @@ func (hs *HttpServer) newMacaron() *macaron.Macaron {
|
|||||||
return m
|
return m
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (hs *HttpServer) metricsEndpoint(ctx *macaron.Context) {
|
||||||
|
if ctx.Req.Method != "GET" || ctx.Req.URL.Path != "/metrics" {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
promhttp.Handler().ServeHTTP(ctx.Resp, ctx.Req.Request)
|
||||||
|
}
|
||||||
|
|
||||||
func (hs *HttpServer) healthHandler(ctx *macaron.Context) {
|
func (hs *HttpServer) healthHandler(ctx *macaron.Context) {
|
||||||
if ctx.Req.Method != "GET" || ctx.Req.URL.Path != "/api/health" {
|
if ctx.Req.Method != "GET" || ctx.Req.URL.Path != "/api/health" {
|
||||||
return
|
return
|
||||||
|
@ -206,6 +206,18 @@ func (b *Bridge) writeMetrics(w io.Writer, mfs []*dto.MetricFamily, prefix strin
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ignoreThisMetric := false
|
||||||
|
for _, v := range ignorePrefix {
|
||||||
|
if strings.HasPrefix(mf.GetName(), v) {
|
||||||
|
ignoreThisMetric = true
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ignoreThisMetric {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
buf := bufio.NewWriter(w)
|
buf := bufio.NewWriter(w)
|
||||||
for _, s := range vec {
|
for _, s := range vec {
|
||||||
if err := writeSanitized(buf, prefix); err != nil {
|
if err := writeSanitized(buf, prefix); err != nil {
|
||||||
|
@ -343,6 +343,46 @@ func TestCounter(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestCanIgnoreSomeMetrics(t *testing.T) {
|
||||||
|
cntVec := prometheus.NewCounter(
|
||||||
|
prometheus.CounterOpts{
|
||||||
|
Name: "http_request_total",
|
||||||
|
Help: "docstring",
|
||||||
|
ConstLabels: prometheus.Labels{"constname": "constvalue"},
|
||||||
|
})
|
||||||
|
|
||||||
|
reg := prometheus.NewRegistry()
|
||||||
|
reg.MustRegister(cntVec)
|
||||||
|
|
||||||
|
cntVec.Inc()
|
||||||
|
|
||||||
|
b, err := NewBridge(&Config{
|
||||||
|
URL: "localhost:8080",
|
||||||
|
Gatherer: reg,
|
||||||
|
CountersAsDelta: true,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("error creating bridge: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// first collect
|
||||||
|
mfs, err := reg.Gather()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("error: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var buf bytes.Buffer
|
||||||
|
err = b.writeMetrics(&buf, mfs, "prefix", model.Time(1477043083))
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("error: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
want := ""
|
||||||
|
if got := buf.String(); want != got {
|
||||||
|
t.Fatalf("wanted \n%s\n, got \n%s\n", want, got)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func TestPush(t *testing.T) {
|
func TestPush(t *testing.T) {
|
||||||
reg := prometheus.NewRegistry()
|
reg := prometheus.NewRegistry()
|
||||||
cntVec := prometheus.NewCounterVec(
|
cntVec := prometheus.NewCounterVec(
|
||||||
|
@ -16,10 +16,12 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
M_Instance_Start prometheus.Counter
|
M_Instance_Start prometheus.Counter
|
||||||
M_Page_Status *prometheus.CounterVec
|
M_Page_Status *prometheus.CounterVec
|
||||||
M_Api_Status *prometheus.CounterVec
|
M_Api_Status *prometheus.CounterVec
|
||||||
M_Proxy_Status *prometheus.CounterVec
|
M_Proxy_Status *prometheus.CounterVec
|
||||||
|
M_Http_Request_Total *prometheus.CounterVec
|
||||||
|
M_Http_Request_Summary *prometheus.SummaryVec
|
||||||
|
|
||||||
M_Api_User_SignUpStarted prometheus.Counter
|
M_Api_User_SignUpStarted prometheus.Counter
|
||||||
M_Api_User_SignUpCompleted prometheus.Counter
|
M_Api_User_SignUpCompleted prometheus.Counter
|
||||||
@ -84,6 +86,22 @@ func init() {
|
|||||||
[]string{"code"},
|
[]string{"code"},
|
||||||
)
|
)
|
||||||
|
|
||||||
|
M_Http_Request_Total = prometheus.NewCounterVec(
|
||||||
|
prometheus.CounterOpts{
|
||||||
|
Name: "http_request_total",
|
||||||
|
Help: "http request counter",
|
||||||
|
},
|
||||||
|
[]string{"code", "method"},
|
||||||
|
)
|
||||||
|
|
||||||
|
M_Http_Request_Summary = prometheus.NewSummaryVec(
|
||||||
|
prometheus.SummaryOpts{
|
||||||
|
Name: "http_request_duration",
|
||||||
|
Help: "http request summary",
|
||||||
|
},
|
||||||
|
[]string{"code", "method"},
|
||||||
|
)
|
||||||
|
|
||||||
M_Api_User_SignUpStarted = prometheus.NewCounter(prometheus.CounterOpts{
|
M_Api_User_SignUpStarted = prometheus.NewCounter(prometheus.CounterOpts{
|
||||||
Name: "api_user_signup_started_total",
|
Name: "api_user_signup_started_total",
|
||||||
Help: "amount of users who started the signup flow",
|
Help: "amount of users who started the signup flow",
|
||||||
@ -214,6 +232,9 @@ func initMetricVars(settings *MetricSettings) {
|
|||||||
M_Instance_Start,
|
M_Instance_Start,
|
||||||
M_Page_Status,
|
M_Page_Status,
|
||||||
M_Api_Status,
|
M_Api_Status,
|
||||||
|
M_Proxy_Status,
|
||||||
|
M_Http_Request_Total,
|
||||||
|
M_Http_Request_Summary,
|
||||||
M_Api_User_SignUpStarted,
|
M_Api_User_SignUpStarted,
|
||||||
M_Api_User_SignUpCompleted,
|
M_Api_User_SignUpCompleted,
|
||||||
M_Api_User_SignUpInvite,
|
M_Api_User_SignUpInvite,
|
||||||
|
@ -2,7 +2,9 @@ package middleware
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/grafana/grafana/pkg/metrics"
|
"github.com/grafana/grafana/pkg/metrics"
|
||||||
"gopkg.in/macaron.v1"
|
"gopkg.in/macaron.v1"
|
||||||
@ -11,10 +13,16 @@ import (
|
|||||||
func RequestMetrics() macaron.Handler {
|
func RequestMetrics() macaron.Handler {
|
||||||
return func(res http.ResponseWriter, req *http.Request, c *macaron.Context) {
|
return func(res http.ResponseWriter, req *http.Request, c *macaron.Context) {
|
||||||
rw := res.(macaron.ResponseWriter)
|
rw := res.(macaron.ResponseWriter)
|
||||||
|
now := time.Now()
|
||||||
c.Next()
|
c.Next()
|
||||||
|
|
||||||
status := rw.Status()
|
status := rw.Status()
|
||||||
|
|
||||||
|
code := sanitizeCode(status)
|
||||||
|
method := sanitizeMethod(req.Method)
|
||||||
|
metrics.M_Http_Request_Total.WithLabelValues(code, method).Inc()
|
||||||
|
metrics.M_Http_Request_Summary.WithLabelValues(code, method).Observe(time.Since(now).Seconds())
|
||||||
|
|
||||||
if strings.HasPrefix(req.RequestURI, "/api/datasources/proxy") {
|
if strings.HasPrefix(req.RequestURI, "/api/datasources/proxy") {
|
||||||
countProxyRequests(status)
|
countProxyRequests(status)
|
||||||
} else if strings.HasPrefix(req.RequestURI, "/api/") {
|
} else if strings.HasPrefix(req.RequestURI, "/api/") {
|
||||||
@ -63,3 +71,130 @@ func countProxyRequests(status int) {
|
|||||||
metrics.M_Proxy_Status.WithLabelValues("unknown").Inc()
|
metrics.M_Proxy_Status.WithLabelValues("unknown").Inc()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func sanitizeMethod(m string) string {
|
||||||
|
switch m {
|
||||||
|
case "GET", "get":
|
||||||
|
return "get"
|
||||||
|
case "PUT", "put":
|
||||||
|
return "put"
|
||||||
|
case "HEAD", "head":
|
||||||
|
return "head"
|
||||||
|
case "POST", "post":
|
||||||
|
return "post"
|
||||||
|
case "DELETE", "delete":
|
||||||
|
return "delete"
|
||||||
|
case "CONNECT", "connect":
|
||||||
|
return "connect"
|
||||||
|
case "OPTIONS", "options":
|
||||||
|
return "options"
|
||||||
|
case "NOTIFY", "notify":
|
||||||
|
return "notify"
|
||||||
|
default:
|
||||||
|
return strings.ToLower(m)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// If the wrapped http.Handler has not set a status code, i.e. the value is
|
||||||
|
// currently 0, santizeCode will return 200, for consistency with behavior in
|
||||||
|
// the stdlib.
|
||||||
|
func sanitizeCode(s int) string {
|
||||||
|
switch s {
|
||||||
|
case 100:
|
||||||
|
return "100"
|
||||||
|
case 101:
|
||||||
|
return "101"
|
||||||
|
|
||||||
|
case 200, 0:
|
||||||
|
return "200"
|
||||||
|
case 201:
|
||||||
|
return "201"
|
||||||
|
case 202:
|
||||||
|
return "202"
|
||||||
|
case 203:
|
||||||
|
return "203"
|
||||||
|
case 204:
|
||||||
|
return "204"
|
||||||
|
case 205:
|
||||||
|
return "205"
|
||||||
|
case 206:
|
||||||
|
return "206"
|
||||||
|
|
||||||
|
case 300:
|
||||||
|
return "300"
|
||||||
|
case 301:
|
||||||
|
return "301"
|
||||||
|
case 302:
|
||||||
|
return "302"
|
||||||
|
case 304:
|
||||||
|
return "304"
|
||||||
|
case 305:
|
||||||
|
return "305"
|
||||||
|
case 307:
|
||||||
|
return "307"
|
||||||
|
|
||||||
|
case 400:
|
||||||
|
return "400"
|
||||||
|
case 401:
|
||||||
|
return "401"
|
||||||
|
case 402:
|
||||||
|
return "402"
|
||||||
|
case 403:
|
||||||
|
return "403"
|
||||||
|
case 404:
|
||||||
|
return "404"
|
||||||
|
case 405:
|
||||||
|
return "405"
|
||||||
|
case 406:
|
||||||
|
return "406"
|
||||||
|
case 407:
|
||||||
|
return "407"
|
||||||
|
case 408:
|
||||||
|
return "408"
|
||||||
|
case 409:
|
||||||
|
return "409"
|
||||||
|
case 410:
|
||||||
|
return "410"
|
||||||
|
case 411:
|
||||||
|
return "411"
|
||||||
|
case 412:
|
||||||
|
return "412"
|
||||||
|
case 413:
|
||||||
|
return "413"
|
||||||
|
case 414:
|
||||||
|
return "414"
|
||||||
|
case 415:
|
||||||
|
return "415"
|
||||||
|
case 416:
|
||||||
|
return "416"
|
||||||
|
case 417:
|
||||||
|
return "417"
|
||||||
|
case 418:
|
||||||
|
return "418"
|
||||||
|
|
||||||
|
case 500:
|
||||||
|
return "500"
|
||||||
|
case 501:
|
||||||
|
return "501"
|
||||||
|
case 502:
|
||||||
|
return "502"
|
||||||
|
case 503:
|
||||||
|
return "503"
|
||||||
|
case 504:
|
||||||
|
return "504"
|
||||||
|
case 505:
|
||||||
|
return "505"
|
||||||
|
|
||||||
|
case 428:
|
||||||
|
return "428"
|
||||||
|
case 429:
|
||||||
|
return "429"
|
||||||
|
case 431:
|
||||||
|
return "431"
|
||||||
|
case 511:
|
||||||
|
return "511"
|
||||||
|
|
||||||
|
default:
|
||||||
|
return strconv.Itoa(s)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user