Submit social housing lettings and sales data (CORE)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

352 lines
11 KiB

3 years ago
class CaseLogValidator < ActiveModel::Validator
# Validations methods need to be called 'validate_' to run on model save
# or form page submission
include Validations::HouseholdValidations
include Validations::PropertyValidations
include Validations::FinancialValidations
include Validations::TenancyValidations
include Validations::DateValidations
3 years ago
def validate(record)
validation_methods = public_methods.select { |method| method.starts_with?("validate_") }
validation_methods.each { |meth| public_send(meth, record) }
3 years ago
end
3 years ago
private
def validate_other_field(record, main_field, other_field)
main_field_label = main_field.humanize(capitalize: false)
other_field_label = other_field.humanize(capitalize: false)
if record[main_field] == "Other" && record[other_field].blank?
record.errors.add other_field.to_sym, "If #{main_field_label} is other then #{other_field_label} must be provided"
end
if record[main_field] != "Other" && record[other_field].present?
record.errors.add other_field.to_sym, "#{other_field_label} must not be provided if #{main_field_label} was not other"
end
end
3 years ago
end
3 years ago
class CaseLog < ApplicationRecord
include Discard::Model
include Validations::SoftValidations
include Constants::CaseLog
include Constants::IncomeRanges
default_scope -> { kept }
validates_with CaseLogValidator
before_save :update_status!
belongs_to :owning_organisation, class_name: "Organisation"
belongs_to :managing_organisation, class_name: "Organisation"
scope :for_organisation, ->(org) { where(owning_organisation: org).or(where(managing_organisation: org)) }
enum status: { "not_started" => 0, "in_progress" => 1, "completed" => 2 }
enum ethnic: ETHNIC
enum national: NATIONAL, _suffix: true
enum ecstat1: ECSTAT, _suffix: true
enum ecstat2: ECSTAT, _suffix: true
enum ecstat3: ECSTAT, _suffix: true
enum ecstat4: ECSTAT, _suffix: true
enum ecstat5: ECSTAT, _suffix: true
enum ecstat6: ECSTAT, _suffix: true
enum ecstat7: ECSTAT, _suffix: true
enum ecstat8: ECSTAT, _suffix: true
enum relat2: RELAT, _suffix: true
enum relat3: RELAT, _suffix: true
enum relat4: RELAT, _suffix: true
enum relat5: RELAT, _suffix: true
enum relat6: RELAT, _suffix: true
enum relat7: RELAT, _suffix: true
enum relat8: RELAT, _suffix: true
enum sex2: GENDER, _suffix: true
enum sex3: GENDER, _suffix: true
enum sex4: GENDER, _suffix: true
enum sex5: GENDER, _suffix: true
enum sex6: GENDER, _suffix: true
enum sex7: GENDER, _suffix: true
enum sex8: GENDER, _suffix: true
enum prevten: PREVIOUS_TENANCY, _suffix: true
enum homeless: HOMELESS, _suffix: true
enum underoccupation_benefitcap: BENEFITCAP, _suffix: true
enum reservist: RESERVIST, _suffix: true
enum leftreg: LEFTREG, _suffix: true
enum illness: ILLNESS, _suffix: true
enum preg_occ: PREGNANCY, _suffix: true
enum override_net_income_validation: POLAR, _suffix: true
enum housingneeds_a: POLAR, _suffix: true
enum housingneeds_b: POLAR, _suffix: true
enum housingneeds_c: POLAR, _suffix: true
enum housingneeds_f: POLAR, _suffix: true
enum housingneeds_g: POLAR, _suffix: true
enum housingneeds_h: POLAR, _suffix: true
enum accessibility_requirements_prefer_not_to_say: POLAR, _suffix: true
enum illness_type_1: POLAR, _suffix: true
enum illness_type_2: POLAR, _suffix: true
enum illness_type_3: POLAR, _suffix: true
enum illness_type_4: POLAR, _suffix: true
enum illness_type_5: POLAR, _suffix: true
enum illness_type_6: POLAR, _suffix: true
enum illness_type_7: POLAR, _suffix: true
enum illness_type_8: POLAR, _suffix: true
enum illness_type_9: POLAR, _suffix: true
enum illness_type_10: POLAR, _suffix: true
enum startertenancy: POLAR2, _suffix: true
enum tenancy: TENANCY, _suffix: true
enum landlord: LANDLORD, _suffix: true
enum rsnvac: RSNVAC, _suffix: true
enum unittype_gn: UNITTYPE_GN, _suffix: true
enum rp_homeless: POLAR, _suffix: true
enum rp_insan_unsat: POLAR, _suffix: true
enum rp_medwel: POLAR, _suffix: true
enum rp_hardship: POLAR, _suffix: true
enum rp_dontknow: POLAR, _suffix: true
enum cbl: POLAR2, _suffix: true
enum chr: POLAR2, _suffix: true
enum cap: POLAR2, _suffix: true
enum wchair: POLAR2, _suffix: true
enum incfreq: INCFREQ, _suffix: true
enum benefits: BENEFITS, _suffix: true
enum period: PERIOD, _suffix: true
enum layear: LATIME, _suffix: true
enum lawaitlist: LATIME, _suffix: true
enum reasonpref: POLAR_WITH_UNKNOWN, _suffix: true
enum reason: REASON, _suffix: true
enum la: ENGLISH_LA, _suffix: true
enum prevloc: UK_LA, _suffix: true
enum majorrepairs: POLAR, _suffix: true
enum hb: HOUSING_BENEFIT, _suffix: true
enum hbrentshortfall: POLAR_WITH_UNKNOWN, _suffix: true
enum property_relet: POLAR, _suffix: true
enum armedforces: ARMED_FORCES, _suffix: true
enum first_time_property_let_as_social_housing: POLAR, _suffix: true
enum unitletas: UNITLETAS, _suffix: true
enum builtype: BUILTYPE, _suffix: true
enum incref: POLAR, _suffix: true
enum renttype: RENT_TYPE, _suffix: true
enum needstype: NEEDS_TYPE, _suffix: true
enum lettype: LET_TYPE, _suffix: true
enum postcode_known: POLAR, _suffix: true
enum la_known: POLAR, _suffix: true
enum net_income_known: NET_INCOME_KNOWN, _suffix: true
AUTOGENERATED_FIELDS = %w[id status created_at updated_at discarded_at renttype lettype is_la_inferred totchild totelder totadult incfreq tcharge].freeze
OPTIONAL_FIELDS = %w[postcode_known
la_known
first_time_property_let_as_social_housing].freeze
def form
FormHandler.instance.get_form(form_name)
end
def form_name
return "2021_2022" unless startdate
window_end_date = Time.zone.local(startdate.year, 4, 1)
if startdate < window_end_date
"#{startdate.year - 1}_#{startdate.year}"
else
"#{startdate.year}_#{startdate.year + 1}"
end
end
def self.editable_fields
attribute_names - AUTOGENERATED_FIELDS
end
def completed?
status == "completed"
end
def not_started?
status == "not_started"
end
def in_progress?
status == "in_progress"
end
def weekly_net_income
return unless earnings && incfreq
case incfreq
when "Weekly"
earnings
when "Monthly"
((earnings * 12) / 52.0).round(0)
when "Yearly"
(earnings / 12.0).round(0)
end
end
def applicable_income_range
return unless ecstat1
ALLOWED_INCOME_RANGES[ecstat1.to_sym]
end
def first_time_property_let_as_social_housing?
first_time_property_let_as_social_housing == "Yes"
end
3 years ago
private
PIO = Postcodes::IO.new
def update_status!
self.status = if all_fields_completed? && errors.empty?
3 years ago
"completed"
elsif all_fields_nil?
"not_started"
else
"in_progress"
end
set_derived_fields
end
def set_derived_fields
if previous_postcode.present?
self.ppostc1 = UKPostcode.parse(previous_postcode).outcode
self.ppostc2 = UKPostcode.parse(previous_postcode).incode
end
if mrcdate.present?
self.mrcday = mrcdate.day
self.mrcmonth = mrcdate.month
self.mrcyear = mrcdate.year
end
if startdate.present?
self.day = startdate.day
self.month = startdate.month
self.year = startdate.year
end
case net_income_known
when "Yes – the household has a weekly income"
self.incfreq = "Weekly"
when "Yes – the household has a monthly income"
self.incfreq = "Monthly"
when "Yes – the household has a yearly income"
self.incfreq = "Yearly"
when "Tenant prefers not to say"
self.incref = 1
end
self.hhmemb = other_hhmemb + 1 if other_hhmemb.present?
self.renttype = RENT_TYPE_MAPPING[rent_type]
self.lettype = "#{renttype} #{needstype} #{owning_organisation['Org type']}" if renttype.present? && needstype.present? && owning_organisation["Org type"].present?
self.is_la_inferred = false if is_la_inferred.nil?
if property_postcode.blank? || postcode_known == "No"
reset_location_fields
else
inferred_la = get_inferred_la(property_postcode)
self.is_la_inferred = inferred_la.present?
self.la = inferred_la if inferred_la.present?
end
if property_postcode.present?
self.postcode = UKPostcode.parse(property_postcode).outcode
self.postcod2 = UKPostcode.parse(property_postcode).incode
else
self.postcode = self.postcod2 = nil
end
self.totchild = get_totchild
self.totelder = get_totelder
self.totadult = get_totadult
self.tcharge = brent.to_i + scharge.to_i + pscharge.to_i + supcharg.to_i
end
def get_totelder
ages = [age1, age2, age3, age4, age5, age6, age7, age8]
ages.count { |x| !x.nil? && x >= 60 }
end
def get_totchild
relationships = [relat2, relat3, relat4, relat5, relat6, relat7, relat8]
relationships.count("Child - includes young adult and grown-up")
end
def get_totadult
total = !age1.nil? && age1 >= 16 && age1 < 60 ? 1 : 0
total + (2..8).count do |i|
age = public_send("age#{i}")
relat = public_send("relat#{i}")
!age.nil? && ((age >= 16 && age < 18 && %w[Partner Other].include?(relat)) || age >= 18 && age < 60)
end
end
def get_inferred_la(postcode)
postcode_lookup = nil
Timeout.timeout(5) { postcode_lookup = PIO.lookup(postcode) }
if postcode_lookup && postcode_lookup.info.present?
postcode_lookup.admin_district
end
end
def reset_location_fields
if is_la_inferred == true
self.la = nil
end
self.is_la_inferred = false
self.property_postcode = nil
end
def all_fields_completed?
mandatory_fields.none? { |_key, val| val.nil? }
end
def all_fields_nil?
init_fields = %w[owning_organisation_id managing_organisation_id]
fields = mandatory_fields.except(*init_fields)
fields.all? { |_key, val| val.nil? }
end
def mandatory_fields
required = attributes.except(*(AUTOGENERATED_FIELDS + OPTIONAL_FIELDS))
dynamically_not_required = []
if reason != "Other"
dynamically_not_required << "other_reason_for_leaving_last_settled_home"
end
if earnings.to_i.zero?
dynamically_not_required << "incfreq"
end
if sale_or_letting == "Letting"
dynamically_not_required << "sale_completion_date"
end
if la.present?
dynamically_not_required << "why_dont_you_know_la"
end
if tenancy == "Secure (including flexible)"
dynamically_not_required << "tenancylength"
end
unless net_income_in_soft_max_range? || net_income_in_soft_min_range?
dynamically_not_required << "override_net_income_validation"
end
unless tenancy == "Other"
dynamically_not_required << "tenancyother"
end
dynamically_not_required << if net_income_known == "Tenant prefers not to say"
"earnings"
else
"incref"
end
3 years ago
start_range = (other_hhmemb || 0) + 2
(start_range..8).each do |n|
dynamically_not_required << "age#{n}"
dynamically_not_required << "sex#{n}"
dynamically_not_required << "relat#{n}"
dynamically_not_required << "ecstat#{n}"
end
required.delete_if { |key, _value| dynamically_not_required.include?(key) }
end
3 years ago
end