2015-09-11 14:41:01 +05:30
|
|
|
require 'spec_helper'
|
|
|
|
|
|
|
|
feature 'Login', feature: true do
|
|
|
|
describe 'with two-factor authentication' do
|
|
|
|
context 'with valid username/password' do
|
|
|
|
let(:user) { create(:user, :two_factor) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
login_with(user)
|
|
|
|
expect(page).to have_content('Two-factor Authentication')
|
|
|
|
end
|
|
|
|
|
|
|
|
def enter_code(code)
|
|
|
|
fill_in 'Two-factor authentication code', with: code
|
|
|
|
click_button 'Verify code'
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not show a "You are already signed in." error message' do
|
|
|
|
enter_code(user.current_otp)
|
|
|
|
expect(page).not_to have_content('You are already signed in.')
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'using one-time code' do
|
|
|
|
it 'allows login with valid code' do
|
|
|
|
enter_code(user.current_otp)
|
|
|
|
expect(current_path).to eq root_path
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'blocks login with invalid code' do
|
|
|
|
enter_code('foo')
|
|
|
|
expect(page).to have_content('Invalid two-factor code')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'allows login with invalid code, then valid code' do
|
|
|
|
enter_code('foo')
|
|
|
|
expect(page).to have_content('Invalid two-factor code')
|
|
|
|
|
|
|
|
enter_code(user.current_otp)
|
|
|
|
expect(current_path).to eq root_path
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'using backup code' do
|
|
|
|
let(:codes) { user.generate_otp_backup_codes! }
|
|
|
|
|
|
|
|
before do
|
|
|
|
expect(codes.size).to eq 10
|
|
|
|
|
|
|
|
# Ensure the generated codes get saved
|
|
|
|
user.save
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with valid code' do
|
|
|
|
it 'allows login' do
|
|
|
|
enter_code(codes.sample)
|
|
|
|
expect(current_path).to eq root_path
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'invalidates the used code' do
|
|
|
|
expect { enter_code(codes.sample) }.
|
|
|
|
to change { user.reload.otp_backup_codes.size }.by(-1)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with invalid code' do
|
|
|
|
it 'blocks login' do
|
|
|
|
code = codes.sample
|
|
|
|
expect(user.invalidate_otp_backup_code!(code)).to eq true
|
|
|
|
|
|
|
|
user.save!
|
|
|
|
expect(user.reload.otp_backup_codes.size).to eq 9
|
|
|
|
|
|
|
|
enter_code(code)
|
|
|
|
expect(page).to have_content('Invalid two-factor code.')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'without two-factor authentication' do
|
|
|
|
let(:user) { create(:user) }
|
|
|
|
|
|
|
|
it 'allows basic login' do
|
|
|
|
login_with(user)
|
|
|
|
expect(current_path).to eq root_path
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not show a "You are already signed in." error message' do
|
|
|
|
login_with(user)
|
|
|
|
expect(page).not_to have_content('You are already signed in.')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'blocks invalid login' do
|
|
|
|
user = create(:user, password: 'not-the-default')
|
|
|
|
|
|
|
|
login_with(user)
|
2015-10-24 18:46:33 +05:30
|
|
|
expect(page).to have_content('Invalid login or password.')
|
2015-09-11 14:41:01 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|