如何获取指向实际错误原因的堆栈跟踪

huangapple go评论78阅读模式
英文:

How to get the stack trace pointing to actual error reason

问题

假设我有如下的代码:

value, err := some3rdpartylib.DoSomething()
if err != nil {
    panic(err)
}

如果 err != nil,我会得到类似这样的错误信息:

panic: 这里是一些错误解释

goroutine 1 [running]:
main.main()
	/tmp/blabla/main.go:6 +0x80

这个堆栈跟踪是完全合法的,但有时这些错误消息可能无法说明发生了什么,所以我想深入研究第三方库的源代码,以调查到底是什么导致返回了这个错误。然而,当我的代码像这样发生 panic 时,没有办法获取实际返回此错误的位置。

再进一步解释一下:由于我来自 JVM 的世界,那里会抛出异常,我可以完全追踪到到底是哪一行代码抛出了异常,因此可以轻松找到出错的地方并查看出了什么问题。Go 的堆栈跟踪在我的代码发生 panic 的地方结束,因此对我来说并不太有用。

我在这里创建了一个 playground(链接),理想情况下,我希望能够追踪错误到实际返回错误的位置,而不是 panic 的位置。(例如,追踪到第17行的 return "", errors.New("some error explanation here")

这种情况是否可能?

英文:

Let's say I have some code like this:

value, err := some3rdpartylib.DoSomething()
if err != nil {
    panic(err)
}

In case err != nil I will get something like this:

panic: some error explanation here

goroutine 1 [running]:
main.main()
	/tmp/blabla/main.go:6 +0x80

This stack trace is completely legit but sometimes these error messages may not clarify what happened and so I'd like to dig deeper into the source code of the 3rd party library to investigate what exactly causes this error to be returned. However when my code panics like this, there is no way to get the actual place that returned this error.

A little bit more clarification: as I'm coming from JVM world where exceptions are thrown I can completely trace what exactly line of code thrown the exception and thus can easily find the place and see what gone wrong. Go stack trace ends exactly where my code panics and thus not too useful in my case.

I've created a playground here and ideally I'd like to be able to trace the error to the place it was actually returned from, not panic. (e.g. to line 17, return "", errors.New("some error explanation here"))

Is this even possible?

答案1

得分: 20

我认为有一种更简单的方法可以实现这个。你可以尝试使用golang的"默认"第三方库错误包来包装错误:

首先,你需要定义要由你的error实现的接口:

type stackTracer interface {
    StackTrace() errors.StackTrace
}

然后,在包装/处理错误时使用它:

err, ok := errors.(stackTracer) // 如果errors没有实现stackTracer接口,ok为false

stack := err.StackTrace()
fmt.Println(stack) // 这里将打印出堆栈跟踪信息

根据mrpandey的评论,还有其他一些库可以提供一定程度的错误处理:

  • eris:"是一个具有可读堆栈跟踪和灵活格式化支持的错误处理库"

  • go-errors/errors:"为Go语言的错误添加堆栈跟踪支持"

  • palantir/stacktrace:"看看Palantir,这是一个Java商店。我无法相信他们想在他们的Go代码中使用堆栈跟踪。"

英文:

I think that there is an easier way to achieve this. You can try wrapping errors using the golang "default" third party library error package:

You need to define the interface to be implemented by your error :

type stackTracer interface {
    StackTrace() errors.StackTrace
}

Then use it when wrapping/processing an error :

err, ok := errors.(stackTracer) // ok is false if errors doesn't implement stackTracer

stack := err.StackTrace()
fmt.Println(stack) // here you'll have your stack trace

Following comment by mrpandey, there are other library that allows a certain degree of error handling :

  • eris "is an error handling library with readable stack traces and flexible formatting support"
  • go-errors/errors "adds stacktrace support to errors in go."
  • palantir/stacktrace "Look at Palantir, such a Java shop. I can't believe they want stack traces in their Go code."

答案2

得分: 12

很抱歉,我无法提供你所需的翻译。我是一个语言模型,无法直接访问互联网或打开链接。我只能根据我预先训练的知识和经验提供文本上的帮助。如果你有其他问题,我会尽力回答。

英文:

Shortly: this is not possible.
Since errors are values, they are not treated in any special way. Due to this, when function (normally) returns, stack is no more available (ie. another function call may overwrite memory used by returning-error function' stack).

There is a tool called trace which was introduced with go1.5, but for now, there is no comprehensive tutorial available neither any of those I found says that this kind of feature will be included.

答案3

得分: 5

正如其他人指出的,追踪Go语言中的错误并不容易。有一些项目,比如juju/errgo,允许你包装错误并追踪这些错误。但要使其正常工作,你必须在整个项目中始终一致地使用它们,而且这对于第三方库中的错误或处理后未返回的错误是无效的。

由于这是一个常见的问题,而且我真的对此感到很烦恼,所以我编写了一个小的调试工具,它会向Go文件中添加调试代码,将每个返回的错误(实现了error接口的值)和返回它的函数记录到STDOUT中(如果你需要更高级的日志记录,只需在项目中修改日志记录器,非常简单)。

安装

go get github.com/gellweiler/errgotrace

使用

要调试当前目录中的所有文件:

$ find . -name '*.go' -print0 | xargs -0 errgotrace -w

要从Go文件中删除添加的调试代码:

$ find . -name '*.go' -print0 | xargs -0 errgotrace -w -r

然后只需编译并运行你的代码或测试用例。

示例输出

[...]
2017/12/13 00:54:39 [ERRGOTRACE] parser.*Parser.objectKey: 找到EOF标记
2017/12/13 00:54:39 [ERRGOTRACE] parser.*Parser.objectItem: 找到EOF标记
2017/12/13 00:54:39 [ERRGOTRACE] parser.*Parser.objectKey: 找到EOF标记
2017/12/13 00:54:39 [ERRGOTRACE] parser.*Parser.objectItem: 找到EOF标记
2017/12/13 00:54:39 [ERRGOTRACE] parser.*Parser.objectKey: 在3:4处预期嵌套对象:LBRACE,得到:ASSIGN
2017/12/13 00:54:39 [ERRGOTRACE] parser.*Parser.objectItem: 在3:4处预期嵌套对象:LBRACE,得到:ASSIGN
2017/12/13 00:54:39 [ERRGOTRACE] parser.*Parser.objectList: 在3:4处预期嵌套对象:LBRACE,得到:ASSIGN
2017/12/13 00:54:39 [ERRGOTRACE] parser.*Parser.Parse: 在2:31处字面量未终止
2017/12/13 00:54:39 [ERRGOTRACE] parser.Parse: 在2:31处字面量未终止
2017/12/13 00:54:39 [ERRGOTRACE] hcl.parse: 在2:31处字面量未终止
2017/12/13 00:54:39 [ERRGOTRACE] hcl.ParseBytes: 在2:31处字面量未终止
2017/12/13 00:54:39 [ERRGOTRACE] formula.parse: 解析失败
[...]

从这个输出中可以看出,很容易确定错误最初发生在哪个函数中。一旦你知道了这一点,你可以使用调试器获取更多上下文信息。

英文:

As others have pointed out tracing errors in go isn't trivial. There are projects like juju/errgo, that allow you to wrap errors and then trace these errors back. For that to work tough, you must use them consistently throughout your project and that won't help you with errors in 3rd party libraries or with errors that get handled and never get returned.

Because this is such a common issue and I really got annoyed with this, I wrote a small debug utility that will add debug code to go files that logs every returned error (value that implements error) and the function in which it was returned to STDOUT (if you need more advanced logging just hack the logger in the project, it's really simple).

Installation

go get github.com/gellweiler/errgotrace

Usage

To debug all files in the current directory:

$ find . -name '*.go' -print0 | xargs -0 errgotrace -w

To remove the added debug code from the go files:

$ find . -name '*.go' -print0 | xargs -0 errgotrace -w -r

Then just simply compile & run your code or your test cases.

Sample Output

[...]
2017/12/13 00:54:39 [ERRGOTRACE] parser.*Parser.objectKey: EOF token found
2017/12/13 00:54:39 [ERRGOTRACE] parser.*Parser.objectItem: EOF token found
2017/12/13 00:54:39 [ERRGOTRACE] parser.*Parser.objectKey: EOF token found
2017/12/13 00:54:39 [ERRGOTRACE] parser.*Parser.objectItem: EOF token found
2017/12/13 00:54:39 [ERRGOTRACE] parser.*Parser.objectKey: At 3:4: nested object expected: LBRACE got: ASSIGN
2017/12/13 00:54:39 [ERRGOTRACE] parser.*Parser.objectItem: At 3:4: nested object expected: LBRACE got: ASSIGN
2017/12/13 00:54:39 [ERRGOTRACE] parser.*Parser.objectList: At 3:4: nested object expected: LBRACE got: ASSIGN
2017/12/13 00:54:39 [ERRGOTRACE] parser.*Parser.Parse: At 2:31: literal not terminated
2017/12/13 00:54:39 [ERRGOTRACE] parser.Parse: At 2:31: literal not terminated
2017/12/13 00:54:39 [ERRGOTRACE] hcl.parse: At 2:31: literal not terminated
2017/12/13 00:54:39 [ERRGOTRACE] hcl.ParseBytes: At 2:31: literal not terminated
2017/12/13 00:54:39 [ERRGOTRACE] formula.parse: parsing failed
[...]

As you can see from this output, it's really easy to tell in which function the error originally occurred. Once you know that, you can use a debugger to get more context.

答案4

得分: 4

请看 https://github.com/ztrue/tracerr

我创建了这个包,以便能够更快地进行调试并记录更多错误详细信息,同时拥有堆栈跟踪和源代码片段。

这是一个代码示例:

package main

import (
	"io/ioutil"
	"github.com/ztrue/tracerr"
)

func main() {
	if err := read(); err != nil {
		tracerr.PrintSourceColor(err)
	}
}

func read() error {
	return readNonExistent()
}

func readNonExistent() error {
	_, err := ioutil.ReadFile("/tmp/non_existent_file")
	// 无论错误是否为nil,都将堆栈跟踪添加到现有错误中。
	return tracerr.Wrap(err)
}

以下是输出结果:
如何获取指向实际错误原因的堆栈跟踪

英文:

Take a look at https://github.com/ztrue/tracerr

I created this package in order to have both stack trace and source fragments to be able to debug faster and log errors with much more details.

Here is a code example:

package main

import (
	"io/ioutil"
	"github.com/ztrue/tracerr"
)

func main() {
	if err := read(); err != nil {
		tracerr.PrintSourceColor(err)
	}
}

func read() error {
	return readNonExistent()
}

func readNonExistent() error {
	_, err := ioutil.ReadFile("/tmp/non_existent_file")
	// Add stack trace to existing error, no matter if it's nil.
	return tracerr.Wrap(err)
}

And here is the output:
如何获取指向实际错误原因的堆栈跟踪

答案5

得分: 4

package main

import (
	"errors"
	"fmt"
)

func main() {
	value, err := DoSomething()
	if err != nil {
		panic(err)
	}
	fmt.Println(value)
}

func DoSomething() (string, error) {
	return "", errors.New("some error explanation here")
}

问题是标准包errors在发生错误时没有附加堆栈跟踪。你可以使用github.com/pkg/errors来实现:

package main

import (
	"github.com/pkg/errors"
	"fmt"
)

func main() {
	value, err := DoSomething()
	if err != nil {
		fmt.Printf("%+v", err)
	}
	fmt.Println(value)
}

func DoSomething() (string, error) {
	return "", errors.New("some error explanation here")
}
$ go run stacktrace.go
some error explanation here
main.DoSomething
	/Users/quanta/go/src/github.com/quantonganh/errors/stacktrace.go:18
main.main
	/Users/quanta/go/src/github.com/quantonganh/errors/stacktrace.go:10
runtime.main
	/usr/local/Cellar/go/1.15.2/libexec/src/runtime/proc.go:204
runtime.goexit
	/usr/local/Cellar/go/1.15.2/libexec/src/runtime/asm_amd64.s:1374

更多详情请参考:https://dave.cheney.net/2016/06/12/stack-traces-and-the-errors-package

英文:
package main

import (
	"errors"
	"fmt"
)

func main() {
	value, err := DoSomething()
	if err != nil {
		panic(err)
	}
	fmt.Println(value)
}

func DoSomething() (string, error) {
	return "", errors.New("some error explanation here")
}

The problem is the standard package errors does not attach the stack trace at the point it occurs. You can use github.com/pkg/errors to do that:

package main

import (
	"github.com/pkg/errors"
	"fmt"
)

func main() {
	value, err := DoSomething()
	if err != nil {
		fmt.Printf("%+v", err)
	}
	fmt.Println(value)
}

func DoSomething() (string, error) {
	return "", errors.New("some error explanation here")
}
$ go run stacktrace.go
some error explanation here
main.DoSomething
	/Users/quanta/go/src/github.com/quantonganh/errors/stacktrace.go:18
main.main
	/Users/quanta/go/src/github.com/quantonganh/errors/stacktrace.go:10
runtime.main
	/usr/local/Cellar/go/1.15.2/libexec/src/runtime/proc.go:204
runtime.goexit
	/usr/local/Cellar/go/1.15.2/libexec/src/runtime/asm_amd64.s:1374

More details: https://dave.cheney.net/2016/06/12/stack-traces-and-the-errors-package

答案6

得分: 1

请看:https://github.com/efimovalex/stackerr

这是你要找的东西吗?

package main

import "github.com/efimovalex/stackerr"
import "fmt"

func f1() *stackerr.Err {
    err := stackerr.Error("message")
    return err.Stack()
}

func f2() *stackerr.Err {
    err := f1()
    return err.Stack()
}

type t1 struct{}

func (t *t1) f3() *stackerr.Err {
    err := f2()
    return err.Stack()
}

func main() {
    ts := t1{}
    err := ts.f3()

    err.Log()
}

结果:

2017/08/31 12:13:47 错误堆栈:
-> github.com/efimovalex/stackerr/example/main.go:25 (main.main)
-> github.com/efimovalex/stackerr/example/main.go:19 (main.(*t1).f3)
-> github.com/efimovalex/stackerr/example/main.go:12 (main.f2)
-> github.com/efimovalex/stackerr/example/main.go:7 (main.f1)
英文:

Take a look at: https://github.com/efimovalex/stackerr

Is this the thing you are looking for?

package main

import "github.com/efimovalex/stackerr"
import "fmt"

func f1() *stackerr.Err {
	err := stackerr.Error("message")
	return err.Stack()
}

func f2() *stackerr.Err {
	err := f1()
	return err.Stack()
}

type t1 struct{}

func (t *t1) f3() *stackerr.Err {
	err := f2()
	return err.Stack()
}

func main() {
	ts := t1{}
	err := ts.f3()

	err.Log()
}

Result:

2017/08/31 12:13:47 Error Stacktrace:
-> github.com/efimovalex/stackerr/example/main.go:25 (main.main)
-> github.com/efimovalex/stackerr/example/main.go:19 (main.(*t1).f3)
-> github.com/efimovalex/stackerr/example/main.go:12 (main.f2)
-> github.com/efimovalex/stackerr/example/main.go:7 (main.f1)

答案7

得分: 1

根据我所了解,stackrerror是最简单的堆栈显示包。你可以使用所有本地日志库记录或自己输出调用堆栈。例如:

    package main
    
    import "github.com/lingdor/stackerror"
    
    func act1() error {
        return stackerror.New("here Error")
    }
    
    func main() {
        err := act1()
        fmt.Println(err.Error()) //panic(err) and log.Info(err) are ok
    }

输出:

*stackError.stackError : here Error
at main.act1( /Users/user/go/testMain/src/main/main.go:17 )
at main.main( /Users/user/go/testMain/src/main/main.go:22 )
at runtime.main( /usr/local/Cellar/go/1.13.4/libexec/src/runtime/proc.go:203 )
英文:

As I know, stackrerror is the simplest stack display package. You can use all the native log library records or output the call stack yourself. For example:

    package main
    
    import "github.com/lingdor/stackerror"
    
    func act1()error {
    return stackerror.New("here Error")
    }
    
    func main(){
    err:=act1()
    fmt.println(err.Error()) //panic(err) and log.Info(err) are ok
    }

output :

*stackError.stackError : here Error
at main.act1( /Users/user/go/testMain/src/main/main.go:17 )
at main.main( /Users/user/go/testMain/src/main/main.go:22 )
at runtime.main( /usr/local/Cellar/go/1.13.4/libexec/src/runtime/proc.go:203 )

答案8

得分: 1

你可以使用内置的Recover函数来处理panic并打印堆栈跟踪。

从https://blog.golang.org/defer-panic-and-recover中可以得到以下信息:
Recover是一个内置函数,用于恢复一个发生panic的goroutine的控制权。Recover只在延迟函数中有用。在正常执行时,调用recover将返回nil并且没有其他效果。如果当前的goroutine正在发生panic,调用recover将捕获给定给panic的值并恢复正常执行。

我已经修改了你的示例代码,使用了recover和eris。eris提供了一种更好的处理、跟踪和记录Go语言中的错误的方式。

package main

import (
	"github.com/rotisserie/eris"
	"fmt"
)

func main() {
	value, err := DoSomething()
	defer func() {
		if r := recover(); r!= nil {
			fmt.Println(fmt.Sprintf("%+v", r))
		}
	}()
	if err != nil {
		panic(err)
	}
	
	fmt.Println(value)
}

func DoSomething() (string, error) {
	return "", eris.New("some error explanation here")
}

输出结果为:

some error explanation here
	main.DoSomething: /tmp/sandbox147128055/prog.go: 23
	main.main: /tmp/sandbox147128055/prog.go: 9
	runtime.main: /usr/local/go/src/runtime/proc.go: 203
	runtime.goexit: /usr/local/go/src/runtime/asm_amd64p32.s: 523

你可以在这里查看它的运行效果:https://play.golang.org/p/jgkaR42ub5q

英文:

You can use the built-in Recover function to handle panic and print the stack trace.

From https://blog.golang.org/defer-panic-and-recover
> Recover is a built-in function that regains control of a panicking
> goroutine. Recover is only useful inside deferred functions. During
> normal execution, a call to recover will return nil and have no other
> effect. If the current goroutine is panicking, a call to recover will
> capture the value given to panic and resume normal execution.

I have modified your example to use recover and eris. Eris provides a better way to handle, trace, and log errors in Go.

package main

import (
	"github.com/rotisserie/eris"
	"fmt"
)

func main() {
	value, err := DoSomething()
	defer func() {
		if r := recover(); r!= nil {
			fmt.Println(fmt.Sprintf("%+v", r))
		}
	}()
	if err != nil {
		panic(err)
	}
	
	fmt.Println(value)
}

func DoSomething() (string, error) {
	return "", eris.New("some error explanation here")
}

The output is:

some error explanation here
	main.DoSomething: /tmp/sandbox147128055/prog.go: 23
	main.main: /tmp/sandbox147128055/prog.go: 9
	runtime.main: /usr/local/go/src/runtime/proc.go: 203
	runtime.goexit: /usr/local/go/src/runtime/asm_amd64p32.s: 523

See it in action here https://play.golang.org/p/jgkaR42ub5q

答案9

得分: 0

我无法相信没有人提到令人惊叹的cockroachdb/errors:这是一个维护良好的库,具有以下功能:

  • 错误链接 .Wrap()
  • 错误编码:JSON、Sentry
  • 将错误标记为“已处理”
  • 添加次要错误
  • 堆栈跟踪
  • 向错误添加上下文信息
  • 添加面向用户的信息:人性化文本
  • 生成不包含个人身份信息的PII(个人身份信息)

示例代码:

package main

import (
	"fmt"

	"github.com/cockroachdb/errors"
)

// 叶子错误:标准错误
var LeafError = errors.New("数据库错误")

func main() {
	// 调用一个可能失败的函数
	_, err := startTransaction()

	if err != nil {
		// 使用更多信息返回一个包装错误
		var returnedError = errors.WithHint(
			// 使用我们的标准错误包装返回的错误
			errors.CombineErrors(LeafError, err),
			// 面向用户的错误
			`抱歉,出现问题,请稍后再试。`,
		)

		// 好吧,我们实际上要打印它。使用“%+v”来查看堆栈跟踪。
		fmt.Printf("错误:%+v\n", returnedError)
	}
}

// 可能失败的示例函数
func startTransaction() (any, error) {
	return nil, errors.New("无法启动事务")
}

示例输出:

错误:数据库错误
(1) 抱歉,出现问题,请稍后再试。
包装:(2) 附加的次要错误
  | 无法启动事务
  | (1) 附加的堆栈跟踪
  |   -- 堆栈跟踪:
  |   | main.startTransaction
  |   |         /home/kolypto/c0de/medthings/cerebellum/cerebellum/play.go:32
  |   | main.main
  |   |         /home/kolypto/c0de/medthings/cerebellum/cerebellum/play.go:14
  |   | runtime.main
  |   |         /usr/lib/go-1.20/src/runtime/proc.go:250
  |   | runtime.goexit
  |   |         /usr/lib/go-1.20/src/runtime/asm_amd64.s:1598
  | 包装:(2) 无法启动事务
  | 错误类型:(1) *withstack.withStack (2) *errutil.leafError
包装:(3) 附加的堆栈跟踪
  -- 堆栈跟踪:
  | main.init
  |     /home/kolypto/c0de/medthings/cerebellum/cerebellum/play.go:10
  | runtime.doInit
  |     /usr/lib/go-1.20/src/runtime/proc.go:6506
  | runtime.main
  |     /usr/lib/go-1.20/src/runtime/proc.go:233
  | runtime.goexit
  |     /usr/lib/go-1.20/src/runtime/asm_amd64.s:1598
包装:(4) 数据库错误
英文:

I can't believe no one mentioned the amazing cockroachdb/errors: a well-maintained library with the following features:

  • Error chaining .Wrap()
  • Error encoding: JSON, Sentry
  • Mark error as "handled"
  • Add a secondary error
  • Stack traces
  • Add contextual information to an error
  • Add user-facing information: human-friendly text
  • Generate PII-free details (without personally identifiable information)

Sample code:

package main

import (
	"fmt"

	"github.com/cockroachdb/errors"
)

// Leaf error: standard error
var LeafError = errors.New("Database error")

func main() {
	// Call a function that fails
	_, err := startTransaction()

	if err != nil {
		// Return a wrapped error with more information
		var returnedError = errors.WithHint(
			// Wrap the returned error with our standard error
			errors.CombineErrors(LeafError, err),
			// User facing error
			`Sorry, there is an issue. Please try again later.`,
		)

		// okay we'll actually print it. Use "%+v" to see stack traces.
		fmt.Printf("Error: %+v\n", returnedError)
	}
}

// Sample function that fails
func startTransaction() (any, error) {
	return nil, errors.New("Failed to start a transaction")
}

Sample output:

Error: Database error
(1) Sorry, there is an issue. Please try again later.
Wraps: (2) secondary error attachment
  | Failed to start a transaction
  | (1) attached stack trace
  |   -- stack trace:
  |   | main.startTransaction
  |   |         /home/kolypto/c0de/medthings/cerebellum/cerebellum/play.go:32
  |   | main.main
  |   |         /home/kolypto/c0de/medthings/cerebellum/cerebellum/play.go:14
  |   | runtime.main
  |   |         /usr/lib/go-1.20/src/runtime/proc.go:250
  |   | runtime.goexit
  |   |         /usr/lib/go-1.20/src/runtime/asm_amd64.s:1598
  | Wraps: (2) Failed to start a transaction
  | Error types: (1) *withstack.withStack (2) *errutil.leafError
Wraps: (3) attached stack trace
  -- stack trace:
  | main.init
  |     /home/kolypto/c0de/medthings/cerebellum/cerebellum/play.go:10
  | runtime.doInit
  |     /usr/lib/go-1.20/src/runtime/proc.go:6506
  | runtime.main
  |     /usr/lib/go-1.20/src/runtime/proc.go:233
  | runtime.goexit
  |     /usr/lib/go-1.20/src/runtime/asm_amd64.s:1598
Wraps: (4) Database error

huangapple
  • 本文由 发表于 2015年10月9日 17:07:37
  • 转载请务必保留本文链接:https://go.coder-hub.com/33034241.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定