cowboy_loop.erl 2.8 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576
  1. %% Copyright (c) 2011-2017, Loïc Hoguin <essen@ninenines.eu>
  2. %%
  3. %% Permission to use, copy, modify, and/or distribute this software for any
  4. %% purpose with or without fee is hereby granted, provided that the above
  5. %% copyright notice and this permission notice appear in all copies.
  6. %%
  7. %% THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
  8. %% WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
  9. %% MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
  10. %% ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
  11. %% WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
  12. %% ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
  13. %% OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  14. -module(cowboy_loop).
  15. -behaviour(cowboy_sub_protocol).
  16. -export([upgrade/4]).
  17. -export([upgrade/5]).
  18. -export([loop/4]).
  19. -callback init(Req, any())
  20. -> {ok | module(), Req, any()}
  21. | {module(), Req, any(), any()}
  22. when Req::cowboy_req:req().
  23. -callback info(any(), Req, State)
  24. -> {ok, Req, State}
  25. | {ok, Req, State, hibernate}
  26. | {stop, Req, State}
  27. when Req::cowboy_req:req(), State::any().
  28. -callback terminate(any(), cowboy_req:req(), any()) -> ok.
  29. -optional_callbacks([terminate/3]).
  30. -spec upgrade(Req, Env, module(), any())
  31. -> {ok, Req, Env} | {suspend, ?MODULE, loop, [any()]}
  32. when Req::cowboy_req:req(), Env::cowboy_middleware:env().
  33. upgrade(Req, Env, Handler, HandlerState) ->
  34. loop(Req, Env, Handler, HandlerState).
  35. -spec upgrade(Req, Env, module(), any(), hibernate)
  36. -> {suspend, ?MODULE, loop, [any()]}
  37. when Req::cowboy_req:req(), Env::cowboy_middleware:env().
  38. upgrade(Req, Env, Handler, HandlerState, hibernate) ->
  39. suspend(Req, Env, Handler, HandlerState).
  40. -spec loop(Req, Env, module(), any())
  41. -> {ok, Req, Env} | {suspend, ?MODULE, loop, [any()]}
  42. when Req::cowboy_req:req(), Env::cowboy_middleware:env().
  43. %% @todo Handle system messages.
  44. loop(Req, Env, Handler, HandlerState) ->
  45. receive
  46. Message ->
  47. call(Req, Env, Handler, HandlerState, Message)
  48. end.
  49. call(Req0, Env, Handler, HandlerState0, Message) ->
  50. try Handler:info(Message, Req0, HandlerState0) of
  51. {ok, Req, HandlerState} ->
  52. loop(Req, Env, Handler, HandlerState);
  53. {ok, Req, HandlerState, hibernate} ->
  54. suspend(Req, Env, Handler, HandlerState);
  55. {stop, Req, HandlerState} ->
  56. terminate(Req, Env, Handler, HandlerState, stop)
  57. catch Class:Reason ->
  58. cowboy_handler:terminate({crash, Class, Reason}, Req0, HandlerState0, Handler),
  59. erlang:raise(Class, Reason, erlang:get_stacktrace())
  60. end.
  61. suspend(Req, Env, Handler, HandlerState) ->
  62. {suspend, ?MODULE, loop, [Req, Env, Handler, HandlerState]}.
  63. terminate(Req, Env, Handler, HandlerState, Reason) ->
  64. Result = cowboy_handler:terminate(Reason, Req, HandlerState, Handler),
  65. {ok, Req, Env#{result => Result}}.