mirror of
https://github.com/m1k1o/neko.git
synced 2024-07-24 14:40:50 +12:00
logger use chi's middleware.
This commit is contained in:
parent
5688be80ba
commit
fe84e218e6
@ -1,7 +1,6 @@
|
|||||||
package http
|
package http
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/http"
|
"net/http"
|
||||||
"time"
|
"time"
|
||||||
@ -10,60 +9,13 @@ import (
|
|||||||
"github.com/rs/zerolog"
|
"github.com/rs/zerolog"
|
||||||
"github.com/rs/zerolog/log"
|
"github.com/rs/zerolog/log"
|
||||||
|
|
||||||
"demodesk/neko/internal/http/auth"
|
|
||||||
"demodesk/neko/internal/types"
|
"demodesk/neko/internal/types"
|
||||||
"demodesk/neko/internal/utils"
|
"demodesk/neko/internal/utils"
|
||||||
)
|
)
|
||||||
|
|
||||||
type logEntryKey int
|
type logFormatter struct{}
|
||||||
|
|
||||||
const logEntryKeyCtx logEntryKey = iota
|
func (l *logFormatter) NewLogEntry(r *http.Request) middleware.LogEntry {
|
||||||
|
|
||||||
func setLogEntry(r *http.Request, data *logEntry) *http.Request {
|
|
||||||
ctx := context.WithValue(r.Context(), logEntryKeyCtx, data)
|
|
||||||
return r.WithContext(ctx)
|
|
||||||
}
|
|
||||||
|
|
||||||
func getLogEntry(r *http.Request) *logEntry {
|
|
||||||
return r.Context().Value(logEntryKeyCtx).(*logEntry)
|
|
||||||
}
|
|
||||||
|
|
||||||
func LoggerMiddleware(next http.Handler) http.Handler {
|
|
||||||
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
||||||
ww := middleware.NewWrapResponseWriter(w, r.ProtoMajor)
|
|
||||||
|
|
||||||
logEntry := newLogEntry(w, r)
|
|
||||||
defer func() {
|
|
||||||
logEntry.Write(ww.Status(), ww.BytesWritten())
|
|
||||||
}()
|
|
||||||
|
|
||||||
next.ServeHTTP(ww, setLogEntry(r, logEntry))
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
type logEntry struct {
|
|
||||||
req struct {
|
|
||||||
Time time.Time
|
|
||||||
Id string
|
|
||||||
Scheme string
|
|
||||||
Proto string
|
|
||||||
Method string
|
|
||||||
Remote string
|
|
||||||
Agent string
|
|
||||||
Uri string
|
|
||||||
}
|
|
||||||
res struct {
|
|
||||||
Time time.Time
|
|
||||||
Code int
|
|
||||||
Bytes int
|
|
||||||
}
|
|
||||||
err error
|
|
||||||
elapsed time.Duration
|
|
||||||
hasSession bool
|
|
||||||
session types.Session
|
|
||||||
}
|
|
||||||
|
|
||||||
func newLogEntry(w http.ResponseWriter, r *http.Request) *logEntry {
|
|
||||||
e := logEntry{}
|
e := logEntry{}
|
||||||
e.req.Time = time.Now()
|
e.req.Time = time.Now()
|
||||||
|
|
||||||
@ -85,30 +37,48 @@ func newLogEntry(w http.ResponseWriter, r *http.Request) *logEntry {
|
|||||||
return &e
|
return &e
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *logEntry) SetResponse(w http.ResponseWriter, r *http.Request) {
|
type logEntry struct {
|
||||||
e.res.Time = time.Now()
|
req struct {
|
||||||
|
Time time.Time
|
||||||
e.elapsed = e.res.Time.Sub(e.req.Time)
|
Id string
|
||||||
e.session, e.hasSession = auth.GetSession(r)
|
Scheme string
|
||||||
|
Proto string
|
||||||
|
Method string
|
||||||
|
Remote string
|
||||||
|
Agent string
|
||||||
|
Uri string
|
||||||
|
}
|
||||||
|
res struct {
|
||||||
|
Time time.Time
|
||||||
|
Code int
|
||||||
|
Bytes int
|
||||||
|
}
|
||||||
|
err error
|
||||||
|
session *types.Session
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *logEntry) SetError(err error) {
|
func (e *logEntry) SetError(err error) {
|
||||||
e.err = err
|
e.err = err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *logEntry) Write(status, bytes int) {
|
func (e *logEntry) SetSession(session types.Session) {
|
||||||
|
e.session = &session
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *logEntry) Write(status, bytes int, header http.Header, elapsed time.Duration, extra interface{}) {
|
||||||
|
e.res.Time = time.Now()
|
||||||
e.res.Code = status
|
e.res.Code = status
|
||||||
e.res.Bytes = bytes
|
e.res.Bytes = bytes
|
||||||
|
|
||||||
logger := log.With().
|
logger := log.With().
|
||||||
Str("module", "http").
|
Str("module", "http").
|
||||||
Float64("elapsed", float64(e.elapsed.Nanoseconds())/1000000.0).
|
Float64("elapsed", float64(elapsed.Nanoseconds())/1000000.0).
|
||||||
Interface("req", e.req).
|
Interface("req", e.req).
|
||||||
Interface("res", e.res).
|
Interface("res", e.res).
|
||||||
Logger()
|
Logger()
|
||||||
|
|
||||||
if e.hasSession {
|
if e.session != nil {
|
||||||
logger = logger.With().Str("session_id", e.session.ID()).Logger()
|
logger = logger.With().Str("session_id", (*e.session).ID()).Logger()
|
||||||
}
|
}
|
||||||
|
|
||||||
if e.err != nil {
|
if e.err != nil {
|
||||||
@ -140,3 +110,12 @@ func (e *logEntry) Write(status, bytes int) {
|
|||||||
|
|
||||||
logger.Debug().Msgf("request complete (%d)", e.res.Code)
|
logger.Debug().Msgf("request complete (%d)", e.res.Code)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (e *logEntry) Panic(v interface{}, stack []byte) {
|
||||||
|
message := fmt.Sprintf("%+v", v)
|
||||||
|
|
||||||
|
log.Fatal().
|
||||||
|
Str("message", message).
|
||||||
|
Str("stack", string(stack)).
|
||||||
|
Msg("got HTTP panic")
|
||||||
|
}
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
package http
|
package http
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"demodesk/neko/internal/http/auth"
|
||||||
"demodesk/neko/internal/types"
|
"demodesk/neko/internal/types"
|
||||||
"demodesk/neko/internal/utils"
|
"demodesk/neko/internal/utils"
|
||||||
"net/http"
|
"net/http"
|
||||||
@ -15,9 +16,9 @@ type RouterCtx struct {
|
|||||||
|
|
||||||
func newRouter() *RouterCtx {
|
func newRouter() *RouterCtx {
|
||||||
router := chi.NewRouter()
|
router := chi.NewRouter()
|
||||||
router.Use(middleware.Recoverer) // Recover from panics without crashing server
|
|
||||||
router.Use(middleware.RequestID) // Create a request ID for each request
|
router.Use(middleware.RequestID) // Create a request ID for each request
|
||||||
router.Use(LoggerMiddleware)
|
router.Use(middleware.RequestLogger(&logFormatter{}))
|
||||||
|
router.Use(middleware.Recoverer) // Recover from panics without crashing server
|
||||||
return &RouterCtx{router}
|
return &RouterCtx{router}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -78,27 +79,37 @@ func errorHandler(err error, w http.ResponseWriter, r *http.Request) {
|
|||||||
|
|
||||||
func routeHandler(fn types.RouterHandler) http.HandlerFunc {
|
func routeHandler(fn types.RouterHandler) http.HandlerFunc {
|
||||||
return func(w http.ResponseWriter, r *http.Request) {
|
return func(w http.ResponseWriter, r *http.Request) {
|
||||||
logEntry := getLogEntry(r)
|
// get custom log entry pointer from context
|
||||||
|
logEntry, _ := r.Context().Value(middleware.LogEntryCtxKey).(*logEntry)
|
||||||
|
|
||||||
if err := fn(w, r); err != nil {
|
if err := fn(w, r); err != nil {
|
||||||
logEntry.SetError(err)
|
logEntry.SetError(err)
|
||||||
errorHandler(err, w, r)
|
errorHandler(err, w, r)
|
||||||
}
|
}
|
||||||
|
|
||||||
logEntry.SetResponse(w, r)
|
// set session if exits
|
||||||
|
if session, ok := auth.GetSession(r); ok {
|
||||||
|
logEntry.SetSession(session)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func middlewareHandler(fn types.MiddlewareHandler) func(http.Handler) http.Handler {
|
func middlewareHandler(fn types.MiddlewareHandler) func(http.Handler) http.Handler {
|
||||||
return func(next http.Handler) http.Handler {
|
return func(next http.Handler) http.Handler {
|
||||||
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
logEntry := getLogEntry(r)
|
// get custom log entry pointer from context
|
||||||
|
logEntry, _ := r.Context().Value(middleware.LogEntryCtxKey).(*logEntry)
|
||||||
|
|
||||||
ctx, err := fn(w, r)
|
ctx, err := fn(w, r)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logEntry.SetError(err)
|
logEntry.SetError(err)
|
||||||
errorHandler(err, w, r)
|
errorHandler(err, w, r)
|
||||||
logEntry.SetResponse(w, r)
|
|
||||||
|
// set session if exits
|
||||||
|
if session, ok := auth.GetSession(r); ok {
|
||||||
|
logEntry.SetSession(session)
|
||||||
|
}
|
||||||
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if ctx != nil {
|
if ctx != nil {
|
||||||
|
Loading…
Reference in New Issue
Block a user