From a65e30bec3f53f54bec289702297c4a75fa11362 Mon Sep 17 00:00:00 2001 From: Eric Tillberg Date: Wed, 27 Nov 2024 10:50:47 -0500 Subject: [PATCH 1/2] Catch more errors in NotificationEmail --- .../simple_forms_api/notification_email.rb | 32 ++++++-- .../spec/services/notification_email_spec.rb | 75 +++++++++++++++++++ 2 files changed, 99 insertions(+), 8 deletions(-) diff --git a/modules/simple_forms_api/app/services/simple_forms_api/notification_email.rb b/modules/simple_forms_api/app/services/simple_forms_api/notification_email.rb index 58c87eeddd5..75814a69eb8 100644 --- a/modules/simple_forms_api/app/services/simple_forms_api/notification_email.rb +++ b/modules/simple_forms_api/app/services/simple_forms_api/notification_email.rb @@ -63,7 +63,10 @@ class NotificationEmail SUPPORTED_FORMS = TEMPLATE_IDS.keys def initialize(config, notification_type: :confirmation, user: nil, user_account: nil) + @notification_type = notification_type + check_missing_keys(config) + check_if_form_is_supported(config) @form_data = config[:form_data] @form_number = config[:form_number] @@ -71,23 +74,22 @@ def initialize(config, notification_type: :confirmation, user: nil, user_account @date_submitted = config[:date_submitted] @expiration_date = config[:expiration_date] @lighthouse_updated_at = config[:lighthouse_updated_at] - @notification_type = notification_type @user = user @user_account = user_account end def send(at: nil) - return unless SUPPORTED_FORMS.include?(form_number) return unless flipper? template_id = TEMPLATE_IDS[form_number][notification_type] return unless template_id - if at - enqueue_email(at, template_id) - else - send_email_now(template_id) - end + sent_to_va_notify = if at + enqueue_email(at, template_id) + else + send_email_now(template_id) + end + StatsD.increment('silent_failure', tags: statsd_tags) if error_notification? && !sent_to_va_notify end private @@ -97,7 +99,17 @@ def check_missing_keys(config) if config[:form_number] == 'vba_21_0966_intent_api' && config[:expiration_date].nil? missing_keys << :expiration_date end - raise ArgumentError, "Missing keys: #{missing_keys.join(', ')}" if missing_keys.any? + if missing_keys.any? + StatsD.increment('silent_failure', tags: statsd_tags) if error_notification? + raise ArgumentError, "Missing keys: #{missing_keys.join(', ')}" + end + end + + def check_if_form_is_supported(config) + unless SUPPORTED_FORMS.include?(config[:form_number]) + StatsD.increment('silent_failure', tags: statsd_tags) if error_notification? + raise ArgumentError, "Unsupported form: given form number was #{config[:form_number]}" + end end def flipper? @@ -397,5 +409,9 @@ def form40_10007_first_name def statsd_tags { 'service' => 'veteran-facing-forms', 'function' => "#{form_number} form submission to Lighthouse" } end + + def error_notification? + notification_type == :error + end end end diff --git a/modules/simple_forms_api/spec/services/notification_email_spec.rb b/modules/simple_forms_api/spec/services/notification_email_spec.rb index f8aae6aeb10..231461abedf 100644 --- a/modules/simple_forms_api/spec/services/notification_email_spec.rb +++ b/modules/simple_forms_api/spec/services/notification_email_spec.rb @@ -26,6 +26,15 @@ it 'fails' do expect { described_class.new(config, notification_type:) }.to raise_error(ArgumentError) end + + context 'error email', if: notification_type == :error do + it 'increments StatsD' do + allow(StatsD).to receive(:increment) + + expect { described_class.new(config, notification_type:) }.to raise_error(ArgumentError) + expect(StatsD).to have_received(:increment).with('silent_failure', tags: anything) + end + end end context 'missing form_number' do @@ -37,6 +46,15 @@ it 'fails' do expect { described_class.new(config, notification_type:) }.to raise_error(ArgumentError) end + + context 'error email', if: notification_type == :error do + it 'increments StatsD' do + allow(StatsD).to receive(:increment) + + expect { described_class.new(config, notification_type:) }.to raise_error(ArgumentError) + expect(StatsD).to have_received(:increment).with('silent_failure', tags: anything) + end + end end context 'missing confirmation_number' do @@ -47,6 +65,15 @@ it 'fails' do expect { described_class.new(config, notification_type:) }.to raise_error(ArgumentError) end + + context 'error email', if: notification_type == :error do + it 'increments StatsD' do + allow(StatsD).to receive(:increment) + + expect { described_class.new(config, notification_type:) }.to raise_error(ArgumentError) + expect(StatsD).to have_received(:increment).with('silent_failure', tags: anything) + end + end end context 'missing date_submitted' do @@ -57,6 +84,35 @@ it 'fails' do expect { described_class.new(config, notification_type:) }.to raise_error(ArgumentError) end + + context 'error email', if: notification_type == :error do + it 'increments StatsD' do + allow(StatsD).to receive(:increment) + + expect { described_class.new(config, notification_type:) }.to raise_error(ArgumentError) + expect(StatsD).to have_received(:increment).with('silent_failure', tags: anything) + end + end + end + + context 'form not supported' do + let(:config) do + { form_data: {}, form_number: 'nonsense', confirmation_number: 'confirmation_number', + date_submitted: Time.zone.today.strftime('%B %d, %Y') } + end + + it 'fails' do + expect { described_class.new(config, notification_type:) }.to raise_error(ArgumentError) + end + + context 'error email', if: notification_type == :error do + it 'increments StatsD' do + allow(StatsD).to receive(:increment) + + expect { described_class.new(config, notification_type:) }.to raise_error(ArgumentError) + expect(StatsD).to have_received(:increment).with('silent_failure', tags: anything) + end + end end end @@ -89,6 +145,25 @@ expect(VANotify::EmailJob).to have_received(:perform_async) end + + context 'did not send to VA Notify because of no first name', if: notification_type == :error do + let(:profile) { double(given_names: []) } + let(:mpi_profile) { double(profile:, error: nil) } + + it 'increments StatsD' do + data['witness_full_name']['first'] = nil + allow(VANotify::EmailJob).to receive(:perform_async) + allow(VANotify::UserAccountJob).to receive(:perform_at) + allow_any_instance_of(MPI::Service).to receive(:find_profile_by_identifier).and_return(mpi_profile) + allow(StatsD).to receive(:increment) + + subject = described_class.new(config, notification_type:) + subject.send + + expect(VANotify::EmailJob).not_to have_received(:perform_async) + expect(StatsD).to have_received(:increment).with('silent_failure', tags: anything) + end + end end context 'flipper is off' do From cb05118dbfea63b185678ff5b79b3cf7fc8ca6ae Mon Sep 17 00:00:00 2001 From: Eric Tillberg Date: Mon, 2 Dec 2024 09:43:54 -0500 Subject: [PATCH 2/2] specs --- .../spec/services/notification_email_spec.rb | 54 +++++-------------- 1 file changed, 14 insertions(+), 40 deletions(-) diff --git a/modules/simple_forms_api/spec/services/notification_email_spec.rb b/modules/simple_forms_api/spec/services/notification_email_spec.rb index 231461abedf..db5b8669a3e 100644 --- a/modules/simple_forms_api/spec/services/notification_email_spec.rb +++ b/modules/simple_forms_api/spec/services/notification_email_spec.rb @@ -3,6 +3,15 @@ require 'rails_helper' require SimpleFormsApi::Engine.root.join('spec', 'spec_helper.rb') +shared_examples 'an error notification email' do + it 'increments StatsD' do + allow(StatsD).to receive(:increment) + + expect { described_class.new(config, notification_type: :error) }.to raise_error(ArgumentError) + expect(StatsD).to have_received(:increment).with('silent_failure', tags: anything) + end +end + describe SimpleFormsApi::NotificationEmail do %i[confirmation error received].each do |notification_type| describe '#initialize' do @@ -27,14 +36,7 @@ expect { described_class.new(config, notification_type:) }.to raise_error(ArgumentError) end - context 'error email', if: notification_type == :error do - it 'increments StatsD' do - allow(StatsD).to receive(:increment) - - expect { described_class.new(config, notification_type:) }.to raise_error(ArgumentError) - expect(StatsD).to have_received(:increment).with('silent_failure', tags: anything) - end - end + it_behaves_like 'an error notification email' if notification_type == :error end context 'missing form_number' do @@ -47,14 +49,7 @@ expect { described_class.new(config, notification_type:) }.to raise_error(ArgumentError) end - context 'error email', if: notification_type == :error do - it 'increments StatsD' do - allow(StatsD).to receive(:increment) - - expect { described_class.new(config, notification_type:) }.to raise_error(ArgumentError) - expect(StatsD).to have_received(:increment).with('silent_failure', tags: anything) - end - end + it_behaves_like 'an error notification email' if notification_type == :error end context 'missing confirmation_number' do @@ -66,14 +61,7 @@ expect { described_class.new(config, notification_type:) }.to raise_error(ArgumentError) end - context 'error email', if: notification_type == :error do - it 'increments StatsD' do - allow(StatsD).to receive(:increment) - - expect { described_class.new(config, notification_type:) }.to raise_error(ArgumentError) - expect(StatsD).to have_received(:increment).with('silent_failure', tags: anything) - end - end + it_behaves_like 'an error notification email' if notification_type == :error end context 'missing date_submitted' do @@ -85,14 +73,7 @@ expect { described_class.new(config, notification_type:) }.to raise_error(ArgumentError) end - context 'error email', if: notification_type == :error do - it 'increments StatsD' do - allow(StatsD).to receive(:increment) - - expect { described_class.new(config, notification_type:) }.to raise_error(ArgumentError) - expect(StatsD).to have_received(:increment).with('silent_failure', tags: anything) - end - end + it_behaves_like 'an error notification email' if notification_type == :error end context 'form not supported' do @@ -105,14 +86,7 @@ expect { described_class.new(config, notification_type:) }.to raise_error(ArgumentError) end - context 'error email', if: notification_type == :error do - it 'increments StatsD' do - allow(StatsD).to receive(:increment) - - expect { described_class.new(config, notification_type:) }.to raise_error(ArgumentError) - expect(StatsD).to have_received(:increment).with('silent_failure', tags: anything) - end - end + it_behaves_like 'an error notification email' if notification_type == :error end end