Browse Source

fix: different cookies for different users

master v1.1.5
Dmitrii Golub 9 years ago
parent
commit
bdb89e50f8
  1. 1
      Gemfile
  2. 2
      app/controllers/devise/two_factor_authentication_controller.rb
  3. 11
      lib/two_factor_authentication/hooks/two_factor_authenticatable.rb
  4. 62
      spec/features/two_factor_authenticatable_spec.rb
  5. 17
      spec/support/features_spec_helper.rb

1
Gemfile

@ -27,4 +27,5 @@ end
group :test do group :test do
gem 'rack_session_access' gem 'rack_session_access'
gem 'ammeter' gem 'ammeter'
gem 'pry'
end end

2
app/controllers/devise/two_factor_authentication_controller.rb

@ -22,7 +22,7 @@ class Devise::TwoFactorAuthenticationController < DeviseController
if expires_seconds && expires_seconds > 0 if expires_seconds && expires_seconds > 0
cookies.signed[TwoFactorAuthentication::REMEMBER_TFA_COOKIE_NAME] = { cookies.signed[TwoFactorAuthentication::REMEMBER_TFA_COOKIE_NAME] = {
value: true, value: "#{resource.class}-#{resource.id}",
expires: expires_seconds.from_now expires: expires_seconds.from_now
} }
end end

11
lib/two_factor_authentication/hooks/two_factor_authenticatable.rb

@ -1,8 +1,15 @@
Warden::Manager.after_authentication do |user, auth, options| Warden::Manager.after_authentication do |user, auth, options|
reset_otp_state_for(user) reset_otp_state_for(user)
if user.respond_to?(:need_two_factor_authentication?) && expected_cookie_value = "#{user.class}-#{user.id}"
!auth.env["action_dispatch.cookies"].signed[TwoFactorAuthentication::REMEMBER_TFA_COOKIE_NAME] actual_cookie_value = auth.env["action_dispatch.cookies"].signed[TwoFactorAuthentication::REMEMBER_TFA_COOKIE_NAME]
if actual_cookie_value.nil?
bypass_by_cookie = false
else
bypass_by_cookie = actual_cookie_value == expected_cookie_value
end
if user.respond_to?(:need_two_factor_authentication?) && !bypass_by_cookie
if auth.session(options[:scope])[TwoFactorAuthentication::NEED_AUTHENTICATION] = user.need_two_factor_authentication?(auth.request) if auth.session(options[:scope])[TwoFactorAuthentication::NEED_AUTHENTICATION] = user.need_two_factor_authentication?(auth.request)
user.send_two_factor_authentication_code user.send_two_factor_authentication_code
end end

62
spec/features/two_factor_authenticatable_spec.rb

@ -138,6 +138,44 @@ feature "User of two factor authentication" do
expect(page).to have_content("You are signed in as Marissa") expect(page).to have_content("You are signed in as Marissa")
expect(page).to have_content("Enter your personal code") expect(page).to have_content("Enter your personal code")
end end
scenario 'TFA should be different for different users' do
visit user_two_factor_authentication_path
fill_in 'code', with: user.otp_code
click_button 'Submit'
tfa_cookie1 = get_tfa_cookie()
logout
reset_session!
user2 = create_user()
login_as(user2)
visit user_two_factor_authentication_path
fill_in 'code', with: user2.otp_code
click_button 'Submit'
tfa_cookie2 = get_tfa_cookie()
expect(tfa_cookie1).not_to eq tfa_cookie2
end
scenario 'TFA should be unique for specific user' do
visit user_two_factor_authentication_path
fill_in 'code', with: user.otp_code
click_button 'Submit'
tfa_cookie1 = get_tfa_cookie()
logout
reset_session!
user2 = create_user()
set_tfa_cookie(tfa_cookie1)
login_as(user2)
visit dashboard_path
expect(page).to have_content('Enter your personal code')
end
end end
it 'sets the warden session need_two_factor_authentication key to true' do it 'sets the warden session need_two_factor_authentication key to true' do
@ -151,7 +189,11 @@ feature "User of two factor authentication" do
let(:user) { create_user } let(:user) { create_user }
scenario 'when UserOtpSender#reset_otp_state is defined' do scenario 'when UserOtpSender#reset_otp_state is defined' do
stub_const 'UserOtpSender', Class.new klass = stub_const 'UserOtpSender', Class.new
klass.class_eval do
def reset_otp_state; end
end
otp_sender = instance_double(UserOtpSender) otp_sender = instance_double(UserOtpSender)
expect(UserOtpSender).to receive(:new).with(user).and_return(otp_sender) expect(UserOtpSender).to receive(:new).with(user).and_return(otp_sender)
@ -162,7 +204,11 @@ feature "User of two factor authentication" do
end end
scenario 'when UserOtpSender#reset_otp_state is not defined' do scenario 'when UserOtpSender#reset_otp_state is not defined' do
stub_const 'UserOtpSender', Class.new klass = stub_const 'UserOtpSender', Class.new
klass.class_eval do
def reset_otp_state; end
end
otp_sender = instance_double(UserOtpSender) otp_sender = instance_double(UserOtpSender)
allow(otp_sender).to receive(:respond_to?).with(:reset_otp_state).and_return(false) allow(otp_sender).to receive(:respond_to?).with(:reset_otp_state).and_return(false)
@ -182,7 +228,11 @@ feature "User of two factor authentication" do
visit new_user_session_path visit new_user_session_path
complete_sign_in_form_for(user) complete_sign_in_form_for(user)
stub_const 'UserOtpSender', Class.new klass = stub_const 'UserOtpSender', Class.new
klass.class_eval do
def reset_otp_state; end
end
otp_sender = instance_double(UserOtpSender) otp_sender = instance_double(UserOtpSender)
expect(UserOtpSender).to receive(:new).with(user).and_return(otp_sender) expect(UserOtpSender).to receive(:new).with(user).and_return(otp_sender)
@ -195,7 +245,11 @@ feature "User of two factor authentication" do
visit new_user_session_path visit new_user_session_path
complete_sign_in_form_for(user) complete_sign_in_form_for(user)
stub_const 'UserOtpSender', Class.new klass = stub_const 'UserOtpSender', Class.new
klass.class_eval do
def reset_otp_state; end
end
otp_sender = instance_double(UserOtpSender) otp_sender = instance_double(UserOtpSender)
allow(otp_sender).to receive(:respond_to?).with(:reset_otp_state).and_return(false) allow(otp_sender).to receive(:respond_to?).with(:reset_otp_state).and_return(false)

17
spec/support/features_spec_helper.rb

@ -10,6 +10,22 @@ module FeaturesSpecHelper
fill_in "Password", with: 'password' fill_in "Password", with: 'password'
find('.actions input').click # 'Sign in' or 'Log in' find('.actions input').click # 'Sign in' or 'Log in'
end end
def set_cookie key, value
page.driver.browser.set_cookie [key, value].join('=')
end
def get_cookie key
Capybara.current_session.driver.request.cookies[key]
end
def set_tfa_cookie value
set_cookie TwoFactorAuthentication::REMEMBER_TFA_COOKIE_NAME, value
end
def get_tfa_cookie
get_cookie TwoFactorAuthentication::REMEMBER_TFA_COOKIE_NAME
end
end end
RSpec.configure do |config| RSpec.configure do |config|
@ -24,4 +40,3 @@ RSpec.configure do |config|
Warden.test_reset! Warden.test_reset!
end end
end end

Loading…
Cancel
Save