欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

golang框架gin的日志處理和zap lumberjack日志使用方式

 更新時(shí)間:2024年01月17日 08:56:25   作者:可以吧可以吧  
這篇文章主要介紹了golang框架gin的日志處理和zap lumberjack日志使用方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教

gin框架好用,輪子也多,我也來(lái)豐富下內(nèi)容,golang框架gin的日志處理和zap lumberjack日志使用

gin自帶日志

新建logger.go

package logs
 
import (
	"fmt"
	"github.com/gin-gonic/gin"
	rotatelogs "github.com/lestrrat-go/file-rotatelogs"
	"github.com/rifflock/lfshook"
	"github.com/sirupsen/logrus"
	"os"
	"runtime"
	"time"
	"upload/config"
)
 
var ostype = runtime.GOOS
 
// 日志記錄到文件
func LoggerToFile() gin.HandlerFunc {
	logFileName := config.LOG_FILE_NAME
	fileName := config.LOG_FILE_PATH_LINUX + "/" + logFileName
 
	if ostype == "windows" {
		fileName = config.LOG_FILE_PATH_WIN + "\\" + logFileName
	} else if ostype == "linux" {
		fileName = config.LOG_FILE_PATH_LINUX + "/" + logFileName
	}
 
	 日志文件
	//fileName := path.Join(logFilePath, logFileName)
 
	// 寫入文件
	src, err := os.OpenFile(fileName, os.O_APPEND|os.O_WRONLY, os.ModeAppend)
	if err != nil {
		fmt.Println("loggerToFile err==>", err)
	}
 
	// 實(shí)例化
	logger := logrus.New()
 
	// 設(shè)置輸出
	logger.Out = src
 
	// 設(shè)置日志級(jí)別
	logger.SetLevel(logrus.DebugLevel)
 
	// 設(shè)置 rotatelogs
	logWriter, err := rotatelogs.New(
		// 分割后的文件名稱
		fileName+".%Y%m%d.log",
 
		// 生成軟鏈,指向最新日志文件
		rotatelogs.WithLinkName(fileName),
 
		// 設(shè)置最大保存時(shí)間(7天)
		rotatelogs.WithMaxAge(7*24*time.Hour),
 
		// 設(shè)置日志切割時(shí)間間隔(1天)
		rotatelogs.WithRotationTime(24*time.Hour),
	)
 
	writeMap := lfshook.WriterMap{
		logrus.InfoLevel:  logWriter,
		logrus.FatalLevel: logWriter,
		logrus.DebugLevel: logWriter,
		logrus.WarnLevel:  logWriter,
		logrus.ErrorLevel: logWriter,
		logrus.PanicLevel: logWriter,
	}
 
	lfHook := lfshook.NewHook(writeMap, &logrus.JSONFormatter{
		TimestampFormat: "2006-01-02 15:04:05",
	})
 
	// 新增鉤子
	logger.AddHook(lfHook)
 
	return func(c *gin.Context) {
		// 開(kāi)始時(shí)間
		startTime := time.Now()
 
		// 處理請(qǐng)求
		c.Next()
 
		// 結(jié)束時(shí)間
		endTime := time.Now()
 
		// 執(zhí)行時(shí)間
		latencyTime := endTime.Sub(startTime)
 
		// 請(qǐng)求方式
		reqMethod := c.Request.Method
 
		// 請(qǐng)求路由
		reqUri := c.Request.RequestURI
 
		// 狀態(tài)碼
		statusCode := c.Writer.Status()
 
		// 請(qǐng)求IP
		clientIP := c.ClientIP()
 
		// 日志格式
		logger.WithFields(logrus.Fields{
			"status_code":  statusCode,
			"latency_time": latencyTime,
			"client_ip":    clientIP,
			"req_method":   reqMethod,
			"req_uri":      reqUri,
		}).Info()
	}
}
 
// 日志記錄到 MongoDB
func LoggerToMongo() gin.HandlerFunc {
	return func(c *gin.Context) {
 
	}
}
 
// 日志記錄到 ES
func LoggerToES() gin.HandlerFunc {
	return func(c *gin.Context) {
 
	}
}
 
// 日志記錄到 MQ
func LoggerToMQ() gin.HandlerFunc {
	return func(c *gin.Context) {
 
	}
}

mian.go 引用

engine.Use(middleware.LoggerToFile())

配置config/config.go

package config
const (
	//[log]
	LOG_FILE_PATH_LINUX = "/home/data/logs/upload/logs"
	LOG_FILE_PATH_WIN   = "\\home\\data\\logs\\upload\\logs\\"
	LOG_FILE_NAME       = "system.log"
)
 
var config = new(Config)
 
 
func Get() Config {
	return *config
}

zap lumberjack接管gin日志

新建logger.go

package logs
 
import (
	"github.com/gin-gonic/gin"
	"github.com/natefinch/lumberjack"
	"go.uber.org/zap"
	"go.uber.org/zap/zapcore"
	"net"
	"net/http/httputil"
	"os"
	"runtime/debug"
	"strings"
	"time"
)
 
// 1 定義一下logger使用的常量
const (
	mode     = "dev"                   //開(kāi)發(fā)模式
	filename = "logs/logs.log" // 日志存放路徑
	//level       = "debug"       // 日志級(jí)別
	level       = zapcore.DebugLevel // 日志級(jí)別
	max_size    = 200                //最大存儲(chǔ)大小
	max_age     = 30                 //最大存儲(chǔ)時(shí)間
	max_backups = 7                  //#備份數(shù)量
)
 
// 2 初始化Logger對(duì)象
func InitLogger() (err error) {
	// 創(chuàng)建Core三大件,進(jìn)行初始化
	writeSyncer := getLogWriter(filename, max_size, max_backups, max_age)
	encoder := getEncoder()
	// 創(chuàng)建核心-->如果是dev模式,就在控制臺(tái)和文件都打印,否則就只寫到文件中
	var core zapcore.Core
	if mode == "dev" {
		// 開(kāi)發(fā)模式,日志輸出到終端
		consoleEncoder := zapcore.NewConsoleEncoder(zap.NewDevelopmentEncoderConfig())
		// NewTee創(chuàng)建一個(gè)核心,將日志條目復(fù)制到兩個(gè)或多個(gè)底層核心中。
		core = zapcore.NewTee(
			zapcore.NewCore(encoder, writeSyncer, level),
			zapcore.NewCore(consoleEncoder, zapcore.Lock(os.Stdout), level),
		)
	} else {
		core = zapcore.NewCore(encoder, writeSyncer, level)
	}
 
	//core := zapcore.NewCore(encoder, writeSyncer, level)
	// 創(chuàng)建 logger 對(duì)象
	log := zap.New(core, zap.AddCaller())
	// 替換全局的 logger, 后續(xù)在其他包中只需使用zap.L()調(diào)用即可
	zap.ReplaceGlobals(log)
	return
}
 
// 獲取Encoder,給初始化logger使用的
func getEncoder() zapcore.Encoder {
	// 使用zap提供的 NewProductionEncoderConfig
	encoderConfig := zap.NewProductionEncoderConfig()
	// 設(shè)置時(shí)間格式
	encoderConfig.EncodeTime = zapcore.ISO8601TimeEncoder
	// 時(shí)間的key
	encoderConfig.TimeKey = "time"
	// 級(jí)別
	encoderConfig.EncodeLevel = zapcore.CapitalLevelEncoder
	// 顯示調(diào)用者信息
	encoderConfig.EncodeCaller = zapcore.ShortCallerEncoder
	// 返回json 格式的 日志編輯器
	return zapcore.NewJSONEncoder(encoderConfig)
}
 
// 獲取切割的問(wèn)題,給初始化logger使用的
func getLogWriter(filename string, maxSize, maxBackup, maxAge int) zapcore.WriteSyncer {
	// 使用 lumberjack 歸檔切片日志
	lumberJackLogger := &lumberjack.Logger{
		Filename:   filename,
		MaxSize:    maxSize,
		MaxBackups: maxBackup,
		MaxAge:     maxAge,
	}
	return zapcore.AddSync(lumberJackLogger)
}
 
// GinLogger 用于替換gin框架的Logger中間件,不傳參數(shù),直接這樣寫
func GinLogger(c *gin.Context) {
	logger := zap.L()
	start := time.Now()
	path := c.Request.URL.Path
	query := c.Request.URL.RawQuery
	c.Next() // 執(zhí)行視圖函數(shù)
	// 視圖函數(shù)執(zhí)行完成,統(tǒng)計(jì)時(shí)間,記錄日志
	cost := time.Since(start)
	logger.Info(path,
		zap.Int("status", c.Writer.Status()),
		zap.String("method", c.Request.Method),
		zap.String("path", path),
		zap.String("query", query),
		zap.String("ip", c.ClientIP()),
		zap.String("user-agent", c.Request.UserAgent()),
		zap.String("errors", c.Errors.ByType(gin.ErrorTypePrivate).String()),
		zap.Duration("cost", cost),
	)
 
}
 
// GinRecovery 用于替換gin框架的Recovery中間件,因?yàn)閭魅雲(yún)?shù),再包一層
func GinRecovery(stack bool) gin.HandlerFunc {
	logger := zap.L()
	return func(c *gin.Context) {
		defer func() {
			// defer 延遲調(diào)用,出了異常,處理并恢復(fù)異常,記錄日志
			if err := recover(); err != nil {
				//  這個(gè)不必須,檢查是否存在斷開(kāi)的連接(broken pipe或者connection reset by peer)---------開(kāi)始--------
				var brokenPipe bool
				if ne, ok := err.(*net.OpError); ok {
					if se, ok := ne.Err.(*os.SyscallError); ok {
						if strings.Contains(strings.ToLower(se.Error()), "broken pipe") || strings.Contains(strings.ToLower(se.Error()), "connection reset by peer") {
							brokenPipe = true
						}
					}
				}
				//httputil包預(yù)先準(zhǔn)備好的DumpRequest方法
				httpRequest, _ := httputil.DumpRequest(c.Request, false)
				if brokenPipe {
					logger.Error(c.Request.URL.Path,
						zap.Any("error", err),
						zap.String("request", string(httpRequest)),
					)
					// 如果連接已斷開(kāi),我們無(wú)法向其寫入狀態(tài)
					c.Error(err.(error))
					c.Abort()
					return
				}
				//  這個(gè)不必須,檢查是否存在斷開(kāi)的連接(broken pipe或者connection reset by peer)---------結(jié)束--------
 
				// 是否打印堆棧信息,使用的是debug.Stack(),傳入false,在日志中就沒(méi)有堆棧信息
				if stack {
					logger.Error("[Recovery from panic]",
						zap.Any("error", err),
						zap.String("request", string(httpRequest)),
						zap.String("stack", string(debug.Stack())),
					)
				} else {
					logger.Error("[Recovery from panic]",
						zap.Any("error", err),
						zap.String("request", string(httpRequest)),
					)
				}
				// 有錯(cuò)誤,直接返回給前端錯(cuò)誤,前端直接報(bào)錯(cuò)
				//c.AbortWithStatus(http.StatusInternalServerError)
				// 該方式前端不報(bào)錯(cuò)
				c.String(200, "訪問(wèn)出錯(cuò)了")
			}
		}()
		c.Next()
	}
}

main.go 調(diào)用

	r := gin.New()
	// 初始化logger
	middleware.InitLogger()
	// 兩個(gè)中間件加入gin中
	r.Use(middleware.GinLogger, middleware.GinRecovery(true))

就可以看到日志了

2022-10-20T14:56:04.998+0800    INFO    middleware/logger.go:92 /ping   {"appid": "", "status": 200, "method": "GET", "path": "/ping", "query": "aa=1", "ip": "127.0.0.1", "user-agent": "PostmanRuntime/7.29.2", "errors": "", "cost":
"0s"}

總結(jié)

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。

相關(guān)文章

  • Go語(yǔ)言結(jié)合正則表達(dá)式實(shí)現(xiàn)高效獲取數(shù)據(jù)

    Go語(yǔ)言結(jié)合正則表達(dá)式實(shí)現(xiàn)高效獲取數(shù)據(jù)

    這篇文章主要為大家詳細(xì)介紹了Go語(yǔ)言如何結(jié)合正則表達(dá)式實(shí)現(xiàn)高效獲取數(shù)據(jù),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2025-04-04
  • Golang switch語(yǔ)句的具體使用

    Golang switch語(yǔ)句的具體使用

    switch 語(yǔ)句提供了一種簡(jiǎn)潔的方式來(lái)執(zhí)行多路分支選擇,本文主要介紹了Golang switch語(yǔ)句的具體使用,具有一定的參考價(jià)值,感興趣的可以了解一下
    2024-08-08
  • 試了下Golang實(shí)現(xiàn)try catch的方法

    試了下Golang實(shí)現(xiàn)try catch的方法

    雖然在使用Golang的時(shí)候發(fā)現(xiàn)沒(méi)有try catch這種錯(cuò)誤處理機(jī)制但是想一想golang作為一門優(yōu)雅的語(yǔ)言,似乎也是情理之中。那么夠怎么捕獲異常呢,本文就來(lái)介紹一下
    2021-07-07
  • Golang打包配置文件的實(shí)現(xiàn)示例

    Golang打包配置文件的實(shí)現(xiàn)示例

    本文主要介紹了Golang打包配置文件的實(shí)現(xiàn)示例,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-12-12
  • Golang編程實(shí)現(xiàn)刪除字符串中出現(xiàn)次數(shù)最少字符的方法

    Golang編程實(shí)現(xiàn)刪除字符串中出現(xiàn)次數(shù)最少字符的方法

    這篇文章主要介紹了Golang編程實(shí)現(xiàn)刪除字符串中出現(xiàn)次數(shù)最少字符的方法,涉及Go語(yǔ)言字符串遍歷與運(yùn)算相關(guān)操作技巧,需要的朋友可以參考下
    2017-01-01
  • 淺談Go語(yǔ)言不提供隱式數(shù)字轉(zhuǎn)換的原因

    淺談Go語(yǔ)言不提供隱式數(shù)字轉(zhuǎn)換的原因

    本文主要介紹了淺談Go語(yǔ)言不提供隱式數(shù)字轉(zhuǎn)換的原因,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-03-03
  • Golang設(shè)計(jì)模式之原型模式詳細(xì)講解

    Golang設(shè)計(jì)模式之原型模式詳細(xì)講解

    如果一個(gè)類的有非常多的屬性,層級(jí)還很深。每次構(gòu)造起來(lái),不管是直接構(gòu)造還是用建造者模式,都要對(duì)太多屬性進(jìn)行復(fù)制,那么有沒(méi)有一種好的方式讓我們創(chuàng)建太的時(shí)候使用體驗(yàn)更好一點(diǎn)呢? 今天的文章里就給大家介紹一種設(shè)計(jì)模式,來(lái)解決這個(gè)問(wèn)題
    2023-01-01
  • 一文帶你了解Golang中的并發(fā)性

    一文帶你了解Golang中的并發(fā)性

    并發(fā)是一個(gè)很酷的話題,一旦你掌握了它,就會(huì)成為一筆巨大的財(cái)富。所以本文就來(lái)和大家一起來(lái)聊聊Golang中的并發(fā)性,感興趣的可以了解一下
    2023-03-03
  • xorm根據(jù)數(shù)據(jù)庫(kù)生成go model文件的操作

    xorm根據(jù)數(shù)據(jù)庫(kù)生成go model文件的操作

    這篇文章主要介紹了xorm根據(jù)數(shù)據(jù)庫(kù)生成go model文件的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2020-12-12
  • Golang語(yǔ)言實(shí)現(xiàn)gRPC的具體使用

    Golang語(yǔ)言實(shí)現(xiàn)gRPC的具體使用

    本文主要介紹了Golang語(yǔ)言實(shí)現(xiàn)gRPC的具體使用,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2022-08-08

最新評(píng)論