mirror of
https://github.com/gogf/gf.git
synced 2025-04-05 03:05:05 +08:00
224 lines
5.8 KiB
Go
224 lines
5.8 KiB
Go
// Copyright GoFrame Author(https://goframe.org). All Rights Reserved.
|
|
//
|
|
// This Source Code Form is subject to the terms of the MIT License.
|
|
// If a copy of the MIT was not distributed with this file,
|
|
// You can obtain one at https://github.com/gogf/gf.
|
|
|
|
package glog
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"time"
|
|
|
|
"github.com/gogf/gf/v2/util/gconv"
|
|
)
|
|
|
|
// Handler is function handler for custom logging content outputs.
|
|
type Handler func(ctx context.Context, in *HandlerInput)
|
|
|
|
// HandlerInput is the input parameter struct for logging Handler.
|
|
//
|
|
// The logging content is consisted in:
|
|
// TimeFormat [LevelFormat] {TraceId} {CtxStr} Prefix CallerFunc CallerPath Content Values Stack
|
|
//
|
|
// The header in the logging content is:
|
|
// TimeFormat [LevelFormat] {TraceId} {CtxStr} Prefix CallerFunc CallerPath
|
|
type HandlerInput struct {
|
|
internalHandlerInfo
|
|
|
|
// Current Logger object.
|
|
Logger *Logger
|
|
|
|
// Buffer for logging content outputs.
|
|
Buffer *bytes.Buffer
|
|
|
|
// (ReadOnly) Logging time, which is the time that logging triggers.
|
|
Time time.Time
|
|
|
|
// Formatted time string for output, like "2016-01-09 12:00:00".
|
|
TimeFormat string
|
|
|
|
// (ReadOnly) Using color constant value, like COLOR_RED, COLOR_BLUE, etc.
|
|
// Example: 34
|
|
Color int
|
|
|
|
// (ReadOnly) Using level, like LEVEL_INFO, LEVEL_ERRO, etc.
|
|
// Example: 256
|
|
Level int
|
|
|
|
// Formatted level string for output, like "DEBU", "ERRO", etc.
|
|
// Example: ERRO
|
|
LevelFormat string
|
|
|
|
// The source function name that calls logging, only available if F_CALLER_FN set.
|
|
CallerFunc string
|
|
|
|
// The source file path and its line number that calls logging,
|
|
// only available if F_FILE_SHORT or F_FILE_LONG set.
|
|
CallerPath string
|
|
|
|
// The retrieved context value string from context, only available if Config.CtxKeys configured.
|
|
// It's empty if no Config.CtxKeys configured.
|
|
CtxStr string
|
|
|
|
// Trace id, only available if OpenTelemetry is enabled, or else it's an empty string.
|
|
TraceId string
|
|
|
|
// Custom prefix string in logging content header part.
|
|
// Note that, it takes no effect if HeaderPrint is disabled.
|
|
Prefix string
|
|
|
|
// Custom logging content for logging.
|
|
Content string
|
|
|
|
// The passed un-formatted values array to logger.
|
|
Values []any
|
|
|
|
// Stack string produced by logger, only available if Config.StStatus configured.
|
|
// Note that there are usually multiple lines in stack content.
|
|
Stack string
|
|
|
|
// IsAsync marks it is in asynchronous logging.
|
|
IsAsync bool
|
|
}
|
|
|
|
type internalHandlerInfo struct {
|
|
index int // Middleware handling index for internal usage.
|
|
handlers []Handler // Handler array calling bu index.
|
|
}
|
|
|
|
// defaultHandler is the default handler for package.
|
|
var defaultHandler Handler
|
|
|
|
// doFinalPrint is a handler for logging content printing.
|
|
// This handler outputs logging content to file/stdout/write if any of them configured.
|
|
func doFinalPrint(ctx context.Context, in *HandlerInput) {
|
|
buffer := in.Logger.doFinalPrint(ctx, in)
|
|
if in.Buffer.Len() == 0 {
|
|
in.Buffer = buffer
|
|
}
|
|
}
|
|
|
|
// SetDefaultHandler sets default handler for package.
|
|
func SetDefaultHandler(handler Handler) {
|
|
defaultHandler = handler
|
|
}
|
|
|
|
// GetDefaultHandler returns the default handler of package.
|
|
func GetDefaultHandler() Handler {
|
|
return defaultHandler
|
|
}
|
|
|
|
// Next calls the next logging handler in middleware way.
|
|
func (in *HandlerInput) Next(ctx context.Context) {
|
|
in.index++
|
|
if in.index < len(in.handlers) {
|
|
in.handlers[in.index](ctx, in)
|
|
}
|
|
}
|
|
|
|
// String returns the logging content formatted by default logging handler.
|
|
func (in *HandlerInput) String(withColor ...bool) string {
|
|
formatWithColor := false
|
|
if len(withColor) > 0 {
|
|
formatWithColor = withColor[0]
|
|
}
|
|
return in.getDefaultBuffer(formatWithColor).String()
|
|
}
|
|
|
|
// ValuesContent converts and returns values as string content.
|
|
func (in *HandlerInput) ValuesContent() string {
|
|
var (
|
|
buffer = bytes.NewBuffer(nil)
|
|
valueContent string
|
|
)
|
|
for _, v := range in.Values {
|
|
valueContent = gconv.String(v)
|
|
if len(valueContent) == 0 {
|
|
continue
|
|
}
|
|
if buffer.Len() == 0 {
|
|
buffer.WriteString(valueContent)
|
|
continue
|
|
}
|
|
if buffer.Bytes()[buffer.Len()-1] != '\n' {
|
|
buffer.WriteString(" " + valueContent)
|
|
continue
|
|
}
|
|
// Remove one blank line(\n\n).
|
|
if valueContent[0] == '\n' {
|
|
valueContent = valueContent[1:]
|
|
}
|
|
buffer.WriteString(valueContent)
|
|
}
|
|
return buffer.String()
|
|
}
|
|
|
|
func (in *HandlerInput) getDefaultBuffer(withColor bool) *bytes.Buffer {
|
|
buffer := bytes.NewBuffer(nil)
|
|
if in.Logger.config.HeaderPrint {
|
|
if in.TimeFormat != "" {
|
|
buffer.WriteString(in.TimeFormat)
|
|
}
|
|
if in.Logger.config.LevelPrint && in.LevelFormat != "" {
|
|
var levelStr = "[" + in.LevelFormat + "]"
|
|
if withColor {
|
|
in.addStringToBuffer(buffer, in.Logger.getColoredStr(
|
|
in.Logger.getColorByLevel(in.Level), levelStr,
|
|
))
|
|
} else {
|
|
in.addStringToBuffer(buffer, levelStr)
|
|
}
|
|
}
|
|
}
|
|
if in.TraceId != "" {
|
|
in.addStringToBuffer(buffer, "{"+in.TraceId+"}")
|
|
}
|
|
if in.CtxStr != "" {
|
|
in.addStringToBuffer(buffer, "{"+in.CtxStr+"}")
|
|
}
|
|
if in.Logger.config.HeaderPrint {
|
|
if in.Prefix != "" {
|
|
in.addStringToBuffer(buffer, in.Prefix)
|
|
}
|
|
if in.CallerFunc != "" {
|
|
in.addStringToBuffer(buffer, in.CallerFunc)
|
|
}
|
|
if in.CallerPath != "" {
|
|
in.addStringToBuffer(buffer, in.CallerPath)
|
|
}
|
|
}
|
|
|
|
if in.Content != "" {
|
|
in.addStringToBuffer(buffer, in.Content)
|
|
}
|
|
|
|
if len(in.Values) > 0 {
|
|
in.addStringToBuffer(buffer, in.ValuesContent())
|
|
}
|
|
|
|
if in.Stack != "" {
|
|
in.addStringToBuffer(buffer, "\nStack:\n"+in.Stack)
|
|
}
|
|
// avoid a single space at the end of a line.
|
|
buffer.WriteString("\n")
|
|
return buffer
|
|
}
|
|
|
|
func (in *HandlerInput) getRealBuffer(withColor bool) *bytes.Buffer {
|
|
if in.Buffer.Len() > 0 {
|
|
return in.Buffer
|
|
}
|
|
return in.getDefaultBuffer(withColor)
|
|
}
|
|
|
|
func (in *HandlerInput) addStringToBuffer(buffer *bytes.Buffer, strings ...string) {
|
|
for _, s := range strings {
|
|
if buffer.Len() > 0 {
|
|
buffer.WriteByte(' ')
|
|
}
|
|
buffer.WriteString(s)
|
|
}
|
|
}
|