Skip to content

Commit

Permalink
Cookie store: test that >4K raises CookieOverflow and that unverifiab…
Browse files Browse the repository at this point in the history
…le cookies are automatically deleted.

git-svn-id: http://svn-commit.rubyonrails.org/rails/trunk@6294 5ecf4fe2-1ee6-0310-87b1-e25e094e27de
  • Loading branch information
jeremy committed Mar 3, 2007
1 parent a7b90b7 commit a0563bf
Show file tree
Hide file tree
Showing 2 changed files with 47 additions and 14 deletions.
5 changes: 4 additions & 1 deletion actionpack/lib/action_controller/session/cookie_store.rb
Expand Up @@ -89,7 +89,10 @@ def marshal(session)
def unmarshal(cookie)
if cookie
data, digest = CGI.unescape(cookie).split('--')
raise TamperedWithCookie unless digest == generate_digest(data)
unless digest == generate_digest(data)
delete
raise TamperedWithCookie
end
Marshal.load(Base64.decode64(data))
end
end
Expand Down
56 changes: 43 additions & 13 deletions actionpack/test/controller/session/cookie_store_test.rb
Expand Up @@ -68,56 +68,86 @@ def test_restore_unmarshals_good_cookies
end
end

def test_restore_deletes_tampered_cookies
set_cookie! 'a--b'
new_session do |session|
assert_raise(CGI::Session::CookieStore::TamperedWithCookie) { session['fail'] }
assert_cookie_deleted session
end
end

def test_close_doesnt_write_cookie_if_data_is_blank
new_session do |session|
assert_nil session.cgi.output_cookies, session.cgi.output_cookies.inspect
assert_no_cookies session
session.close
assert_nil session.cgi.output_cookies, session.cgi.output_cookies.inspect
assert_no_cookies session
end
end

def test_close_doesnt_write_cookie_if_data_is_unchanged
set_cookie! Cookies::TYPICAL.first
new_session do |session|
assert_nil session.cgi.output_cookies, session.cgi.output_cookies.inspect
assert_no_cookies session
session['user_id'] = session['user_id']
session.close
assert_nil session.cgi.output_cookies, session.cgi.output_cookies.inspect
assert_no_cookies session
end
end

def test_close_raises_when_data_overflows
set_cookie! Cookies::EMPTY.first
new_session do |session|
session['overflow'] = 'bye!' * 1024
assert_raise(CGI::Session::CookieStore::CookieOverflow) { session.close }
assert_no_cookies session
end
end

def test_close_marshals_and_writes_cookie
set_cookie! Cookies::TYPICAL.first
new_session do |session|
assert_nil session.cgi.output_cookies, session.cgi.output_cookies.inspect
assert_no_cookies session
session['flash'] = {}
assert_nil session.cgi.output_cookies, session.cgi.output_cookies.inspect
assert_no_cookies session
session.close
assert_equal 1, session.cgi.output_cookies.size
cookie = session.cgi.output_cookies.first
assert_equal ['_myapp_session', [Cookies::FLASHED.first]],
[cookie.name, cookie.value]
assert_cookie cookie, Cookies::FLASHED.first
end
end

def test_delete_writes_expired_empty_cookie_and_sets_data_to_nil
set_cookie! Cookies::TYPICAL.first
new_session do |session|
assert_nil session.cgi.output_cookies, session.cgi.output_cookies.inspect
assert_no_cookies session
session.delete
assert_equal 1, session.cgi.output_cookies.size
cookie = session.cgi.output_cookies.first
assert_equal ['_myapp_session', [], 1.year.ago.to_date],
[cookie.name, cookie.value, cookie.expires.to_date]
assert_cookie_deleted session

# @data is set to nil so #close doesn't send another cookie.
session.close
assert_equal ['_myapp_session', [], 1.year.ago.to_date],
[cookie.name, cookie.value, cookie.expires.to_date]
assert_cookie_deleted session
end
end

private
def assert_no_cookies(session)
assert_nil session.cgi.output_cookies, session.cgi.output_cookies.inspect
end

def assert_cookie_deleted(session, message = 'Expected session deletion cookie to be set')
assert_equal 1, session.cgi.output_cookies.size
cookie = session.cgi.output_cookies.first
assert_cookie cookie, nil, 1.year.ago.to_date, message
end

def assert_cookie(cookie, value = nil, expires = nil, message = nil)
assert_equal '_myapp_session', cookie.name, message
assert_equal [value].compact, cookie.value, message
assert_equal expires, cookie.expires ? cookie.expires.to_date : cookie.expires, message
end

def set_cookie!(value)
ENV['HTTP_COOKIE'] = "_myapp_session=#{value}"
end
Expand Down

0 comments on commit a0563bf

Please sign in to comment.