Errors🔗
How to Handle Errors: The Basics🔗
- Go handles errors by returning a value of type
error
. If fuction executes as expected,nil
is returned for error parameter. If something goes wrong, error value is returned. - Calling function checks the error return value by comparing it to
nil
, handling the error or returning another error of its own.
func calcRemainderAndMod(numerator, denominator int) (int, int, error) {
if denominator == 0 {
return 0, 0, errors.New("denominator is 0")
}
return numerator / denominator, numerator % denominator, nil
}
- NOTE: Error messages should not be capitalized nor should they end with punctuations. In most cases set return value to their zero values (except in case of sentinel errors).
// example of checking error
num := 20
den := 3
rem, mod, err := calcRemaindereAndMod(num, den)
if err != nil {
fmt.Println(err)
os.Exit(1)
}
fmt.Println(remainder, mod)
error
is a built-in interface that defines a single method.
- Anything that implements this interface is considered an error. The reason you return
nil
from a function to indicate that no error occurred is thatnil
is zero value of any interface type.
Why go uses returned error instead of thrown exception ?🔗
- exception add at least one new code path through code. These paths are sometimes unclear.This produces code that crashes in surprising ways when exception is not handled correctly.
- Go compiler requires all variables to be read. Making returned values forces devs to either check and handle error condition or make it explicit that they are ignoring errors by using
_
Use Strings for Simple Errors🔗
- Go standard library has two ways to create error from string
errors.New
function takes string and returnserror
. String you provide will be returned when you callError
method on returned error instance.fmt.Println
automatically callsError
method.fmt.Errorf
function allows to include runtime information in the error message by usingfmt.Printf
verbs to format an error string.
Sentinel Errors🔗
- some errors signal that processing cannot continue due to problem with current state.
- sentinel errors are few variables that are declared at package level. By convention, their names start with
Err
(exceptio.EOF
) They should be treated as read-only. - Example:
ErrFormat
is raised whenarchive/zip
encounters data which doesn’t have zip data.
data := []byte("This is not a zip file")
notAZipFile := bytes.NewReader(data)
_, err := zip.NewReader(notAZipFile, int64(len(data)))
if err == zip.ErrFormat {
fmt.Println("Told you so")
}
Errors are values🔗
- Since
error
is an interface, you can define your own errors that include additional information for logging or error handling. For example implementing error codes.
// enumeration to represent status codes
type Status int
const (
InvalidLogin Status = iota + 1
NotFound
)
// define StatusErr to hold this value
type StatusErr struct {
Status Status
Message string
}
func (se StatusErr) Error() string {
return se.Message
}
// usage
token, err := login(uid, pwd)
if err != nil {
return nil, StatusErr{
Status: InvalidLogin,
Message: fmt.Sprintf("invalid credentials for user %s", uid),
}
}
NOTE: When using custom errors, never define a variable to be of the type of your custom error. Either explicitly return nil
when no error occurs or define the variable to be of type error
.
Wrapping Errors🔗
- Preserving an error and adding information to it is called as wrapping an error.
- A series of wrapped errors, it is called an error tree
- Ex - The
fmt.Errorf
function has special verb,%w
. Its used to create an error whose formatted string includes the formatted string of another error. - Standard library has a function for unwrapping functions in
errors
package.
func fileChecker(name string) error {
f, err := os.Open(name)
if err != nil {
return fmt.Errorf("in fileChecker: %w", err)
}
f.Close()
return nil
}
func main() {
err := fileChecker("not_here.txt")
if err != nil {
fmt.Println(err)
if wrappedErr := errors.Unwrap(err); wrappedErr != nil {
fmt.Println(wrappedErr)
}
}
}
- If you want to wrap an error with your custom error type, your error type needs to implement the method
Unwrap
- NOTE: You don’t usually call
errors.Unwrap
directly. Instead, you useerrors.Is
anderrors.As
to find a specific wrapped error. I’ll talk about these two functions in the next section.
- It is not necessary to wrap all errors, there could be implementation details which are not relevant for the current context. You could use
fmt.Errorf
with the%v
verb instead of%w
Wrapping Multiple Errors🔗
- Ex- in case of fieldValidator we need to validate and return a single error containing all invalid fields.
- Since standard function returns an
error
rather than an array of error, we can merge multiple errors into single error usingerrors.Join
function.
type Person struct {
FirstName string
LastName string
Age int
}
func ValidatePerson(p Person) error {
var errs []error // never use your own err types in declartion, this is fine
if len(p.FirstName) == 0 {
errs = append(errs, errors.New("field FirstName cannot be empty"))
}
if len(p.LastName) == 0 {
errs = append(errs, errors.New("field LastName cannot be empty"))
}
if p.Age < 0 {
errs = append(errs, errors.New("field Age cannot be negative"))
}
if len(errs) > 0 {
return errors.Join(errs...)
}
return nil // always return nil values.
}
- Another way to merge errors is just use
fmt.Errorf
with multiple%w
verbs. - You could create your own version of Unwrap that takes an array of errors
[] error
. But since Go doesn’t support overloading you can’t create single type and provide both implementation :(. But you could do something like this.
var err error
err = funcThatReturnsAnError()
switch err := err.(type) {
case interface {Unwrap() error}:
// handle single error
innerErr := err.Unwrap()
// process innerErr
case interface {Unwrap() []error}:
//handle multiple wrapped errors
innerErrs := err.Unwrap()
for _, innerErr := range innerErrs {
// process each innerErr
}
default:
// handle no wrapped error
}
Is and As🔗
- If sentinel errors are wrapped, we can’t use
==
to compare them. Go providesIs
andAs
for this. errors.Is
: checks if any error in error tree matches the the provided sentinel
func fileChecker(name string) error {
f, err := os.Open(name)
if err != nil {
return fmt.Errorf("in fileChecker: %w", err)
}
f.Close()
return nil
}
func main() {
err := fileChecker("not_here.txt")
if err != nil {
if errors.Is(err, os.ErrNotExist) {
fmt.Println("That file doesn't exist")
}
}
}
errors.As
: function allows you to check whether a returned error (on any error it wraps) matches a specific type. It takes two parameters, first the error being examined, second is pointer to variable of type that you are looking for.
err := AFunctionThatReturnsAnError()
var myErr MyErr
if errors.As(err, &myErr) {
fmt.Println(myErr.Codes)
}
- we can even pass pointer to inteface as well instead of the variable.
err := AFunctionThatReturnsAnError()
// anonymous interface
var coder interface {
CodeVals() []int
}
if errors.As(err, &coder) {
fmt.Println(coder.CodeVals())
}
Wrapping Errors with defer🔗
Some times we find ourselves wrapping multiple errors with same message, which can be simplified using defer
func DoSomeThings(val1 int, val2 string) (string, error) {
val3, err := doThing1(val1)
if err != nil {
return "", fmt.Errorf("in DoSomeThings: %w", err)
}
val4, err := doThing2(val2)
if err != nil {
return "", fmt.Errorf("in DoSomeThings: %w", err)
}
result, err := doThing3(val3, val4)
if err != nil {
return "", fmt.Errorf("in DoSomeThings: %w", err)
}
return result, nil
}
// using defer
func DoSomeThings(val1 int, val2 string) (_ string, err error) {
// this clousure checks whether an error was returned. So it reassins wrapping the error
defer func() {
if err != nil {
err = fmt.Errorf("in DoSomeThings: %w", err)
}
}()
val3, err := doThing1(val1)
if err != nil {
return "", err
}
val4, err := doThing2(val2)
if err != nil {
return "", err
}
return doThing3(val3, val4)
}
panic and recover🔗
- A panic is similar to an
Error
in Java or Python. It is state generated by Go runtime whenever it is not able to figure out what to do next.
- Go runtime panics if it detects bugs in itself, such as GC misbehaving. Or it could be due to user like, accessing slices beyond capacity or declaring negative size slices using
make
.
- If there is a panic, then most likely runtime is last to blame.
- As soon as a panic happens, the current function exits immediately, and any
defer
s attached to the current function start running. When thosedefer
s complete, thedefer
s attached to the calling function run, and so on, untilmain
is reached. The program then exits with a message and a stack trace.
- You can create a custom
panic
using built-in which takes input as any type. Usually it is string. - We can capture a
panic
to provide more graceful shutdown or prevent shutdown at all. The built-inrecover
function is called from within adefer
to check whether a panic happened. Once arecover
happens, execution continue normally.
func div60(i int) {
defer func() {
if v := recover(); v != nil {
fmt.Println(v)
}
}()
fmt.Println(60 / i)
}
func main() {
for _, val := range []int{1, 2, 0, 6} {
div60(val)
}
}
- Use panic for fatal errors and recover to handle them gracefully, but avoid relying on them for general error handling. Explicitly check for errors instead. If writing a library, use recover to prevent panics from escaping public APIs and convert them into errors.
Getting a Stack Trace from an Error🔗
- Go doesn’t provide stack traces by default, but you can use error wrapping or third-party libraries like CockroachDB’s to generate them. To print a stack trace, use fmt.Printf with %+v. To hide file paths in errors, compile with the -trimpath flag.