This package brings Haskell-style type classes to Erlang, including monads, applicatives, functors, and traversables. It provides implementations of commonly used class instances, as well as useful utility functions.
To install the latest release from hex, add
do to the deps in your rebar3 config file:
{do, "2.0.2"}
The do package implements either,
list, and
maybe monads. For a complete overview
of types and functions, refer to do_types.hrl and
do's docs on hex.
Use the ?fmap macro or do:fmap/2 to map functions over functors:
-include_lib("do/include/do.hrl").
increment(N) ->
N + 1.
fmap_either() ->
{ok, 2} = ?fmap(fun increment/1, {ok, 1}),
{error, reason} = ?fmap(fun increment/1, {error, reason}).
fmap_maybe() ->
{just, 2} = ?fmap(fun increment/1, {just, 1}),
nothing = ?fmap(fun increment/1, nothing).
fmap_list() ->
[2, 3, 4] = ?fmap(fun increment/1, [1, 2, 3]).
fmap_map() ->
#{a => 2} = ?fmap(fun increment/1, #{a => 1}).Use the ?sequence macro or do:sequence/1 to sequence a traversable of
applicatives. For example:
-include_lib("do/include/do.hrl").
sequence_list() ->
{ok, [1, 2, 3]} = ?sequence([{ok, 1}, {ok, 2}, {ok, 3}]),
{error, reason} = ?sequence([{ok, 1}, {error, reason}, {ok, 3}]).
sequence_map() ->
{just, #{a => 1, b => 2}} = ?sequence(#{a => {just, 1}, b => {just, 2}}),
nothing = ?sequence(#{a => {just, 1}, b => nothing}).
sequence_either() ->
[{ok, 1}] = ?sequence({ok, [1]}),
[] = ?sequence({ok, []}).
Use the ?bind macro or do:bind/2 to bind (>>=) a function that returns a
monad to a monad of the same type. In case of the
either monad:
-include_lib("do/include/do.hrl").
increment_either(N) when is_integer(N) ->
{ok, N + 1};
increment_either(_) ->
{error, no_int}.
bind_either() ->
{ok, 2} = ?bind({ok, 1}, fun increment_either/1),
{error, no_int} = ?bind({ok, foo}, fun increment_either/1),
{error, reason} = ?bind({error, reason}, fun increment_either/1).For the maybe monad:
increment_maybe(N) when is_integer(N) ->
{just, N + 1};
increment_maybe(_) ->
nothing.
bind_maybe() ->
{just, 2} = ?bind({just, 1}, fun increment_maybe/1),
nothing = ?bind({just, foo}, fun increment_maybe/1),
nothing = ?bind(nothing, fun increment_maybe/1).For the list monad:
increment_list(N) when is_integer(N) ->
[N + 1];
increment_maybe(_) ->
[].
bind_list() ->
[2] = ?bind([1], fun increment_list/1),
[] = ?bind([foo], fun increment_list/1),
[] = ?bind([], fun increment_maybe/1).Use the ?then macro or do:then/2 to chain (>>) monadic expressions of the
same type. The second argument to ?then is wrapped in a thunk that will only
be executed if the first argument indicates success. For example:
-include_lib("do/include/do.hrl").
increment(N) when is_integer(N) ->
[N + 1];
increment(_) ->
[].
then_list() ->
[2] = ?then([5], increment(1)),
[] = ?then([5], increment(foo)),
[] = ?then([], increment(1)).Use the ?liftm macro to lift a function into a monad. For example:
-include_lib("do/include/do.hrl").
liftm_either() ->
{ok, 3} = ?liftm(fun erlang:'+'/2, {ok, 1}, {ok, 2}),
{error, reason} = ?liftm(fun erlang:'+'/2, {ok, 1}, {error, reason}).Arguments to ?liftm are evaluated lazily. In the following example
1 + 1 will never be evaluated:
-include_lib("do/include/do.hrl").
liftm_either() ->
{error, reason} = ?liftm(fun erlang:'+'/2, {error, reason}, {ok, 1 + 1}).Use the ?do macro or do:do/2 to consecutively bind (>>= or >>) monads
and functions. The macro takes a start value (a monad), and a list of functions.
The functions must each take either 0 or 1 argument(s) and must return a monad.
On execution, the start value is passed to the first function, and is then
piped through consecutive functions using bind or then. For example (with
either monad):
-include_lib("do/include/do.hrl").
increment(N) when is_integer(N) ->
{ok, N + 1};
increment(_) ->
{error, no_int}.
int_to_bin(N) when is_integer(N) ->
{ok, integer_to_binary(N)};
int_to_bin(_) ->
{error, no_int}.
do_either() ->
{ok, 4} = ?do({ok, 1}, [ fun increment/1,
fun increment/1,
fun increment/1 ]),
{error, no_int} = ?do({ok, 1}, [ fun increment/1,
fun int_to_bin/1,
fun increment/1 ]).