grafana/pkg/api/frontend_logging.go
2023-08-10 10:32:12 +02:00

128 lines
4.5 KiB
Go

package api
import (
"net/http"
"time"
"golang.org/x/time/rate"
"github.com/grafana/grafana/pkg/api/frontendlogging"
"github.com/grafana/grafana/pkg/infra/log"
"github.com/grafana/grafana/pkg/web"
)
var frontendLogger = log.New("frontend")
type frontendLogMessageHandler func(hs *HTTPServer, c *web.Context)
const grafanaJavascriptAgentEndpointPath = "/log-grafana-javascript-agent"
func GrafanaJavascriptAgentLogMessageHandler(store *frontendlogging.SourceMapStore) frontendLogMessageHandler {
return func(hs *HTTPServer, c *web.Context) {
event := frontendlogging.FrontendGrafanaJavascriptAgentEvent{}
if err := web.Bind(c.Req, &event); err != nil {
c.Resp.WriteHeader(http.StatusBadRequest)
_, err = c.Resp.Write([]byte("bad request data"))
if err != nil {
hs.log.Error("could not write to response", "err", err)
}
}
// Meta object is standard across event types, adding it globally.
if event.Logs != nil && len(event.Logs) > 0 {
for _, logEntry := range event.Logs {
var ctx = frontendlogging.CtxVector{}
ctx = event.AddMetaToContext(ctx)
ctx = append(ctx, "kind", "log", "original_timestamp", logEntry.Timestamp)
for k, v := range frontendlogging.KeyValToInterfaceMap(logEntry.KeyValContext()) {
ctx = append(ctx, k, v)
}
switch logEntry.LogLevel {
case frontendlogging.LogLevelDebug, frontendlogging.LogLevelTrace:
{
ctx = append(ctx, "original_log_level", logEntry.LogLevel)
frontendLogger.Debug(logEntry.Message, ctx...)
}
case frontendlogging.LogLevelError:
{
ctx = append(ctx, "original_log_level", logEntry.LogLevel)
frontendLogger.Error(logEntry.Message, ctx...)
}
case frontendlogging.LogLevelWarning:
{
ctx = append(ctx, "original_log_level", logEntry.LogLevel)
frontendLogger.Warn(logEntry.Message, ctx...)
}
default:
{
ctx = append(ctx, "original_log_level", logEntry.LogLevel)
frontendLogger.Info(logEntry.Message, ctx...)
}
}
}
}
if event.Measurements != nil && len(event.Measurements) > 0 {
for _, measurementEntry := range event.Measurements {
for measurementName, measurementValue := range measurementEntry.Values {
var ctx = frontendlogging.CtxVector{}
ctx = event.AddMetaToContext(ctx)
ctx = append(ctx, measurementName, measurementValue)
ctx = append(ctx, "kind", "measurement", "original_timestamp", measurementEntry.Timestamp)
frontendLogger.Info("Measurement: "+measurementEntry.Type, ctx...)
}
}
}
if event.Exceptions != nil && len(event.Exceptions) > 0 {
for _, exception := range event.Exceptions {
var ctx = frontendlogging.CtxVector{}
ctx = event.AddMetaToContext(ctx)
exception := exception
transformedException := frontendlogging.TransformException(c.Req.Context(), &exception, store)
ctx = append(ctx, "kind", "exception", "type", transformedException.Type, "value", transformedException.Value, "stacktrace", transformedException.String())
ctx = append(ctx, "original_timestamp", exception.Timestamp)
frontendLogger.Error(exception.Message(), ctx...)
}
}
c.Resp.WriteHeader(http.StatusAccepted)
_, err := c.Resp.Write([]byte("OK"))
if err != nil {
hs.log.Error("could not write to response", "err", err)
}
}
}
// setupFrontendLogHandlers will set up handlers for logs incoming from frontend.
// handlers are setup even if frontend logging is disabled, but in this case do nothing
// this is to avoid reporting errors in case config was changes but there are browser
// sessions still open with older config
func (hs *HTTPServer) frontendLogEndpoints() web.Handler {
if !(hs.Cfg.GrafanaJavascriptAgent.Enabled) {
return func(ctx *web.Context) {
if ctx.Req.Method == http.MethodPost && ctx.Req.URL.Path == grafanaJavascriptAgentEndpointPath {
ctx.Resp.WriteHeader(http.StatusAccepted)
_, err := ctx.Resp.Write([]byte("OK"))
if err != nil {
hs.log.Error("could not write to response", "err", err)
}
}
}
}
sourceMapStore := frontendlogging.NewSourceMapStore(hs.Cfg, hs.pluginStaticRouteResolver, frontendlogging.ReadSourceMapFromFS)
rateLimiter := rate.NewLimiter(rate.Limit(hs.Cfg.GrafanaJavascriptAgent.EndpointRPS), hs.Cfg.GrafanaJavascriptAgent.EndpointBurst)
handler := GrafanaJavascriptAgentLogMessageHandler(sourceMapStore)
return func(ctx *web.Context) {
if ctx.Req.Method == http.MethodPost && ctx.Req.URL.Path == grafanaJavascriptAgentEndpointPath {
if !rateLimiter.AllowN(time.Now(), 1) {
ctx.Resp.WriteHeader(http.StatusTooManyRequests)
return
}
handler(hs, ctx)
}
}
}