错误堆栈

标准库的 error 错误实现比较简单,无法进行堆栈追溯,对于产生错误时的上层调用者来讲不是很友好,无法获得错误的调用链详细信息。 gerror 支持错误堆栈记录,通过 New/NewfWrap/Wrapf, WrapSkip/WrapSkipf 等方法均会自动记录当前错误产生时的堆栈信息。

示例:

  1. package main
  2. import (
  3. "fmt"
  4. "github.com/gogf/gf/v2/errors/gerror"
  5. )
  6. func OpenFile() error {
  7. return gerror.New("permission denied")
  8. }
  9. func OpenConfig() error {
  10. return gerror.Wrap(OpenFile(), "configuration file opening failed")
  11. }
  12. func ReadConfig() error {
  13. return gerror.Wrap(OpenConfig(), "reading configuration failed")
  14. }
  15. func main() {
  16. fmt.Printf("%+v", ReadConfig())
  17. }

执行后,输出示例:

  1. reading configuration failed: configuration file opening failed: permission denied
  2. 1. reading configuration failed
  3. 1). main.ReadConfig
  4. /Users/john/Workspace/Go/GOPATH/src/github.com/gogf/gf/v2/geg/errors/gerror/gerror2.go:18
  5. 2). main.main
  6. /Users/john/Workspace/Go/GOPATH/src/github.com/gogf/gf/v2/geg/errors/gerror/gerror2.go:25
  7. 2. configuration file opening failed
  8. 1). main.OpenConfig
  9. /Users/john/Workspace/Go/GOPATH/src/github.com/gogf/gf/v2/geg/errors/gerror/gerror2.go:14
  10. 2). main.ReadConfig
  11. /Users/john/Workspace/Go/GOPATH/src/github.com/gogf/gf/v2/geg/errors/gerror/gerror2.go:18
  12. 3). main.main
  13. /Users/john/Workspace/Go/GOPATH/src/github.com/gogf/gf/v2/geg/errors/gerror/gerror2.go:25
  14. 3. permission denied
  15. 1). main.OpenFile
  16. /Users/john/Workspace/Go/GOPATH/src/github.com/gogf/gf/v2/geg/errors/gerror/gerror2.go:10
  17. 2). main.OpenConfig
  18. /Users/john/Workspace/Go/GOPATH/src/github.com/gogf/gf/v2/geg/errors/gerror/gerror2.go:14
  19. 3). main.ReadConfig
  20. /Users/john/Workspace/Go/GOPATH/src/github.com/gogf/gf/v2/geg/errors/gerror/gerror2.go:18
  21. 4). main.main
  22. /Users/john/Workspace/Go/GOPATH/src/github.com/gogf/gf/v2/geg/errors/gerror/gerror2.go:25

错误处理-堆栈特性 - 图1提示

其中打印的错误堆栈路径为笔者本地执行的代码路径。

可以看到,调用端可以通过 Wrap 方法将底层的错误信息进行层级叠加,并且包含完整的错误堆栈信息。

HasStack 判断错误是否带堆栈

  • 说明:通过 HasStack 方法我们可以判断给定的 error 接口对象是否实现(包含)了堆栈信息。

  • 格式:

    1. // HasStack checks and reports whether `err` implemented interface `gerror.IStack`.
    2. HasStack(err error) bool
  • 示例:

    1. func ExampleHasStack() {
    2. err1 := errors.New("sql error")
    3. err2 := gerror.New("write error")
    4. fmt.Println(gerror.HasStack(err1))
    5. fmt.Println(gerror.HasStack(err2))
    6. // Output:
    7. // false
    8. // true
    9. }

错误处理-堆栈特性 - 图2注意

目前该方法是通过判断错误对象是否实现gerror.IStack接口来实现的,如果错误对象未实现该接口那么将会无法识别其是否带有堆栈信息。

Stack 获取堆栈信息

  • 说明:通过 Stack 方法我们可以获得 error 对象的完整堆栈信息,返回堆栈列表字符串。

  • 格式:

    1. // Stack returns the stack callers as string.
    2. // It returns the error string directly if the `err` does not support stacks.
    3. Stack(err error) string
  • 示例:

    1. func ExampleStack() {
    2. var err error
    3. err = errors.New("sql error")
    4. err = gerror.Wrap(err, "adding failed")
    5. err = gerror.Wrap(err, "api calling failed")
    6. fmt.Println(gerror.Stack(err))
    7. }

    执行后,输出示例信息:

    1. 1. api calling failed
    2. 1). main.main
    3. /Users/john/Workspace/Go/GOPATH/src/github.com/gogf/gf/.example/other/test.go:14
    4. 2. adding failed
    5. 1). main.main
    6. /Users/john/Workspace/Go/GOPATH/src/github.com/gogf/gf/.example/other/test.go:13
    7. 3. sql error

错误处理-堆栈特性 - 图3注意

目前该方法是通过获取错误对象实现的gerror.IStack接口来实现的,如果错误对象未实现该接口那么该方法将会返回空字符串。

Current 获取当前 error

  • 说明: Current 方法用于获取当前层级的错误信息,通过 error 接口对象返回。

  • 格式:

    1. // Current creates and returns the current level error.
    2. // It returns nil if current level error is nil.
    3. Current(err error) error
  • 示例:

    1. func ExampleCurrent() {
    2. var err error
    3. err = errors.New("sql error")
    4. err = gerror.Wrap(err, "adding failed")
    5. err = gerror.Wrap(err, "api calling failed")
    6. fmt.Println(err)
    7. fmt.Println(gerror.Current(err))
    8. // Output:
    9. // api calling failed: adding failed: sql error
    10. // api calling failed
    11. }

Unwrap 获取下一层 error

  • 说明: Unwrap 方法用于获取层级错误的下一级错误 error 接口对象。当下一层级不存在时,返回 nil

  • 格式:

    1. // Unwrap returns the next level error.
    2. // It returns nil if current level error or the next level error is nil.
    3. Unwrap(err error) error
  • 示例1:简单的错误层级访问示例。

    1. func ExampleUnwrap() {
    2. var err error
    3. err = errors.New("sql error")
    4. err = gerror.Wrap(err, "adding failed")
    5. err = gerror.Wrap(err, "api calling failed")
    6. fmt.Println(err)
    7. err = gerror.Unwrap(err)
    8. fmt.Println(err)
    9. err = gerror.Unwrap(err)
    10. fmt.Println(err)
    11. // Output:
    12. // api calling failed: adding failed: sql error
    13. // adding failed: sql error
    14. // sql error
    15. }
  • 示例2:常见遍历逻辑代码示例。

    1. func IsGrpcErrorNotFound(err error) bool {
    2. if err != nil {
    3. for e := err; e != nil; e = gerror.Unwrap(e) {
    4. if s, ok := status.FromError(e); ok && s != nil && s.Code() == codes.NotFound {
    5. return true
    6. }
    7. }
    8. }
    9. return false
    10. }

Cause 获取根错误 error

  • 说明:通过 Cause 方法我们可以获得 error 对象的根错误信息(原始错误)。

  • 格式:

    1. // Cause returns the root cause error of `err`.
    2. Cause(err error) error
  • 示例:

    1. package main
    2. import (
    3. "fmt"
    4. "github.com/gogf/gf/v2/errors/gerror"
    5. )
    6. func OpenFile() error {
    7. return gerror.New("permission denied")
    8. }
    9. func OpenConfig() error {
    10. return gerror.Wrap(OpenFile(), "configuration file opening failed")
    11. }
    12. func ReadConfig() error {
    13. return gerror.Wrap(OpenConfig(), "reading configuration failed")
    14. }
    15. func main() {
    16. fmt.Println(gerror.Cause(ReadConfig()))
    17. }
    18. // Output:
    19. // permission denied