Go 优雅的错误处理: 支持错误码, 错误堆栈, 错误链的工具库 - V2EX
V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
The Go Programming Language
http://golang.org/
Go Playground
Go Projects
Revel Web Framework
atVoid
V2EX    Go 编程语言

Go 优雅的错误处理: 支持错误码, 错误堆栈, 错链的工具库

  •  
  •   atVoid
    morrisxyang 2023-07-11 14:16:38 +08:00 1736 次点击
    这是一个创建于 832 天前的主题,其中的信息可能已经有所发展或是发生改变。

    https://github.com/morrisxyang/errors

    在业务开发中需要一个支持错误码对外返回, 堆栈打印等能力的错误工具库, 先开始使用 pkg/errors, 但该项目已经只读, 上次更新是几年前, 而且有一些点比如调整堆栈打印深度, 打印格式等能力没有支持, 后续根据业务的需要抽取了一个通用库, 做了一些优化, 详见下方.

    如果觉得有用欢迎 Star 和 PR, 有问题请直接提 issue

    errors

    Go Reference Static Badge Coverage Status Go Report Card Static Badge

    简单的支持错误堆栈, 错误码, 错误链的工具库:

    • 支持携带堆栈, 嵌套构造错误链

    • 支持携带错误码, 方便接口返回

    • 支持自定义堆栈打印深度和错误链打印格式

    • 使用 CallersFrames 替代 FuncForPC 生成堆栈, 避免特殊情况line number 错误等问题, 详见runtime: strongly encourage using CallersFrames over FuncForPC with Callers result

    • 简化堆栈信息, 一条链路多次Wrap操作只保留最深层堆栈, 只打印一次

    安装和文档

    安装使用 go get github.com/morrisxyang/errors

    文档地址是 https://pkg.go.dev/github.com/morrisxyang/errors

    快速开始

    构造错误链

    func a() error { err := b() err = Wrap(err, "a failed reason") return err } func b() error { err := c() err = Wrap(err, "b failed reason") return err } func c() error { _, err := os.Open("test") if err != nil { return WrapWithCode(err, 123, "c failed reason") } return nil } 

    打印错误信息, %+v会打印堆栈, %v只打印错误信息

    a failed reason Caused by: b failed reason Caused by: 123, c failed reason Caused by: open test: no such file or directory github.com/morrisxyang/errors.c /Users/morrisyang/Nutstore Files/go-proj/githuberrors/errors_test.go:94 github.com/morrisxyang/errors.b /Users/morrisyang/Nutstore Files/go-proj/githuberrors/errors_test.go:86 github.com/morrisxyang/errors.a /Users/morrisyang/Nutstore Files/go-proj/githuberrors/errors_test.go:80 ....堆栈信息省略 

    核心方法

    错误封装

    错误解析

    配置

    FAQ

    1. 多次 Wrap 错误会携带多次堆栈吗?

      可在调用链路上多次 Wrap, 添加说明信息, 但只有最深层的 Wrap 操作会设置堆栈, 继续 Wrap, return err 等操作不会影响堆栈信息

    2. 在链路中某个错误设置了合适的错误码, 然后继续 Wrap 时没有设置, 如何获取?

      建议在合适的清晰的时机设置有效的错误码, 可以使用EffectiveCode获取链路中外层第一个有效的非 0 错误码, 由于系统调用等情况, 同一链路中可能有多个错误携带错误码, 此时默认外层的错误码应该对外暴露, 屏蔽了内层的详细信息.

    4 条回复    2023-07-19 18:45:25 +08:00
    tairan2006
        1
    tairan2006  
       2023-07-11 17:49:23 +08:00
    标准库里面内置的就有 errors 啊
    atVoid
        2
    atVoid  
    OP
       2023-07-13 14:12:15 +08:00
    @tairan2006
    标准库的 errors 没有堆栈, 这里主要是增加了堆栈, 以及微服务常用的错误码等支持
    tairan2006
        3
    tairan2006  
       2023-07-13 16:22:19 +08:00   1
    @atVoid #2 我的意思是,你的包名能不能换一个…
    atVoid
        4
    atVoid  
    OP
       2023-07-19 18:45:25 +08:00
    @tairan2006 啊, 确实, 我知道了. 我想想
    关于     帮助文档     自助推广系统     博客     API     FAQ     Solana     2702 人在线   最高记录 6679       Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 25ms UTC 14:58 PVG 22:58 LAX 07:58 JFK 10:58
    Do have faith in what you're doing.
    ubao msn snddm index pchome yahoo rakuten mypaper meadowduck bidyahoo youbao zxmzxm asda bnvcg cvbfg dfscv mmhjk xxddc yybgb zznbn ccubao uaitu acv GXCV ET GDG YH FG BCVB FJFH CBRE CBC GDG ET54 WRWR RWER WREW WRWER RWER SDG EW SF DSFSF fbbs ubao fhd dfg ewr dg df ewwr ewwr et ruyut utut dfg fgd gdfgt etg dfgt dfgd ert4 gd fgg wr 235 wer3 we vsdf sdf gdf ert xcv sdf rwer hfd dfg cvb rwf afb dfh jgh bmn lgh rty gfds cxv xcv xcs vdas fdf fgd cv sdf tert sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf shasha9178 shasha9178 shasha9178 shasha9178 shasha9178 liflif2 liflif2 liflif2 liflif2 liflif2 liblib3 liblib3 liblib3 liblib3 liblib3 zhazha444 zhazha444 zhazha444 zhazha444 zhazha444 dende5 dende denden denden2 denden21 fenfen9 fenf619 fen619 fenfe9 fe619 sdf sdf sdf sdf sdf zhazh90 zhazh0 zhaa50 zha90 zh590 zho zhoz zhozh zhozho zhozho2 lislis lls95 lili95 lils5 liss9 sdf0ty987 sdft876 sdft9876 sdf09876 sd0t9876 sdf0ty98 sdf0976 sdf0ty986 sdf0ty96 sdf0t76 sdf0876 df0ty98 sf0t876 sd0ty76 sdy76 sdf76 sdf0t76 sdf0ty9 sdf0ty98 sdf0ty987 sdf0ty98 sdf6676 sdf876 sd876 sd876 sdf6 sdf6 sdf9876 sdf0t sdf06 sdf0ty9776 sdf0ty9776 sdf0ty76 sdf8876 sdf0t sd6 sdf06 s688876 sd688 sdf86