golang框架gin的日志處理和zap lumberjack日志使用方式
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ù)
這篇文章主要為大家詳細(xì)介紹了Go語(yǔ)言如何結(jié)合正則表達(dá)式實(shí)現(xiàn)高效獲取數(shù)據(jù),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2025-04-04試了下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-07Golang編程實(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)換的原因,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-03-03Golang設(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-01xorm根據(jù)數(shù)據(jù)庫(kù)生成go model文件的操作
這篇文章主要介紹了xorm根據(jù)數(shù)據(jù)庫(kù)生成go model文件的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-12-12Golang語(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