mirror of https://github.com/grafana/grafana.git
				
				
				
			
		
			
				
	
	
		
			194 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			194 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			Go
		
	
	
	
| package api
 | |
| 
 | |
| import (
 | |
| 	"net/http"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/getsentry/sentry-go"
 | |
| 	"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 sentryLogEndpointPath = "/log"
 | |
| const grafanaJavascriptAgentEndpointPath = "/log-grafana-javascript-agent"
 | |
| 
 | |
| /** @deprecated will be removed in the next major version */
 | |
| func NewFrontendLogMessageHandler(store *frontendlogging.SourceMapStore) frontendLogMessageHandler {
 | |
| 	return func(hs *HTTPServer, c *web.Context) {
 | |
| 		event := frontendlogging.FrontendSentryEvent{}
 | |
| 		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)
 | |
| 			}
 | |
| 			return
 | |
| 		}
 | |
| 
 | |
| 		var msg = "unknown"
 | |
| 
 | |
| 		if len(event.Message) > 0 {
 | |
| 			msg = event.Message
 | |
| 		} else if event.Exception != nil && len(event.Exception.Values) > 0 {
 | |
| 			msg = event.Exception.Values[0].FmtMessage()
 | |
| 		}
 | |
| 
 | |
| 		var ctx = event.ToLogContext(store)
 | |
| 
 | |
| 		switch event.Level {
 | |
| 		case sentry.LevelError:
 | |
| 			frontendLogger.Error(msg, ctx...)
 | |
| 		case sentry.LevelWarning:
 | |
| 			frontendLogger.Warn(msg, ctx...)
 | |
| 		case sentry.LevelDebug:
 | |
| 			frontendLogger.Debug(msg, ctx...)
 | |
| 		default:
 | |
| 			frontendLogger.Info(msg, 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)
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| 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(&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 || hs.Cfg.Sentry.Enabled) {
 | |
| 		return func(ctx *web.Context) {
 | |
| 			if ctx.Req.Method == http.MethodPost && (ctx.Req.URL.Path == sentryLogEndpointPath || 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)
 | |
| 
 | |
| 	var rateLimiter *rate.Limiter
 | |
| 	var handler frontendLogMessageHandler
 | |
| 	handlerEndpoint := ""
 | |
| 	dummyEndpoint := ""
 | |
| 
 | |
| 	if hs.Cfg.GrafanaJavascriptAgent.Enabled {
 | |
| 		rateLimiter = rate.NewLimiter(rate.Limit(hs.Cfg.GrafanaJavascriptAgent.EndpointRPS), hs.Cfg.GrafanaJavascriptAgent.EndpointBurst)
 | |
| 		handler = GrafanaJavascriptAgentLogMessageHandler(sourceMapStore)
 | |
| 		handlerEndpoint = grafanaJavascriptAgentEndpointPath
 | |
| 		dummyEndpoint = sentryLogEndpointPath
 | |
| 	} else {
 | |
| 		rateLimiter = rate.NewLimiter(rate.Limit(hs.Cfg.Sentry.EndpointRPS), hs.Cfg.Sentry.EndpointBurst)
 | |
| 		handler = NewFrontendLogMessageHandler(sourceMapStore)
 | |
| 		handlerEndpoint = sentryLogEndpointPath
 | |
| 		dummyEndpoint = grafanaJavascriptAgentEndpointPath
 | |
| 	}
 | |
| 
 | |
| 	return func(ctx *web.Context) {
 | |
| 		if ctx.Req.Method == http.MethodPost && ctx.Req.URL.Path == dummyEndpoint {
 | |
| 			ctx.Resp.WriteHeader(http.StatusAccepted)
 | |
| 			_, err := ctx.Resp.Write([]byte("OK"))
 | |
| 			if err != nil {
 | |
| 				hs.log.Error("could not write to response", "err", err)
 | |
| 			}
 | |
| 		}
 | |
| 		if ctx.Req.Method == http.MethodPost && ctx.Req.URL.Path == handlerEndpoint {
 | |
| 			if !rateLimiter.AllowN(time.Now(), 1) {
 | |
| 				ctx.Resp.WriteHeader(http.StatusTooManyRequests)
 | |
| 				return
 | |
| 			}
 | |
| 			handler(hs, ctx)
 | |
| 		}
 | |
| 	}
 | |
| }
 |