網(wǎng)站首頁(yè) 編程語言 正文
問題描述
在使用 gin-swagger
的過程中, 經(jīng)常會(huì)發(fā)生因?yàn)槿鄙?json
等 tag
而導(dǎo)致的異常。 由于 gin-swagger
是并發(fā)執(zhí)行的, 輸出的日志本身是錯(cuò)位的。 這就導(dǎo)致無法定義是哪一個(gè)結(jié)構(gòu)體缺少 tag
導(dǎo)致的。
一般而言, 這種時(shí)候只能一個(gè)個(gè)點(diǎn)開去檢查。
解決方案
思路 : 要是每行日志帶當(dāng)前 goroutine_id
的話, 是不是就可以準(zhǔn)確定位到報(bào)錯(cuò)的 goroutine
他打印的日志是哪些了呢!
說做就做
實(shí)現(xiàn)思路
- 查看當(dāng)前日志是怎么打印的
發(fā)現(xiàn) gin-swagger
日志直接調(diào)用的 golang
的標(biāo)準(zhǔn)庫(kù) log
由于沒有對(duì)log初始化, 所以默認(rèn)使用的是 stdout
。
log.Printf("Picking operation from %s\n", aurora.Blue(funType.FullName()))
- 如果想要給日志中添加
goroutine_id
的話, 就需要在調(diào)用log.Printf
的時(shí)候獲取當(dāng)前goroutine
的 id , 所以首先要解決的是怎么獲取goroutine_id
的問題。
調(diào)研后發(fā)現(xiàn)了集中常見的獲取 goroutine_id
的方法:
2.1 通過棧信息解析后獲取
func GetGID() uint64 { b := make([]byte, 64) b = b[:runtime.Stack(b, false)] b = bytes.TrimPrefix(b, []byte("goroutine ")) b = b[:bytes.IndexByte(b, ' ')] n, _ := strconv.ParseUint(string(b), 10, 64) return n }
2.2 修改 Go 源碼獲取
# src/runtime/runtime2.go func Goid() int64 { _g_ := getg() return _g_.goid }
2.3 通過 CGO 獲取
文件 id.c
#include "runtime.h" int64 ·Id(void) { return g->goid; }
文件 id.go
package id func Id() int64
另外還可以通過匯編獲取 goroutine_id
由于go的版本不同,goroutine的結(jié)構(gòu)也可能不同, 所以此處我直接調(diào)用一個(gè)開源實(shí)現(xiàn):
https://github.com/petermattis/goid
- 修改
gin-swagger
main.go
源碼, 修改log
Write
的實(shí)現(xiàn)
修改前
func main() { cmd.Execute() }
修改后
type Out os.File func (o Out) Write(b []byte) (int, error) { prefix := fmt.Sprintf("gid-%d: ", goid.Get()) all := make([]byte, len(b)+len(prefix)) all = []byte(prefix) all = append(all, b...) f := os.File(o) return f.Write(all) } func main() { var out Out out = Out(os.Stdout) log.SetOutput(out) cmd.Execute()
這樣修改后,每次 gin-swagger
調(diào)用 log
打印日志都時(shí)候, 都會(huì)使用我定義的 Write
方法, Write
方法中每次打印前都會(huì)先查詢出當(dāng)前的 goroutine_id
,然后作為日志的前綴。
修改后的日志效果
從中可以看到每行日志開頭,都已經(jīng)加上了 goroutine_id
。 只要使用 panic
goroutine
的 id
做一次 grep
, 即可篩選出需要的日志了,極大的方便了定位問題。
cat /tmp/gin-swagger.log | grep 7647
原文鏈接:https://juejin.cn/post/7075159546393542686
相關(guān)推薦
- 2022-10-27 python使用pika庫(kù)調(diào)用rabbitmq參數(shù)使用詳情_python
- 2022-07-15 在SQL?Server中使用子查詢更新語句_MsSql
- 2022-01-17 類組件與函數(shù)組件的區(qū)別 react中class創(chuàng)建的組件與function創(chuàng)建的組件有什么區(qū)別
- 2022-06-21 Android?使用flow實(shí)現(xiàn)倒計(jì)時(shí)的方式_Android
- 2022-10-28 Redis?RESP?協(xié)議實(shí)現(xiàn)實(shí)例詳解_Redis
- 2022-10-07 C語言順序查找算法介紹及示例_C 語言
- 2022-12-06 Python基礎(chǔ)之文件操作及光標(biāo)移動(dòng)詳解_python
- 2022-05-13 hiveserver2 連接報(bào):root is not allowed to impersonate
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運(yùn)行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲(chǔ)小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運(yùn)算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯(cuò)誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實(shí)現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡(jiǎn)單動(dòng)態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對(duì)象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊(duì)列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支