Go错误处理库cockroachdb/errors

简介

CockroachDB errors 是Go标准库errors、github.com/pkg/errors的通用替代品

特性

  • 提供简单错误信息和详细堆栈跟踪信息
  • 添加消息前缀
  • 次要错误
  • 检查错误类型
  • 支持自定义错误详细信息

使用

基本使用

import "github.com/cockroachdb/errors"

err := errors.New("msg")
fmt.Println(err)

运行,输出结果:

$ msg

错误堆栈跟踪

err := errors.New("msg")
fmt.Printf("%+v", err)

运行,输出结果:

$ msg
(1) attached stack trace
  -- stack trace:
  | main.main
  |     /Your Path/main.go:37
  | runtime.main
  |     /usr/local/Cellar/go/1.16/libexec/src/runtime/proc.go:225
  | runtime.goexit
  |     /usr/local/Cellar/go/1.16/libexec/src/runtime/asm_amd64.s:1371
Wraps: (2) err
Error types: (1) *withstack.withStack (2) *errutil.leafError% 

添加消息前缀

err1 := errors.New("msg")
err2 := errors.Wrap(err1, "another err")
fmt.Printf("%+v", err2)

运行,输出结果:

another err: msg
(1) attached stack trace
  -- stack trace:
  | main.main
  |     /YourPath/main.go:38
  | [...repeated from below...]
Wraps: (2) another err
Wraps: (3) attached stack trace
  -- stack trace:
  | main.main
  |     /YourPath/main.go:37
  | runtime.main
  |     /usr/local/Cellar/go/1.16/libexec/src/runtime/proc.go:225
  | runtime.goexit
  |     /usr/local/Cellar/go/1.16/libexec/src/runtime/asm_amd64.s:1371
Wraps: (4) err
Error types: (1) *withstack.withStack (2) *errutil.withPrefix (3) *withstack.withStack (4) *errutil.leafError% 

检查错误类型

// CustomErr 实现error interface
type CustomErr struct{}

func (*CustomErr) Error() string {
    return "custom err"
}

err1 := &CustomErr{}
err2 := errors.Wrap(err1, "msg")
err3 := errors.New("msg")
fmt.Println(errors.Is(err2, err1), errors.Is(err3, err1), errors.Is(err2, err1))

运行,输出结果:

true false true

在go1.13以后错误都是链表,这个方法可以很方便的检查错误类型

次要错误

err := errors.New("msg")
err2 := errors.WithSecondaryError(err, errors.New("secondary err"))
fmt.Printf("%+v", err2)

运行,输出结果:

$ msg
(1) secondary error attachment
  | secondary err
  | (1) attached stack trace
  |   -- stack trace:
  |   | main.main
  |   |         /YourPath/main.go:38
  |   | runtime.main
  |   |         /usr/local/Cellar/go/1.16/libexec/src/runtime/proc.go:225
  |   | runtime.goexit
  |   |         /usr/local/Cellar/go/1.16/libexec/src/runtime/asm_amd64.s:1371
  | Wraps: (2) secondary err
  | Error types: (1) *withstack.withStack (2) *errutil.leafError
Wraps: (2) attached stack trace
  -- stack trace:
  | main.main
  |     /YourPath/main.go:37
  | runtime.main
  |     /usr/local/Cellar/go/1.16/libexec/src/runtime/proc.go:225
  | runtime.goexit
  |     /usr/local/Cellar/go/1.16/libexec/src/runtime/asm_amd64.s:1371
Wraps: (3) err
Error types: (1) *secondary.withSecondaryError (2) *withstack.withStack (3) *errutil.leafError

次要错误一般是在处理错误过程中产生的错误

自定义详细错误信息

type CustomErr struct {
    Cause error
    Code  int
}

func (*CustomErr) Error() string {
    return "custom err"
}

func (c *CustomErr) Format(s fmt.State, verb rune) { errors.FormatError(c, s, verb) }

func (c *CustomErr) SafeFormatError(p errors.Printer) (next error) {
    if p.Detail() {
    p.Printf("http code: %d", errors.Safe(c.Code))
    }
    return c.Cause
}

err := errors.New("msg")
err1 := &CustomErr{
    Cause: err,
    Code:  500,
}
fmt.Printf("%+v", err1)

运行,结果:

(1) http code: 500
Error types: (1) *main.CustomErr

总结

Go1.13开始将错误视为链表,对Unwrap()方法进行标准化,并使用了Is()和As()函数增强了包,但是fmt.Print不知道如何打印这种新形式的错误, CockroachDB error 库在错误打印方面花费了大量精力,我们可以使用他来更好的输出错误信息。

更多用法参考官方API

京ICP备16046576号-1