|
|
|
class SalesLogValidator < ActiveModel::Validator
|
|
|
|
include Validations::Sales::SetupValidations
|
|
|
|
include Validations::Sales::HouseholdValidations
|
|
|
|
include Validations::Sales::PropertyValidations
|
|
|
|
include Validations::Sales::FinancialValidations
|
|
|
|
include Validations::Sales::SaleInformationValidations
|
|
|
|
include Validations::SharedValidations
|
|
|
|
include Validations::LocalAuthorityValidations
|
|
|
|
|
|
|
|
def validate(record)
|
|
|
|
validation_methods = public_methods.select { |method| method.starts_with?("validate_") }
|
|
|
|
validation_methods.each { |meth| public_send(meth, record) }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
class SalesLog < Log
|
|
|
|
include DerivedVariables::SalesLogVariables
|
|
|
|
include Validations::Sales::SoftValidations
|
|
|
|
include Validations::SoftValidations
|
|
|
|
include MoneyFormattingHelper
|
|
|
|
|
|
|
|
self.inheritance_column = :_type_disabled
|
|
|
|
|
|
|
|
has_paper_trail
|
|
|
|
|
|
|
|
validates_with SalesLogValidator
|
|
|
|
before_validation :recalculate_start_year!, if: :saledate_changed?
|
|
|
|
before_validation :process_postcode_changes!, if: :postcode_full_changed?
|
|
|
|
before_validation :process_previous_postcode_changes!, if: :ppostcode_full_changed?
|
|
|
|
before_validation :reset_invalidated_dependent_fields!
|
|
|
|
before_validation :reset_location_fields!, unless: :postcode_known?
|
|
|
|
before_validation :reset_previous_location_fields!, unless: :previous_postcode_known?
|
|
|
|
before_validation :set_derived_fields!
|
|
|
|
before_validation :process_uprn_change!, if: :should_process_uprn_change?
|
|
|
|
before_validation :process_address_change!, if: :should_process_address_change?
|
|
|
|
|
|
|
|
belongs_to :managing_organisation, class_name: "Organisation", optional: true
|
|
|
|
|
|
|
|
scope :filter_by_year, ->(year) { where(saledate: Time.zone.local(year.to_i, 4, 1)...Time.zone.local(year.to_i + 1, 4, 1)) }
|
|
|
|
scope :filter_by_years_or_nil, lambda { |years, _user = nil|
|
|
|
|
first_year = years.shift
|
|
|
|
query = filter_by_year(first_year)
|
|
|
|
years.each { |year| query = query.or(filter_by_year(year)) }
|
|
|
|
query = query.or(where(saledate: nil))
|
|
|
|
query.all
|
|
|
|
}
|
|
|
|
scope :filter_by_purchaser_code, ->(purchid) { where("purchid ILIKE ?", "%#{purchid}%") }
|
|
|
|
scope :search_by, lambda { |param|
|
|
|
|
sanitized_param = ActiveRecord::Base.sanitize_sql(param)
|
|
|
|
param_without_spaces = sanitized_param.delete(" ")
|
|
|
|
|
|
|
|
by_id = Arel.sql("CASE WHEN id = ? THEN 0 ELSE 1 END")
|
|
|
|
by_purchaser_code = Arel.sql("CASE WHEN purchid = ? THEN 0 WHEN purchid ILIKE ? THEN 1 ELSE 2 END")
|
|
|
|
by_postcode = Arel.sql("CASE WHEN REPLACE(postcode_full, ' ', '') = ? THEN 0 WHEN REPLACE(postcode_full, ' ', '') ILIKE ? THEN 1 ELSE 2 END")
|
|
|
|
|
|
|
|
filter_by_purchaser_code(param)
|
|
|
|
.or(filter_by_postcode(param))
|
|
|
|
.or(filter_by_id(param.gsub(/log/i, "")))
|
|
|
|
.order([by_id, sanitized_param.to_i],
|
|
|
|
[by_purchaser_code, sanitized_param, sanitized_param],
|
|
|
|
[by_postcode, param_without_spaces, param_without_spaces])
|
|
|
|
}
|
|
|
|
scope :age1_answered, -> { where.not(age1: nil).or(where(age1_known: [1, 2])) }
|
|
|
|
scope :ecstat1_answered, -> { where.not(ecstat1: nil).or(where("saledate >= ?", Time.zone.local(2025, 4, 1))) }
|
|
|
|
scope :duplicate_logs, lambda { |log|
|
|
|
|
visible.where(log.slice(*DUPLICATE_LOG_ATTRIBUTES))
|
|
|
|
.where.not(id: log.id)
|
|
|
|
.where.not(saledate: nil)
|
|
|
|
.where.not(sex1: nil)
|
|
|
|
.where.not(postcode_full: nil)
|
|
|
|
.ecstat1_answered
|
|
|
|
.age1_answered
|
|
|
|
}
|
|
|
|
scope :after_date, ->(date) { where("saledate >= ?", date) }
|
|
|
|
|
|
|
|
scope :duplicate_sets, lambda { |assigned_to_id = nil|
|
|
|
|
scope = visible
|
|
|
|
.group(*DUPLICATE_LOG_ATTRIBUTES)
|
|
|
|
.where.not(saledate: nil)
|
|
|
|
.where.not(sex1: nil)
|
|
|
|
.where.not(postcode_full: nil)
|
|
|
|
.age1_answered
|
|
|
|
.ecstat1_answered
|
|
|
|
.having("COUNT(*) > 1")
|
|
|
|
|
|
|
|
if assigned_to_id
|
|
|
|
scope = scope.having("MAX(CASE WHEN assigned_to_id = ? THEN 1 ELSE 0 END) >= 1", assigned_to_id)
|
|
|
|
end
|
|
|
|
|
|
|
|
scope.pluck("ARRAY_AGG(id)")
|
|
|
|
}
|
|
|
|
|
|
|
|
OPTIONAL_FIELDS = %w[purchid othtype buyers_organisations].freeze
|
|
|
|
DUPLICATE_LOG_ATTRIBUTES = %w[owning_organisation_id purchid saledate age1_known age1 sex1 ecstat1 postcode_full].freeze
|
|
|
|
|
|
|
|
def lettings?
|
|
|
|
false
|
|
|
|
end
|
|
|
|
|
|
|
|
def sales?
|
|
|
|
true
|
|
|
|
end
|
|
|
|
|
|
|
|
def startdate
|
|
|
|
saledate
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.editable_fields
|
|
|
|
attribute_names
|
|
|
|
end
|
|
|
|
|
|
|
|
def purchaser_code
|
|
|
|
purchid
|
|
|
|
end
|
|
|
|
|
|
|
|
def form_name
|
|
|
|
return unless saledate
|
|
|
|
|
|
|
|
FormHandler.instance.form_name_from_start_year(collection_start_year, "sales")
|
|
|
|
end
|
|
|
|
|
|
|
|
def form
|
|
|
|
FormHandler.instance.get_form(form_name) || FormHandler.instance.current_sales_form
|
|
|
|
end
|
|
|
|
|
|
|
|
def optional_fields
|
|
|
|
OPTIONAL_FIELDS + dynamically_not_required
|
|
|
|
end
|
|
|
|
|
|
|
|
def dynamically_not_required
|
|
|
|
not_required = []
|
|
|
|
not_required << "deposit" if form.start_year_2024_or_later? && stairowned_100?
|
|
|
|
not_required += %w[address_line2 county]
|
|
|
|
|
|
|
|
not_required
|
|
|
|
end
|
|
|
|
|
|
|
|
def unresolved
|
|
|
|
false
|
|
|
|
end
|
|
|
|
|
|
|
|
LONDON_BOROUGHS = %w[
|
|
|
|
E09000001
|
|
|
|
E09000033
|
|
|
|
E09000020
|
|
|
|
E09000013
|
|
|
|
E09000032
|
|
|
|
E09000022
|
|
|
|
E09000028
|
|
|
|
E09000030
|
|
|
|
E09000012
|
|
|
|
E09000019
|
|
|
|
E09000007
|
|
|
|
E09000005
|
|
|
|
E09000009
|
|
|
|
E09000018
|
|
|
|
E09000027
|
|
|
|
E09000021
|
|
|
|
E09000024
|
|
|
|
E09000029
|
|
|
|
E09000008
|
|
|
|
E09000006
|
|
|
|
E09000023
|
|
|
|
E09000011
|
|
|
|
E09000004
|
|
|
|
E09000016
|
|
|
|
E09000002
|
|
|
|
E09000026
|
|
|
|
E09000025
|
|
|
|
E09000031
|
|
|
|
E09000014
|
|
|
|
E09000010
|
|
|
|
E09000003
|
|
|
|
E09000015
|
|
|
|
E09000017
|
|
|
|
].freeze
|
|
|
|
|
|
|
|
def london_property?
|
|
|
|
la && LONDON_BOROUGHS.include?(la)
|
|
|
|
end
|
|
|
|
|
|
|
|
def property_not_in_london?
|
|
|
|
!london_property?
|
|
|
|
end
|
|
|
|
|
|
|
|
def income1_used_for_mortgage?
|
|
|
|
inc1mort == 1
|
|
|
|
end
|
|
|
|
|
|
|
|
def buyers_will_live_in?
|
|
|
|
buylivein == 1
|
|
|
|
end
|
|
|
|
|
|
|
|
def buyers_will_not_live_in?
|
|
|
|
buylivein == 2
|
|
|
|
end
|
|
|
|
|
|
|
|
def buyer_two_will_live_in_property?
|
|
|
|
buy2livein == 1
|
|
|
|
end
|
|
|
|
|
|
|
|
def buyer_two_will_not_live_in_property?
|
|
|
|
buy2livein == 2
|
|
|
|
end
|
|
|
|
|
|
|
|
def buyer_one_will_not_live_in_property?
|
|
|
|
buy1livein == 2
|
|
|
|
end
|
|
|
|
|
|
|
|
def buyer_two_not_already_living_in_property?
|
|
|
|
buy2living == 2
|
|
|
|
end
|
|
|
|
|
|
|
|
def income2_used_for_mortgage?
|
|
|
|
inc2mort == 1
|
|
|
|
end
|
|
|
|
|
|
|
|
def right_to_buy?
|
|
|
|
[9, 14, 27].include?(type)
|
|
|
|
end
|
|
|
|
|
|
|
|
def rent_to_buy_full_ownership?
|
|
|
|
type == 29
|
|
|
|
end
|
|
|
|
|
|
|
|
def outright_sale_or_discounted_with_full_ownership?
|
|
|
|
ownershipsch == 3 || (ownershipsch == 2 && rent_to_buy_full_ownership?)
|
|
|
|
end
|
|
|
|
|
|
|
|
def social_homebuy?
|
|
|
|
type == 18
|
|
|
|
end
|
|
|
|
|
|
|
|
def ppostcode_full=(postcode)
|
|
|
|
if postcode
|
|
|
|
super UKPostcode.parse(postcode).to_s
|
|
|
|
else
|
|
|
|
super nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def previous_postcode_known?
|
|
|
|
ppcodenk&.zero?
|
|
|
|
end
|
|
|
|
|
|
|
|
def postcode_known?
|
|
|
|
pcodenk&.zero?
|
|
|
|
end
|
|
|
|
|
|
|
|
def postcode_full=(postcode)
|
|
|
|
if postcode
|
|
|
|
super UKPostcode.parse(postcode).to_s
|
|
|
|
else
|
|
|
|
super nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def expected_shared_ownership_deposit_value
|
|
|
|
return unless value && equity
|
|
|
|
|
|
|
|
value * equity / 100
|
|
|
|
end
|
|
|
|
|
|
|
|
def stairbought_part_of_value
|
|
|
|
return unless value && stairbought
|
|
|
|
|
|
|
|
value * stairbought / 100
|
|
|
|
end
|
|
|
|
|
|
|
|
def mortgage_deposit_and_discount_total
|
|
|
|
mortgage_amount = mortgage || 0
|
|
|
|
deposit_amount = deposit || 0
|
|
|
|
cashdis_amount = cashdis || 0
|
|
|
|
|
|
|
|
mortgage_amount + deposit_amount + cashdis_amount
|
|
|
|
end
|
|
|
|
|
|
|
|
def deposit_and_discount_total
|
|
|
|
deposit_amount = deposit || 0
|
|
|
|
cashdis_amount = cashdis || 0
|
|
|
|
|
|
|
|
deposit_amount + cashdis_amount
|
|
|
|
end
|
|
|
|
|
|
|
|
def value_times_equity
|
|
|
|
return unless value && equity
|
|
|
|
|
|
|
|
value * equity / 100
|
|
|
|
end
|
|
|
|
|
|
|
|
def mortgage_deposit_and_discount_error_fields
|
|
|
|
[
|
|
|
|
"mortgage",
|
|
|
|
"deposit",
|
|
|
|
cashdis.present? ? "cash discount" : nil,
|
|
|
|
].compact.to_sentence
|
|
|
|
end
|
|
|
|
|
|
|
|
def mortgage_and_deposit_total
|
|
|
|
return unless mortgage && deposit
|
|
|
|
|
|
|
|
mortgage + deposit
|
|
|
|
end
|
|
|
|
|
|
|
|
def outright_sale?
|
|
|
|
ownershipsch == 3
|
|
|
|
end
|
|
|
|
|
|
|
|
def discounted_ownership_sale?
|
|
|
|
ownershipsch == 2
|
|
|
|
end
|
|
|
|
|
|
|
|
def mortgage_used?
|
|
|
|
mortgageused == 1
|
|
|
|
end
|
|
|
|
|
|
|
|
def mortgage_not_used?
|
|
|
|
mortgageused == 2
|
|
|
|
end
|
|
|
|
|
|
|
|
def mortgage_use_unknown?
|
|
|
|
mortgageused == 3
|
|
|
|
end
|
|
|
|
|
|
|
|
def process_postcode_changes!
|
|
|
|
self.postcode_full = upcase_and_remove_whitespace(postcode_full)
|
|
|
|
return if postcode_full.blank?
|
|
|
|
|
|
|
|
self.pcodenk = 0
|
|
|
|
inferred_la = get_inferred_la(postcode_full)
|
|
|
|
self.is_la_inferred = inferred_la.present?
|
|
|
|
self.la = inferred_la if inferred_la.present?
|
|
|
|
end
|
|
|
|
|
|
|
|
def process_previous_postcode_changes!
|
|
|
|
self.ppostcode_full = upcase_and_remove_whitespace(ppostcode_full)
|
|
|
|
return if ppostcode_full.blank?
|
|
|
|
|
|
|
|
self.ppcodenk = 0
|
|
|
|
inferred_la = get_inferred_la(ppostcode_full)
|
|
|
|
self.is_previous_la_inferred = inferred_la.present?
|
|
|
|
self.prevloc = inferred_la if inferred_la.present?
|
|
|
|
end
|
|
|
|
|
|
|
|
def reset_assigned_to!
|
|
|
|
return unless updated_by&.support?
|
|
|
|
return if owning_organisation.blank? || managing_organisation.blank? || assigned_to.blank?
|
|
|
|
return if assigned_to&.organisation == owning_organisation || assigned_to&.organisation == managing_organisation
|
|
|
|
return if assigned_to&.organisation == owning_organisation.absorbing_organisation || assigned_to&.organisation == managing_organisation.absorbing_organisation
|
|
|
|
|
|
|
|
update!(assigned_to: nil)
|
|
|
|
end
|
|
|
|
|
|
|
|
def joint_purchase?
|
|
|
|
jointpur == 1
|
|
|
|
end
|
|
|
|
|
|
|
|
def not_joint_purchase?
|
|
|
|
jointpur == 2
|
|
|
|
end
|
|
|
|
|
|
|
|
def buyer_has_seen_privacy_notice?
|
|
|
|
privacynotice == 1
|
|
|
|
end
|
|
|
|
|
|
|
|
def buyer_not_interviewed?
|
|
|
|
noint == 1
|
|
|
|
end
|
|
|
|
|
|
|
|
def old_persons_shared_ownership?
|
|
|
|
type == 24
|
|
|
|
end
|
|
|
|
|
|
|
|
def is_bedsit?
|
|
|
|
proptype == 2
|
|
|
|
end
|
|
|
|
|
|
|
|
def is_beds_inferred?
|
|
|
|
form.start_year_2025_or_later? && is_bedsit?
|
|
|
|
end
|
|
|
|
|
|
|
|
def shared_ownership_scheme?
|
|
|
|
ownershipsch == 1
|
|
|
|
end
|
|
|
|
|
|
|
|
def company_buyer?
|
|
|
|
companybuy == 1
|
|
|
|
end
|
|
|
|
|
|
|
|
def no_monthly_leasehold_charges?
|
|
|
|
has_mscharge&.zero?
|
|
|
|
end
|
|
|
|
|
|
|
|
def no_buyer_organisation?
|
|
|
|
pregyrha&.zero? &&
|
|
|
|
pregla&.zero? &&
|
|
|
|
pregghb&.zero? &&
|
|
|
|
pregother&.zero?
|
|
|
|
end
|
|
|
|
|
|
|
|
def buyers_age_for_old_persons_shared_ownership_invalid?
|
|
|
|
return unless old_persons_shared_ownership?
|
|
|
|
|
|
|
|
(joint_purchase? && ages_unknown_or_under_64?([1, 2])) || (not_joint_purchase? && ages_unknown_or_under_64?([1]))
|
|
|
|
end
|
|
|
|
|
|
|
|
def ages_unknown_or_under_64?(person_indexes)
|
|
|
|
person_indexes.all? { |person_num| self["age#{person_num}"].present? && self["age#{person_num}"] < 64 || self["age#{person_num}_known"] == 1 }
|
|
|
|
end
|
|
|
|
|
|
|
|
def purchase_price_soft_min
|
|
|
|
LaSaleRange.find_by(start_year: collection_start_year, la:, bedrooms: beds).soft_min
|
|
|
|
end
|
|
|
|
|
|
|
|
def purchase_price_soft_max
|
|
|
|
LaSaleRange.find_by(start_year: collection_start_year, la:, bedrooms: beds).soft_max
|
|
|
|
end
|
|
|
|
|
|
|
|
def income_soft_min_for_ecstat(ecstat_field)
|
|
|
|
economic_status_code = public_send(ecstat_field)
|
|
|
|
|
|
|
|
return unless ALLOWED_INCOME_RANGES_SALES && ALLOWED_INCOME_RANGES_SALES[economic_status_code]
|
|
|
|
|
|
|
|
soft_min = ALLOWED_INCOME_RANGES_SALES[economic_status_code].soft_min
|
|
|
|
format_as_currency(soft_min)
|
|
|
|
end
|
|
|
|
|
|
|
|
def should_process_uprn_change?
|
|
|
|
return unless uprn
|
|
|
|
return unless saledate
|
|
|
|
|
|
|
|
uprn_changed? || saledate_changed?
|
|
|
|
end
|
|
|
|
|
|
|
|
def should_process_address_change?
|
|
|
|
return unless uprn_selection || select_best_address_match
|
|
|
|
return unless saledate
|
|
|
|
return unless form.start_year_2024_or_later?
|
|
|
|
|
|
|
|
if select_best_address_match
|
|
|
|
address_line1_input.present? && postcode_full_input.present?
|
|
|
|
else
|
|
|
|
uprn_selection_changed? || saledate_changed?
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def value_with_discount
|
|
|
|
return if value.blank?
|
|
|
|
|
|
|
|
discount_amount = discount ? value * discount / 100 : 0
|
|
|
|
value - discount_amount
|
|
|
|
end
|
|
|
|
|
|
|
|
def mortgage_deposit_and_grant_total
|
|
|
|
return if deposit.blank?
|
|
|
|
|
|
|
|
grant_amount = grant || 0
|
|
|
|
mortgage_amount = mortgage || 0
|
|
|
|
mortgage_amount + deposit + grant_amount
|
|
|
|
end
|
|
|
|
|
|
|
|
def beds_for_la_sale_range
|
|
|
|
beds.nil? ? nil : [beds, LaSaleRange::MAX_BEDS].min
|
|
|
|
end
|
|
|
|
|
|
|
|
def ownership_scheme(uppercase: false)
|
|
|
|
ownership_scheme = case ownershipsch
|
|
|
|
when 1 then "shared ownership"
|
|
|
|
when 2 then "discounted ownership"
|
|
|
|
when 3 then "outright or other sale"
|
|
|
|
end
|
|
|
|
uppercase ? ownership_scheme.capitalize : ownership_scheme
|
|
|
|
end
|
|
|
|
|
|
|
|
def combined_income
|
|
|
|
buyer_1_income = income1 || 0
|
|
|
|
buyer_2_income = income2 || 0
|
|
|
|
|
|
|
|
buyer_1_income + buyer_2_income
|
|
|
|
end
|
|
|
|
|
|
|
|
def blank_compound_invalid_non_setup_fields!
|
|
|
|
super
|
|
|
|
|
|
|
|
self.pcodenk = nil if errors.attribute_names.include? :postcode_full
|
|
|
|
end
|
|
|
|
|
|
|
|
def duplicate_check_question_ids
|
|
|
|
["owning_organisation_id",
|
|
|
|
"saledate",
|
|
|
|
"purchid",
|
|
|
|
"age1",
|
|
|
|
"sex1",
|
|
|
|
"ecstat1",
|
|
|
|
uprn.blank? ? "postcode_full" : "uprn"].compact
|
|
|
|
end
|
|
|
|
|
|
|
|
def soctenant_is_inferred?
|
|
|
|
form.start_year_2024_or_later?
|
|
|
|
end
|
|
|
|
|
|
|
|
def duplicates
|
|
|
|
return SalesLog.none if duplicate_set_id.nil?
|
|
|
|
|
|
|
|
SalesLog.where(duplicate_set_id:).where.not(id:)
|
|
|
|
end
|
|
|
|
|
|
|
|
def nationality2_uk_or_prefers_not_to_say?
|
|
|
|
nationality_all_buyer2_group&.zero? || nationality_all_buyer2_group == 826
|
|
|
|
end
|
|
|
|
|
|
|
|
def is_staircase?
|
|
|
|
staircase == 1
|
|
|
|
end
|
|
|
|
|
|
|
|
def discount_value
|
|
|
|
return unless discount && value
|
|
|
|
|
|
|
|
value * discount / 100
|
|
|
|
end
|
|
|
|
|
|
|
|
def is_not_staircasing?
|
|
|
|
staircase == 2 || staircase == 3
|
|
|
|
end
|
|
|
|
|
|
|
|
def stairowned_100?
|
|
|
|
stairowned == 100
|
|
|
|
end
|
|
|
|
|
|
|
|
def address_search_given?
|
|
|
|
address_line1_input.present? && postcode_full_input.present?
|
|
|
|
end
|
|
|
|
|
|
|
|
def is_resale?
|
|
|
|
resale == 1
|
|
|
|
end
|
|
|
|
|
|
|
|
def is_firststair?
|
|
|
|
firststair == 1
|
|
|
|
end
|
|
|
|
end
|