-
Notifications
You must be signed in to change notification settings - Fork 66
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge branch 'master' into vebt-747-additional-logging
- Loading branch information
Showing
50 changed files
with
1,097 additions
and
8 deletions.
There are no files selected for viewing
Validating CODEOWNERS rules …
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,21 @@ | ||
# frozen_string_literal: true | ||
|
||
class CreateExcelFilesMailer < ApplicationMailer | ||
def build(filename) | ||
date = Time.zone.now.strftime('%m/%d/%Y') | ||
file_contents = File.read("tmp/#{filename}") | ||
headers['Content-Disposition'] = "attachment; filename=#{filename}" | ||
|
||
# rubocop:disable Layout/LineLength | ||
recipients = Settings.vsp_environment.eql?('production') ? Settings.edu.production_excel_contents.emails : Settings.edu.staging_excel_contents.emails | ||
subject = Settings.vsp_environment.eql?('production') ? "22-10282 Form CSV file for #{date}" : "Staging CSV file for #{date}" | ||
# rubocop:enable Layout/LineLength | ||
|
||
mail( | ||
to: recipients, | ||
subject: subject, | ||
content_type: 'text/csv', | ||
body: file_contents | ||
) | ||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,20 @@ | ||
# frozen_string_literal: true | ||
|
||
class ExcelFileEvent < ApplicationRecord | ||
validates :filename, uniqueness: true | ||
|
||
# Look for an existing row with same filename | ||
# and increase retry attempt if wasn't successful from previous attempt | ||
# Otherwise create a new event | ||
def self.build_event(filename) | ||
filename_date = filename.match(/(.+)_/)[1] | ||
event = find_by('filename like ?', "#{filename_date}%") | ||
|
||
if event.present? | ||
event.update(retry_attempt: event.retry_attempt + 1) if event.successful_at.nil? | ||
return event | ||
end | ||
|
||
create(filename: filename) | ||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,11 @@ | ||
# frozen_string_literal: true | ||
|
||
class FormProfiles::VA10282 < FormProfile | ||
def metadata | ||
{ | ||
version: 0, | ||
prefill: true, | ||
returnUrl: '/applicant/information' | ||
} | ||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,29 @@ | ||
# frozen_string_literal: true | ||
|
||
class SavedClaim::EducationBenefits::VA10282 < SavedClaim::EducationBenefits | ||
add_form_and_validation('22-10282') | ||
|
||
def after_submit(_user) | ||
return unless Flipper.enabled?(:form22_10282_confirmation_email) | ||
|
||
parsed_form_data = JSON.parse(form) | ||
email = parsed_form_data['email'] | ||
return if email.blank? | ||
|
||
send_confirmation_email(parsed_form_data, email) | ||
end | ||
|
||
private | ||
|
||
def send_confirmation_email(parsed_form_data, email) | ||
VANotify::EmailJob.perform_async( | ||
email, | ||
Settings.vanotify.services.va_gov.template_id.form22_10282_confirmation_email, | ||
{ | ||
'first_name' => parsed_form_data.dig('veteranFullName', 'first')&.upcase.presence, | ||
'date_submitted' => Time.zone.today.strftime('%B %d, %Y'), | ||
'confirmation_number' => education_benefits_claim.confirmation_number | ||
} | ||
) | ||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,194 @@ | ||
# frozen_string_literal: true | ||
|
||
require 'sentry_logging' | ||
require 'sftp_writer/factory' | ||
|
||
module EducationForm | ||
class DailyExcelFileError < StandardError | ||
end | ||
|
||
class CreateDailyExcelFiles | ||
MAX_RETRIES = 5 | ||
STATSD_KEY = 'worker.education_benefits_claim' | ||
STATSD_FAILURE_METRIC = "#{STATSD_KEY}.failed_excel_file".freeze | ||
LIVE_FORM_TYPES = ['22-10282'].freeze | ||
AUTOMATED_DECISIONS_STATES = [nil, 'denied', 'processed'].freeze | ||
EXCEL_FIELDS = %w[ | ||
name | ||
first_name | ||
last_name | ||
military_affiliation | ||
phone_number | ||
email_address | ||
country | ||
state | ||
race_ethnicity | ||
gender | ||
education_level | ||
employment_status | ||
salary | ||
technology_industry | ||
].freeze | ||
HEADERS = ['Name', 'First Name', 'Last Name', 'Select Military Affiliation', | ||
'Phone Number', 'Email Address', 'Country', 'State', 'Race/Ethnicity', | ||
'Gender of Applicant', 'What is your highest level of education?', | ||
'Are you currently employed?', 'What is your current salary?', | ||
'Are you currently working in the technology industry? (If so, please select one)'].freeze | ||
include Sidekiq::Job | ||
include SentryLogging | ||
sidekiq_options queue: 'default', | ||
unique_for: 30.minutes, | ||
retry: 5 | ||
|
||
# rubocop:disable Metrics/MethodLength | ||
def perform | ||
retry_count = 0 | ||
filename = "22-10282_#{Time.zone.now.strftime('%m%d%Y_%H%M%S')}.csv" | ||
excel_file_event = ExcelFileEvent.build_event(filename) | ||
begin | ||
records = EducationBenefitsClaim | ||
.unprocessed | ||
.joins(:saved_claim) | ||
.where( | ||
saved_claims: { | ||
form_id: LIVE_FORM_TYPES | ||
} | ||
) | ||
return false if federal_holiday? | ||
|
||
if records.count.zero? | ||
log_info('No records to process.') | ||
return true | ||
elsif retry_count.zero? | ||
log_info("Processing #{records.count} application(s)") | ||
end | ||
|
||
# Format the records and write to CSV file | ||
formatted_records = format_records(records) | ||
write_csv_file(formatted_records, filename) | ||
|
||
email_excel_files(filename) | ||
|
||
# Make records processed and add excel file event for rake job | ||
records.each { |r| r.update(processed_at: Time.zone.now) } | ||
excel_file_event.update(number_of_submissions: records.count, successful_at: Time.zone.now) | ||
rescue => e | ||
StatsD.increment("#{STATSD_FAILURE_METRIC}.general") | ||
if retry_count < MAX_RETRIES | ||
log_exception(DailyExcelFileError.new("Error creating excel files.\n\n#{e} | ||
Retry count: #{retry_count}. Retrying..... ")) | ||
retry_count += 1 | ||
sleep(10 * retry_count) # exponential backoff for retries | ||
retry | ||
else | ||
log_exception(DailyExcelFileError.new("Error creating excel files. | ||
Job failed after #{MAX_RETRIES} retries \n\n#{e}")) | ||
end | ||
end | ||
true | ||
end | ||
|
||
def write_csv_file(records, filename) | ||
retry_count = 0 | ||
|
||
begin | ||
# Generate CSV string content instead of writing to file | ||
csv_contents = CSV.generate do |csv| | ||
# Add headers | ||
csv << HEADERS | ||
|
||
# Add data rows | ||
records.each_with_index do |record, index| | ||
log_info("Processing record #{index + 1}: #{record.inspect}") | ||
|
||
begin | ||
row_data = EXCEL_FIELDS.map do |field| | ||
value = record.public_send(field) | ||
value.is_a?(Hash) ? value.to_s : value | ||
end | ||
|
||
csv << row_data | ||
rescue => e | ||
log_exception(DailyExcelFileError.new("Failed to add row #{index + 1}:\n")) | ||
log_exception(DailyExcelFileError.new("#{e.message}\nRecord: #{record.inspect}")) | ||
next | ||
end | ||
end | ||
end | ||
|
||
# Write to file for backup/audit purposes | ||
File.write("tmp/#{filename}", csv_contents) | ||
log_info('Successfully created CSV file') | ||
|
||
# Return the CSV contents | ||
csv_contents | ||
rescue => e | ||
StatsD.increment("#{STATSD_FAILURE_METRIC}.general") | ||
log_exception(DailyExcelFileError.new('Error creating CSV files.')) | ||
|
||
if retry_count < MAX_RETRIES | ||
log_exception(DailyExcelFileError.new("Retry count: #{retry_count}. Retrying..... ")) | ||
retry_count += 1 | ||
sleep(5) | ||
retry | ||
else | ||
log_exception(DailyExcelFileError.new("Job failed after #{MAX_RETRIES} retries \n\n#{e}")) | ||
end | ||
end | ||
end | ||
# rubocop:enable Metrics/MethodLength | ||
|
||
def format_records(records) | ||
records.map do |record| | ||
format_application(record) | ||
end.compact | ||
end | ||
|
||
def format_application(data) | ||
form = EducationForm::Forms::Base.build(data) | ||
track_form_type("22-#{data.form_type}") | ||
form | ||
rescue => e | ||
inform_on_error(data, e) | ||
nil | ||
end | ||
|
||
def inform_on_error(claim, error = nil) | ||
StatsD.increment("#{STATSD_KEY}.failed_formatting.22-#{claim.form_type}") | ||
exception = if error.present? | ||
FormattingError.new("Could not format #{claim.confirmation_number}.\n\n#{error}") | ||
else | ||
FormattingError.new("Could not format #{claim.confirmation_number}") | ||
end | ||
log_exception(exception) | ||
end | ||
|
||
private | ||
|
||
def federal_holiday? | ||
holiday = Holidays.on(Time.zone.today, :us, :observed) | ||
if holiday.empty? | ||
false | ||
else | ||
log_info("Skipping on a Holiday: #{holiday.first[:name]}") | ||
true | ||
end | ||
end | ||
|
||
def track_form_type(type) | ||
StatsD.gauge("#{STATSD_KEY}.transmissions.#{type}", 1) | ||
end | ||
|
||
def log_exception(exception) | ||
log_exception_to_sentry(exception) | ||
end | ||
|
||
def log_info(message) | ||
logger.info(message) | ||
end | ||
|
||
def email_excel_files(contents) | ||
CreateExcelFilesMailer.build(contents).deliver_now | ||
end | ||
end | ||
end |
Oops, something went wrong.