require " rails_helper "
RSpec . describe Location , type : :model do
describe " # new " do
let ( :location ) { FactoryBot . build ( :location ) }
before do
stub_request ( :get , / api.postcodes.io / )
. to_return ( status : 200 , body : " { \" status \" :200, \" result \" :{ \" admin_district \" : \" Manchester \" , \" codes \" :{ \" admin_district \" : \" E08000003 \" }}} " , headers : { } )
end
it " belongs to an organisation " do
expect ( location . scheme ) . to be_a ( Scheme )
end
it " infers the local authority " do
location . postcode = " M1 1AE "
location . save!
expect ( location . location_code ) . to eq ( " E08000003 " )
end
end
describe " # validate_postcode " do
let ( :location ) { FactoryBot . build ( :location ) }
it " does not add an error if postcode is valid " do
location . postcode = " M1 1AE "
location . save!
expect ( location . errors ) . to be_empty
end
it " does add an error when the postcode is invalid " do
location . postcode = " invalid "
expect { location . save! }
. to raise_error ( ActiveRecord :: RecordInvalid , " Validation failed: Postcode #{ I18n . t ( 'validations.postcode' ) } " )
end
end
describe " # units " do
let ( :location ) { FactoryBot . build ( :location ) }
it " does add an error when the postcode is invalid " do
location . units = nil
expect { location . save! }
. to raise_error ( ActiveRecord :: RecordInvalid , " Validation failed: Units #{ I18n . t ( 'activerecord.errors.models.location.attributes.units.blank' ) } " )
end
end
describe " # type_of_unit " do
let ( :location ) { FactoryBot . build ( :location ) }
it " does add an error when the postcode is invalid " do
location . type_of_unit = nil
expect { location . save! }
. to raise_error ( ActiveRecord :: RecordInvalid , " Validation failed: Type of unit #{ I18n . t ( 'activerecord.errors.models.location.attributes.type_of_unit.blank' ) } " )
end
end
describe " paper trail " do
let ( :location ) { FactoryBot . create ( :location ) }
let! ( :name ) { location . name }
it " creates a record of changes to a log " do
expect { location . update! ( name : " new test name " ) } . to change ( location . versions , :count ) . by ( 1 )
end
it " allows lettings logs to be restored to a previous version " do
location . update! ( name : " new test name " )
expect ( location . paper_trail . previous_version . name ) . to eq ( name )
end
end
describe " scopes " do
before do
FactoryBot . create ( :location , name : " ABC " , postcode : " NW1 8RR " , startdate : Time . zone . today )
FactoryBot . create ( :location , name : " XYZ " , postcode : " SE1 6HJ " , startdate : Time . zone . today + 1 . day )
FactoryBot . create ( :location , name : " GHQ " , postcode : " EW1 7JK " , startdate : Time . zone . today - 1 . day , confirmed : false )
FactoryBot . create ( :location , name : " GHQ " , postcode : " EW1 7JK " , startdate : nil )
end
context " when searching by name " do
it " returns case insensitive matching records " do
expect ( described_class . search_by_name ( " abc " ) . count ) . to eq ( 1 )
expect ( described_class . search_by_name ( " AbC " ) . count ) . to eq ( 1 )
end
end
context " when searching by postcode " do
it " returns case insensitive matching records " do
expect ( described_class . search_by_postcode ( " se1 6hj " ) . count ) . to eq ( 1 )
expect ( described_class . search_by_postcode ( " SE1 6HJ " ) . count ) . to eq ( 1 )
end
end
context " when searching by all searchable field " do
it " returns case insensitive matching records " do
expect ( described_class . search_by ( " aBc " ) . count ) . to eq ( 1 )
expect ( described_class . search_by ( " nw18rr " ) . count ) . to eq ( 1 )
end
end
context " when filtering by started locations " do
it " returns only locations that started today or earlier " do
expect ( described_class . started . count ) . to eq ( 3 )
end
end
context " when filtering by active locations " do
it " returns only locations that started today or earlier and have been confirmed " do
expect ( described_class . active . count ) . to eq ( 2 )
end
end
end
describe " status " do
let ( :location ) { FactoryBot . build ( :location , startdate : Time . zone . local ( 2022 , 4 , 1 ) ) }
before do
Timecop . freeze ( 2022 , 6 , 7 )
end
after do
Timecop . unfreeze
end
context " when there have not been any previous deactivations " do
it " returns active if the location has no deactivation records " do
expect ( location . status ) . to eq ( :active )
end
it " returns deactivating soon if deactivation_date is in the future " do
FactoryBot . create ( :location_deactivation_period , deactivation_date : Time . zone . local ( 2022 , 8 , 8 ) , location : )
location . save!
expect ( location . status ) . to eq ( :deactivating_soon )
end
it " returns deactivated if deactivation_date is in the past " do
FactoryBot . create ( :location_deactivation_period , deactivation_date : Time . zone . local ( 2022 , 6 , 6 ) , location : )
location . save!
expect ( location . status ) . to eq ( :deactivated )
end
it " returns deactivated if deactivation_date is today " do
FactoryBot . create ( :location_deactivation_period , deactivation_date : Time . zone . local ( 2022 , 6 , 7 ) , location : )
location . save!
expect ( location . status ) . to eq ( :deactivated )
end
it " returns reactivating soon if the location has a future reactivation date " do
FactoryBot . create ( :location_deactivation_period , deactivation_date : Time . zone . local ( 2022 , 6 , 7 ) , reactivation_date : Time . zone . local ( 2022 , 6 , 8 ) , location : )
location . save!
expect ( location . status ) . to eq ( :reactivating_soon )
end
it " returns activating soon if the location has a future startdate " do
location . startdate = Time . zone . local ( 2022 , 7 , 7 )
location . save!
expect ( location . status ) . to eq ( :activating_soon )
end
end
context " when there have been previous deactivations " do
before do
FactoryBot . create ( :location_deactivation_period , deactivation_date : Time . zone . local ( 2022 , 5 , 4 ) , reactivation_date : Time . zone . local ( 2022 , 6 , 5 ) , location : )
location . save!
end
it " returns active if the location has no relevant deactivation records " do
expect ( location . status ) . to eq ( :active )
end
it " returns deactivating soon if deactivation_date is in the future " do
FactoryBot . create ( :location_deactivation_period , deactivation_date : Time . zone . local ( 2022 , 8 , 8 ) , location : )
location . save!
expect ( location . status ) . to eq ( :deactivating_soon )
end
it " returns deactivated if deactivation_date is in the past " do
FactoryBot . create ( :location_deactivation_period , deactivation_date : Time . zone . local ( 2022 , 6 , 6 ) , location : )
location . save!
expect ( location . status ) . to eq ( :deactivated )
end
it " returns deactivated if deactivation_date is today " do
FactoryBot . create ( :location_deactivation_period , deactivation_date : Time . zone . local ( 2022 , 6 , 7 ) , location : )
location . save!
expect ( location . status ) . to eq ( :deactivated )
end
it " returns reactivating soon if the location has a future reactivation date " do
Timecop . freeze ( 2022 , 6 , 8 )
FactoryBot . create ( :location_deactivation_period , deactivation_date : Time . zone . local ( 2022 , 6 , 7 ) , reactivation_date : Time . zone . local ( 2022 , 6 , 9 ) , location : )
location . save!
expect ( location . status ) . to eq ( :reactivating_soon )
end
it " returns reactivating soon if the location had a deactivation during another deactivation " do
Timecop . freeze ( 2022 , 6 , 4 )
FactoryBot . create ( :location_deactivation_period , deactivation_date : Time . zone . local ( 2022 , 5 , 5 ) , reactivation_date : Time . zone . local ( 2022 , 6 , 2 ) , location : )
location . save!
expect ( location . status ) . to eq ( :reactivating_soon )
end
it " returns activating soon if the location has a future startdate " do
location . startdate = Time . zone . local ( 2022 , 7 , 7 )
location . save!
expect ( location . status ) . to eq ( :activating_soon )
end
end
end
describe " available_from " do
context " when there is a startdate " do
let ( :location ) { FactoryBot . build ( :location , startdate : Time . zone . local ( 2022 , 4 , 6 ) ) }
it " returns the startdate " do
expect ( location . available_from ) . to eq ( Time . zone . local ( 2022 , 4 , 6 ) )
end
end
context " when there is no start date " do
context " and the location was created at the start of the 2022/23 collection window " do
let ( :location ) { FactoryBot . build ( :location , created_at : Time . zone . local ( 2022 , 4 , 6 ) , startdate : nil ) }
it " returns the beginning of 22/23 collection window " do
expect ( location . available_from ) . to eq ( Time . zone . local ( 2022 , 4 , 1 ) )
end
end
context " and the location was created at the end of the 2022/23 collection window " do
let ( :location ) { FactoryBot . build ( :location , created_at : Time . zone . local ( 2023 , 2 , 6 ) , startdate : nil ) }
it " returns the beginning of 22/23 collection window " do
expect ( location . available_from ) . to eq ( Time . zone . local ( 2022 , 4 , 1 ) )
end
end
context " and the location was created at the start of the 2021/22 collection window " do
let ( :location ) { FactoryBot . build ( :location , created_at : Time . zone . local ( 2021 , 4 , 6 ) , startdate : nil ) }
it " returns the beginning of 21/22 collection window " do
expect ( location . available_from ) . to eq ( Time . zone . local ( 2021 , 4 , 1 ) )
end
end
context " and the location was created at the end of the 2021/22 collection window " do
let ( :location ) { FactoryBot . build ( :location , created_at : Time . zone . local ( 2022 , 2 , 6 ) , startdate : nil ) }
it " returns the beginning of 21/22 collection window " do
expect ( location . available_from ) . to eq ( Time . zone . local ( 2021 , 4 , 1 ) )
end
end
end
end
end