How to write a custom middleware
View SourceProblem
You need behaviour that runs before or after every handler in a stack: auth, CORS, rate limiting, feature flags, request mutation.
Solution
Implement the livery_middleware behaviour. One callback:
call(Req, Next, State) -> Resp.
-module(my_cors).
-behaviour(livery_middleware).
-export([call/3]).
call(Req, Next, #{origins := Allowed} = State) ->
case livery_req:header(<<"origin">>, Req) of
undefined ->
Next(Req);
Origin ->
case lists:member(Origin, Allowed) of
true ->
Resp = Next(Req),
livery_resp:with_header(
<<"access-control-allow-origin">>, Origin, Resp);
false ->
livery_resp:text(403, <<"origin not allowed">>)
end
end.Wire it into a stack as {my_cors, #{origins => [...]}}.
Sugar helpers
For simple shapes, use the constructors instead of a full module:
%% Mutate the request, then continue.
livery_middleware:before(fun(R) ->
livery_req:set_meta(start, erlang:monotonic_time(), R)
end).
%% Mutate the response on the way out.
livery_middleware:after_response(fun(R) ->
livery_resp:with_header(<<"X-Server">>, <<"livery">>, R)
end).
%% Catch downstream exceptions and turn them into a response.
livery_middleware:wrap(fun(Class, Reason, _Stack) ->
livery_resp:text(500,
iolist_to_binary(io_lib:format("~p: ~p", [Class, Reason])))
end).Three shapes a middleware can take
- Pass-through. Transform request or response. Always call
Next. Example:livery_request_id,livery_access_log. - Short-circuit. Skip
Nextand return a response directly. Example: auth failures, rate limit hits. - Wrapper. Run
Nextinsidetry/catchor a monitor. Example:livery_middleware:wrap,livery_timeout.
Storing state on the request
Use livery_req:set_meta/3 to thread values from middleware to
handler:
call(Req, Next, _State) ->
{ok, User} = verify(Req),
Next(livery_req:set_meta(user, User, Req)).The handler reads it back with livery_req:meta(user, Req).
Ordering
The first entry in the stack list is outermost. Put auth before business logic. Put request id and error wrappers at the very top so every response carries them.
Testing
denies_when_origin_missing_test() ->
Cap = livery_test_adapter:run(
[{my_cors, #{origins => [<<"https://app">>]}}],
fun (_R) -> livery_resp:text(200, <<>>) end,
#{headers => [{<<"origin">>, <<"https://evil">>}]}),
?assertEqual(403, livery_test_adapter:status(Cap)).See also
- Tutorial: Compose a middleware stack
- Reference:
livery_middleware - Reference:
livery_request_id,livery_body_limit,livery_timeout,livery_access_log