require " rails_helper "
RSpec . describe BulkUpload :: Sales :: Year2023 :: RowParser do
subject ( :parser ) { described_class . new ( attributes ) }
let ( :now ) { Time . zone . parse ( " 01/05/2023 " ) }
let ( :attributes ) { { bulk_upload : } }
let ( :bulk_upload ) { create ( :bulk_upload , :sales , user : , year : 2023 ) }
let ( :user ) { create ( :user , organisation : owning_org ) }
let ( :owning_org ) { create ( :organisation , :with_old_visible_id ) }
let ( :setup_section_params ) do
{
bulk_upload : ,
field_1 : owning_org . old_visible_id , # organisation
field_2 : user . email , # user
field_3 : now . day . to_s , # sale day
field_4 : now . month . to_s , # sale month
field_5 : now . strftime ( " %g " ) , # sale year
field_6 : " test id " , # purchase id
field_7 : " 1 " , # owhershipsch
field_8 : " 2 " , # shared ownership sale type
field_13 : " 1 " , # will the buyers live in the property
field_14 : " 2 " , # joint purchase
}
end
let ( :valid_attributes ) do
{
bulk_upload : ,
field_1 : owning_org . old_visible_id ,
field_3 : " 22 " ,
field_4 : " 2 " ,
field_5 : " 24 " ,
field_6 : " test id " ,
field_7 : " 1 " ,
field_8 : " 2 " ,
field_13 : " 1 " ,
field_14 : " 1 " ,
field_15 : " 2 " ,
field_16 : " 2 " ,
field_17 : " 1 " ,
field_18 : " 1 " ,
field_19 : " 12 " ,
field_24 : " CR0 " ,
field_25 : " 4BB " ,
field_26 : " E09000008 " ,
field_27 : " 3 " ,
field_28 : " 1 " ,
field_29 : " 1 " ,
field_30 : " 32 " ,
field_31 : " M " ,
field_32 : " 12 " ,
field_33 : " 18 " ,
field_35 : " 1 " ,
field_36 : " 1 " ,
field_37 : " R " ,
field_38 : " 32 " ,
field_39 : " F " ,
field_40 : " 17 " ,
field_41 : " 13 " ,
field_43 : " 2 " ,
field_44 : " 1 " ,
field_45 : " 0 " ,
field_62 : " 1 " ,
field_63 : " 1 " ,
field_64 : " A1 " ,
field_65 : " 1AA " ,
field_66 : " E09000008 " ,
field_69 : " 1 " ,
field_70 : " 1 " ,
field_71 : " 3 " ,
field_73 : " 3 " ,
field_75 : " 5 " ,
field_76 : " 3 " ,
field_77 : " 3 " ,
field_78 : " 30000 " ,
field_79 : " 1 " ,
field_80 : " 15000 " ,
field_81 : " 1 " ,
field_82 : " 4 " ,
field_83 : " 20000 " ,
field_84 : " 3 " ,
field_86 : " 5 " ,
field_87 : " 1 " ,
field_88 : " 10 " ,
field_89 : " 10 " ,
field_90 : " 1 " ,
field_92 : " 30 " ,
field_93 : " 3 " ,
field_94 : " 2022 " ,
field_95 : " 24 " ,
field_96 : " 3 " ,
field_97 : " 2022 " ,
field_98 : " 3 " ,
field_99 : " 1 " ,
field_100 : " 1 " ,
field_101 : " 1 " ,
field_102 : " 1 " ,
field_103 : " 250000 " ,
field_104 : " 25 " ,
field_105 : " 1 " ,
field_106 : " 42500 " ,
field_107 : " 1 " ,
field_109 : " 20 " ,
field_110 : " 3 " ,
field_111 : " 20000 " ,
field_113 : " 800 " ,
field_114 : " 200 " ,
}
end
around do | example |
Timecop . freeze ( Time . zone . local ( 2024 , 2 , 22 ) ) do
Singleton . __init__ ( FormHandler )
example . run
end
end
describe " # blank_row? " do
context " when a new object " do
it " returns true " do
expect ( parser ) . to be_blank_row
end
end
context " when any field is populated " do
before do
parser . field_1 = " 1 "
end
it " returns false " do
expect ( parser ) . not_to be_blank_row
end
end
context " when the only populated fields are empty strings or whitespace " do
before do
parser . field_6 = " "
parser . field_19 = " "
end
it " returns true " do
expect ( parser ) . to be_blank_row
end
end
end
describe " purchaser_code " do
before do
def purch_id_field
described_class :: QUESTIONS . key ( " What is the purchaser code? " ) . to_s
end
end
let ( :attributes ) do
{
bulk_upload : ,
purch_id_field = > " some purchaser code " ,
}
end
it " is linked to the correct field " do
expect ( parser . purchaser_code ) . to eq ( " some purchaser code " )
end
end
describe " previous postcode known " do
context " when field_63 is 1 " do
let ( :attributes ) do
{
bulk_upload : ,
field_63 : 1 ,
}
end
it " sets previous postcode known to yes " do
expect ( parser . log . ppcodenk ) . to eq ( 0 )
end
end
context " when field_63 is 2 " do
let ( :attributes ) do
{
bulk_upload : ,
field_63 : 2 ,
}
end
it " sets previous postcode known to no " do
expect ( parser . log . ppcodenk ) . to eq ( 1 )
end
end
end
describe " income and savings fields " do
context " when set to R " do
let ( :attributes ) do
{
bulk_upload : ,
field_78 : " R " , # income 1
field_80 : " R " , # income 2
field_83 : " R " , # savings
}
end
it " sets the not known field as not known " do
expect ( parser . log . income1nk ) . to be ( 1 )
expect ( parser . log . income2nk ) . to be ( 1 )
expect ( parser . log . savingsnk ) . to be ( 1 )
end
it " leaves the value field nil " do
expect ( parser . log . income1 ) . to be_nil
expect ( parser . log . income2 ) . to be_nil
expect ( parser . log . savings ) . to be_nil
end
end
context " when set to a number " do
let ( :attributes ) do
{
bulk_upload : ,
field_78 : " 30000 " , # income 1
field_80 : " 0 " , # income 2
field_83 : " 12420 " , # savings
}
end
it " sets the not known field as known " do
expect ( parser . log . income1nk ) . to be ( 0 )
expect ( parser . log . income2nk ) . to be ( 0 )
expect ( parser . log . savingsnk ) . to be ( 0 )
end
it " sets the values " do
expect ( parser . log . income1 ) . to be ( 30_000 )
expect ( parser . log . income2 ) . to be ( 0 )
expect ( parser . log . savings ) . to be ( 12_420 )
end
end
end
describe " validations " do
before do
stub_request ( :get , / api \ .postcodes \ .io / )
. to_return ( status : 200 , body : " { \" status \" :200, \" result \" :{ \" admin_district \" : \" Manchester \" , \" codes \" :{ \" admin_district \" : \" E08000003 \" }}} " , headers : { } )
parser . valid?
end
describe " # valid? " do
context " when the row is blank " do
let ( :attributes ) { { bulk_upload : } }
it " returns true " do
expect ( parser ) . to be_valid
end
end
context " when calling the method multiple times " do
let ( :attributes ) { { bulk_upload : , field_7 : 2 } }
it " does not add keep adding errors to the pile " do
expect { parser . valid? } . not_to change ( parser . errors , :count )
end
end
context " when valid row " do
let ( :attributes ) { valid_attributes }
it " returns true " do
expect ( parser ) . to be_valid
end
it " instantiates a log with everything completed " , aggregate_failures : true do
questions = parser . send ( :questions ) . reject do | q |
parser . send ( :log ) . optional_fields . include? ( q . id ) || q . completed? ( parser . send ( :log ) )
end
expect ( questions . map ( & :id ) . size ) . to eq ( 0 )
expect ( questions . map ( & :id ) ) . to eql ( [ ] )
end
end
end
context " when setup section not complete and type is not given " do
let ( :attributes ) do
{
bulk_upload : ,
field_6 : " test id " ,
}
end
it " has errors on correct setup fields " do
errors = parser . errors . select { | e | e . options [ :category ] == :setup } . map ( & :attribute ) . sort
expect ( errors ) . to eql ( % i [ field_1 field_29 field_3 field_4 field_5 field_7 ] )
end
end
context " when setup section not complete and type is shared ownership " do
let ( :attributes ) do
{
bulk_upload : ,
field_6 : " test id " ,
field_7 : " 1 " ,
}
end
it " has errors on correct setup fields " do
errors = parser . errors . select { | e | e . options [ :category ] == :setup } . map ( & :attribute ) . sort
expect ( errors ) . to eql ( % i [ field_1 field_14 field_29 field_3 field_4 field_5 field_8 ] )
end
end
context " when setup section not complete it's shared ownership joint purchase " do
let ( :attributes ) do
{
bulk_upload : ,
field_6 : " test id " ,
field_7 : " 1 " ,
field_8 : " 2 " ,
field_14 : " 1 " ,
}
end
it " has errors on correct setup fields " do
errors = parser . errors . select { | e | e . options [ :category ] == :setup } . map ( & :attribute ) . sort
expect ( errors ) . to eql ( % i [ field_1 field_15 field_29 field_3 field_4 field_5 ] )
end
end
context " when setup section not complete and type is discounted ownership " do
let ( :attributes ) do
{
bulk_upload : ,
field_6 : " test id " ,
field_7 : " 2 " ,
}
end
it " has errors on correct setup fields " do
errors = parser . errors . select { | e | e . options [ :category ] == :setup } . map ( & :attribute ) . sort
expect ( errors ) . to eql ( % i [ field_1 field_14 field_29 field_3 field_4 field_5 field_9 ] )
end
end
context " when setup section not complete it's discounted ownership joint purchase " do
let ( :attributes ) do
{
bulk_upload : ,
field_28 : " test id " ,
field_30 : " 2 " ,
field_47 : " 8 " ,
field_39 : " 1 " ,
}
end
it " has errors on correct setup fields " do
errors = parser . errors . select { | e | e . options [ :category ] == :setup } . map ( & :attribute ) . sort
expect ( errors ) . to eql ( % i [ field_1 field_29 field_3 field_4 field_5 field_7 ] )
end
end
context " when setup section not complete and type is outright sale " do
let ( :attributes ) do
{
bulk_upload : ,
field_6 : " test id " ,
field_7 : " 3 " ,
}
end
it " has errors on correct setup fields " do
errors = parser . errors . select { | e | e . options [ :category ] == :setup } . map ( & :attribute ) . sort
expect ( errors ) . to eql ( % i [ field_1 field_10 field_12 field_13 field_29 field_3 field_4 field_5 ] )
end
end
context " when setup section not complete outright sale buyer is not company " do
let ( :attributes ) do
{
bulk_upload : ,
field_6 : " test id " ,
field_7 : " 3 " ,
field_10 : " 12 " ,
field_12 : " 2 " ,
}
end
it " has errors on correct setup fields " do
errors = parser . errors . select { | e | e . options [ :category ] == :setup } . map ( & :attribute ) . sort
expect ( errors ) . to eql ( % i [ field_1 field_11 field_13 field_14 field_29 field_3 field_4 field_5 ] )
end
end
describe " # field_1 " do # owning org
context " when no data given " do
let ( :attributes ) { setup_section_params . merge ( field_1 : nil ) }
it " is not permitted as setup error " do
expect ( parser . errors . where ( :field_1 , category : :setup ) . map ( & :message ) ) . to eql ( [ I18n . t ( " validations.sales.2023.bulk_upload.not_answered " , question : " owning organisation. " ) ] )
end
it " blocks log creation " do
expect ( parser ) . to be_block_log_creation
end
end
context " when cannot find owning org " do
let ( :attributes ) { { bulk_upload : , field_1 : " donotexist " } }
it " is not permitted as a setup error " do
expect ( parser . errors . where ( :field_1 , category : :setup ) . map ( & :message ) ) . to eql ( [ I18n . t ( " validations.sales.2023.bulk_upload.not_answered " , question : " owning organisation. " ) ] )
end
it " blocks log creation " do
expect ( parser ) . to be_block_log_creation
end
end
context " when not affiliated with owning org " do
let ( :unaffiliated_org ) { create ( :organisation , :with_old_visible_id ) }
let ( :attributes ) { { bulk_upload : , field_1 : unaffiliated_org . old_visible_id } }
it " is not permitted as setup error " do
expect ( parser . errors . where ( :field_1 , category : :setup ) . map ( & :message ) ) . to eql ( [ I18n . t ( " validations.sales.2023.bulk_upload.owning_organisation.not_permitted " ) ] )
end
it " blocks log creation " do
expect ( parser ) . to be_block_log_creation
end
end
context " when user's org has absorbed owning organisation with stock owners " do
let ( :merged_org ) { create ( :organisation , :with_old_visible_id , holds_own_stock : true ) }
let ( :merged_org_stock_owner ) { create ( :organisation , :with_old_visible_id , holds_own_stock : true ) }
let ( :attributes ) { { bulk_upload : , field_1 : merged_org_stock_owner . old_visible_id } }
before do
create ( :organisation_relationship , parent_organisation : merged_org_stock_owner , child_organisation : merged_org )
merged_org . update! ( absorbing_organisation : user . organisation , merge_date : Time . zone . today )
merged_org . reload
user . organisation . reload
end
it " is permitted " do
parser . valid?
expect ( parser . errors . where ( :field_1 ) ) . not_to be_present
end
end
context " when user's org has absorbed owning organisation " do
let ( :merged_org ) { create ( :organisation , :with_old_visible_id , holds_own_stock : true ) }
let ( :attributes ) { { bulk_upload : , field_1 : merged_org . old_visible_id , field_2 : user . email } }
before do
merged_org . update! ( absorbing_organisation : user . organisation , merge_date : Time . zone . today )
merged_org . reload
user . organisation . reload
user . reload
end
it " is permitted " do
parser = described_class . new ( attributes )
parser . valid?
expect ( parser . errors . where ( :field_1 ) ) . not_to be_present
expect ( parser . errors . where ( :field_2 ) ) . not_to be_present
end
end
context " when user's org has absorbed owning organisation before the startdate " do
let ( :merged_org ) { create ( :organisation , :with_old_visible_id , holds_own_stock : true ) }
let ( :attributes ) { setup_section_params . merge ( { field_1 : merged_org . old_visible_id , field_2 : user . email } ) }
before do
merged_org . update! ( absorbing_organisation : user . organisation , merge_date : Time . zone . today - 3 . years )
merged_org . reload
user . organisation . reload
user . reload
end
it " is not permitted " do
parser = described_class . new ( attributes )
parser . valid?
expect ( parser . errors [ :field_1 ] ) . to include ( / The owning organisation must be active on the sale completion date / )
expect ( parser . errors [ :field_3 ] ) . to include ( / Enter a date when the owning organisation was active / )
expect ( parser . errors [ :field_4 ] ) . to include ( / Enter a date when the owning organisation was active / )
expect ( parser . errors [ :field_5 ] ) . to include ( / Enter a date when the owning organisation was active / )
end
end
end
describe " # field_2 " do # username for assigned_to
context " when blank " do
let ( :attributes ) { setup_section_params . merge ( bulk_upload : , field_2 : nil ) }
it " is permitted " do
expect ( parser . errors [ :field_2 ] ) . to be_blank
end
it " sets assigned to to bulk upload user " do
expect ( parser . log . assigned_to ) . to eq ( bulk_upload . user )
end
it " sets created by to bulk upload user " do
expect ( parser . log . created_by ) . to eq ( bulk_upload . user )
end
end
context " when user could not be found " do
let ( :attributes ) { { bulk_upload : , field_2 : " idonotexist@example.com " } }
it " is not permitted " do
expect ( parser . errors [ :field_2 ] ) . to be_present
end
end
context " when an unaffiliated user " do
let ( :other_user ) { create ( :user ) }
let ( :attributes ) { { bulk_upload : , field_1 : owning_org . old_visible_id , field_2 : other_user . email } }
it " is not permitted as a setup error " do
expect ( parser . errors . where ( :field_2 , category : :setup ) ) . to be_present
end
it " blocks log creation " do
expect ( parser ) . to be_block_log_creation
end
end
context " when a user part of owning org " do
let ( :other_user ) { create ( :user , organisation : owning_org ) }
let ( :attributes ) { { bulk_upload : , field_1 : owning_org . old_visible_id , field_2 : other_user . email } }
it " is permitted " do
expect ( parser . errors [ :field_2 ] ) . to be_blank
end
it " sets assigned to to the user " do
expect ( parser . log . assigned_to ) . to eq ( other_user )
end
it " sets created by to bulk upload user " do
expect ( parser . log . created_by ) . to eq ( bulk_upload . user )
end
end
context " when email matches other than casing " do
let ( :other_user ) { create ( :user , organisation : owning_org ) }
let ( :attributes ) { { bulk_upload : , field_1 : owning_org . old_visible_id , field_2 : other_user . email . upcase! } }
it " is permitted " do
expect ( parser . errors [ :field_2 ] ) . to be_blank
end
end
end
describe " fields 3, 4, 5 => saledate " do
context " when all of these fields are blank " do
let ( :attributes ) { setup_section_params . merge ( { field_3 : nil , field_4 : nil , field_5 : nil } ) }
it " returns them as setup errors " do
expect ( parser . errors . where ( :field_3 , category : :setup ) ) . to be_present
expect ( parser . errors . where ( :field_4 , category : :setup ) ) . to be_present
expect ( parser . errors . where ( :field_5 , category : :setup ) ) . to be_present
end
end
context " when one of these fields is blank " do
let ( :attributes ) { setup_section_params . merge ( { field_3 : " 1 " , field_4 : " 1 " , field_5 : nil } ) }
it " returns an error only on blank field as setup error " do
expect ( parser . errors [ :field_3 ] ) . to be_blank
expect ( parser . errors [ :field_4 ] ) . to be_blank
expect ( parser . errors . where ( :field_5 , category : :setup ) ) . to be_present
end
end
context " when field 5 is 4 digits instead of 2 " do
let ( :attributes ) { setup_section_params . merge ( { bulk_upload : , field_5 : " 2023 " , field_4 : " 4 " , field_3 : " 3 " } ) }
it " correctly sets the date " do
parser . valid?
expect ( parser . errors . where ( :field_5 , category : :setup ) ) . to be_empty
expect ( parser . log . saledate ) . to eq ( Time . zone . local ( 2023 , 4 , 3 ) )
end
end
context " when field 5 is not 2 or 4 digits " do
let ( :attributes ) { setup_section_params . merge ( { bulk_upload : , field_5 : " 202 " } ) }
it " returns a setup error " do
parser . valid?
expect ( parser . errors . where ( :field_5 , category : :setup ) . map ( & :message ) ) . to include ( I18n . t ( " validations.sales.2023.bulk_upload.saledate.year_not_two_or_four_digits " ) )
end
end
context " when invalid date given " do
let ( :attributes ) { setup_section_params . merge ( { field_3 : " a " , field_4 : " 12 " , field_5 : " 2022 " } ) }
it " does not raise an error " do
expect { parser . valid? } . not_to raise_error
end
end
context " when inside of collection year " do
around do | example |
Timecop . freeze ( Date . new ( 2023 , 10 , 1 ) ) do
example . run
end
end
let ( :attributes ) { setup_section_params . merge ( { field_3 : " 1 " , field_4 : " 10 " , field_5 : " 23 " } ) }
let ( :bulk_upload ) { create ( :bulk_upload , :sales , user : , year : 2023 ) }
it " does not return errors " do
expect ( parser . errors [ :field_3 ] ) . not_to be_present
expect ( parser . errors [ :field_4 ] ) . not_to be_present
expect ( parser . errors [ :field_5 ] ) . not_to be_present
end
end
context " when outside of collection year " do
around do | example |
Timecop . freeze ( Date . new ( 2022 , 4 , 2 ) ) do
example . run
end
end
let ( :attributes ) { setup_section_params . merge ( { field_3 : " 1 " , field_4 : " 1 " , field_5 : " 22 " } ) }
let ( :bulk_upload ) { create ( :bulk_upload , :sales , user : , year : 2022 ) }
it " returns setup errors " do
expect ( parser . errors . where ( :field_3 , category : :setup ) ) . to be_present
expect ( parser . errors . where ( :field_4 , category : :setup ) ) . to be_present
expect ( parser . errors . where ( :field_5 , category : :setup ) ) . to be_present
end
end
end
context " when the log already exists in the db " do
let ( :attributes ) { valid_attributes }
before do
parser . log . save!
parser . instance_variable_set ( :@valid , nil )
end
it " is not a valid row " do
expect ( parser ) . not_to be_valid
end
it " adds an error to all (and only) the fields used to determine duplicates " do
parser . valid?
error_message = I18n . t ( " validations.sales.2023.bulk_upload.duplicate " )
[
:field_1 , # Owning org
:field_3 , # Sale completion date
:field_4 , # Sale completion date
:field_5 , # Sale completion date
:field_24 , # Postcode
:field_25 , # Postcode
:field_30 , # Buyer 1 age
:field_31 , # Buyer 1 gender
:field_35 , # Buyer 1 working situation
:field_6 , # Purchaser code
] . each do | field |
expect ( parser . errors [ field ] ) . to include ( error_message )
end
end
end
context " when a hidden log already exists in db " do
before do
parser . log . status = " pending "
parser . log . skip_update_status = true
parser . log . save!
end
it " is a valid row " do
expect ( parser ) . to be_valid
end
it " does not add duplicate errors " do
parser . valid?
[
:field_1 , # Owning org
:field_3 , # Sale completion date
:field_4 , # Sale completion date
:field_5 , # Sale completion date
:field_24 , # Postcode
:field_25 , # Postcode
:field_30 , # Buyer 1 age
:field_31 , # Buyer 1 gender
:field_35 , # Buyer 1 working situation
:field_6 , # Purchaser code
] . each do | field |
expect ( parser . errors [ field ] ) . to be_blank
end
end
end
describe " # field_7 " do # ownership scheme
context " when an invalid option " do
let ( :attributes ) { setup_section_params . merge ( { field_7 : " 100 " } ) }
it " returns setup error " do
expect ( parser . errors . where ( :field_7 , category : :setup ) ) . to be_present
end
end
end
describe " # field_8 " do # type for shared ownership sale
context " when an invalid option " do
let ( :attributes ) { setup_section_params . merge ( { field_8 : " 100 " } ) }
it " returns setup error " do
expect ( parser . errors . where ( :field_8 , category : :setup ) ) . to be_present
end
end
end
describe " # field_9 " do # type for discounted sale
context " when an invalid option " do
let ( :attributes ) { setup_section_params . merge ( { field_9 : " 100 " } ) }
it " returns setup error " do
expect ( parser . errors . where ( :field_9 , category : :setup ) ) . to be_present
end
end
end
describe " # field_10 " do # type for outright sale
context " when an invalid option " do
let ( :attributes ) { setup_section_params . merge ( { field_10 : " 100 " } ) }
it " returns setup error " do
expect ( parser . errors . where ( :field_10 , category : :setup ) ) . to be_present
end
end
end
describe " # field_11 " do # type of other outright sale
context " when an invalid option " do
let ( :attributes ) { setup_section_params . merge ( { field_10 : " 12 " , field_11 : nil } ) }
it " returns setup error " do
expect ( parser . errors . where ( :field_11 , category : :setup ) ) . to be_present
end
end
end
describe " # field_12 " do # buyer a company?
context " when an invalid option " do
let ( :attributes ) { setup_section_params . merge ( { field_7 : " 3 " , field_12 : " 100 " } ) }
it " returns setup error " do
expect ( parser . errors . where ( :field_12 ) . map ( & :message ) ) . to include ( " Enter a valid value for is the buyer a company? " )
expect ( parser . errors . where ( :field_12 , category : :setup ) ) . to be_present
end
end
end
describe " # field_13 " do # will buyers live in the property?
context " when an invalid option " do
let ( :attributes ) { setup_section_params . merge ( { field_7 : " 3 " , field_13 : " 100 " } ) }
it " returns setup error " do
expect ( parser . errors . where ( :field_13 ) . map ( & :message ) ) . to eql ( [ " Enter a valid value for will the buyers live in the property? " ] )
expect ( parser . errors . where ( :field_13 , category : :setup ) ) . to be_present
end
end
end
describe " # field_14 " do # joint purchase
context " when an invalid option " do
let ( :attributes ) { setup_section_params . merge ( { field_14 : " 100 " } ) }
it " returns a setup error " do
expect ( parser . errors . where ( :field_14 , category : :setup ) ) . to be_present
end
end
end
describe " # field_15 " do # more than 2 joint buyers?
context " when invalid option and must be answered " do
let ( :attributes ) { setup_section_params . merge ( { field_14 : " 1 " , field_15 : " 100 " } ) }
it " returns a setup error " do
expect ( parser . errors . where ( :field_15 , category : :setup ) ) . to be_present
end
end
end
describe " # field_19 " do # UPRN
context " when UPRN known and lookup found " do
let ( :attributes ) { setup_section_params . merge ( { field_19 : " 12 " } ) }
it " is valid " do
expect ( parser . errors [ :field_19 ] ) . to be_blank
end
it " sets UPRN and UPRN known " do
expect ( parser . log . uprn ) . to eq ( " 12 " )
expect ( parser . log . uprn_known ) . to eq ( 1 )
expect ( parser . log . uprn_confirmed ) . to eq ( 1 )
end
end
context " when UPRN not known but address known " do
let ( :attributes ) { setup_section_params . merge ( { field_19 : nil , field_20 : " some street " , field_22 : " some town " , field_24 : " EC1N " , field_25 : " 2TD " } ) }
it " is valid " do
expect ( parser . errors [ :field_19 ] ) . to be_blank
end
end
context " when neither UPRN or address known " do
let ( :attributes ) { setup_section_params . merge ( { field_19 : nil , field_20 : nil , field_22 : nil , field_24 : nil , field_25 : nil } ) }
it " is not valid " do
expect ( parser . errors [ :field_19 ] ) . to be_present
end
end
context " when UPRN entered but no lookup found " do
let ( :attributes ) { setup_section_params . merge ( { field_19 : " 1234567890123 " } ) }
it " is not valid " do
expect ( parser . errors [ :field_19 ] ) . to be_present
end
end
end
[
{ field : :field_20 , name : " address line 1 " } ,
{ field : :field_22 , name : " town or city " } ,
] . each do | data |
describe " # #{ data [ :field ] } ( #{ data [ :name ] } ) " do
context " when UPRN present " do
context " when UPRN valid " do
let ( :attributes ) { setup_section_params . merge ( { field_19 : " 12 " , data [ :field ] = > nil } ) }
it " can be blank " do
expect ( parser . errors [ data [ :field ] ] ) . to be_blank
end
end
context " when UPRN invalid " do
let ( :attributes ) { setup_section_params . merge ( { field_19 : " 1234567890123 " , data [ :field ] = > nil } ) }
it " cannot be blank " do
expect ( parser . errors [ data [ :field ] ] ) . to be_present
end
end
end
context " when UPRN not present " do
let ( :attributes ) { setup_section_params . merge ( { field_19 : nil , data [ :field ] = > nil } ) }
it " cannot be blank " do
expect ( parser . errors [ data [ :field ] ] ) . to be_present
end
end
end
end
describe " # field_29 " do # data protection
let ( :attributes ) { setup_section_params . merge ( { field_29 : nil } ) }
context " when not answered " do
it " returns a setup error " do
expect ( parser . errors . where ( :field_29 , category : :setup ) ) . to be_present
end
end
end
[
%w[ age1_known details_known_1 age1 field_30 field_37 field_39 ] ,
%w[ age2_known details_known_2 age2 field_38 field_37 field_39 ] ,
%w[ age3_known details_known_3 age3 field_47 field_46 field_48 ] ,
%w[ age4_known details_known_4 age4 field_51 field_50 field_52 ] ,
%w[ age5_known details_known_5 age5 field_55 field_54 field_56 ] ,
%w[ age6_known details_known_6 age6 field_59 field_58 field_60 ] ,
] . each do | known , details_known , age , field , relationship , gender |
describe " # #{ known } and # #{ age } " do
context " when #{ field } is blank " do
context " and person details are blank " do
let ( :attributes ) { setup_section_params . merge ( { field . to_s = > nil , relationship . to_sym = > nil , gender . to_sym = > nil , field_29 : " 1 " , field_45 : " 5 " } ) }
it " does not set # #{ known } " do
unless known == " age1_known "
expect ( parser . log . public_send ( known ) ) . to be_nil
end
end
it " sets # #{ details_known } to no " do
unless details_known == " details_known_1 "
expect ( parser . log . public_send ( details_known ) ) . to eq ( 2 )
end
end
it " sets # #{ age } to nil " do
expect ( parser . log . public_send ( age ) ) . to be_nil
end
end
context " and person details are given " do
let ( :attributes ) { setup_section_params . merge ( { field . to_sym = > nil , relationship . to_sym = > " C " , gender . to_sym = > " X " , field_29 : " 1 " , field_45 : " 5 " } ) }
it " does not set # #{ age } " do
parser . valid?
expect ( parser . errors [ field . to_sym ] ) . to include ( / #{ I18n . t ( " validations.sales.2023.bulk_upload.not_answered " , question : " " ) } / )
end
end
end
context " when #{ field } is R " do
let ( :attributes ) { setup_section_params . merge ( { field . to_s = > " R " , field_28 : " 1 " , field_45 : " 5 " , field_29 : " 1 " } ) }
it " sets # #{ known } 1 " do
expect ( parser . log . public_send ( known ) ) . to be ( 1 )
end
it " sets # #{ age } to nil " do
expect ( parser . log . public_send ( age ) ) . to be_nil
end
end
context " when #{ field } is a number " do
let ( :attributes ) { setup_section_params . merge ( { field . to_s = > " 50 " , field_28 : " 1 " , field_45 : " 5 " , field_29 : " 1 " } ) }
it " sets # #{ known } to 0 " do
expect ( parser . log . public_send ( known ) ) . to be ( 0 )
end
it " sets # #{ age } to given age " do
expect ( parser . log . public_send ( age ) ) . to be ( 50 )
end
end
context " when #{ field } is a non-sensical value " do
let ( :attributes ) { setup_section_params . merge ( { field . to_s = > " A " , field_28 : " 1 " , field_45 : " 5 " , field_29 : " 1 " } ) }
it " sets # #{ known } to 0 " do
expect ( parser . log . public_send ( known ) ) . to be ( 0 )
end
it " sets # #{ age } to nil " do
expect ( parser . log . public_send ( age ) ) . to be_nil
end
end
end
end
describe " field_35 " do # ecstat1
context " when buyer 1 is marked as a child " do
let ( :attributes ) { valid_attributes . merge ( { field_35 : " 9 " } ) }
it " a custom validation is applied " do
validation_message = I18n . t ( " validations.sales.2023.bulk_upload.ecstat1.child_under_16 " )
expect ( parser . errors [ :field_35 ] ) . to include validation_message
end
end
end
describe " # field_36 " do # will buyer1 live in property?
context " when not a possible value " do
let ( :attributes ) { valid_attributes . merge ( { field_36 : " 3 " } ) }
it " is not valid " do
expect ( parser . errors ) . to include ( :field_36 )
end
end
end
describe " # field_67 - 70 " do # buyers organisations
let ( :empty_organisation_params ) { setup_section_params . merge ( field_67 : nil , field_68 : nil , field_69 : nil , field_70 : nil ) }
context " when all empty " do
let ( :attributes ) { empty_organisation_params }
it " sets pregblank field " do
expect ( parser . log . pregblank ) . to be ( 1 )
end
end
% i [ field_67 field_68 field_69 field_70 ] . each do | field_number |
context " when #{ field_number } present " do
let ( :attributes ) { empty_organisation_params . merge ( { field_number = > 1 } ) }
it " does not set pregblank field " do
attributes [ :field_number ] = 1
expect ( parser . log . pregblank ) . to be ( 0 )
end
end
end
end
describe " soft validations " do
context " when soft validation is triggered " do
let ( :attributes ) { valid_attributes . merge ( { field_30 : 22 , field_35 : 5 } ) }
it " adds an error to the relevant fields " do
expect ( parser . errors . where ( :field_30 , category : :soft_validation ) ) . to be_present
expect ( parser . errors . where ( :field_35 , category : :soft_validation ) ) . to be_present
end
it " populates with correct error message " do
expect ( parser . errors . where ( :field_30 , category : :soft_validation ) . first . message ) . to eql ( " You told us this person is aged 22 years and retired. The minimum expected retirement age in England is 66. " )
expect ( parser . errors . where ( :field_30 , category : :soft_validation ) . first . message ) . to eql ( " You told us this person is aged 22 years and retired. The minimum expected retirement age in England is 66. " )
end
end
context " when a soft validation is triggered that relates both to fields that are and are not routed to " do
let ( :attributes ) { valid_attributes . merge ( { field_103 : " 300000 " } ) }
it " adds errors to fields that are routed to " do
expect ( parser . errors . where ( :field_103 , category : :soft_validation ) ) . to be_present
end
it " does not add errors to fields that are not routed to " do
expect ( parser . errors . where ( :field_112 , category : :soft_validation ) ) . not_to be_present
end
end
end
describe " # field_105 " do
context " when invalid value " do
let ( :attributes ) { setup_section_params . merge ( field_105 : " 4 " ) }
it " returns correct errors " do
expect ( parser . errors [ :field_105 ] ) . to include ( I18n . t ( " validations.sales.2023.bulk_upload.invalid_option " , question : " was a mortgage used for the purchase of this property? - Shared ownership. " ) )
parser . log . blank_invalid_non_setup_fields!
parser . log . save!
expect ( parser . log . mortgageused ) . to be_nil
end
end
context " when value is 3 and stairowned is not 100 " do
let ( :attributes ) { setup_section_params . merge ( field_105 : " 3 " , field_87 : " 1 " , field_88 : " 50 " , field_89 : " 99 " , field_111 : nil ) }
it " returns correct errors " do
expect ( parser . errors [ :field_105 ] ) . to include ( " The percentage owned has to be 100% if the mortgage used is 'Don’t know' " )
parser . log . blank_invalid_non_setup_fields!
parser . log . save!
expect ( parser . log . mortgageused ) . to be_nil
end
end
context " when value is 3 and stairowned is not answered " do
let ( :attributes ) { setup_section_params . merge ( field_105 : " 3 " , field_87 : " 1 " , field_88 : " 50 " , field_89 : nil , field_111 : nil ) }
it " does not add errors " do
expect ( parser . errors [ :field_105 ] ) . not_to include ( " Enter a valid value for Was a mortgage used for the purchase of this property? - Shared ownership " )
expect ( parser . errors [ :field_105 ] ) . not_to include ( " The percentage owned has to be 100% if the mortgage used is 'Don’t know' " )
end
end
context " when value is 3 and stairowned is 100 " do
let ( :attributes ) { setup_section_params . merge ( field_105 : " 3 " , field_87 : " 1 " , field_88 : " 50 " , field_89 : " 100 " , field_111 : nil ) }
it " does not add errors " do
expect ( parser . errors [ :field_105 ] ) . to be_empty
expect ( parser . errors [ :field_119 ] ) . to be_empty
expect ( parser . errors [ :field_128 ] ) . to be_empty
parser . log . blank_invalid_non_setup_fields!
parser . log . save!
expect ( parser . log . mortgageused ) . to eq ( 3 )
end
end
context " when it is not a staircasing transaction " do
context " when value is 3 and stairowned is not answered " do
let ( :attributes ) { setup_section_params . merge ( field_105 : " 3 " , field_87 : " 2 " , field_88 : " 50 " , field_89 : nil , field_111 : nil ) }
it " returns correct errors " do
expect ( parser . errors [ :field_105 ] ) . to include ( " Enter a valid value for was a mortgage used for the purchase of this property? " )
parser . log . blank_invalid_non_setup_fields!
parser . log . save!
expect ( parser . log . mortgageused ) . to be_nil
end
end
context " when value is 3 and stairowned is 100 " do
let ( :attributes ) { setup_section_params . merge ( field_105 : " 3 " , field_87 : " 2 " , field_88 : " 50 " , field_89 : " 100 " , field_111 : nil ) }
it " returns correct errors " do
expect ( parser . errors [ :field_105 ] ) . to include ( " Enter a valid value for was a mortgage used for the purchase of this property? " )
parser . log . blank_invalid_non_setup_fields!
parser . log . save!
expect ( parser . log . mortgageused ) . to be_nil
end
end
end
end
describe " # field_119 " do
let ( :attributes ) { valid_attributes . merge ( { field_7 : " 2 " , field_9 : " 8 " , field_119 : " 3 " } ) }
it " does not allow 3 (don't know) as an option for discounted ownership " do
expect ( parser . errors [ :field_119 ] ) . to include ( " Enter a valid value for was a mortgage used for the purchase of this property? " )
parser . log . blank_invalid_non_setup_fields!
parser . log . save!
expect ( parser . log . mortgageused ) . to be_nil
end
context " when validate_discounted_ownership_value is triggered " do
let ( :attributes ) { setup_section_params . merge ( field_116 : 100 , field_125 : 100 , field_7 : 2 , field_9 : 9 , field_119 : 2 , field_118 : 10 ) }
it " only adds errors to the discounted ownership field " do
expect ( parser . errors [ :field_105 ] ) . to be_empty
expect ( parser . errors [ :field_119 ] ) . to include ( " Mortgage, deposit, and grant total must equal £90.00. Your given mortgage, deposit and grant total is £100.00. " )
expect ( parser . errors [ :field_128 ] ) . to be_empty
end
end
end
describe " # field_128 " do
let ( :attributes ) { valid_attributes . merge ( { field_7 : " 3 " , field_10 : " 10 " , field_128 : " 3 " , field_12 : " 2 " } ) }
it " does not allow 3 (don't know) as an option for outright sale " do
expect ( parser . errors [ :field_128 ] ) . to include ( " Enter a valid value for was a mortgage used for the purchase of this property? " )
parser . log . blank_invalid_non_setup_fields!
parser . log . save!
expect ( parser . log . mortgageused ) . to be_nil
end
end
end
describe " # log " do
describe " # noint " do
context " when field is set to 1 " do
let ( :attributes ) { valid_attributes . merge ( { field_28 : 1 } ) }
it " is correctly set " do
expect ( parser . log . noint ) . to be ( 1 )
end
end
context " when field is set to 2 " do
let ( :attributes ) { valid_attributes . merge ( { field_28 : 2 } ) }
it " is correctly set " do
expect ( parser . log . noint ) . to be ( 2 )
end
end
end
describe " # uprn " do
let ( :attributes ) { setup_section_params . merge ( { field_19 : " 12 " } ) }
it " is correctly set " do
expect ( parser . log . uprn ) . to eql ( " 12 " )
end
end
describe " # uprn_known " do
context " when uprn known " do
let ( :attributes ) { setup_section_params . merge ( { field_19 : " 12 " } ) }
it " is correctly set " do
expect ( parser . log . uprn_known ) . to be ( 1 )
end
end
context " when uprn not known " do
let ( :attributes ) { setup_section_params . merge ( { field_19 : nil } ) }
it " is correctly set " do
expect ( parser . log . uprn_known ) . to be ( 0 )
end
end
end
describe " # address_line1 " do
let ( :attributes ) { setup_section_params . merge ( { field_20 : " some street " } ) }
it " is correctly set " do
expect ( parser . log . address_line1 ) . to eql ( " some street " )
end
end
describe " # address_line2 " do
let ( :attributes ) { setup_section_params . merge ( { field_21 : " some other street " } ) }
it " is correctly set " do
expect ( parser . log . address_line2 ) . to eql ( " some other street " )
end
end
describe " # town_or_city " do
let ( :attributes ) { setup_section_params . merge ( { field_22 : " some town " } ) }
it " is correctly set " do
expect ( parser . log . town_or_city ) . to eql ( " some town " )
end
end
describe " # county " do
let ( :attributes ) { setup_section_params . merge ( { field_23 : " some county " } ) }
it " is correctly set " do
expect ( parser . log . county ) . to eql ( " some county " )
end
end
describe " # ethnic_group2 " do
let ( :attributes ) { setup_section_params . merge ( { field_40 : " 1 " } ) }
it " is correctly set " do
expect ( parser . log . ethnic_group2 ) . to be ( 0 )
end
end
describe " # ethnicbuy2 " do
let ( :attributes ) { setup_section_params . merge ( { field_40 : " 1 " } ) }
it " is correctly set " do
expect ( parser . log . ethnicbuy2 ) . to be ( 1 )
end
end
describe " # nationalbuy2 " do
let ( :attributes ) { setup_section_params . merge ( { field_41 : " 18 " } ) }
it " is correctly set " do
expect ( parser . log . nationalbuy2 ) . to be ( 18 )
end
end
describe " # buy2living " do
let ( :attributes ) { setup_section_params . merge ( { field_71 : " 1 " } ) }
it " is correctly set " do
expect ( parser . log . buy2living ) . to be ( 1 )
end
end
describe " # prevtenbuy2 " do
let ( :attributes ) { setup_section_params . merge ( { field_72 : " R " } ) }
it " is correctly set " do
expect ( parser . log . prevtenbuy2 ) . to be ( 0 )
end
end
describe " # hhregres " do
let ( :attributes ) { setup_section_params . merge ( { field_73 : " 1 " } ) }
it " is correctly set " do
expect ( parser . log . hhregres ) . to be ( 1 )
end
end
describe " # hhregresstill " do
let ( :attributes ) { setup_section_params . merge ( { field_74 : " 4 " } ) }
it " is correctly set " do
expect ( parser . log . hhregresstill ) . to be ( 4 )
end
end
describe " # prevshared " do
let ( :attributes ) { setup_section_params . merge ( { field_85 : " 3 " } ) }
it " is correctly set " do
expect ( parser . log . prevshared ) . to be ( 3 )
end
end
describe " # staircasesale " do
let ( :attributes ) { setup_section_params . merge ( { field_90 : " 1 " } ) }
it " is correctly set " do
expect ( parser . log . staircasesale ) . to be ( 1 )
end
end
describe " # soctenant " do
let ( :attributes ) { setup_section_params . merge ( { field_99 : " 1 " } ) }
it " is correctly set " do
expect ( parser . log . soctenant ) . to be ( 1 )
end
end
describe " with living before purchase years for shared ownership more than 0 " do
let ( :attributes ) { setup_section_params . merge ( { field_7 : " 1 " , field_86 : " 1 " } ) }
it " is sets living before purchase asked to yes and sets the correct living before purchase years " do
expect ( parser . log . proplen_asked ) . to be ( 0 )
expect ( parser . log . proplen ) . to be ( 1 )
end
end
describe " with living before purchase years for discounted ownership more than 0 " do
let ( :attributes ) { setup_section_params . merge ( { field_7 : " 2 " , field_115 : " 1 " } ) }
it " is sets living before purchase asked to yes and sets the correct living before purchase years " do
expect ( parser . log . proplen_asked ) . to be ( 0 )
expect ( parser . log . proplen ) . to be ( 1 )
end
end
describe " with living before purchase years for shared ownership set to 0 " do
let ( :attributes ) { setup_section_params . merge ( { field_7 : " 1 " , field_86 : " 0 " } ) }
it " is sets living before purchase asked to no " do
expect ( parser . log . proplen_asked ) . to be ( 1 )
expect ( parser . log . proplen ) . to be_nil
end
end
describe " with living before purchase 0 years for discounted ownership set to 0 " do
let ( :attributes ) { setup_section_params . merge ( { field_7 : " 2 " , field_115 : " 0 " } ) }
it " is sets living before purchase asked to no " do
expect ( parser . log . proplen_asked ) . to be ( 1 )
expect ( parser . log . proplen ) . to be_nil
end
end
context " when mscharge is given, but is set to 0 for shared ownership " do
let ( :attributes ) { valid_attributes . merge ( field_114 : " 0 " ) }
it " does not override variables correctly " do
log = parser . log
expect ( log [ " has_mscharge " ] ) . to eq ( 0 ) # no
expect ( log [ " mscharge " ] ) . to be_nil
end
end
context " when mscharge is given, but is set to 0 for discounted ownership " do
let ( :attributes ) { valid_attributes . merge ( field_7 : " 2 " , field_126 : " 0 " ) }
it " does not override variables correctly " do
log = parser . log
expect ( log [ " has_mscharge " ] ) . to eq ( 0 ) # no
expect ( log [ " mscharge " ] ) . to be_nil
end
end
context " when mscharge is given, but is set to 0 for outright sale " do
let ( :attributes ) { valid_attributes . merge ( field_7 : " 3 " , field_135 : " 0 " ) }
it " does not override variables correctly " do
log = parser . log
expect ( log [ " has_mscharge " ] ) . to eq ( 0 ) # no
expect ( log [ " mscharge " ] ) . to be_nil
end
end
describe " shared ownership sale type " do
context " when 32 is selected for shared ownership type " do
let ( :attributes ) { valid_attributes . merge ( field_8 : " 32 " ) }
it " sets the value correctly " do
log = parser . log
expect ( log . type ) . to eq ( 32 )
end
end
end
describe " # managing_organisation_id " do
let ( :attributes ) { setup_section_params }
context " when user is part of the owning organisation " do
it " sets managing organisation to the users organisation " do
parser . valid?
expect ( parser . log . owning_organisation_id ) . to be ( owning_org . id )
expect ( parser . log . managing_organisation_id ) . to be ( owning_org . id )
end
end
context " when user is part of an organisation affiliated with owning org " do
let ( :managing_agent ) { create ( :organisation ) }
let ( :user ) { create ( :user , organisation : managing_agent ) }
let ( :attributes ) { setup_section_params }
before do
create ( :organisation_relationship , child_organisation : managing_agent , parent_organisation : owning_org )
end
it " is not permitted as setup error " do
parser . valid?
expect ( parser . log . owning_organisation_id ) . to be ( owning_org . id )
expect ( parser . log . managing_organisation_id ) . to be ( managing_agent . id )
end
end
context " when user is part of an organisation not affiliated with owning org " do
let ( :unaffiliated_org ) { create ( :organisation ) }
let ( :user ) { create ( :user , organisation : unaffiliated_org ) }
let ( :attributes ) { setup_section_params }
it " is not permitted as setup error " do
parser . valid?
setup_errors = parser . errors . select { | e | e . options [ :category ] == :setup }
expect ( setup_errors . find { | e | e . attribute == :field_2 } . message ) . to eql ( I18n . t ( " validations.sales.2023.bulk_upload.assigned_to.managing_organisation_not_related " ) )
end
it " blocks log creation " do
parser . valid?
expect ( parser ) . to be_block_log_creation
end
end
end
end
describe " # owning_organisation_id " do
let ( :attributes ) { setup_section_params }
context " when owning organisation does not own stock " do
let ( :owning_org ) { create ( :organisation , :with_old_visible_id , holds_own_stock : false ) }
let ( :attributes ) { setup_section_params }
it " is not permitted as setup error " do
parser . valid?
setup_errors = parser . errors . select { | e | e . options [ :category ] == :setup }
expect ( setup_errors . find { | e | e . attribute == :field_1 } . message ) . to eql ( I18n . t ( " validations.sales.2023.bulk_upload.owning_organisation.not_stock_owner " ) )
end
it " blocks log creation " do
parser . valid?
expect ( parser ) . to be_block_log_creation
end
end
end
describe " # spreadsheet_duplicate_hash " do
it " returns a hash " do
expect ( parser . spreadsheet_duplicate_hash ) . to be_a ( Hash )
end
end
describe " # add_duplicate_found_in_spreadsheet_errors " do
it " adds errors " do
expect { parser . add_duplicate_found_in_spreadsheet_errors } . to change ( parser . errors , :size )
end
end
end