2019-11-25 17:33:58 -05:00
|
|
|
package log
|
2019-06-20 18:36:40 -05:00
|
|
|
|
|
|
|
import (
|
2021-05-25 03:38:21 -05:00
|
|
|
"encoding/base64"
|
2019-07-10 00:23:59 -05:00
|
|
|
"net/http"
|
2019-06-20 18:36:40 -05:00
|
|
|
"os"
|
2021-09-17 19:00:59 -05:00
|
|
|
"runtime"
|
|
|
|
"strconv"
|
2021-05-25 03:38:21 -05:00
|
|
|
"strings"
|
2019-06-20 18:36:40 -05:00
|
|
|
"time"
|
|
|
|
|
2019-07-10 00:23:59 -05:00
|
|
|
"github.com/gorilla/mux"
|
2019-06-20 18:36:40 -05:00
|
|
|
"github.com/rs/zerolog"
|
|
|
|
)
|
|
|
|
|
2020-05-11 17:13:24 -05:00
|
|
|
// Logger extends zerolog's Logger.
|
2019-11-25 17:33:58 -05:00
|
|
|
type Logger struct {
|
|
|
|
zerolog.Logger
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l Logger) Println(v ...interface{}) {
|
|
|
|
l.Logger.Error().Msg("panic recovered")
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewLogger(level string, output string) Logger {
|
2019-06-20 18:36:40 -05:00
|
|
|
zerolog.TimeFieldFormat = time.RFC3339Nano
|
2019-11-25 17:33:58 -05:00
|
|
|
lvl, err := zerolog.ParseLevel(level)
|
2019-12-13 00:53:18 -05:00
|
|
|
|
2019-06-20 18:36:40 -05:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2019-12-13 00:53:18 -05:00
|
|
|
|
2019-06-20 18:36:40 -05:00
|
|
|
zerolog.SetGlobalLevel(lvl)
|
2019-12-13 00:53:18 -05:00
|
|
|
|
2019-06-20 18:36:40 -05:00
|
|
|
var log zerolog.Logger
|
2019-12-13 00:53:18 -05:00
|
|
|
|
2019-11-25 17:33:58 -05:00
|
|
|
if output == "" {
|
2019-06-20 18:36:40 -05:00
|
|
|
log = zerolog.New(os.Stdout)
|
|
|
|
} else {
|
2019-11-25 17:33:58 -05:00
|
|
|
file, err := os.OpenFile(output, os.O_APPEND|os.O_WRONLY|os.O_CREATE, 0600)
|
2019-06-20 18:36:40 -05:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
log = zerolog.New(file)
|
|
|
|
}
|
2019-12-13 00:53:18 -05:00
|
|
|
|
2021-09-17 19:00:59 -05:00
|
|
|
return Logger{Logger: log.Hook(goroutineHook{}).With().Caller().Timestamp().Logger()}
|
2019-06-20 18:36:40 -05:00
|
|
|
}
|
|
|
|
|
2021-05-25 03:38:21 -05:00
|
|
|
func NewAuditLogger(level string, audit string) *Logger {
|
|
|
|
zerolog.TimeFieldFormat = time.RFC3339Nano
|
|
|
|
lvl, err := zerolog.ParseLevel(level)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
zerolog.SetGlobalLevel(lvl)
|
|
|
|
|
|
|
|
var auditLog zerolog.Logger
|
|
|
|
|
|
|
|
auditFile, err := os.OpenFile(audit, os.O_APPEND|os.O_WRONLY|os.O_CREATE, 0600)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
auditLog = zerolog.New(auditFile)
|
|
|
|
|
|
|
|
return &Logger{Logger: auditLog.With().Timestamp().Logger()}
|
|
|
|
}
|
|
|
|
|
2019-07-10 00:23:59 -05:00
|
|
|
type statusWriter struct {
|
|
|
|
http.ResponseWriter
|
|
|
|
status int
|
|
|
|
length int
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *statusWriter) WriteHeader(status int) {
|
|
|
|
w.status = status
|
|
|
|
w.ResponseWriter.WriteHeader(status)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *statusWriter) Write(b []byte) (int, error) {
|
|
|
|
if w.status == 0 {
|
|
|
|
w.status = 200
|
|
|
|
}
|
2019-12-13 00:53:18 -05:00
|
|
|
|
2019-07-10 00:23:59 -05:00
|
|
|
n, err := w.ResponseWriter.Write(b)
|
|
|
|
w.length += n
|
2019-12-13 00:53:18 -05:00
|
|
|
|
2019-07-10 00:23:59 -05:00
|
|
|
return n, err
|
|
|
|
}
|
|
|
|
|
2021-05-25 03:38:21 -05:00
|
|
|
// SessionLogger logs session details.
|
2019-11-25 17:33:58 -05:00
|
|
|
func SessionLogger(log Logger) mux.MiddlewareFunc {
|
2019-06-20 18:36:40 -05:00
|
|
|
l := log.With().Str("module", "http").Logger()
|
2019-12-13 00:53:18 -05:00
|
|
|
|
2019-07-10 00:23:59 -05:00
|
|
|
return func(next http.Handler) http.Handler {
|
|
|
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
// Start timer
|
|
|
|
start := time.Now()
|
|
|
|
path := r.URL.Path
|
|
|
|
raw := r.URL.RawQuery
|
2019-06-20 18:36:40 -05:00
|
|
|
|
2019-07-10 00:23:59 -05:00
|
|
|
sw := statusWriter{ResponseWriter: w}
|
2019-06-20 18:36:40 -05:00
|
|
|
|
2019-07-10 00:23:59 -05:00
|
|
|
// Process request
|
|
|
|
next.ServeHTTP(&sw, r)
|
|
|
|
|
|
|
|
// Stop timer
|
|
|
|
end := time.Now()
|
|
|
|
latency := end.Sub(start)
|
|
|
|
if latency > time.Minute {
|
|
|
|
// Truncate in a golang < 1.8 safe way
|
|
|
|
latency -= latency % time.Second
|
|
|
|
}
|
|
|
|
clientIP := r.RemoteAddr
|
|
|
|
method := r.Method
|
2019-10-24 16:16:41 -05:00
|
|
|
headers := map[string][]string{}
|
2021-05-25 03:38:21 -05:00
|
|
|
username := ""
|
|
|
|
log := l.Info()
|
2019-10-24 16:16:41 -05:00
|
|
|
for key, value := range r.Header {
|
|
|
|
if key == "Authorization" { // anonymize from logs
|
2021-05-25 03:38:21 -05:00
|
|
|
s := strings.SplitN(value[0], " ", 2)
|
|
|
|
if len(s) == 2 && strings.EqualFold(s[0], "basic") {
|
|
|
|
b, err := base64.StdEncoding.DecodeString(s[1])
|
|
|
|
if err == nil {
|
|
|
|
pair := strings.SplitN(string(b), ":", 2)
|
|
|
|
// nolint:gomnd
|
|
|
|
if len(pair) == 2 {
|
|
|
|
username = pair[0]
|
|
|
|
log = log.Str("username", username)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-10-24 16:16:41 -05:00
|
|
|
value = []string{"******"}
|
|
|
|
}
|
|
|
|
headers[key] = value
|
|
|
|
}
|
2019-07-10 00:23:59 -05:00
|
|
|
statusCode := sw.status
|
|
|
|
bodySize := sw.length
|
|
|
|
if raw != "" {
|
|
|
|
path = path + "?" + raw
|
|
|
|
}
|
2019-06-20 18:36:40 -05:00
|
|
|
|
2021-05-25 03:38:21 -05:00
|
|
|
log.Str("clientIP", clientIP).
|
2019-07-10 00:23:59 -05:00
|
|
|
Str("method", method).
|
|
|
|
Str("path", path).
|
|
|
|
Int("statusCode", statusCode).
|
|
|
|
Str("latency", latency.String()).
|
|
|
|
Int("bodySize", bodySize).
|
|
|
|
Interface("headers", headers).
|
|
|
|
Msg("HTTP API")
|
|
|
|
})
|
2019-06-20 18:36:40 -05:00
|
|
|
}
|
|
|
|
}
|
2021-05-25 03:38:21 -05:00
|
|
|
|
|
|
|
func SessionAuditLogger(audit *Logger) mux.MiddlewareFunc {
|
|
|
|
return func(next http.Handler) http.Handler {
|
|
|
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
path := r.URL.Path
|
|
|
|
raw := r.URL.RawQuery
|
|
|
|
|
|
|
|
sw := statusWriter{ResponseWriter: w}
|
|
|
|
|
|
|
|
// Process request
|
|
|
|
next.ServeHTTP(&sw, r)
|
|
|
|
|
|
|
|
clientIP := r.RemoteAddr
|
|
|
|
method := r.Method
|
|
|
|
username := ""
|
|
|
|
|
|
|
|
for key, value := range r.Header {
|
|
|
|
if key == "Authorization" { // anonymize from logs
|
|
|
|
s := strings.SplitN(value[0], " ", 2)
|
|
|
|
if len(s) == 2 && strings.EqualFold(s[0], "basic") {
|
|
|
|
b, err := base64.StdEncoding.DecodeString(s[1])
|
|
|
|
if err == nil {
|
|
|
|
pair := strings.SplitN(string(b), ":", 2)
|
|
|
|
// nolint:gomnd
|
|
|
|
if len(pair) == 2 {
|
|
|
|
username = pair[0]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
statusCode := sw.status
|
|
|
|
if raw != "" {
|
|
|
|
path = path + "?" + raw
|
|
|
|
}
|
|
|
|
|
|
|
|
if (method == http.MethodPost || method == http.MethodPut ||
|
|
|
|
method == http.MethodPatch || method == http.MethodDelete) &&
|
|
|
|
(statusCode == http.StatusOK || statusCode == http.StatusCreated || statusCode == http.StatusAccepted) {
|
|
|
|
audit.Info().
|
|
|
|
Str("clientIP", clientIP).
|
|
|
|
Str("subject", username).
|
|
|
|
Str("action", method).
|
|
|
|
Str("object", path).
|
|
|
|
Int("status", statusCode).
|
|
|
|
Msg("HTTP API Audit")
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2021-09-17 19:00:59 -05:00
|
|
|
|
|
|
|
// goroutineID adds goroutine-id to logs to help debug concurrency issues.
|
|
|
|
func goroutineID() int {
|
|
|
|
var buf [64]byte
|
|
|
|
n := runtime.Stack(buf[:], false)
|
|
|
|
idField := strings.Fields(strings.TrimPrefix(string(buf[:n]), "goroutine "))[0]
|
|
|
|
|
|
|
|
id, err := strconv.Atoi(idField)
|
|
|
|
if err != nil {
|
|
|
|
return -1
|
|
|
|
}
|
|
|
|
|
|
|
|
return id
|
|
|
|
}
|
|
|
|
|
|
|
|
type goroutineHook struct{}
|
|
|
|
|
|
|
|
func (h goroutineHook) Run(e *zerolog.Event, level zerolog.Level, msg string) {
|
|
|
|
if level != zerolog.NoLevel {
|
|
|
|
e.Int("goroutine", goroutineID())
|
|
|
|
}
|
|
|
|
}
|