2007-12-21 08:48:59 +01:00
|
|
|
require 'action_controller/session/cookie_store'
|
2007-01-22 14:43:50 +01:00
|
|
|
require 'action_controller/session/drb_store'
|
|
|
|
require 'action_controller/session/mem_cache_store'
|
|
|
|
if Object.const_defined?(:ActiveRecord)
|
|
|
|
require 'action_controller/session/active_record_store'
|
|
|
|
end
|
|
|
|
|
|
|
|
module ActionController #:nodoc:
|
|
|
|
module SessionManagement #:nodoc:
|
|
|
|
def self.included(base)
|
2007-12-21 08:48:59 +01:00
|
|
|
base.class_eval do
|
|
|
|
extend ClassMethods
|
|
|
|
alias_method_chain :process, :session_management_support
|
|
|
|
alias_method_chain :process_cleanup, :session_management_support
|
|
|
|
end
|
2007-01-22 14:43:50 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
module ClassMethods
|
2008-05-18 06:22:34 +02:00
|
|
|
# Set the session store to be used for keeping the session data between requests.
|
|
|
|
# By default, sessions are stored in browser cookies (<tt>:cookie_store</tt>),
|
|
|
|
# but you can also specify one of the other included stores (<tt>:active_record_store</tt>,
|
|
|
|
# <tt>:p_store</tt>, <tt>:drb_store</tt>, <tt>:mem_cache_store</tt>, or
|
|
|
|
# <tt>:memory_store</tt>) or your own custom class.
|
2007-01-22 14:43:50 +01:00
|
|
|
def session_store=(store)
|
|
|
|
ActionController::CgiRequest::DEFAULT_SESSION_OPTIONS[:database_manager] =
|
|
|
|
store.is_a?(Symbol) ? CGI::Session.const_get(store == :drb_store ? "DRbStore" : store.to_s.camelize) : store
|
|
|
|
end
|
|
|
|
|
|
|
|
# Returns the session store class currently used.
|
|
|
|
def session_store
|
|
|
|
ActionController::CgiRequest::DEFAULT_SESSION_OPTIONS[:database_manager]
|
|
|
|
end
|
|
|
|
|
|
|
|
# Returns the hash used to configure the session. Example use:
|
|
|
|
#
|
|
|
|
# ActionController::Base.session_options[:session_secure] = true # session only available over HTTPS
|
|
|
|
def session_options
|
|
|
|
ActionController::CgiRequest::DEFAULT_SESSION_OPTIONS
|
|
|
|
end
|
|
|
|
|
|
|
|
# Specify how sessions ought to be managed for a subset of the actions on
|
|
|
|
# the controller. Like filters, you can specify <tt>:only</tt> and
|
|
|
|
# <tt>:except</tt> clauses to restrict the subset, otherwise options
|
|
|
|
# apply to all actions on this controller.
|
|
|
|
#
|
|
|
|
# The session options are inheritable, as well, so if you specify them in
|
|
|
|
# a parent controller, they apply to controllers that extend the parent.
|
|
|
|
#
|
|
|
|
# Usage:
|
|
|
|
#
|
|
|
|
# # turn off session management for all actions.
|
|
|
|
# session :off
|
|
|
|
#
|
|
|
|
# # turn off session management for all actions _except_ foo and bar.
|
|
|
|
# session :off, :except => %w(foo bar)
|
|
|
|
#
|
|
|
|
# # turn off session management for only the foo and bar actions.
|
|
|
|
# session :off, :only => %w(foo bar)
|
|
|
|
#
|
|
|
|
# # the session will only work over HTTPS, but only for the foo action
|
|
|
|
# session :only => :foo, :session_secure => true
|
|
|
|
#
|
2008-10-27 07:47:01 +01:00
|
|
|
# # the session by default uses HttpOnly sessions for security reasons.
|
|
|
|
# # this can be switched off.
|
|
|
|
# session :only => :foo, :session_http_only => false
|
|
|
|
#
|
2007-01-22 14:43:50 +01:00
|
|
|
# # the session will only be disabled for 'foo', and only if it is
|
|
|
|
# # requested as a web service
|
|
|
|
# session :off, :only => :foo,
|
|
|
|
# :if => Proc.new { |req| req.parameters[:ws] }
|
|
|
|
#
|
2007-12-21 08:48:59 +01:00
|
|
|
# # the session will be disabled for non html/ajax requests
|
|
|
|
# session :off,
|
|
|
|
# :if => Proc.new { |req| !(req.format.html? || req.format.js?) }
|
|
|
|
#
|
2008-05-18 06:22:34 +02:00
|
|
|
# # turn the session back on, useful when it was turned off in the
|
|
|
|
# # application controller, and you need it on in another controller
|
|
|
|
# session :on
|
|
|
|
#
|
2007-01-22 14:43:50 +01:00
|
|
|
# All session options described for ActionController::Base.process_cgi
|
|
|
|
# are valid arguments.
|
|
|
|
def session(*args)
|
2007-12-21 08:48:59 +01:00
|
|
|
options = args.extract_options!
|
2007-01-22 14:43:50 +01:00
|
|
|
|
2008-05-18 06:22:34 +02:00
|
|
|
options[:disabled] = false if args.delete(:on)
|
2007-01-22 14:43:50 +01:00
|
|
|
options[:disabled] = true if !args.empty?
|
|
|
|
options[:only] = [*options[:only]].map { |o| o.to_s } if options[:only]
|
|
|
|
options[:except] = [*options[:except]].map { |o| o.to_s } if options[:except]
|
|
|
|
if options[:only] && options[:except]
|
|
|
|
raise ArgumentError, "only one of either :only or :except are allowed"
|
|
|
|
end
|
|
|
|
|
2008-10-27 07:47:01 +01:00
|
|
|
write_inheritable_array(:session_options, [options])
|
2007-01-22 14:43:50 +01:00
|
|
|
end
|
|
|
|
|
2007-12-21 08:48:59 +01:00
|
|
|
# So we can declare session options in the Rails initializer.
|
|
|
|
alias_method :session=, :session
|
|
|
|
|
2007-01-22 14:43:50 +01:00
|
|
|
def cached_session_options #:nodoc:
|
2008-10-27 07:47:01 +01:00
|
|
|
@session_options ||= read_inheritable_attribute(:session_options) || []
|
2007-01-22 14:43:50 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
def session_options_for(request, action) #:nodoc:
|
|
|
|
if (session_options = cached_session_options).empty?
|
|
|
|
{}
|
|
|
|
else
|
|
|
|
options = {}
|
|
|
|
|
|
|
|
action = action.to_s
|
|
|
|
session_options.each do |opts|
|
|
|
|
next if opts[:if] && !opts[:if].call(request)
|
|
|
|
if opts[:only] && opts[:only].include?(action)
|
|
|
|
options.merge!(opts)
|
|
|
|
elsif opts[:except] && !opts[:except].include?(action)
|
|
|
|
options.merge!(opts)
|
|
|
|
elsif !opts[:only] && !opts[:except]
|
|
|
|
options.merge!(opts)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
if options.empty? then options
|
|
|
|
else
|
|
|
|
options.delete :only
|
|
|
|
options.delete :except
|
|
|
|
options.delete :if
|
|
|
|
options[:disabled] ? false : options
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def process_with_session_management_support(request, response, method = :perform_action, *arguments) #:nodoc:
|
|
|
|
set_session_options(request)
|
|
|
|
process_without_session_management_support(request, response, method, *arguments)
|
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
def set_session_options(request)
|
|
|
|
request.session_options = self.class.session_options_for(request, request.parameters["action"] || "index")
|
|
|
|
end
|
|
|
|
|
|
|
|
def process_cleanup_with_session_management_support
|
|
|
|
clear_persistent_model_associations
|
2007-02-09 09:04:31 +01:00
|
|
|
process_cleanup_without_session_management_support
|
2007-01-22 14:43:50 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
# Clear cached associations in session data so they don't overflow
|
|
|
|
# the database field. Only applies to ActiveRecordStore since there
|
|
|
|
# is not a standard way to iterate over session data.
|
|
|
|
def clear_persistent_model_associations #:doc:
|
2007-02-09 09:04:31 +01:00
|
|
|
if defined?(@_session) && @_session.respond_to?(:data)
|
|
|
|
session_data = @_session.data
|
2007-01-22 14:43:50 +01:00
|
|
|
|
|
|
|
if session_data && session_data.respond_to?(:each_value)
|
|
|
|
session_data.each_value do |obj|
|
|
|
|
obj.clear_association_cache if obj.respond_to?(:clear_association_cache)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|