/src/middleware/ewgi_session/ewgi_session_server_store.erl

http://github.com/skarab/ewgi · Erlang · 130 lines · 87 code · 13 blank · 30 comment · 1 complexity · 7e412a7163eaf6c7a78687344a3b4533 MD5 · raw file

  1. %% @author Davide Marquês <nesrait@gmail.com>
  2. %% @copyright 2009 Davide Marquês <nesrait@gmail.com>
  3. %%
  4. %% @doc Interface for server-side session stores.
  5. %%
  6. %% This store serves as an interface for stores that manage
  7. %% client sessions on the server side using session_id's to
  8. %% between differenciate the sessions.
  9. %%
  10. %% Currently cookies are needed for the client to hold the session_id.
  11. %% BUT/TODO: we could also encode the session_id on the urls so it might
  12. %% be a good idea to swap cookie_headers/4 for a inject_session_data_on_response/?
  13. %% function defined on the ewgi_session module.
  14. %% @end
  15. %%
  16. %% Licensed under the MIT license:
  17. %% http://www.opensource.org/licenses/mit-license.php
  18. -module(ewgi_session_server_store).
  19. -author('Davide Marquês <nesrait@gmail.com>').
  20. %% Session Store API
  21. -export([load_session/2, delete_session/2, store_session/2]).
  22. %% Usage examples
  23. -export([create_example/1, delete_example/1]).
  24. -import(ewgi_util_cookie, [cookie_headers/5, cookie_safe_encode/1, cookie_safe_decode/1]).
  25. -include("ewgi.hrl").
  26. -define(SESSION_ID, "ewgi.session_server_store.session_id").
  27. -define(SESSION_SERVER_MODULE, ewgi_session_server).
  28. %%====================================================================
  29. %% Session Store API
  30. %%====================================================================
  31. load_session(Ctx, [ServerId, CookieName, _CookiePath, _SecureCookie, Timeout, IncludeIp] = StoreArgs) ->
  32. case ewgi_api:get_header_value("cookie", Ctx) of
  33. undefined ->
  34. ewgi_session2:new_session(Ctx);
  35. Cookies ->
  36. CookieValues = ewgi_util_cookie:parse_cookie(Cookies),
  37. case proplists:get_value(CookieName, CookieValues) of
  38. undefined ->
  39. ewgi_session:new_session(Ctx);
  40. SidB64 ->
  41. BinSid = cookie_safe_decode(SidB64),
  42. case (catch(binary_to_term(BinSid))) of
  43. Sid when is_list(Sid) ->
  44. Ctx1 = ewgi_api:store_data(?SESSION_ID, Sid, Ctx),
  45. case ?SESSION_SERVER_MODULE:get_session(ServerId, Sid) of
  46. undefined ->
  47. ewgi_session:new_session(Ctx1);
  48. Session ->
  49. case ewgi_session:init_session(Ctx1, Session, Timeout, IncludeIp) of
  50. invalid_session ->
  51. %% DISPLAY ERROR!?
  52. Ctx_2 = ?MODULE:delete_session(Ctx, StoreArgs),
  53. ewgi_session:new_session(Ctx_2);
  54. Ctx_2 ->
  55. Ctx_2
  56. end
  57. end;
  58. _ ->
  59. %% DISPLAY ERROR! {?MODULE, cookie_tampered}
  60. ewgi_session:new_session(Ctx)
  61. end
  62. end
  63. end.
  64. store_session(Ctx, [ServerId, CookieName, CookiePath, SecureCookie, _Timeout, IncludeIp]) ->
  65. Sid = ewgi_api:find_data(?SESSION_ID, Ctx),
  66. case Sid of
  67. undefined ->
  68. %% New session
  69. Session = ewgi_session:get_session(Ctx, IncludeIp),
  70. NewId = ?SESSION_SERVER_MODULE:save_new_session(ServerId, Session),
  71. SidB64 = cookie_safe_encode(term_to_binary(NewId)),
  72. cookie_headers(Ctx, CookieName, SidB64, CookiePath, SecureCookie);
  73. Sid ->
  74. %% existing session
  75. Updated = ewgi_session:session_updated(Ctx),
  76. if Updated ->
  77. Session = ewgi_session:get_session(Ctx, IncludeIp),
  78. ?SESSION_SERVER_MODULE:save_session(ServerId, Sid, Session);
  79. true -> ok %% nothing to do!
  80. end,
  81. Ctx
  82. end.
  83. delete_session(Ctx, [ServerId, CookieName, CookiePath, SecureCookie]) ->
  84. Sid = ewgi_api:find_data(?SESSION_ID, Ctx),
  85. case Sid of
  86. undefined -> ok;
  87. _ -> ?SESSION_SERVER_MODULE:delete_session(ServerId, Sid)
  88. end,
  89. Ctx1 = ewgi_api:store_data(?SESSION_ID, undefined, Ctx),
  90. cookie_headers(Ctx1, CookieName, [], CookiePath, SecureCookie).
  91. %%====================================================================
  92. %% example functions on how to use the session middleware
  93. %%====================================================================
  94. %% The server reference :: Pid | LocalName | {Node,Name} | {global,Name}
  95. -define(SESSION_SERVER_REF, ewgi_session_server).
  96. -define(COOKIE_PATH, "/").
  97. -define(SECURE_COOKIE, false).
  98. -define(INCLUDE_IP, true).
  99. -define(SESSION_TIMEOUT, 15 * 60 * 1000). %% 15 minutes
  100. -define(SESSION_STORE_ARGS, [
  101. ?SESSION_SERVER_REF,
  102. "server_session_id",
  103. ?COOKIE_PATH,
  104. ?SECURE_COOKIE,
  105. ?SESSION_TIMEOUT,
  106. ?INCLUDE_IP
  107. ]).
  108. create_example(Ctx) ->
  109. SessionApp = fun ewgi_session:session_create_app/1,
  110. Ctx1 = ?MODULE:load_session(Ctx, ?SESSION_STORE_ARGS),
  111. Ctx2 = SessionApp(Ctx1),
  112. ?MODULE:store_session(Ctx2, ?SESSION_STORE_ARGS).
  113. delete_example(Ctx) ->
  114. SessionApp = fun ewgi_session:session_delete_app/1,
  115. Ctx1 = ?MODULE:load_session(Ctx, ?SESSION_STORE_ARGS),
  116. Ctx2 = SessionApp(Ctx1),
  117. ?MODULE:store_session(Ctx2, ?SESSION_STORE_ARGS).