Inspired by Failure is your Domain and Error handling in upspin (comparison here), package e
is designed to meet the needs of web applications by maintaining a clean separation between the error data consumed by end-users, clients, and operators.
e
focuses on three principles:
-
Simplicity: Small, fluent interface for developer ergonomics
-
Safety: Explicitly separate end-user messages from internal errors
-
Compatibility: Follows error-wrapping conventions and can be adopted incrementally
e.NewError()
populates the inner error message with the calling function name at runtime.
const CodeInvalidError = "invalid_error"
func Foo(bar string) error {
if bar == nil {
return e.NewError(CodeInvalidError, "bar cannot be nil")
// "Foo: [invalid_error] bar cannot be nil"
}
return nil
}
e.NewErrorf()
allows for formatted strings.
e.Wrap()
also automatically injects calling function name, reducing the burden on developers of providing context to the error and helping keep the error stack free of redundant or unhelpful messages.
In most cases it is enough to simply e.Wrap(err)
.
func Foo(bar string) error {
err := db.GetBar(bar) // "GetBar: [database_error] cannot find bar"
if err != nil {
return e.Wrap(err)
// "Foo: GetBar: [database_error] cannot find bar"
}
return nil
}
Wrap()
can take an optionalInfo
param to inject additional context into the error stack.
func Foo(bar string) error {
err := db.GetBar(bar) // "GetBar: [database_error] cannot find bar"
if err != nil {
return e.Wrap(err, fmt.Sprintf("bar id: %s", bar.Id))
// "Foo: (bar id: 2hs8qh9): GetBar: [database_error] cannot find bar"
}
return nil
}
Wrapf()
allows for formatted strings.
e.Wrap()
can be chained with SetCode()
to provide a new code for errors from another package (or default go errors).
// db.GetBar returns sentinel error:
// var ErrNotFound = errors.New("not found")
const CodeInternalError = "internal_error"
func Foo(bar string) error {
err := db.GetBar(bar) // "not found"
if err != nil {
return e.Wrap(err).SetCode(CodeInternalError)
// "Foo: [internal_error] not found
}
}
ErrorMessage()
is used to display a user-friendly error message to the end-user. NewError()
and Wrap()
do not have a message
param (intentional design). SetMessage()
should be called to assign an intentional and meaningful message.
func doSomething(r *http.Request ctx context.Context) error {
err := Foo()
if err != nil {
return e.Wrap(err).SetMessage("Oh no! An Error has occurred.")
// this message will not show in Error() and can only be retrieved
// by e.Message() or ErrorMessage()
}
return nil
}
This behaviour should be combined with a default error message at the handler level to make message
an optional field that should only be set to give additional context to the end-user.
func (h Handler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
bytes, err := doSomething(r)
if err != nil {
// Log full error stack
logger.Error(err)
// Extract the first error message from the stack
userMsg := e.ErrorMessage(err)
if userMsg == "" {
userMsg = "Unexpected error has occurred"
}
http.Error(w, userMsg, http.StatusInternalServerError)
return
}
w.Write(bytes)
}
ErrorCode()
is intended for use by any clients such as front-end applications, other libraries, and even callers within your own application. In the context of a go codebase, code
provides an alternative way of introspecting error types without comparing Error()
strings or using type assertions.
const (
CodeInvalidError = "invalid_error"
CodeDatabaseError = "database_error"
)
func Foo(bar string) error {
if bar == nil {
return e.NewError(CodeInvalidError, "bar cannot be nil")
}
err := db.GetBar(bar)
if err != nil {
return e.Wrap(err).SetCode(CodeDatabaseError)
// SetCode() may not be necessary if we control the err
// returned from db.GetBar().
}
return nil
}
func doSomething(r *http.Request) error {
err := Foo(r.FormValue("bar"))
if err != nil {
var info string
switch e.ErrorCode(err) {
case CodeInvalidError:
info = "Invalid Request"
case CodeDatabaseError:
err2 := RetryFoo(r.FormValue("bar"))
if err2 != nil {
info = "Database error has occurred. Please try again."
} else {
return nil
}
default:
info = "Unexpected error has occurred. Please try again"
}
return e.Wrap(err).SetMessage(info)
}
return nil
}
Operators are usually the developers or application support who are concerned with logging the logical stack trace of the error. By automatically injecting the calling function name when we e.NewError()
or e.Wrap()
, we can easily build a chain of functions that were called down the stack to the error site.
Logging Error()
is sufficient to write the stack in this format:
op: (additionalInfo): [code] cause
Examples:
Foo: cannot find bar
DoSomething: Foo: [database_error] cannot find bar
ServeHTTP: DoSomething: Foo: (cannot find bar 2hjk7d): GetBar: getBarById: [database_error] cannot find bar
// anonymous functions
Foo.func1: cannot find bar
// goroutines
Foo.func1.1: cannot find bar
In addition, both e.NewError()
and e.Wrap()
populate a more detailed stacktrace that can be retrieved with ErrorStack(err)
(also compatible with any error type that fulfils HasStacktrace):
err := handleSomething()
if err != nil {
logger.Error({
error: err,
stack: ErrorStack(err), // may be ""
})
}
Package e
leverages the const op = "FuncName"
pattern introduced by Rob Pike and Andrew Gerrard in Upspin to build a logical stacktrace but uses runtime libraries to automatically extract the function name.
In place of Upspin's multi-purpose E(args ...interface{})
function, e
uses the familiar verbs New
and Wrap
to provide better type safety and simpler implementation.
Upspin did not have a clear separation between messages for end-users and the error stack, making it unsuitable for a web application which needs to hide internal details.
Package e
is heavily influenced by Ben's approach to error-handling outlined in his blog post Failure is your Domain. These are some key differences:
-
e
does not rely on struct initialization to create errors and instead usesNewError()
,Wrap()
and other helper methods to guarantee valid internal state -
e
does not require the error stack to have uniform type, and is compatible with other error types -
e
keepsmessage
logically distinct from the error stack, making it suitable for displaying to an end-user