Error Handling
Although more common to return the {:error, reason}
tuple, Elixir supports exceptions and in this lesson we’ll look at how to handle errors and the different mechanisms available to us.
In general the convention in Elixir is to create a function (example/1
) which returns {:ok, result}
and {:error, reason}
and a separate function (example!/1
) that returns the unwrapped result
or raises an error.
This lesson will focus on interacting with the latter
General conventions
At the moment, the Elixir community has come to some conventions regarding returning errors:
-
For errors that are part of the regular operation of a function (e.g. a user entered a wrong type of date), a function returns
{:ok, result}
and{:error, reason}
accordingly. - For errors that are not part of normal operations (e.g. being unable to parse configuration data) you throw an exception.
We generally handle standard flow errors by Pattern Matching, but in this lesson, we are focusing on the second case - on the exceptions.
Often, in public APIs, you can also find a second version of the function with a ! (example!/1) that returns the unwrapped result or raises an error.
Error Handling
Before we can handle errors we need to create them and the simplest way to do so is with raise/1
:
iex> raise "Oh no!"
** (RuntimeError) Oh no!
If we want to specify the type and message, we need to use raise/2
:
iex> raise ArgumentError, message: "the argument value is invalid"
** (ArgumentError) the argument value is invalid
When we know an error may occur, we can handle it using try/rescue
and pattern matching:
iex> try do
...> raise "Oh no!"
...> rescue
...> e in RuntimeError -> IO.puts("An error occurred: " <> e.message)
...> end
An error occurred: Oh no!
:ok
It’s possible to match multiple errors in a single rescue:
try do
opts
|> Keyword.fetch!(:source_file)
|> File.read!()
rescue
e in KeyError -> IO.puts("missing :source_file option")
e in File.Error -> IO.puts("unable to read source file")
end
After
At times it may be necessary to perform some action after our try/rescue
regardless of error.
For this we have try/after
.
If you’re familiar with Ruby this is akin to begin/rescue/ensure
or in Java try/catch/finally
:
iex> try do
...> raise "Oh no!"
...> rescue
...> e in RuntimeError -> IO.puts("An error occurred: " <> e.message)
...> after
...> IO.puts "The end!"
...> end
An error occurred: Oh no!
The end!
:ok
This is most commonly used with files or connections that should be closed:
{:ok, file} = File.open("example.json")
try do
# Do hazardous work
after
File.close(file)
end
New Errors
While Elixir includes a number of builtin error types like RuntimeError
, we maintain the ability to create our own if we need something specific.
Creating a new error with the defexception/1
macro conveniently accepts the :message
option to set a default error message:
defmodule ExampleError do
defexception message: "an example error has occurred"
end
Let’s take our new error for a spin:
iex> try do
...> raise ExampleError
...> rescue
...> e in ExampleError -> e
...> end
%ExampleError{message: "an example error has occurred"}
Throws
Another mechanism for working with errors in Elixir is throw
and catch
.
In practice these occur very infrequently in newer Elixir code but it’s important to know and understand them nonetheless.
The throw/1
function gives us the ability to exit execution with a specific value we can catch
and use:
iex> try do
...> for x <- 0..10 do
...> if x == 5, do: throw(x)
...> IO.puts(x)
...> end
...> catch
...> x -> "Caught: #{x}"
...> end
0
1
2
3
4
"Caught: 5"
As mentioned, throw/catch
are quite uncommon and typically exist as stopgaps when libraries fail to provide adequate APIs.
Exiting
The final error mechanism Elixir provides us with is exit
.
Exit signals occur whenever a process dies and are an important part of the fault tolerance of Elixir.
To explicitly exit we can use exit/1
:
iex> spawn_link fn -> exit("oh no") end
** (EXIT from #PID<0.101.0>) evaluator process exited with reason: "oh no"
While it is possible to catch an exit with try/catch
doing so is extremely rare.
In almost all cases it is advantageous to let the supervisor handle the process exit:
iex> try do
...> exit "oh no!"
...> catch
...> :exit, _ -> "exit blocked"
...> end
"exit blocked"
Caught a mistake or want to contribute to the lesson? Edit this lesson on GitHub!