Skip to content
/ terr Public
forked from alnvdl/terr

terr is a minimalistic library for adding error tracing in Go 1.20+.

License

Notifications You must be signed in to change notification settings

enzalito/terr

 
 

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

17 Commits
 
 
 
 
 
 
 
 
 
 

Repository files navigation

terr

Go Reference Test workflow

terr (short for traced error) is a minimalistic library for adding error tracing in Go 1.20+.

Go's error representation primitives introduced in Go 1.131 are quite sufficient, but the lack of tracing capabilities makes it hard to confidently debug errors across layers in complex applications.

To help with that, terr fully embraces the native Go error handling paradigms, but it adds two features:

  • file and line information for tracing errors;
  • the ability to print error trees using the fmt package and the %@ verb;

This library introduces the concept of traced errors: a wrapper for errors that includes the location where they were created (errors.New), passed along (return err), wrapped (%w) or masked (%v). Traced errors keep track of children traced errors that relate to them. An error is a traced error if it was returned by one of the functions of this library.

Traced errors work seamlessly with errors.Is, errors.As and errors.Unwrap just as if terr were not being used.

Without terr With terr
errors.New("error") terr.Newf("error")
fmt.Errorf("error: %w", err) terr.Newf("error: %w", err)
[return] err terr.Trace(err) (annotates file and line)

Under the hood

Starting with Go 1.20, wrapped errors are kept as a n-ary tree. terr works by build a parallel tree containing tracing information, leaving the Go error tree untouched, as if terr weren't being used. Each traced error is thus a node of the parallel traced error tree.

In the glorious day error tracing is added to Go, and assuming it gets done in a way that respects error handling as defined in Go 1.13+, removing terr from a codebase should be a matter of replacing the terr function calls with the equivalent documented expressions.

terr.Newf can wrap multiple errors. In fact, it is just a very slim wrapper around fmt.Errorf. Any traced error passed to terr.Newf will be included in the traced error tree, regardless of the fmt verb used.

terr.Trace on the other hand does nothing with the error it receives (no wrapping and no masking), but it adds one level to the parallel error tracing tree.

To obtain the full trace, terr functions must be used consistently. If fmt.Errorf is used at one point, the error tracing information will be reset at that point, but Go's wrapped error tree will be preserved even in that case.

Printing errors

A traced error tree can be printed with the special %@ formatting verb. For example:

err := terr.Newf("base")
traced := terr.Trace(err)
wrapped := terr.Newf("wrapped: %w", traced)
masked := terr.Newf("masked: %v", wrapped)
fmt.Printf("%@\n", masked)

Will output:

masked: wrapped: base @ /mygomod/file.go:14
        wrapped: base @ /mygomod/file.go:13
                base @ /mygomod/file.go:12
                        base @ /mygomod/file.go:11

%@ prints the error tree in a tab-indented, multi-line representation. If a custom format is needed (e.g., JSON), it is possible to implement a function that walks the error tree and generates that tree in the desired format. See the next subsection.

Walking the traced error tree

terr.TraceTree(err) TracedError can be used to obtain the root of an n-ary traced error tree, which can be navigated using the following methods:

type TracedError interface {
	Error() string
	Location() string
	Children() []TracedError
}

Note that this is not the tree of wrapped errors built by Go's standard library, because:

  • if non-traced errors are provided to terr.Newf, even if wrapped, they will not be a part of the traced error tree;
  • even masked (%v) errors will be part of the traced error tree if terr.Newf was used to mask them.

Methods provided by the by the Go standard library should be used to walk Go's wrapped error tree, which would includes non-traced errors and ignores masked errors (e.g., errors.Unwrap).

Adopting terr

Run the following commands in a folder to recursively adopt terr in all its files (make sure goimports2 is installed first):

$ go get github.com/alnvdl/terr
$ gofmt -w -r 'errors.New(a) -> terr.Newf(a)' .
$ gofmt -w -r 'fmt.Errorf -> terr.Newf' .
$ goimports -w .

Adopting terr.Trace is harder, as it's impossible write a simple gofmt rewrite rule that works for all cases. terr.Trace has to be applied as needed in a code base, typically in cases where return err is used, turning it into return terr.Trace(err).

Getting rid of terr

Run the following commands in a folder to recursively get rid of terr in all its files (make sure goimports2 is installed first):

$ gofmt -w -r 'terr.Newf(a) -> errors.New(a)' .
$ gofmt -w -r 'terr.Newf -> fmt.Errorf' .
$ gofmt -w -r 'terr.Trace(a) -> a' .
$ goimports -w .
$ go mod tidy

Footnotes

  1. https://go.dev/blog/go1.13-errors

  2. https://pkg.go.dev/golang.org/x/tools/cmd/goimports 2

About

terr is a minimalistic library for adding error tracing in Go 1.20+.

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • Go 100.0%