Skip to content

Commit

Permalink
magic link
Browse files Browse the repository at this point in the history
  • Loading branch information
alkesh committed Aug 2, 2024
1 parent fe1ee8e commit ef4defb
Show file tree
Hide file tree
Showing 6 changed files with 115 additions and 9 deletions.
10 changes: 10 additions & 0 deletions app/controllers/claims_controller.rb
Original file line number Diff line number Diff line change
Expand Up @@ -79,9 +79,19 @@ def check_page_is_in_sequence

raise ActionController::RoutingError.new("Not Found for #{params[:slug]}") unless page_sequence.in_sequence?(params[:slug])

handle_magic_link if page_sequence.magic_link?
redirect_to claim_path(current_journey_routing_name, next_required_slug) unless page_sequence.has_completed_journey_until?(params[:slug])
end

def handle_magic_link
otp = OneTimePassword::Validator.new(params[:code], answers.sent_one_time_password_at)
if otp.valid?
journey_session.answers.assign_attributes(email_verified: true)
journey_session.save!
session[:slugs] << page_sequence.next_required_slug
end
end

def initialize_session_slug_history
session[:slugs] ||= []
end
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,9 @@ class EmailAddressForm < Form

def save
journey_session.answers.assign_attributes(
email_address: email_address
email_address: email_address,
sent_one_time_password_at: Time.now,
email_verified: email_verified
)
journey_session.save!

Expand All @@ -15,6 +17,15 @@ def save

private

def email_address_changed?
email_address != answers.email_address
end

def email_verified
return nil if email_address_changed?
answers.email_verified
end

def otp_code
@otp_code ||= OneTimePassword::Generator.new.code
end
Expand Down
5 changes: 5 additions & 0 deletions app/models/journeys/page_sequence.rb
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ class PageSequence

DEAD_END_SLUGS = %w[complete existing-session eligible-later future-eligibility ineligible]
OPTIONAL_SLUGS = %w[postcode-search select-home-address reset-claim]
MAGIC_LINK_SLUGS = %w[consent]

def initialize(slug_sequence, completed_slugs, current_slug, journey_session)
@current_slug = current_slug
Expand Down Expand Up @@ -61,6 +62,10 @@ def next_required_slug
(slugs - completed_slugs - OPTIONAL_SLUGS).first
end

def magic_link?
MAGIC_LINK_SLUGS.include?(current_slug)
end

private

delegate :answers, to: :@journey_session
Expand Down
17 changes: 10 additions & 7 deletions spec/features/early_years_payment/provider/happy_path_spec.rb
Original file line number Diff line number Diff line change
@@ -1,6 +1,8 @@
require "rails_helper"

RSpec.feature "Early years payment provider" do
let(:journey_session) { Journeys::EarlyYearsPayment::Provider::Session.last }

scenario "happy path claim" do
when_early_years_payment_provider_journey_configuration_exists

Expand All @@ -17,13 +19,14 @@
expect(page).to have_content("We have sent an email to [email protected]")

mail = ActionMailer::Base.deliveries.last
mail_personalisation = mail[:personalisation].unparsed_value
expect(mail_personalisation[:one_time_password]).to match(/\A\d{6}\Z/)

# TODO - uncomment below when magic link functionality in place
# expect(page).to have_content("Declaration of Employee Consent")
# check "I confirm that I have obtained consent from my employee and have provided them with the relevant privacy notice."
# click_button "Continue"
otp = mail[:personalisation].unparsed_value[:one_time_password]
expect(otp).to match(/\A\d{6}\Z/)

visit claim_path(Journeys::EarlyYearsPayment::Provider::ROUTING_NAME, :consent, code: otp)
expect(journey_session.reload.answers.email_verified).to be true
expect(page).to have_content("Declaration of Employee Consent")
check "I confirm that I have obtained consent from my employee and have provided them with the relevant privacy notice."
click_button "Continue"
end

scenario "send another link"
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,6 @@

let(:journey) { Journeys::EarlyYearsPayment::Provider }
let(:journey_session) { build(:early_years_payment_provider_session) }
# let(:params) { ActionController::Parameters.new({journey: "test-journey", slug: "test_slug", claim: claim_params}) }

let(:params) do
ActionController::Parameters.new(claim: {email_address: email_address})
Expand All @@ -18,6 +17,22 @@
describe "#save" do
subject { form.save }

around do |example|
travel_to DateTime.new(2024, 1, 1, 12, 0, 0) do
example.run
end
end

before do
allow(OneTimePassword::Generator).to receive(:new).and_return(
instance_double(OneTimePassword::Generator, code: "111111")
)
end

let(:policy) { journey_session.answers.policy }
let(:claim_subject) { I18n.t("#{policy.locale_key}.claim_subject") }
let(:email_subject) { claim_subject }

it { should be_truthy }

it "sets the email address" do
Expand All @@ -26,5 +41,51 @@
eq(email_address)
)
end

it "sends an email" do
subject

expect(email_address).to have_received_email(
"e0b78a08-601b-40ba-a97f-61fb00a7c951",
email_subject: email_subject,
one_time_password: "111111"
)
end

it "updates sent_one_time_password_at" do
subject
expect(journey_session.answers.sent_one_time_password_at).to(
eq(DateTime.new(2024, 1, 1, 12, 0, 0))
)
end

it "resets email_verified" do
subject
expect(journey_session.answers.email_verified).to be_nil
end

context "when the email address has been previously verified, and a new one is submitted" do
before do
journey_session.answers.assign_attributes(email_address: "[email protected]", email_verified: true)
journey_session.save!
end

it "resets email_verified" do
subject
expect(journey_session.answers.email_verified).to be_nil
end
end

context "when the email address submitted has been previously verified, and is the same" do
before do
journey_session.answers.assign_attributes(email_address: email_address, email_verified: true)
journey_session.save!
end

it "returns email_verified" do
subject
expect(journey_session.answers.email_verified).to be true
end
end
end
end
16 changes: 16 additions & 0 deletions spec/models/journeys/page_sequence_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -219,4 +219,20 @@
expect(page_sequence.next_required_slug).to eq("second-slug")
end
end

describe "#magic_link?" do
subject { page_sequence.magic_link? }

context "when the current slug is not a magic link" do
let(:current_slug) { "whatever" }

it { is_expected.to be false }
end

context "when the current slug is not a magic link" do
let(:current_slug) { "consent" }

it { is_expected.to be true }
end
end
end

0 comments on commit ef4defb

Please sign in to comment.