|
|
|
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
|
|
|
|
include Validations::LocalAuthorityValidations
|
|
|
|
include Validations::SubmissionValidations
|
|
|
|
|
|
|
|
def validate(record)
|
|
|
|
validation_methods = public_methods.select { |method| method.starts_with?("validate_") }
|
|
|
|
validation_methods.each { |meth| public_send(meth, record) }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
class CaseLog < ApplicationRecord
|
|
|
|
include Validations::SoftValidations
|
|
|
|
include Constants::CaseLog
|
|
|
|
include Constants::IncomeRanges
|
|
|
|
|
|
|
|
has_paper_trail
|
|
|
|
|
|
|
|
validates_with CaseLogValidator
|
|
|
|
before_validation :process_postcode_changes!, if: :property_postcode_changed?
|
|
|
|
before_validation :process_previous_postcode_changes!, if: :previous_postcode_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_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: STATUS
|
|
|
|
enum letting_in_sheltered_accommodation: SHELTERED_ACCOMMODATION
|
|
|
|
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 sex1: GENDER, _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: POLAR, _suffix: true
|
|
|
|
enum chr: POLAR, _suffix: true
|
|
|
|
enum cap: POLAR, _suffix: true
|
|
|
|
enum letting_allocation_unknown: POLAR, _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 previous_postcode_known: POLAR, _suffix: true
|
|
|
|
enum la_known: POLAR, _suffix: true
|
|
|
|
enum previous_la_known: POLAR, _suffix: true
|
|
|
|
enum net_income_known: NET_INCOME_KNOWN, _suffix: true
|
|
|
|
enum household_charge: POLAR, _suffix: true
|
|
|
|
enum is_carehome: POLAR, _suffix: true
|
|
|
|
enum nocharge: POLAR, _suffix: true
|
|
|
|
enum referral: REFERRAL, _suffix: true
|
|
|
|
enum declaration: POLAR, _suffix: true
|
|
|
|
|
|
|
|
AUTOGENERATED_FIELDS = %w[id status created_at updated_at discarded_at].freeze
|
|
|
|
|
|
|
|
def form
|
|
|
|
FormHandler.instance.get_form(form_name) || FormHandler.instance.forms.first.second
|
|
|
|
end
|
|
|
|
|
|
|
|
def form_name
|
|
|
|
return 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 postcode_known?
|
|
|
|
postcode_known == "Yes"
|
|
|
|
end
|
|
|
|
|
|
|
|
def previous_postcode_known?
|
|
|
|
previous_postcode_known == "Yes"
|
|
|
|
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
|
|
|
|
|
|
|
|
private
|
|
|
|
|
|
|
|
PIO = Postcodes::IO.new
|
|
|
|
|
|
|
|
def update_status!
|
|
|
|
self.status = if all_fields_completed? && errors.empty?
|
|
|
|
"completed"
|
|
|
|
elsif all_fields_nil?
|
|
|
|
"not_started"
|
|
|
|
else
|
|
|
|
"in_progress"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def reset_not_routed_questions
|
|
|
|
form.invalidated_page_questions(self).each do |question|
|
|
|
|
enabled = form.enabled_page_questions(self)
|
|
|
|
answer_options = enabled.map(&:id).include?(question.id) ? enabled.find { |q| q.id == question.id }.answer_options : []
|
|
|
|
contains_selected_answer_option = answer_options.present? ? answer_options.values.map { |x| x["value"] }.include?(public_send(question.id)) : false
|
|
|
|
if !contains_selected_answer_option && respond_to?(question.id.to_s)
|
|
|
|
public_send("#{question.id}=", nil)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def reset_derived_questions
|
|
|
|
dependent_questions = { layear: [{ key: :renewal, value: "No" }],
|
|
|
|
homeless: [{ key: :renewal, value: "No" }],
|
|
|
|
referral: [{ key: :renewal, value: "No" }] }
|
|
|
|
|
|
|
|
dependent_questions.each do |dependent, conditions|
|
|
|
|
condition_key = conditions.first[:key]
|
|
|
|
condition_value = conditions.first[:value]
|
|
|
|
if public_send("#{condition_key}_changed?") && condition_value == public_send(condition_key) && !public_send("#{dependent}_changed?")
|
|
|
|
self[dependent] = nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def reset_invalidated_dependent_fields!
|
|
|
|
return unless form
|
|
|
|
|
|
|
|
reset_not_routed_questions
|
|
|
|
reset_derived_questions
|
|
|
|
end
|
|
|
|
|
|
|
|
def dynamically_not_required
|
|
|
|
(form.invalidated_questions(self) + form.readonly_questions).map(&:id).uniq
|
|
|
|
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
|
|
|
|
self.incref = 1 if net_income_known == "Tenant prefers not to say"
|
|
|
|
self.hhmemb = other_hhmemb + 1 if other_hhmemb.present?
|
|
|
|
self.renttype = RENT_TYPE_MAPPING[rent_type]
|
|
|
|
self.lettype = "#{renttype} #{needstype} #{owning_organisation[:provider_type]}" if renttype.present? && needstype.present? && owning_organisation[:provider_type].present?
|
|
|
|
self.totchild = get_totchild
|
|
|
|
self.totelder = get_totelder
|
|
|
|
self.totadult = get_totadult
|
|
|
|
if %i[brent scharge pscharge supcharg].any? { |f| public_send(f).present? }
|
|
|
|
self.brent ||= 0
|
|
|
|
self.scharge ||= 0
|
|
|
|
self.pscharge ||= 0
|
|
|
|
self.supcharg ||= 0
|
|
|
|
self.tcharge = brent.to_f + scharge.to_f + pscharge.to_f + supcharg.to_f
|
|
|
|
end
|
|
|
|
self.has_benefits = get_has_benefits
|
|
|
|
self.nocharge = household_charge == "Yes" ? "No" : "Yes"
|
|
|
|
self.underoccupation_benefitcap = "No" if renewal == "Yes" && year == 2021
|
|
|
|
if renewal == "Yes"
|
|
|
|
self.homeless = "No"
|
|
|
|
self.referral = "Internal transfer"
|
|
|
|
self.layear = "Less than 1 year"
|
|
|
|
self.reasonpref = nil if reasonpref == "Yes"
|
|
|
|
end
|
|
|
|
if needstype == "General needs"
|
|
|
|
self.prevten = "Fixed-term private registered provider (PRP) general needs tenancy" if managing_organisation.provider_type == "PRP"
|
|
|
|
self.prevten = "Fixed-term local authority general needs tenancy" if managing_organisation.provider_type == "LA"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def process_postcode_changes!
|
|
|
|
process_postcode(property_postcode, "postcode_known", "is_la_inferred", "la", "postcode", "postcod2")
|
|
|
|
end
|
|
|
|
|
|
|
|
def process_previous_postcode_changes!
|
|
|
|
process_postcode(previous_postcode, "previous_postcode_known", "is_previous_la_inferred", "prevloc", "ppostc1", "ppostc2")
|
|
|
|
end
|
|
|
|
|
|
|
|
def process_postcode(postcode, postcode_known_key, la_inferred_key, la_key, outcode_key, incode_key)
|
|
|
|
return if postcode.blank?
|
|
|
|
|
|
|
|
self[postcode_known_key] = "Yes"
|
|
|
|
inferred_la = get_inferred_la(postcode)
|
|
|
|
self[la_inferred_key] = inferred_la.present?
|
|
|
|
self[la_key] = inferred_la if inferred_la.present?
|
|
|
|
self[outcode_key] = UKPostcode.parse(postcode).outcode
|
|
|
|
self[incode_key] = UKPostcode.parse(postcode).incode
|
|
|
|
end
|
|
|
|
|
|
|
|
def reset_location_fields!
|
|
|
|
reset_location(is_la_inferred, "la", "is_la_inferred", "property_postcode", "postcode", "postcod2", la_known)
|
|
|
|
end
|
|
|
|
|
|
|
|
def reset_previous_location_fields!
|
|
|
|
reset_location(is_previous_la_inferred, "prevloc", "is_previous_la_inferred", "previous_postcode", "ppostc1", "ppostc2", previous_la_known)
|
|
|
|
end
|
|
|
|
|
|
|
|
def reset_location(is_inferred, la_key, is_inferred_key, postcode_key, incode_key, outcode_key, is_la_known)
|
|
|
|
if is_inferred || is_la_known != "Yes"
|
|
|
|
self[la_key] = nil
|
|
|
|
end
|
|
|
|
self[is_inferred_key] = false
|
|
|
|
self[postcode_key] = nil
|
|
|
|
self[incode_key] = nil
|
|
|
|
self[outcode_key] = nil
|
|
|
|
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 get_has_benefits
|
|
|
|
return "Yes" if HAS_BENEFITS_OPTIONS.include?(hb)
|
|
|
|
end
|
|
|
|
|
|
|
|
def all_fields_completed?
|
|
|
|
mandatory_fields.none? { |field| public_send(field).nil? if respond_to?(field) }
|
|
|
|
end
|
|
|
|
|
|
|
|
def all_fields_nil?
|
|
|
|
init_fields = %w[owning_organisation_id managing_organisation_id]
|
|
|
|
fields = mandatory_fields.difference(init_fields)
|
|
|
|
fields.none? { |field| public_send(field).present? if respond_to?(field) }
|
|
|
|
end
|
|
|
|
|
|
|
|
def mandatory_fields
|
|
|
|
form.questions.map(&:id).difference(OPTIONAL_FIELDS, dynamically_not_required)
|
|
|
|
end
|
|
|
|
end
|