require "csv" class BulkUpload::Lettings::Validator COLUMN_PERCENTAGE_ERROR_THRESHOLD = 0.6 COLUMN_ABSOLUTE_ERROR_THRESHOLD = 16 include ActiveModel::Validations attr_reader :bulk_upload, :path validate :validate_file_not_empty validate :validate_min_columns validate :validate_max_columns def initialize(bulk_upload:, path:) @bulk_upload = bulk_upload @path = path end def call row_parsers.each_with_index do |row_parser, index| row_parser.valid? 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 update_expected_log_count end def create_logs? return false if any_setup_errors? return false if over_column_error_threshold? return false if row_parsers.any?(&:block_log_creation?) row_parsers.all? { |row_parser| row_parser.log.valid? } 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 private def update_expected_log_count bulk_upload.update!(expected_log_count:) end def expected_log_count row_parsers.count { |row| !row.blank_row? } end def over_column_error_threshold? fields = ("field_1".."field_134").to_a percentage_threshold = (row_parsers.size * COLUMN_PERCENTAGE_ERROR_THRESHOLD).ceil fields.any? do |field| count = row_parsers.count { |row_parser| row_parser.errors[field].present? } next if count < COLUMN_ABSOLUTE_ERROR_THRESHOLD count > percentage_threshold end end def csv_parser @csv_parser ||= case bulk_upload.year when 2022 BulkUpload::Lettings::Year2022::CsvParser.new(path:) when 2023 BulkUpload::Lettings::Year2023::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? errors.add(:file, :blank) halt_validations! end end def validate_min_columns return if halt_validations? column_count = rows.map(&:size).min errors.add(:base, :under_min_column_count) if column_count < csv_parser.class::MIN_COLUMNS end def validate_max_columns return if halt_validations? column_count = rows.map(&:size).max errors.add(:base, :over_max_column_count) if column_count > csv_parser.class::MAX_COLUMNS end def halt_validations! @halt_validations = true end def halt_validations? @halt_validations ||= false end end