diff --git a/app/models/automated_checks/claim_verifiers/early_years_payments/identity.rb b/app/models/automated_checks/claim_verifiers/early_years_payments/identity.rb new file mode 100644 index 0000000000..0037fe2beb --- /dev/null +++ b/app/models/automated_checks/claim_verifiers/early_years_payments/identity.rb @@ -0,0 +1,74 @@ +module AutomatedChecks + module ClaimVerifiers + module EarlyYearsPayments + class Identity < AutomatedChecks::ClaimVerifiers::Identity + TASK_NAME = "identity_confirmation".freeze + + def intialize(claim:, admin_user: nil) + @claim = claim + @admin_user = admin_user + end + + def perform + return unless awaiting_task?(TASK_NAME) + + if one_login_idv_match? + create_task(match: nil, passed: true) + elsif one_login_idv_partial_match? + create_task(match: :any, passed: nil) + + create_note( + body: <<-HTML + [GOV UK One Login Name] - Names partially match: +
+                Provider: "#{claim.eligibility.practitioner_entered_full_name}"
+                GOV.UK One Login: "#{claim.onelogin_idv_full_name}"
+              
+ HTML + ) + elsif claim.one_login_idv_match? + create_task(match: nil, passed: false) + + create_note( + body: <<-HTML + [GOV UK One Login Name] - Names do not match: +
+                Provider: "#{claim.eligibility.practitioner_entered_full_name}"
+                GOV.UK One Login: "#{claim.onelogin_idv_full_name}"
+              
+ HTML + ) + else + create_task(match: :none, passed: false) + + create_note( + body: <<-HTML + [GOV UK One Login] - IDV mismatch: +
+                GOV.UK One Login Name: "#{claim.onelogin_idv_full_name}"
+                GOV.UK One Login DOB: "#{claim.onelogin_idv_date_of_birth}"
+              
+ HTML + ) + end + end + + private + + attr_accessor :claim, :admin_user + + def one_login_idv_match? + return false unless claim.one_login_idv_match? + + claim.eligibility.practitioner_and_provider_entered_names_match? + end + + def one_login_idv_partial_match? + return false unless claim.one_login_idv_match? + + claim.eligibility.practitioner_and_provider_entered_names_partial_match? + end + end + end + end +end diff --git a/app/models/claim.rb b/app/models/claim.rb index 1fbb6016b0..255c2feca5 100644 --- a/app/models/claim.rb +++ b/app/models/claim.rb @@ -443,6 +443,10 @@ def one_login_idv_mismatch? !one_login_idv_name_match? || !one_login_idv_dob_match? end + def one_login_idv_match? + one_login_idv_name_match? && one_login_idv_dob_match? + end + def awaiting_provider_verification? return false unless has_further_education_policy? diff --git a/app/models/policies/early_years_payments.rb b/app/models/policies/early_years_payments.rb index bd2634c6a5..9fd7838fba 100644 --- a/app/models/policies/early_years_payments.rb +++ b/app/models/policies/early_years_payments.rb @@ -7,6 +7,7 @@ module EarlyYearsPayments MIN_QA_THRESHOLD = 10 VERIFIERS = [ + AutomatedChecks::ClaimVerifiers::EarlyYearsPayments::Identity, AutomatedChecks::ClaimVerifiers::StudentLoanPlan # TODO - spec ] diff --git a/app/models/policies/early_years_payments/admin_tasks_presenter.rb b/app/models/policies/early_years_payments/admin_tasks_presenter.rb new file mode 100644 index 0000000000..f409bc557d --- /dev/null +++ b/app/models/policies/early_years_payments/admin_tasks_presenter.rb @@ -0,0 +1,31 @@ +module Policies + module EarlyYearsPayments + class AdminTasksPresenter + attr_reader :claim + + def initialize(claim) + @claim = claim + end + + def identity_confirmation + [] + end + + def provider_entered_claimant_name + claim.eligibility.practitioner_entered_full_name + end + + def one_login_claimant_name + claim.onelogin_idv_full_name + end + + def practitioner_journey_completed? + claim.submitted_at.present? + end + + def qualifications + [] + end + end + end +end diff --git a/app/models/policies/early_years_payments/eligibility.rb b/app/models/policies/early_years_payments/eligibility.rb index 9629cf1a75..54b7b8be61 100644 --- a/app/models/policies/early_years_payments/eligibility.rb +++ b/app/models/policies/early_years_payments/eligibility.rb @@ -20,6 +20,20 @@ def eligible_ey_provider def provider_claim_submitted? provider_claim_submitted_at.present? end + + def practitioner_entered_full_name + "#{practitioner_first_name} #{practitioner_surname}" + end + + def practitioner_and_provider_entered_names_match? + practitioner_first_name.downcase == claim.onelogin_idv_first_name.downcase && + practitioner_surname.downcase == claim.onelogin_idv_last_name.downcase + end + + def practitioner_and_provider_entered_names_partial_match? + practitioner_first_name.downcase == claim.onelogin_idv_first_name.downcase || + practitioner_surname.downcase == claim.onelogin_idv_last_name.downcase + end end end end diff --git a/app/views/admin/tasks/early_years_payments/identity_confirmation.html.erb b/app/views/admin/tasks/early_years_payments/identity_confirmation.html.erb new file mode 100644 index 0000000000..c8146aa0c0 --- /dev/null +++ b/app/views/admin/tasks/early_years_payments/identity_confirmation.html.erb @@ -0,0 +1,60 @@ +<% content_for(:page_title) { page_title("Claim #{@claim.reference} identity confirmation check for #{@claim.policy.short_name}") } %> + +<% content_for :back_link do %> + <%= govuk_back_link href: admin_claim_tasks_path(@claim) %> +<% end %> + +<%= render "shared/error_summary", instance: @task, errored_field_id_overrides: { "passed": "task_passed_true" } if @task.errors.any? %> + +
+ <%= render claim_summary_view, claim: @claim, heading: "Identity confirmation" %> + +
+

<%= @current_task_name.humanize %>

+
+ +
+

+ <%= I18n.t( + "admin.tasks.identity_confirmation.title", + claim_full_name: @claim.full_name + ) %> +

+ + + + + + + + + + + + +
+ Provider entered claimant name + + <%= @tasks_presenter.provider_entered_claimant_name %> +
+ Claimant name from One login + + <%= @tasks_presenter.one_login_claimant_name %> +
+ + <% if @tasks_presenter.practitioner_journey_completed? %> + <% if @task.claim_verifier_match_any? && @task.passed.nil? %> + <%= render "form", task_name: "identity_confirmation", claim: @claim %> + <% else %> + <%= render "task_outcome", task: @task %> + <% end %> + <% else %> +
+ This task is not available until the claimant has submitted their + claim. +
+ <% end %> + + <%= render partial: "admin/task_pagination", locals: { task_pagination: @task_pagination } %> +
+
diff --git a/config/locales/en.yml b/config/locales/en.yml index a62a5b1183..35ec2a005c 100644 --- a/config/locales/en.yml +++ b/config/locales/en.yml @@ -1443,6 +1443,9 @@ en: provider_submitted_at: Provider submitted at practitioner_started_at: Claimant started at submitted_at: Claimant submitted at + task_questions: + identity_confirmation: + title: "Do these names match?" early_years_payment_practitioner: journey_name: Claim an early years financial incentive payment - practitioner feedback_email: "help@opsteam.education.gov.uk" diff --git a/spec/features/admin/admin_ey_identity_task_spec.rb b/spec/features/admin/admin_ey_identity_task_spec.rb new file mode 100644 index 0000000000..7f7cb92967 --- /dev/null +++ b/spec/features/admin/admin_ey_identity_task_spec.rb @@ -0,0 +1,379 @@ +require "rails_helper" + +RSpec.describe "Admin EY identity task" do + around do |example| + travel_to DateTime.new(2024, 10, 30, 9, 0, 0) do + example.run + end + end + + context "when the practitioner hasn't completed their half of the claim" do + it "shows that the task is unavailable" do + claim = complete_provider_journey + + sign_in_as_service_operator + + visit admin_claim_tasks_path(claim) + + expect(task_status("Identity confirmation")).to eq("Incomplete") + + click_on "Confirm the claimant made the claim" + + expect(page).to have_content( + "Provider entered claimant name Bobby Bobberson" + ) + + expect(page).to have_content( + "This task is not available until the claimant has submitted their claim" + ) + end + end + + context "when the practitioner has completed their half of the claim" do + context "when OL IDV is a pass and the names match" do + it "passes the task" do + claim = complete_provider_journey + + complete_practitioner_journey( + claim: claim, + date_of_birth: Date.new(1986, 1, 1), + one_login_first_name: "Bobby", + one_login_last_name: "Bobberson", + one_login_date_of_birth: Date.new(1986, 1, 1) + ) + + sign_in_as_service_operator + + visit admin_claim_tasks_path(claim) + + expect(task_status("Identity confirmation")).to eq("Passed") + + click_on "Confirm the claimant made the claim" + + expect(page).to have_content( + "Provider entered claimant name Bobby Bobberson" + ) + + expect(page).to have_content( + "Claimant name from One login Bobby Bobberson" + ) + + expect(page).to have_content( + "This task was performed by GOV.UK One Login on " \ + "30 October 2024 9:00am" + ) + end + end + + context "when OL IDV is a pass and the names don't match" do + context "when the names are a parital match" do + let(:claim) { complete_provider_journey } + + before do + complete_practitioner_journey( + claim: claim, + one_login_first_name: "Robby", + one_login_last_name: "Bobberson", + one_login_date_of_birth: Date.new(1986, 1, 1), + date_of_birth: Date.new(1986, 1, 1) + ) + + sign_in_as_service_operator + end + + it "shows the task as a partial match" do + visit admin_claim_tasks_path(claim) + + expect(task_status("Identity confirmation")).to eq("Partial match") + + click_on "Confirm the claimant made the claim" + + expect(page).to have_content( + "Provider entered claimant name Bobby Bobberson" + ) + + expect(page).to have_content( + "Claimant name from One login Robby Bobberson" + ) + + expect(page).to have_content( + "[GOV UK One Login Name] - Names partially match" + ) + + expect(page).to have_content('Provider: "Bobby Bobberson"') + + expect(page).to have_content('GOV.UK One Login: "Robby Bobberson"') + end + + it "allows the admin to mark the task as passed" do + visit admin_claim_task_path(claim, name: "identity_confirmation") + + choose "Yes" + + click_on "Save and continue" + + visit admin_claim_task_path(claim, name: "identity_confirmation") + + expect(page).to have_content( + "This task was performed by Aaron Admin" + ) + + visit admin_claim_tasks_path(claim) + + expect(task_status("Identity confirmation")).to eq("Passed") + end + + it "allows the admin to mark the task as failed" do + visit admin_claim_task_path(claim, name: "identity_confirmation") + + choose "No" + + click_on "Save and continue" + + visit admin_claim_task_path(claim, name: "identity_confirmation") + + expect(page).to have_content( + "This task was performed by Aaron Admin" + ) + + visit admin_claim_tasks_path(claim) + + expect(task_status("Identity confirmation")).to eq("Failed") + end + end + + context "when the names don't match" do + let(:claim) { complete_provider_journey } + + before do + complete_practitioner_journey( + claim: claim, + one_login_first_name: "Robby", + one_login_last_name: "Robberson", + one_login_date_of_birth: Date.new(1986, 1, 1), + date_of_birth: Date.new(1986, 1, 1) + ) + + sign_in_as_service_operator + end + + it "shows the task as failed" do + visit admin_claim_tasks_path(claim) + + expect(task_status("Identity confirmation")).to eq("Failed") + + click_on "Confirm the claimant made the claim" + + expect(page).to have_content( + "Provider entered claimant name Bobby Bobberson" + ) + + expect(page).to have_content( + "Claimant name from One login Robby Robberson" + ) + + expect(page).to have_content( + "[GOV UK One Login Name] - Names do not match" + ) + + expect(page).to have_content('Provider: "Bobby Bobberson"') + + expect(page).to have_content('GOV.UK One Login: "Robby Robberson"') + end + + it "doesn't allow the admin to complete the task" do + visit admin_claim_task_path(claim, name: "identity_confirmation") + + expect(page).not_to have_button("Save and continue") + end + end + end + + context "when OL IDV is a fail" do + it "fails the task" do + claim = complete_provider_journey + + complete_practitioner_journey( + claim: claim, + one_login_first_name: "Bobby", + one_login_last_name: "Bobberson", + date_of_birth: Date.new(1986, 1, 11), + one_login_date_of_birth: Date.new(1986, 1, 1) + ) + + sign_in_as_service_operator + + visit admin_claim_tasks_path(claim) + + expect(task_status("Identity confirmation")).to eq("Failed") + + click_on "Confirm the claimant made the claim" + + expect(page).to have_content( + "[GOV UK One Login] - IDV mismatch:" + ) + + expect(page).to have_content('GOV.UK One Login Name: "Bobby Bobberson"') + + expect(page).to have_content('GOV.UK One Login DOB: "1986-01-01"') + + expect(page).not_to have_button("Save and continue") + end + end + end + + def complete_provider_journey + create(:journey_configuration, :early_years_payment_provider_start) + + create(:journey_configuration, :early_years_payment_provider_authenticated) + + nursery = create( + :eligible_ey_provider, + primary_key_contact_email_address: "johndoe@example.com", + secondary_contact_email_address: "janedoe@example.com" + ) + + visit landing_page_path( + Journeys::EarlyYearsPayment::Provider::Start::ROUTING_NAME + ) + + click_link "Start now" + + fill_in "Email address", with: "johndoe@example.com" + click_on "Submit" + + mail = ActionMailer::Base.deliveries.last + magic_link = mail[:personalisation].unparsed_value[:magic_link] + + visit magic_link + + check( + "I confirm that I have obtained consent from my employee and have " \ + "provided them with the relevant privacy notice." + ) + click_button "Continue" + + choose nursery.nursery_name + click_button "Continue" + + fill_in "claim-paye-reference-field", with: "123/123456SE90" + click_button "Continue" + + fill_in "First name", with: "Bobby" + fill_in "Last name", with: "Bobberson" + click_button "Continue" + + date = Date.yesterday + fill_in("Day", with: date.day) + fill_in("Month", with: date.month) + fill_in("Year", with: date.year) + click_button "Continue" + + # /early-years-payment-provider/child-facing + choose "Yes" + click_button "Continue" + + # /early-years-payment-provider/returner + choose "Yes" + click_button "Continue" + + # /early-years-payment-provider/returner-worked-with-children + choose "Yes" + click_button "Continue" + + # /early-years-payment-provider/returner-contract-type + choose "casual or temporary" + click_button "Continue" + + # /early-years-payment-provider/employee-email + fill_in( + "claim-practitioner-email-address-field", + with: "practitioner@example.com" + ) + + click_button "Continue" + + # /early-years-payment-provider/check-your-answers + fill_in "claim-provider-contact-name-field", with: "John Doe" + click_button "Accept and send" + + Claim.last + end + + def complete_practitioner_journey( + claim:, + date_of_birth:, + one_login_first_name:, + one_login_last_name:, + one_login_date_of_birth: + ) + stub_const( + "OmniauthCallbacksController::ONE_LOGIN_TEST_USER", + { + first_name: one_login_first_name, + last_name: one_login_last_name, + date_of_birth: one_login_date_of_birth + } + ) + + create(:journey_configuration, :early_years_payment_practitioner) + + visit "/early-years-payment-practitioner/find-reference?skip_landing_page=true&email=practitioner@example.com" + + fill_in "Claim reference number", with: claim.reference + + click_button "Submit" + + sign_in_with_one_login + + click_on "Continue" + + expect(page).to have_content("Personal details") + fill_in "Day", with: date_of_birth.day + fill_in "Month", with: date_of_birth.month + fill_in "Year", with: date_of_birth.year + fill_in "National Insurance number", with: "PX321499A" + click_on "Continue" + + expect(page).to have_content("What is your address?") + fill_in "House number or name", with: "57" + fill_in "Building and street", with: "Walthamstow Drive" + fill_in "Town or city", with: "Derby" + fill_in "County", with: "City of Derby" + fill_in "Postcode", with: "DE22 4BS" + click_on "Continue" + + expect(page).to have_content("Your email address") + fill_in "claim-email-address-field", with: "johndoe@example.com" + click_on "Continue" + + expect(page).to have_content("Enter the 6-digit passcode") + mail = ActionMailer::Base.deliveries.last + otp_in_mail_sent = mail[:personalisation].unparsed_value[:one_time_password] + fill_in "claim-one-time-password-field", with: otp_in_mail_sent + click_on "Confirm" + + expect(page).to have_content("Would you like to provide your mobile number?") + choose "No" + click_on "Continue" + + fill_in "Name on your account", with: "#{claim.first_name} #{claim.surname}" + fill_in "Sort code", with: "123456" + fill_in "Account number", with: "87654321" + click_on "Continue" + + expect(page).to have_text(I18n.t("forms.gender.questions.payroll_gender")) + choose "Male" + click_on "Continue" + + expect(page).to have_content("Check your answers before submitting this claim") + + perform_enqueued_jobs { click_on "Accept and send" } + end + + def task_status(task_name) + find("h2.app-task-list__section", text: task_name) + .find(:xpath, 'following-sibling::ul//strong[contains(@class, "govuk-tag")]') + .text + end +end