nitro_n2o.erl 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104
  1. -module(nitro_n2o).
  2. -description('N2O Nitrogen Web Framework Protocol').
  3. -include_lib("nitro/include/n2o.hrl").
  4. -export([info/3,render_actions/1,io/1,io/2,event/1]).
  5. % Nitrogen pickle handler
  6. info({text,<<"N2O,",Auth/binary>>}, Req, State) ->
  7. info(#init{token=Auth},Req,State);
  8. info(#init{token=Auth}, Req, State) ->
  9. {'Token', Token} = n2o_session:authenticate([], Auth),
  10. Sid = case n2o:depickle(Token) of {{S,_},_} -> S; X -> X end,
  11. New = State#cx{session = Sid, token = Auth},
  12. put(context,New),
  13. {reply,{bert,case io(init, State) of
  14. {io,_,{stack,_}} = Io -> Io;
  15. {io,Code,_} -> {io,Code,{'Token',Token}} end},
  16. Req,New};
  17. info(#client{data=Message}, Req, State) ->
  18. nitro:actions([]),
  19. {reply,{bert,io(#client{data=Message},State)},Req,State};
  20. info(#pickle{}=Event, Req, State) ->
  21. nitro:actions([]),
  22. {reply,{bert,html_events(Event,State)},Req,State};
  23. info(#flush{data=Actions}, Req, State) ->
  24. nitro:actions(Actions),
  25. {reply,{bert,io(<<>>)},Req,State};
  26. info(#direct{data=Message}, Req, State) ->
  27. nitro:actions([]),
  28. {reply,{bert,case io(Message, State) of
  29. {io,_,{stack,_}} = Io -> Io;
  30. {io,Code,Res} -> {io,Code,{direct,Res}} end},
  31. Req,State};
  32. info(Message,Req,State) -> {unknown,Message,Req,State}.
  33. % double render: actions could generate actions
  34. render_actions(Actions) ->
  35. nitro:actions([]),
  36. First = nitro:render(Actions),
  37. Second = nitro:render(nitro:actions()),
  38. nitro:actions([]),
  39. nitro:to_binary([First,Second]).
  40. % n2o events
  41. html_events(#pickle{source=Source,pickled=Pickled,args=Linked}, State=#cx{token = Token}) ->
  42. Ev = n2o:depickle(Pickled),
  43. L = n2o_session:prolongate(),
  44. Res = case Ev of
  45. #ev{} when L =:= false -> render_ev(Ev,Source,Linked,State), <<>>;
  46. #ev{} -> render_ev(Ev,Source,Linked,State), n2o_session:authenticate([], Token);
  47. _CustomEnvelop -> %?LOG_ERROR("EV expected: ~p~n",[CustomEnvelop]),
  48. {error,"EV expected"} end,
  49. io(Res).
  50. % calling user code in exception-safe manner
  51. -ifdef(OTP_RELEASE).
  52. render_ev(#ev{module=M,name=F,msg=P,trigger=T},_Source,Linked,State) ->
  53. try case F of
  54. api_event -> M:F(P,Linked,State);
  55. event -> [erlang:put(K,V) || {K,V} <- Linked], M:F(P);
  56. _ -> M:F(P,T,State) end
  57. catch E:R:S -> ?LOG_EXCEPTION(E,R,S), {stack,S} end.
  58. io(Event, #cx{module=Module}) ->
  59. try X = Module:event(Event), {io,render_actions(nitro:actions()),X}
  60. catch E:R:S -> ?LOG_EXCEPTION(E,R,S), {io,[],{stack,S}} end.
  61. io(Data) ->
  62. try {io,render_actions(nitro:actions()),Data}
  63. catch E:R:S -> ?LOG_EXCEPTION(E,R,S), {io,[],{stack,S}} end.
  64. -else.
  65. render_ev(#ev{module=M,name=F,msg=P,trigger=T},_Source,Linked,State) ->
  66. try case F of
  67. api_event -> M:F(P,Linked,State);
  68. event -> [erlang:put(K,V) || {K,V} <- Linked], M:F(P);
  69. _ -> M:F(P,T,State) end
  70. catch E:R -> S = erlang:get_stacktrace(), ?LOG_EXCEPTION(E,R,S), {stack,S} end.
  71. io(Event, #cx{module=Module}) ->
  72. try X = Module:event(Event), {io,render_actions(nitro:actions()),X}
  73. catch E:R -> S = erlang:get_stacktrace(), ?LOG_EXCEPTION(E,R,S), {io,<<>>,{stack,S}} end.
  74. io(Data) ->
  75. try {io,render_actions(nitro:actions()),Data}
  76. catch E:R -> S = erlang:get_stacktrace(), ?LOG_EXCEPTION(E,R,S), {io,<<>>,{stack,S}} end.
  77. -endif.
  78. % event Nitrogen Web Framework protocol
  79. event(_) -> [].