2020-10-29 22:12:33 +01:00
|
|
|
package http
|
2020-10-22 16:54:50 +02:00
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/go-chi/chi/middleware"
|
2021-09-17 00:58:50 +02:00
|
|
|
"github.com/rs/zerolog"
|
2020-10-22 16:54:50 +02:00
|
|
|
|
2022-07-14 00:58:22 +02:00
|
|
|
"github.com/demodesk/neko/pkg/types"
|
|
|
|
"github.com/demodesk/neko/pkg/utils"
|
2021-09-17 00:24:33 +02:00
|
|
|
)
|
2020-10-22 16:54:50 +02:00
|
|
|
|
2021-09-17 19:27:21 +02:00
|
|
|
type logFormatter struct {
|
|
|
|
logger zerolog.Logger
|
|
|
|
}
|
2021-03-19 15:04:01 +01:00
|
|
|
|
2021-09-17 19:22:49 +02:00
|
|
|
func (l *logFormatter) NewLogEntry(r *http.Request) middleware.LogEntry {
|
2022-07-04 18:09:47 +02:00
|
|
|
// exclude health & metrics from logs
|
|
|
|
if r.RequestURI == "/health" || r.RequestURI == "/metrics" {
|
2021-09-17 20:35:41 +02:00
|
|
|
return &nulllog{}
|
|
|
|
}
|
|
|
|
|
2022-07-28 12:20:20 +02:00
|
|
|
req := map[string]any{}
|
2020-10-22 16:54:50 +02:00
|
|
|
|
2021-09-17 00:24:33 +02:00
|
|
|
if reqID := middleware.GetReqID(r.Context()); reqID != "" {
|
2021-09-17 20:20:10 +02:00
|
|
|
req["id"] = reqID
|
2021-09-17 00:24:33 +02:00
|
|
|
}
|
2020-10-22 16:54:50 +02:00
|
|
|
|
2021-09-17 00:24:33 +02:00
|
|
|
scheme := "http"
|
|
|
|
if r.TLS != nil {
|
|
|
|
scheme = "https"
|
2020-10-22 16:54:50 +02:00
|
|
|
}
|
2021-09-17 00:24:33 +02:00
|
|
|
|
2021-09-17 20:20:10 +02:00
|
|
|
req["scheme"] = scheme
|
|
|
|
req["proto"] = r.Proto
|
|
|
|
req["method"] = r.Method
|
|
|
|
req["remote"] = r.RemoteAddr
|
|
|
|
req["agent"] = r.UserAgent()
|
|
|
|
req["uri"] = fmt.Sprintf("%s://%s%s", scheme, r.Host, r.RequestURI)
|
|
|
|
|
|
|
|
return &logEntry{
|
|
|
|
logger: l.logger.With().Interface("req", req).Logger(),
|
|
|
|
}
|
2020-10-22 16:54:50 +02:00
|
|
|
}
|
|
|
|
|
2021-09-17 19:22:49 +02:00
|
|
|
type logEntry struct {
|
2021-09-17 20:20:10 +02:00
|
|
|
logger zerolog.Logger
|
2021-09-17 19:22:49 +02:00
|
|
|
err error
|
2021-09-17 20:20:10 +02:00
|
|
|
panic *logPanic
|
2022-08-26 20:16:40 +02:00
|
|
|
session types.Session
|
2020-10-22 16:54:50 +02:00
|
|
|
}
|
|
|
|
|
2021-09-17 20:20:10 +02:00
|
|
|
type logPanic struct {
|
|
|
|
message string
|
|
|
|
stack string
|
|
|
|
}
|
|
|
|
|
2022-07-28 12:20:20 +02:00
|
|
|
func (e *logEntry) Panic(v any, stack []byte) {
|
2021-09-17 20:20:10 +02:00
|
|
|
e.panic = &logPanic{
|
|
|
|
message: fmt.Sprintf("%+v", v),
|
|
|
|
stack: string(stack),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *logEntry) Error(err error) {
|
2021-09-17 00:24:33 +02:00
|
|
|
e.err = err
|
2020-10-22 16:54:50 +02:00
|
|
|
}
|
|
|
|
|
2021-09-17 19:22:49 +02:00
|
|
|
func (e *logEntry) SetSession(session types.Session) {
|
2022-08-26 20:16:40 +02:00
|
|
|
e.session = session
|
2021-09-17 19:22:49 +02:00
|
|
|
}
|
|
|
|
|
2022-07-28 12:20:20 +02:00
|
|
|
func (e *logEntry) Write(status, bytes int, header http.Header, elapsed time.Duration, extra any) {
|
|
|
|
res := map[string]any{}
|
2021-09-17 20:20:10 +02:00
|
|
|
res["time"] = time.Now().UTC().Format(time.RFC1123)
|
|
|
|
res["status"] = status
|
|
|
|
res["bytes"] = bytes
|
|
|
|
res["elapsed"] = float64(elapsed.Nanoseconds()) / 1000000.0
|
2021-09-17 00:58:50 +02:00
|
|
|
|
2021-09-17 20:20:10 +02:00
|
|
|
logger := e.logger.With().Interface("res", res).Logger()
|
2021-09-17 00:24:33 +02:00
|
|
|
|
2021-09-17 20:20:10 +02:00
|
|
|
// add session ID to logs (if exists)
|
2021-09-17 19:22:49 +02:00
|
|
|
if e.session != nil {
|
2022-08-26 20:16:40 +02:00
|
|
|
logger = logger.With().Str("session_id", e.session.ID()).Logger()
|
2021-09-17 00:24:33 +02:00
|
|
|
}
|
|
|
|
|
2021-09-17 20:20:10 +02:00
|
|
|
// handle panic error message
|
|
|
|
if e.panic != nil {
|
|
|
|
logger.WithLevel(zerolog.PanicLevel).
|
|
|
|
Err(e.err).
|
|
|
|
Str("stack", e.panic.stack).
|
|
|
|
Msgf("request failed (%d): %s", status, e.panic.message)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// handle panic error message
|
2021-09-17 00:24:33 +02:00
|
|
|
if e.err != nil {
|
|
|
|
httpErr, ok := e.err.(*utils.HTTPError)
|
|
|
|
if !ok {
|
2021-09-17 20:20:10 +02:00
|
|
|
logger.Err(e.err).Msgf("request failed (%d)", status)
|
2021-09-17 00:24:33 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if httpErr.Message == "" {
|
|
|
|
httpErr.Message = http.StatusText(httpErr.Code)
|
|
|
|
}
|
|
|
|
|
2021-09-17 20:20:10 +02:00
|
|
|
var logLevel zerolog.Level
|
2021-09-17 00:58:50 +02:00
|
|
|
if httpErr.Code < 500 {
|
2021-09-17 20:20:10 +02:00
|
|
|
logLevel = zerolog.WarnLevel
|
2021-09-17 00:58:50 +02:00
|
|
|
} else {
|
2021-09-17 20:20:10 +02:00
|
|
|
logLevel = zerolog.ErrorLevel
|
2021-09-17 00:58:50 +02:00
|
|
|
}
|
2021-09-17 00:24:33 +02:00
|
|
|
|
|
|
|
message := httpErr.Message
|
|
|
|
if httpErr.InternalMsg != "" {
|
|
|
|
message = httpErr.InternalMsg
|
|
|
|
}
|
|
|
|
|
2021-09-17 20:20:10 +02:00
|
|
|
logger.WithLevel(logLevel).Err(httpErr.InternalErr).Msgf("request failed (%d): %s", status, message)
|
2021-09-17 00:24:33 +02:00
|
|
|
return
|
|
|
|
}
|
2020-10-22 16:54:50 +02:00
|
|
|
|
2021-09-17 20:20:10 +02:00
|
|
|
logger.Debug().Msgf("request complete (%d)", status)
|
2021-09-17 19:22:49 +02:00
|
|
|
}
|
2021-09-17 20:35:41 +02:00
|
|
|
|
|
|
|
type nulllog struct{}
|
|
|
|
|
2022-07-28 12:20:20 +02:00
|
|
|
func (e *nulllog) Panic(v any, stack []byte) {}
|
|
|
|
func (e *nulllog) Error(err error) {}
|
|
|
|
func (e *nulllog) SetSession(session types.Session) {}
|
|
|
|
func (e *nulllog) Write(status, bytes int, header http.Header, elapsed time.Duration, extra any) {
|
2021-09-17 20:35:41 +02:00
|
|
|
}
|