class Location < ApplicationRecord validates :postcode, on: :postcode, presence: { message: I18n.t("validations.location.postcode_blank") } validate :validate_postcode, on: :postcode, if: proc { |model| model.postcode.presence } validates :location_admin_district, on: :location_admin_district, presence: { message: I18n.t("validations.location_admin_district") } validates :units, on: :units, presence: { message: I18n.t("validations.location.units") } validates :type_of_unit, on: :type_of_unit, presence: { message: I18n.t("validations.location.type_of_unit") } validates :mobility_type, on: :mobility_type, presence: { message: I18n.t("validations.location.mobility_standards") } validates :startdate, on: :startdate, presence: { message: I18n.t("validations.location.startdate_invalid") } validate :validate_startdate, on: :startdate, if: proc { |model| model.startdate.presence } validate :validate_confirmed belongs_to :scheme has_many :lettings_logs, class_name: "LettingsLog" has_many :location_deactivation_periods, class_name: "LocationDeactivationPeriod" has_paper_trail before_validation :lookup_postcode!, if: :postcode_changed? auto_strip_attributes :name, squish: true scope :search_by_postcode, ->(postcode) { where("REPLACE(locations.postcode, ' ', '') ILIKE ?", "%#{postcode.delete(' ')}%") } scope :search_by_name, ->(name) { where("locations.name ILIKE ?", "%#{name}%") } scope :search_by, ->(param) { search_by_name(param).or(search_by_postcode(param)) } scope :started, -> { where("locations.startdate <= ?", Time.zone.today).or(where(startdate: nil)) } scope :started_in_2_weeks, -> { where("locations.startdate <= ?", Time.zone.today + 2.weeks).or(where(startdate: nil)) } scope :active_in_2_weeks, -> { where(confirmed: true).and(started_in_2_weeks) } scope :confirmed, -> { where(confirmed: true) } scope :unconfirmed, -> { where.not(confirmed: true) } scope :filter_by_status, lambda { |statuses, _user = nil| filtered_records = all scopes = [] statuses.each do |status| if respond_to?(status, true) scopes << (status == "active" ? send("active_status") : send(status)) end end if scopes.any? filtered_records = filtered_records .left_outer_joins(:location_deactivation_periods) .joins(scheme: [:owning_organisation]) .left_outer_joins(scheme: :scheme_deactivation_periods) .order("location_deactivation_periods.created_at DESC") .merge(scopes.reduce(&:or)) end filtered_records } scope :incomplete, lambda { where.not(confirmed: true) .or(where(confirmed: nil)) } scope :deactivated, lambda { |date = Time.zone.now| deactivated_by_organisation(date) .or(deactivated_directly(date)) .or(deactivated_by_scheme(date)) } scope :deactivated_by_organisation, lambda { |date = Time.zone.now| merge(Organisation.filter_by_inactive.or(Organisation.where("merge_date <= ?", date))) } scope :deactivated_by_scheme, lambda { |date = Time.zone.now| merge(Scheme.deactivated_directly(date)) } scope :deactivated_directly, lambda { |date = Time.zone.now| merge(LocationDeactivationPeriod.deactivations_without_reactivation) .where("location_deactivation_periods.deactivation_date <= ?", date) } scope :deactivating_soon_directly, lambda { |date = Time.zone.now| merge(LocationDeactivationPeriod.deactivations_without_reactivation) .where("location_deactivation_periods.deactivation_date > ?", date) } scope :deactivating_soon, lambda { |date = Time.zone.now| deactivating_soon_directly .or(deactivating_soon_by_scheme(date)) } scope :deactivating_soon_by_scheme, lambda { |date = Time.zone.now| merge(Scheme.deactivating_soon(date)) } scope :reactivating_soon, lambda { |date = Time.zone.now| where.not("location_deactivation_periods.reactivation_date IS NULL") .where("location_deactivation_periods.reactivation_date > ?", date) .where.not(id: joins(scheme: [:owning_organisation]).deactivated_by_organisation.pluck(:id)) } scope :reactivating_soon_by_scheme, lambda { |date = Time.zone.now| merge(Scheme.reactivating_soon(date)) } scope :activating_soon, lambda { |date = Time.zone.now| where("locations.startdate > ?", date) } scope :active_status, lambda { where.not(id: joins(:location_deactivation_periods).reactivating_soon.pluck(:id)) .where.not(id: joins(scheme: [:scheme_deactivation_periods]).reactivating_soon_by_scheme.pluck(:id)) .where.not(id: joins(:location_deactivation_periods).merge(Location.deactivated_directly).pluck(:id)) .where.not(id: joins(scheme: [:scheme_deactivation_periods]).merge(Location.deactivated_by_scheme).pluck(:id)) .where.not(id: joins(scheme: [:owning_organisation]).merge(Location.deactivated_by_organisation).pluck(:id)) .where.not(id: incomplete.pluck(:id)) .where.not(id: joins(:location_deactivation_periods).merge(Location.deactivating_soon_directly).pluck(:id)) .where.not(id: joins(scheme: %i[owning_organisation scheme_deactivation_periods]).merge(Location.deactivating_soon_by_scheme).pluck(:id)) .where.not(id: activating_soon.pluck(:id)) } scope :active, lambda { |date = Time.zone.now| where.not(id: joins(:location_deactivation_periods).merge(Location.deactivated_directly(date)).pluck(:id)) .where.not(id: incomplete.pluck(:id)) .where.not(id: activating_soon(date).pluck(:id)) .where(scheme: Scheme.active(date)) } scope :visible, -> { where(discarded_at: nil) } scope :duplicate_sets, lambda { scope = visible .group(*DUPLICATE_LOCATION_ATTRIBUTES) .where.not(scheme_id: nil) .where.not(postcode: nil) .where.not(mobility_type: nil) .having( "COUNT(*) > 1", ) scope.pluck("ARRAY_AGG(id)") } scope :duplicate_sets_within_given_schemes, lambda { scope = visible .group(*DUPLICATE_LOCATION_ATTRIBUTES - %w[scheme_id]) .where.not(postcode: nil) .where.not(mobility_type: nil) .having( "COUNT(*) > 1", ) scope.pluck("ARRAY_AGG(id)") } DUPLICATE_LOCATION_ATTRIBUTES = %w[scheme_id postcode mobility_type].freeze LOCAL_AUTHORITIES = LocalAuthority.all.map { |la| [la.name, la.code] }.to_h enum local_authorities: LOCAL_AUTHORITIES def self.local_authorities_for_current_year LocalAuthority.all.active(Time.zone.today).england.map { |la| [la.code, la.name] }.to_h end MOBILITY_TYPE = { "Wheelchair-user standard": "W", "Fitted with equipment and adaptations": "A", "Property designed to accessible general standard": "M", "None": "N", "Missing": "X", }.freeze enum mobility_type: MOBILITY_TYPE TYPE_OF_UNIT = { "Bungalow": 6, "Self-contained flat or bedsit": 1, "Self-contained flat or bedsit with common facilities": 2, "Self-contained house": 7, "Shared flat": 3, "Shared house or hostel": 4, }.freeze enum type_of_unit: TYPE_OF_UNIT def self.find_by_id_on_multiple_fields(id) return if id.nil? where(id:).or(where("ltrim(old_visible_id, '0') = ?", id.to_i.to_s)).first end def postcode=(postcode) if postcode super UKPostcode.parse(postcode).to_s else super nil end end def available_from startdate || FormHandler.instance.earliest_open_collection_start_date(now: created_at) end def open_deactivation location_deactivation_periods.deactivations_without_reactivation.first end def last_deactivation_before(date) location_deactivation_periods.where("deactivation_date <= ?", date).order("created_at").last end def status @status ||= status_at(Time.zone.now) end def status_at(date) return :deleted if discarded_at.present? return :incomplete unless confirmed return :deactivated if scheme.owning_organisation.status_at(date) == :deactivated || scheme.owning_organisation.status_at(date) == :merged || open_deactivation&.deactivation_date.present? && date >= open_deactivation.deactivation_date || scheme.status_at(date) == :deactivated return :deactivating_soon if open_deactivation&.deactivation_date.present? && date < open_deactivation.deactivation_date || scheme.status_at(date) == :deactivating_soon return :activating_soon if startdate.present? && date < startdate return :reactivating_soon if last_deactivation_before(date)&.reactivation_date.present? && date < last_deactivation_before(date).reactivation_date || scheme.status_at(date) == :reactivating_soon :active end def active? status == :active end def deactivated? status == :deactivated end def reactivating_soon? status == :reactivating_soon end def deactivates_in_a_long_time? status_at(6.months.from_now) == :deactivating_soon end def deactivated_by_scheme? status == :deactivated && scheme.status == :deactivated end def deactivating_soon_by_scheme? status == :deactivating_soon && scheme.status == :deactivating_soon end def reactivating_soon_by_scheme? status == :reactivating_soon && scheme.status == :reactivating_soon end def validate_postcode if !postcode&.match(POSTCODE_REGEXP) error_message = I18n.t("validations.postcode") errors.add :postcode, error_message else self.postcode = PostcodeService.clean(postcode) end end def validate_startdate unless startdate.between?(scheme.available_from, Time.zone.local(2200, 1, 1)) error_message = I18n.t("validations.location.startdate_out_of_range", date: scheme.available_from.to_formatted_s(:govuk_date)) errors.add :startdate, error_message end end def validate_confirmed self.confirmed = [postcode, location_admin_district, location_code, units, type_of_unit, mobility_type].all?(&:present?) end def linked_local_authorities la = LocalAuthority.find_by(code: location_code) return LocalAuthority.none unless la LocalAuthority.where(id: [la.id] + la.linked_local_authority_ids) end def discard! update!(discarded_at: Time.zone.now) end private PIO = PostcodeService.new def lookup_postcode! result = PIO.lookup(postcode) unless location_admin_district_changed? self.location_admin_district = nil self.location_code = nil self.is_la_inferred = false end if result self.location_code = result[:location_code] self.location_admin_district = result[:location_admin_district] self.is_la_inferred = true else self.is_la_inferred = false end end end