|
|
|
package internal
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"sync"
|
|
|
|
|
|
|
|
"github.com/zeromicro/go-zero/core/logx"
|
|
|
|
"github.com/zeromicro/go-zero/rest/httpx"
|
|
|
|
)
|
|
|
|
|
|
|
|
// LogContext is a context key.
|
|
|
|
var LogContext = contextKey("request_logs")
|
|
|
|
|
|
|
|
// A LogCollector is used to collect logs.
|
|
|
|
type LogCollector struct {
|
|
|
|
Messages []string
|
|
|
|
lock sync.Mutex
|
|
|
|
}
|
|
|
|
|
|
|
|
// Append appends msg into log context.
|
|
|
|
func (lc *LogCollector) Append(msg string) {
|
|
|
|
lc.lock.Lock()
|
|
|
|
lc.Messages = append(lc.Messages, msg)
|
|
|
|
lc.lock.Unlock()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Flush flushes collected logs.
|
|
|
|
func (lc *LogCollector) Flush() string {
|
|
|
|
var buffer bytes.Buffer
|
|
|
|
|
|
|
|
start := true
|
|
|
|
for _, message := range lc.takeAll() {
|
|
|
|
if start {
|
|
|
|
start = false
|
|
|
|
} else {
|
|
|
|
buffer.WriteByte('\n')
|
|
|
|
}
|
|
|
|
buffer.WriteString(message)
|
|
|
|
}
|
|
|
|
|
|
|
|
return buffer.String()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (lc *LogCollector) takeAll() []string {
|
|
|
|
lc.lock.Lock()
|
|
|
|
messages := lc.Messages
|
|
|
|
lc.Messages = nil
|
|
|
|
lc.lock.Unlock()
|
|
|
|
|
|
|
|
return messages
|
|
|
|
}
|
|
|
|
|
|
|
|
// Error logs the given v along with r in error log.
|
|
|
|
func Error(r *http.Request, v ...any) {
|
|
|
|
logx.WithContext(r.Context()).Error(format(r, v...))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Errorf logs the given v with format along with r in error log.
|
|
|
|
func Errorf(r *http.Request, format string, v ...any) {
|
|
|
|
logx.WithContext(r.Context()).Error(formatf(r, format, v...))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Info logs the given v along with r in access log.
|
|
|
|
func Info(r *http.Request, v ...any) {
|
|
|
|
appendLog(r, format(r, v...))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Infof logs the given v with format along with r in access log.
|
|
|
|
func Infof(r *http.Request, format string, v ...any) {
|
|
|
|
appendLog(r, formatf(r, format, v...))
|
|
|
|
}
|
|
|
|
|
|
|
|
func appendLog(r *http.Request, message string) {
|
|
|
|
logs := r.Context().Value(LogContext)
|
|
|
|
if logs != nil {
|
|
|
|
logs.(*LogCollector).Append(message)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func format(r *http.Request, v ...any) string {
|
|
|
|
return formatWithReq(r, fmt.Sprint(v...))
|
|
|
|
}
|
|
|
|
|
|
|
|
func formatf(r *http.Request, format string, v ...any) string {
|
|
|
|
return formatWithReq(r, fmt.Sprintf(format, v...))
|
|
|
|
}
|
|
|
|
|
|
|
|
func formatWithReq(r *http.Request, v string) string {
|
|
|
|
return fmt.Sprintf("(%s - %s) %s", r.RequestURI, httpx.GetRemoteAddr(r), v)
|
|
|
|
}
|
|
|
|
|
|
|
|
type contextKey string
|
|
|
|
|
|
|
|
func (c contextKey) String() string {
|
|
|
|
return "rest/internal context key " + string(c)
|
|
|
|
}
|