zap日志和gin使用zap实战演示

zap

效果:

[GIN_ZVCJ] 2021/08/31 - 13:08:01.478 [INFO] [config/ginLog.go:24] /test {“status”: 200, “method”: “GET”, “path”: “/test”, “query”: “”, “ip”: “127.0.0.1”, “user-agent”: “Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/15.6.1 Safari/605.1.15”, “errors”: “”, “cost”: 0.00010775}

日志配置:

func getZapConfig() zapcore.EncoderConfig {
	c := zap.NewProductionEncoderConfig()
	c.EncodeTime = func(t time.Time, pae zapcore.PrimitiveArrayEncoder) {
		pae.AppendString(t.Format(("[GIN_ZVCJ] " + "2006/01/02 - 15:04:05.000")))
	}
	c.EncodeLevel = func(l zapcore.Level, pae zapcore.PrimitiveArrayEncoder) {
		pae.AppendString("[" + l.CapitalString() + "]")
	}
	c.EncodeCaller = func(ec zapcore.EntryCaller, pae zapcore.PrimitiveArrayEncoder) {
		pae.AppendString("[" + ec.TrimmedPath() + "]")
	}
	return c
}

日志分割

func getZapWrite(filename string) zapcore.WriteSyncer {
	lumberjack := &lumberjack.Logger{
		Filename:   filename,
		MaxSize:    10,
		MaxAge:     30,
		MaxBackups: 5,
		Compress:   true,
	}
	return zapcore.AddSync(lumberjack)
}

创建core

func newZapCore(filename string, level zapcore.LevelEnabler) zapcore.Core {
	return zapcore.NewCore(getZapEncode(), getZapWrite(filename), level)
}

func getZapEncode() zapcore.Encoder {
	return zapcore.NewConsoleEncoder(getZapConfig())
}

日志初始化

func ZapInit() *zap.Logger {
	info := zap.LevelEnablerFunc(func(l zapcore.Level) bool {
		return l == zap.InfoLevel
	})
	debug := zap.LevelEnablerFunc(func(l zapcore.Level) bool {
		return l == zap.DebugLevel
	})
	warn := zap.LevelEnablerFunc(func(l zapcore.Level) bool {
		return l == zap.WarnLevel
	})
	error := zap.LevelEnablerFunc(func(l zapcore.Level) bool {
		return l >= zap.ErrorLevel
	})

	t := time.Now().Format("2006-01-02")
	cores := [...]zapcore.Core{
		newZapCore(fmt.Sprintf("%s/%s/info.log", "log", t), info),
		newZapCore(fmt.Sprintf("%s/%s/debug.log", "debug", t), debug),
		newZapCore(fmt.Sprintf("%s/%s/warn.log", "warn", t), warn),
		newZapCore(fmt.Sprintf("%s/%s/error.log", "error", t), error),
	}
	return zap.New(zapcore.NewTee(cores[:]...), zap.AddCaller())
}

gin使用zap日志

配置gin处理

// 日志处理
func GinLogger() gin.HandlerFunc {
	return func(c *gin.Context) {
		start := time.Now()
		path := c.Request.URL.Path
		query := c.Request.URL.RawQuery
		c.Next()
		cost := time.Since(start)
		global.GZV_LOG.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),
		)
	}
}

// 错误处理,也可以不写自己处理错误使用gin写好的错误处理
func GinRecovery(stack bool) gin.HandlerFunc {
	return func(c *gin.Context) {
		defer func() {
			if err := recover(); err != nil {
				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
						}
					}
				}

				httpRequest, _ := httputil.DumpRequest(c.Request, false)
				if brokenPipe {
					global.GZV_LOG.Error(c.Request.URL.Path,
						zap.Any("error", err),
						zap.String("request", string(httpRequest)),
					)
					c.Error(err.(error))
					c.Abort()
					return
				}

				if stack {
					global.GZV_LOG.Error("[Recovery from panic]",
						zap.Any("error", err),
						zap.String("request", string(httpRequest)),
						zap.String("stack", string(debug.Stack())),
					)
				} else {
					global.GZV_LOG.Error("[Recovery from panic]",
						zap.Any("error", err),
						zap.String("request", string(httpRequest)),
					)
				}
				c.AbortWithStatus(http.StatusInternalServerError)
			}
		}()
		c.Next()
	}
}

使用

e := gin.Default()
e.Use(config.GinRecovery(true), config.GinLogger())
// 如果没有写错误处理函数则:
e.Use(gin.Recovery(), config.GinLogger())
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值