|
|
|
class BulkUpload::Lettings::Validator
|
|
|
|
include ActiveModel::Validations
|
|
|
|
include Rails.application.routes.url_helpers
|
|
|
|
|
|
|
|
attr_reader :bulk_upload, :path
|
|
|
|
|
|
|
|
validate :validate_file_not_empty
|
|
|
|
validate :validate_field_numbers_count
|
|
|
|
validate :validate_missing_required_headers
|
|
|
|
validate :validate_max_columns_count_if_no_headers
|
|
|
|
validate :validate_correct_template
|
|
|
|
|
|
|
|
def initialize(bulk_upload:, path:)
|
|
|
|
@bulk_upload = bulk_upload
|
|
|
|
@path = path
|
|
|
|
end
|
|
|
|
|
|
|
|
def call
|
|
|
|
row_parsers.each(&:valid?)
|
|
|
|
|
|
|
|
validate_duplicate_rows if FeatureToggle.bulk_upload_duplicate_log_check_enabled?
|
|
|
|
|
|
|
|
row_parsers.each_with_index do |row_parser, index|
|
|
|
|
row = index + row_offset + 1
|
|
|
|
|
|
|
|
row_parser.errors.each do |error|
|
|
|
|
col = csv_parser.column_for_field(error.attribute.to_s)
|
|
|
|
|
|
|
|
bulk_upload.bulk_upload_errors.create!(
|
|
|
|
field: error.attribute,
|
|
|
|
error: error.message,
|
|
|
|
tenant_code: row_parser.tenant_code,
|
|
|
|
property_ref: row_parser.property_ref,
|
|
|
|
row:,
|
|
|
|
cell: "#{col}#{row}",
|
|
|
|
col:,
|
|
|
|
category: error.options[:category],
|
|
|
|
)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def block_log_creation_reason
|
|
|
|
return "setup_errors" if any_setup_errors?
|
|
|
|
|
|
|
|
if row_parsers.any?(&:block_log_creation?)
|
|
|
|
Sentry.capture_message("Bulk upload log creation blocked: #{bulk_upload.id}.")
|
|
|
|
return "row_parser_block_log_creation"
|
|
|
|
end
|
|
|
|
|
|
|
|
if any_logs_already_exist? && FeatureToggle.bulk_upload_duplicate_log_check_enabled?
|
|
|
|
return "duplicate_logs"
|
|
|
|
end
|
|
|
|
|
|
|
|
row_parsers.each do |row_parser|
|
|
|
|
row_parser.log.blank_invalid_non_setup_fields!
|
|
|
|
end
|
|
|
|
if any_logs_invalid?
|
|
|
|
Sentry.capture_message("Bulk upload log creation blocked due to invalid logs after blanking non setup fields: #{bulk_upload.id}.")
|
|
|
|
"logs_invalid"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.question_for_field(field)
|
|
|
|
QUESTIONS[field]
|
|
|
|
end
|
|
|
|
|
|
|
|
def any_setup_errors?
|
|
|
|
bulk_upload
|
|
|
|
.bulk_upload_errors
|
|
|
|
.where(category: "setup")
|
|
|
|
.count
|
|
|
|
.positive?
|
|
|
|
end
|
|
|
|
|
|
|
|
def soft_validation_errors_only?
|
|
|
|
errors = bulk_upload.bulk_upload_errors
|
|
|
|
errors.count == errors.where(category: "soft_validation").count && errors.count.positive?
|
|
|
|
end
|
|
|
|
|
|
|
|
def any_logs_already_exist?
|
|
|
|
row_parsers.any?(&:log_already_exists?)
|
|
|
|
end
|
|
|
|
|
|
|
|
def total_logs_count
|
|
|
|
csv_parser.body_rows.count
|
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
|
|
|
|
# n^2 algo
|
|
|
|
def validate_duplicate_rows
|
|
|
|
row_parsers.each do |rp|
|
|
|
|
dupe = row_parsers.reject { |r| r.object_id.equal?(rp.object_id) }.any? do |rp_counter|
|
|
|
|
rp.spreadsheet_duplicate_hash == rp_counter.spreadsheet_duplicate_hash
|
|
|
|
end
|
|
|
|
|
|
|
|
if dupe
|
|
|
|
rp.add_duplicate_found_in_spreadsheet_errors
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def any_logs_invalid?
|
|
|
|
row_parsers.any? { |row_parser| row_parser.log.invalid? }
|
|
|
|
end
|
|
|
|
|
|
|
|
def csv_parser
|
|
|
|
@csv_parser ||= case bulk_upload.year
|
|
|
|
when 2023
|
|
|
|
BulkUpload::Lettings::Year2023::CsvParser.new(path:)
|
|
|
|
when 2024
|
|
|
|
BulkUpload::Lettings::Year2024::CsvParser.new(path:)
|
|
|
|
else
|
|
|
|
raise "csv parser not found"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def row_offset
|
|
|
|
csv_parser.row_offset
|
|
|
|
end
|
|
|
|
|
|
|
|
def col_offset
|
|
|
|
csv_parser.col_offset
|
|
|
|
end
|
|
|
|
|
|
|
|
def field_number_for_attribute(attribute)
|
|
|
|
attribute.to_s.split("_").last.to_i
|
|
|
|
end
|
|
|
|
|
|
|
|
def cols
|
|
|
|
csv_parser.cols
|
|
|
|
end
|
|
|
|
|
|
|
|
def row_parsers
|
|
|
|
return @row_parsers if @row_parsers
|
|
|
|
|
|
|
|
@row_parsers = csv_parser.row_parsers
|
|
|
|
|
|
|
|
@row_parsers.each do |row_parser|
|
|
|
|
row_parser.bulk_upload = bulk_upload
|
|
|
|
end
|
|
|
|
|
|
|
|
@row_parsers
|
|
|
|
end
|
|
|
|
|
|
|
|
def rows
|
|
|
|
csv_parser.rows
|
|
|
|
end
|
|
|
|
|
|
|
|
def body_rows
|
|
|
|
csv_parser.body_rows
|
|
|
|
end
|
|
|
|
|
|
|
|
def validate_file_not_empty
|
|
|
|
if File.size(path).zero? || csv_parser.body_rows.flatten.compact.empty?
|
|
|
|
errors.add(:base, I18n.t("validations.lettings.#{@bulk_upload.year}.bulk_upload.blank_file"))
|
|
|
|
|
|
|
|
halt_validations!
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def validate_field_numbers_count
|
|
|
|
return if halt_validations?
|
|
|
|
|
|
|
|
unless csv_parser.correct_field_count?
|
|
|
|
errors.add(:base, I18n.t("validations.lettings.#{@bulk_upload.year}.bulk_upload.wrong_template.wrong_field_numbers_count"))
|
|
|
|
halt_validations!
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def validate_max_columns_count_if_no_headers
|
|
|
|
return if halt_validations?
|
|
|
|
|
|
|
|
if csv_parser.too_many_columns?
|
|
|
|
errors.add(:base, I18n.t("validations.lettings.#{@bulk_upload.year}.bulk_upload.wrong_template.over_max_column_count"))
|
|
|
|
halt_validations!
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def validate_correct_template
|
|
|
|
return if halt_validations?
|
|
|
|
|
|
|
|
errors.add(:base, I18n.t("validations.lettings.#{@bulk_upload.year}.bulk_upload.wrong_template.wrong_template")) if csv_parser.wrong_template_for_year?
|
|
|
|
end
|
|
|
|
|
|
|
|
def validate_missing_required_headers
|
|
|
|
return if halt_validations?
|
|
|
|
|
|
|
|
if csv_parser.missing_required_headers?
|
|
|
|
errors.add :base, I18n.t("validations.lettings.#{@bulk_upload.year}.bulk_upload.wrong_template.no_headers", guidance_link: bulk_upload_lettings_log_url(id: "guidance", form: { year: bulk_upload.year }, host: ENV["APP_HOST"], anchor: "using-the-bulk-upload-template"))
|
|
|
|
halt_validations!
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def halt_validations!
|
|
|
|
@halt_validations = true
|
|
|
|
end
|
|
|
|
|
|
|
|
def halt_validations?
|
|
|
|
@halt_validations ||= false
|
|
|
|
end
|
|
|
|
end
|