2018-11-14 10:18:03 +01:00
|
|
|
package accesslog
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net/http"
|
|
|
|
"time"
|
|
|
|
|
2022-11-21 18:36:05 +01:00
|
|
|
"github.com/rs/zerolog/log"
|
2023-02-03 15:24:05 +01:00
|
|
|
"github.com/traefik/traefik/v3/pkg/logs"
|
|
|
|
"github.com/traefik/traefik/v3/pkg/middlewares/capture"
|
2022-11-21 18:36:05 +01:00
|
|
|
"github.com/vulcand/oxy/v2/utils"
|
2018-11-14 10:18:03 +01:00
|
|
|
)
|
|
|
|
|
2020-05-11 12:06:07 +02:00
|
|
|
// FieldApply function hook to add data in accesslog.
|
2018-11-14 10:18:03 +01:00
|
|
|
type FieldApply func(rw http.ResponseWriter, r *http.Request, next http.Handler, data *LogData)
|
|
|
|
|
|
|
|
// FieldHandler sends a new field to the logger.
|
|
|
|
type FieldHandler struct {
|
|
|
|
next http.Handler
|
|
|
|
name string
|
|
|
|
value string
|
|
|
|
applyFn FieldApply
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewFieldHandler creates a Field handler.
|
2020-07-07 14:42:03 +02:00
|
|
|
func NewFieldHandler(next http.Handler, name, value string, applyFn FieldApply) http.Handler {
|
2018-11-14 10:18:03 +01:00
|
|
|
return &FieldHandler{next: next, name: name, value: value, applyFn: applyFn}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *FieldHandler) ServeHTTP(rw http.ResponseWriter, req *http.Request) {
|
|
|
|
table := GetLogData(req)
|
|
|
|
if table == nil {
|
|
|
|
f.next.ServeHTTP(rw, req)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
table.Core[f.name] = f.value
|
|
|
|
|
|
|
|
if f.applyFn != nil {
|
|
|
|
f.applyFn(rw, req, f.next, table)
|
|
|
|
} else {
|
|
|
|
f.next.ServeHTTP(rw, req)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-11 12:06:07 +02:00
|
|
|
// AddServiceFields add service fields.
|
2018-11-14 10:18:03 +01:00
|
|
|
func AddServiceFields(rw http.ResponseWriter, req *http.Request, next http.Handler, data *LogData) {
|
|
|
|
start := time.Now().UTC()
|
|
|
|
|
2022-09-12 17:10:09 +02:00
|
|
|
next.ServeHTTP(rw, req)
|
2018-11-14 10:18:03 +01:00
|
|
|
|
|
|
|
// use UTC to handle switchover of daylight saving correctly
|
|
|
|
data.Core[OriginDuration] = time.Now().UTC().Sub(start)
|
2022-09-12 17:10:09 +02:00
|
|
|
// make copy of headers, so we can ensure there is no subsequent mutation
|
|
|
|
// during response processing
|
2018-11-14 10:18:03 +01:00
|
|
|
data.OriginResponse = make(http.Header)
|
2022-09-12 17:10:09 +02:00
|
|
|
utils.CopyHeaders(data.OriginResponse, rw.Header())
|
|
|
|
|
|
|
|
ctx := req.Context()
|
|
|
|
capt, err := capture.FromContext(ctx)
|
|
|
|
if err != nil {
|
2022-11-21 18:36:05 +01:00
|
|
|
log.Ctx(ctx).Error().Err(err).Str(logs.MiddlewareType, "AccessLogs").Msg("Could not get Capture")
|
2022-09-12 17:10:09 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
data.Core[OriginStatus] = capt.StatusCode()
|
|
|
|
data.Core[OriginContentSize] = capt.ResponseSize()
|
2018-11-14 10:18:03 +01:00
|
|
|
}
|
2023-09-27 15:22:06 +02:00
|
|
|
|
|
|
|
// InitServiceFields init service fields.
|
|
|
|
func InitServiceFields(rw http.ResponseWriter, req *http.Request, next http.Handler, data *LogData) {
|
|
|
|
// Because they are expected to be initialized when the logger is processing the data table,
|
|
|
|
// the origin fields are initialized in case the response is returned by Traefik itself, and not a service.
|
|
|
|
data.Core[OriginDuration] = time.Duration(0)
|
|
|
|
data.Core[OriginStatus] = 0
|
|
|
|
data.Core[OriginContentSize] = int64(0)
|
|
|
|
|
|
|
|
next.ServeHTTP(rw, req)
|
|
|
|
}
|