Skip to content

Commit

Permalink
Refactor cookie_only option to survive multiple requests and add regr…
Browse files Browse the repository at this point in the history
…ession tests. References #10048. [theflow]

git-svn-id: http://svn-commit.rubyonrails.org/rails/trunk@8176 5ecf4fe2-1ee6-0310-87b1-e25e094e27de
  • Loading branch information
NZKoz committed Nov 21, 2007
1 parent 6967b42 commit 41fb490
Show file tree
Hide file tree
Showing 2 changed files with 95 additions and 3 deletions.
10 changes: 7 additions & 3 deletions actionpack/lib/action_controller/cgi_process.rb
Expand Up @@ -33,21 +33,21 @@ def process_cgi(cgi, session_options = {}) #:nodoc:
end end


class CgiRequest < AbstractRequest #:nodoc: class CgiRequest < AbstractRequest #:nodoc:
attr_accessor :cgi, :session_options, :cookie_only attr_accessor :cgi, :session_options
class SessionFixationAttempt < StandardError; end #:nodoc: class SessionFixationAttempt < StandardError; end #:nodoc:


DEFAULT_SESSION_OPTIONS = { DEFAULT_SESSION_OPTIONS = {
:database_manager => CGI::Session::CookieStore, # store data in cookie :database_manager => CGI::Session::CookieStore, # store data in cookie
:prefix => "ruby_sess.", # prefix session file names :prefix => "ruby_sess.", # prefix session file names
:session_path => "/", # available to all paths in app :session_path => "/", # available to all paths in app
:session_key => "_session_id",
:cookie_only => true :cookie_only => true
} unless const_defined?(:DEFAULT_SESSION_OPTIONS) } unless const_defined?(:DEFAULT_SESSION_OPTIONS)


def initialize(cgi, session_options = {}) def initialize(cgi, session_options = {})
@cgi = cgi @cgi = cgi
@session_options = session_options @session_options = session_options
@env = @cgi.send!(:env_table) @env = @cgi.send!(:env_table)
@cookie_only = session_options.delete :cookie_only
super() super()
end end


Expand Down Expand Up @@ -112,7 +112,7 @@ def session
@session = Hash.new @session = Hash.new
else else
stale_session_check! do stale_session_check! do
if @cookie_only && request_parameters[session_options_with_string_keys['session_key']] if cookie_only? && query_parameters[session_options_with_string_keys['session_key']]
raise SessionFixationAttempt raise SessionFixationAttempt
end end
case value = session_options_with_string_keys['new_session'] case value = session_options_with_string_keys['new_session']
Expand Down Expand Up @@ -158,6 +158,10 @@ def new_session
end end
end end


def cookie_only?
session_options_with_string_keys['cookie_only']
end

def stale_session_check! def stale_session_check!
yield yield
rescue ArgumentError => argument_error rescue ArgumentError => argument_error
Expand Down
88 changes: 88 additions & 0 deletions actionpack/test/controller/session_fixation_test.rb
@@ -0,0 +1,88 @@
require File.dirname(__FILE__) + '/../abstract_unit'

class SessionFixationTest < Test::Unit::TestCase
class MockCGI < CGI #:nodoc:
attr_accessor :stdoutput, :env_table

def initialize(env, data = '')
self.env_table = env
self.stdoutput = StringIO.new
super(nil, StringIO.new(data))
end
end

class TestController < ActionController::Base
session :session_key => '_myapp_session_id', :secret => 'secret', :except => :default_session_key
session :cookie_only => false, :only => :allow_session_fixation

def default_session_key
render :text => "default_session_key"
end

def custom_session_key
render :text => "custom_session_key: #{params[:id]}"
end

def allow_session_fixation
render :text => "allow_session_fixation"
end

def rescue_action(e) raise end
end

def setup
@controller = TestController.new
end

def test_should_be_able_to_make_a_successful_request
cgi = mock_cgi_for_request_to(:custom_session_key, :id => 1)

assert_nothing_raised do
@controller.send(:process, ActionController::CgiRequest.new(cgi, {}), ActionController::CgiResponse.new(cgi))
end
assert_equal 'custom_session_key: 1', @controller.response.body
assert_not_nil @controller.session
end

def test_should_catch_session_fixation_attempt
cgi = mock_cgi_for_request_to(:custom_session_key, :_myapp_session_id => 42)

assert_raises ActionController::CgiRequest::SessionFixationAttempt do
@controller.send(:process, ActionController::CgiRequest.new(cgi, {}), ActionController::CgiResponse.new(cgi))
end
assert_nil @controller.session
end

def test_should_not_catch_session_fixation_attempt_when_cookie_only_setting_is_disabled
cgi = mock_cgi_for_request_to(:allow_session_fixation, :_myapp_session_id => 42)

assert_nothing_raised do
@controller.send(:process, ActionController::CgiRequest.new(cgi, {}), ActionController::CgiResponse.new(cgi))
end
assert ! @controller.response.body.blank?
assert_not_nil @controller.session
end

def test_should_catch_session_fixation_attempt_with_default_session_key
ActionController::Base.session_store = :p_store # using the default session_key is not possible with cookie store
cgi = mock_cgi_for_request_to(:default_session_key, :_session_id => 42)

assert_raises ActionController::CgiRequest::SessionFixationAttempt do
@controller.send(:process, ActionController::CgiRequest.new(cgi, {}), ActionController::CgiResponse.new(cgi))
end
assert @controller.response.body.blank?
assert_nil @controller.session
end

private

def mock_cgi_for_request_to(action, params = {})
MockCGI.new({
"REQUEST_METHOD" => "GET",
"QUERY_STRING" => "action=#{action}&#{params.to_query}",
"REQUEST_URI" => "/",
"SERVER_PORT" => "80",
"HTTP_HOST" => "testdomain.com" }, '')
end

end

0 comments on commit 41fb490

Please sign in to comment.