Bug 33817: Enforce server-side
[koha.git] / t / db_dependent / Koha / Item.t
1 #!/usr/bin/perl
2
3 # Copyright 2019 Koha Development team
4 #
5 # This file is part of Koha
6 #
7 # Koha is free software; you can redistribute it and/or modify it
8 # under the terms of the GNU General Public License as published by
9 # the Free Software Foundation; either version 3 of the License, or
10 # (at your option) any later version.
11 #
12 # Koha is distributed in the hope that it will be useful, but
13 # WITHOUT ANY WARRANTY; without even the implied warranty of
14 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 # GNU General Public License for more details.
16 #
17 # You should have received a copy of the GNU General Public License
18 # along with Koha; if not, see <http://www.gnu.org/licenses>.
19
20 use Modern::Perl;
21 use utf8;
22
23 use Test::More tests => 29;
24 use Test::Exception;
25 use Test::MockModule;
26
27 use C4::Biblio qw( GetMarcSubfieldStructure );
28 use C4::Circulation qw( AddIssue AddReturn );
29
30 use Koha::Caches;
31 use Koha::Items;
32 use Koha::Database;
33 use Koha::DateUtils qw( dt_from_string );
34 use Koha::Old::Items;
35 use Koha::Recalls;
36 use Koha::AuthorisedValues;
37
38 use List::MoreUtils qw(all);
39
40 use t::lib::TestBuilder;
41 use t::lib::Mocks;
42 use t::lib::Dates;
43
44 my $schema  = Koha::Database->new->schema;
45 my $builder = t::lib::TestBuilder->new;
46
47 subtest 'return_claims relationship' => sub {
48     plan tests => 3;
49
50     $schema->storage->txn_begin;
51
52     my $biblio = $builder->build_sample_biblio();
53     my $item   = $builder->build_sample_item({
54         biblionumber => $biblio->biblionumber,
55     });
56     my $return_claims = $item->return_claims;
57     is( ref($return_claims), 'Koha::Checkouts::ReturnClaims', 'return_claims returns a Koha::Checkouts::ReturnClaims object set' );
58     is($item->return_claims->count, 0, "Empty Koha::Checkouts::ReturnClaims set returned if no return_claims");
59     my $claim1 = $builder->build({ source => 'ReturnClaim', value => { itemnumber => $item->itemnumber }});
60     my $claim2 = $builder->build({ source => 'ReturnClaim', value => { itemnumber => $item->itemnumber }});
61
62     is($item->return_claims()->count,2,"Two ReturnClaims found for item");
63
64     $schema->storage->txn_rollback;
65 };
66
67 subtest 'return_claim accessor' => sub {
68     plan tests => 5;
69
70     $schema->storage->txn_begin;
71
72     my $biblio = $builder->build_sample_biblio();
73     my $item   = $builder->build_sample_item({
74         biblionumber => $biblio->biblionumber,
75     });
76     my $return_claim = $item->return_claim;
77     is( $return_claim, undef, 'return_claim returned undefined if there are no claims for this item' );
78
79     my $claim1 = $builder->build_object(
80         {
81             class => 'Koha::Checkouts::ReturnClaims',
82             value => { itemnumber => $item->itemnumber, resolution => undef, created_on => dt_from_string()->subtract( minutes => 10 ) }
83         }
84     );
85     my $claim2 = $builder->build_object(
86         {
87             class => 'Koha::Checkouts::ReturnClaims',
88             value  => { itemnumber => $item->itemnumber, resolution => undef, created_on => dt_from_string()->subtract( minutes => 5 ) }
89         }
90     );
91
92     $return_claim = $item->return_claim;
93     is( ref($return_claim), 'Koha::Checkouts::ReturnClaim', 'return_claim returned a Koha::Checkouts::ReturnClaim object' );
94     is( $return_claim->id, $claim2->id, 'return_claim returns the most recent unresolved claim');
95
96     $claim2->resolution('test')->store();
97     $return_claim = $item->return_claim;
98     is( $return_claim->id, $claim1->id, 'return_claim returns the only unresolved claim');
99
100     $claim1->resolution('test')->store();
101     $return_claim = $item->return_claim;
102     is( $return_claim, undef, 'return_claim returned undefined if there are no active claims for this item' );
103
104     $schema->storage->txn_rollback;
105 };
106
107 subtest 'tracked_links relationship' => sub {
108     plan tests => 3;
109
110     my $biblio = $builder->build_sample_biblio();
111     my $item   = $builder->build_sample_item({
112         biblionumber => $biblio->biblionumber,
113     });
114     my $tracked_links = $item->tracked_links;
115     is( ref($tracked_links), 'Koha::TrackedLinks', 'tracked_links returns a Koha::TrackedLinks object set' );
116     is($item->tracked_links->count, 0, "Empty Koha::TrackedLinks set returned if no tracked_links");
117     my $link1 = $builder->build({ source => 'Linktracker', value => { itemnumber => $item->itemnumber }});
118     my $link2 = $builder->build({ source => 'Linktracker', value => { itemnumber => $item->itemnumber }});
119
120     is($item->tracked_links()->count,2,"Two tracked links found");
121 };
122
123 subtest 'is_bundle tests' => sub {
124     plan tests => 2;
125
126     $schema->storage->txn_begin;
127
128     my $item   = $builder->build_sample_item();
129
130     my $is_bundle = $item->is_bundle;
131     is($is_bundle, 0, 'is_bundle returns 0 when there are no items attached');
132
133     my $item2 = $builder->build_sample_item();
134     $schema->resultset('ItemBundle')
135       ->create( { host => $item->itemnumber, item => $item2->itemnumber } );
136
137     $is_bundle = $item->is_bundle;
138     is($is_bundle, 1, 'is_bundle returns 1 when there is at least one item attached');
139
140     $schema->storage->txn_rollback;
141 };
142
143 subtest 'in_bundle tests' => sub {
144     plan tests => 2;
145
146     $schema->storage->txn_begin;
147
148     my $item   = $builder->build_sample_item();
149
150     my $in_bundle = $item->in_bundle;
151     is($in_bundle, 0, 'in_bundle returns 0 when the item is not in a bundle');
152
153     my $host_item = $builder->build_sample_item();
154     $schema->resultset('ItemBundle')
155       ->create( { host => $host_item->itemnumber, item => $item->itemnumber } );
156
157     $in_bundle = $item->in_bundle;
158     is($in_bundle, 1, 'in_bundle returns 1 when the item is in a bundle');
159
160     $schema->storage->txn_rollback;
161 };
162
163 subtest 'bundle_items tests' => sub {
164     plan tests => 3;
165
166     $schema->storage->txn_begin;
167
168     my $host_item = $builder->build_sample_item();
169     my $bundle_items = $host_item->bundle_items;
170     is( ref($bundle_items), 'Koha::Items',
171         'bundle_items returns a Koha::Items object set' );
172     is( $bundle_items->count, 0,
173         'bundle_items set is empty when no items are bundled' );
174
175     my $bundle_item1 = $builder->build_sample_item();
176     my $bundle_item2 = $builder->build_sample_item();
177     my $bundle_item3 = $builder->build_sample_item();
178     $schema->resultset('ItemBundle')
179       ->create(
180         { host => $host_item->itemnumber, item => $bundle_item1->itemnumber } );
181     $schema->resultset('ItemBundle')
182       ->create(
183         { host => $host_item->itemnumber, item => $bundle_item2->itemnumber } );
184     $schema->resultset('ItemBundle')
185       ->create(
186         { host => $host_item->itemnumber, item => $bundle_item3->itemnumber } );
187
188     $bundle_items = $host_item->bundle_items;
189     is( $bundle_items->count, 3,
190         'bundle_items returns all the bundled items in the set' );
191
192     $schema->storage->txn_rollback;
193 };
194
195 subtest 'bundle_host tests' => sub {
196     plan tests => 3;
197
198     $schema->storage->txn_begin;
199
200     my $host_item = $builder->build_sample_item();
201     my $bundle_item1 = $builder->build_sample_item();
202     my $bundle_item2 = $builder->build_sample_item();
203     $schema->resultset('ItemBundle')
204       ->create(
205         { host => $host_item->itemnumber, item => $bundle_item2->itemnumber } );
206
207     my $bundle_host = $bundle_item1->bundle_host;
208     is( $bundle_host, undef, 'bundle_host returns undefined when the item it not part of a bundle');
209     $bundle_host = $bundle_item2->bundle_host;
210     is( ref($bundle_host), 'Koha::Item', 'bundle_host returns a Koha::Item object when the item is in a bundle');
211     is( $bundle_host->id, $host_item->id, 'bundle_host returns the host item when called against an item in a bundle');
212
213     $schema->storage->txn_rollback;
214 };
215
216 subtest 'add_to_bundle tests' => sub {
217     plan tests => 12;
218
219     $schema->storage->txn_begin;
220
221     t::lib::Mocks::mock_preference( 'BundleNotLoanValue', 1 );
222
223     my $library = $builder->build_object({ class => 'Koha::Libraries' });
224     t::lib::Mocks::mock_userenv({
225         branchcode => $library->branchcode
226     });
227
228     my $host_item = $builder->build_sample_item();
229     my $bundle_item1 = $builder->build_sample_item();
230     my $bundle_item2 = $builder->build_sample_item();
231     my $bundle_item3 = $builder->build_sample_item();
232
233     my $patron = $builder->build_object( { class => 'Koha::Patrons' } );
234
235     throws_ok { $host_item->add_to_bundle($host_item) }
236     'Koha::Exceptions::Item::Bundle::IsBundle',
237       'Exception thrown if you try to add the item to itself';
238
239     my $reserve_id = C4::Reserves::AddReserve(
240         {
241             branchcode     => $library->branchcode,
242             borrowernumber => $patron->borrowernumber,
243             biblionumber   => $bundle_item3->biblionumber,
244             itemnumber     => $bundle_item3->itemnumber,
245         }
246     );
247     throws_ok { $host_item->add_to_bundle($bundle_item3) }
248     'Koha::Exceptions::Item::Bundle::ItemHasHolds',
249       'Exception thrown if you try to add an item with holds to a bundle';
250
251     ok($host_item->add_to_bundle($bundle_item1), 'bundle_item1 added to bundle');
252     is($bundle_item1->notforloan, 1, 'add_to_bundle sets notforloan to BundleNotLoanValue');
253
254     throws_ok { $host_item->add_to_bundle($bundle_item1) }
255     'Koha::Exceptions::Object::DuplicateID',
256       'Exception thrown if you try to add the same item twice';
257
258     throws_ok { $bundle_item1->add_to_bundle($bundle_item2) }
259     'Koha::Exceptions::Item::Bundle::IsBundle',
260       'Exception thrown if you try to add an item to a bundled item';
261
262     throws_ok { $bundle_item2->add_to_bundle($host_item) }
263     'Koha::Exceptions::Item::Bundle::IsBundle',
264       'Exception thrown if you try to add a bundle host to a bundle item';
265
266     C4::Circulation::AddIssue( $patron->unblessed, $host_item->barcode );
267     throws_ok { $host_item->add_to_bundle($bundle_item2) }
268     'Koha::Exceptions::Item::Bundle::BundleIsCheckedOut',
269       'Exception thrown if you try to add an item to a checked out bundle';
270     C4::Circulation::AddReturn( $host_item->barcode, $host_item->homebranch );
271     $host_item->discard_changes;
272
273     C4::Circulation::AddIssue( $patron->unblessed, $bundle_item2->barcode );
274     throws_ok { $host_item->add_to_bundle($bundle_item2) }
275     'Koha::Exceptions::Item::Bundle::ItemIsCheckedOut',
276       'Exception thrown if you try to add a checked out item';
277
278     $bundle_item2->withdrawn(1)->store;
279     t::lib::Mocks::mock_preference( 'BlockReturnOfWithdrawnItems', 1 );
280     throws_ok { $host_item->add_to_bundle( $bundle_item2, { force_checkin => 1 } ) }
281     'Koha::Exceptions::Checkin::FailedCheckin',
282       'Exception thrown if you try to add a checked out item using
283       "force_checkin" and the return is not possible';
284
285     $bundle_item2->withdrawn(0)->store;
286     lives_ok { $host_item->add_to_bundle( $bundle_item2, { force_checkin => 1 } ) }
287     'No exception if you try to add a checked out item using "force_checkin" and the return is possible';
288
289     $bundle_item2->discard_changes;
290     ok( !$bundle_item2->checkout, 'Item is not checked out after being added to a bundle' );
291
292     $schema->storage->txn_rollback;
293 };
294
295 subtest 'remove_from_bundle tests' => sub {
296     plan tests => 4;
297
298     $schema->storage->txn_begin;
299
300     my $host_item = $builder->build_sample_item();
301     my $bundle_item1 = $builder->build_sample_item({ notforloan => 1 });
302     $host_item->add_to_bundle($bundle_item1);
303
304     my $patron = $builder->build_object( { class => 'Koha::Patrons' } );
305     t::lib::Mocks::mock_userenv( { branchcode => $patron->branchcode } );
306
307     C4::Circulation::AddIssue( $patron->unblessed, $host_item->barcode );
308     throws_ok { $bundle_item1->remove_from_bundle }
309     'Koha::Exceptions::Item::Bundle::BundleIsCheckedOut',
310       'Exception thrown if you try to add an item to a checked out bundle';
311     my ( $doreturn, $messages, $issue ) = C4::Circulation::AddReturn( $host_item->barcode, $host_item->homebranch );
312     $bundle_item1->discard_changes;
313
314     is($bundle_item1->remove_from_bundle(), 1, 'remove_from_bundle returns 1 when item is removed from a bundle');
315     is($bundle_item1->notforloan, 0, 'remove_from_bundle resets notforloan to 0');
316     $bundle_item1->discard_changes;
317     is($bundle_item1->remove_from_bundle(), 0, 'remove_from_bundle returns 0 when item is not in a bundle');
318
319     $schema->storage->txn_rollback;
320 };
321
322 subtest 'hidden_in_opac() tests' => sub {
323
324     plan tests => 4;
325
326     $schema->storage->txn_begin;
327
328     my $item  = $builder->build_sample_item({ itemlost => 2 });
329     my $rules = {};
330
331     # disable hidelostitems as it interteres with OpachiddenItems for the calculation
332     t::lib::Mocks::mock_preference( 'hidelostitems', 0 );
333
334     ok( !$item->hidden_in_opac, 'No rules passed, shouldn\'t hide' );
335     ok( !$item->hidden_in_opac({ rules => $rules }), 'Empty rules passed, shouldn\'t hide' );
336
337     # enable hidelostitems to verify correct behaviour
338     t::lib::Mocks::mock_preference( 'hidelostitems', 1 );
339     ok( $item->hidden_in_opac, 'Even with no rules, item should hide because of hidelostitems syspref' );
340
341     # disable hidelostitems
342     t::lib::Mocks::mock_preference( 'hidelostitems', 0 );
343     my $withdrawn = $item->withdrawn + 1; # make sure this attribute doesn't match
344
345     $rules = { withdrawn => [$withdrawn], itype => [ $item->itype ] };
346
347     ok( $item->hidden_in_opac({ rules => $rules }), 'Rule matching itype passed, should hide' );
348
349
350
351     $schema->storage->txn_rollback;
352 };
353
354 subtest 'has_pending_hold() tests' => sub {
355
356     plan tests => 2;
357
358     $schema->storage->txn_begin;
359
360     my $dbh = C4::Context->dbh;
361     my $item  = $builder->build_sample_item({ itemlost => 0 });
362     my $itemnumber = $item->itemnumber;
363
364     $dbh->do("INSERT INTO tmp_holdsqueue (surname,borrowernumber,itemnumber) VALUES ('Clamp',42,$itemnumber)");
365     ok( $item->has_pending_hold, "Yes, we have a pending hold");
366     $dbh->do("DELETE FROM tmp_holdsqueue WHERE itemnumber=$itemnumber");
367     ok( !$item->has_pending_hold, "We don't have a pending hold if nothing in the tmp_holdsqueue");
368
369     $schema->storage->txn_rollback;
370 };
371
372 subtest "as_marc_field() tests" => sub {
373
374     my $mss = C4::Biblio::GetMarcSubfieldStructure( '' );
375     my ( $itemtag, $itemtagsubfield) = C4::Biblio::GetMarcFromKohaField( "items.itemnumber" );
376
377     my @schema_columns = $schema->resultset('Item')->result_source->columns;
378     my @mapped_columns = grep { exists $mss->{'items.'.$_} } @schema_columns;
379
380     plan tests => scalar @mapped_columns + 5;
381
382     $schema->storage->txn_begin;
383
384     my $item = $builder->build_sample_item;
385     # Make sure it has at least one undefined attribute
386     $item->set({ replacementprice => undef })->store->discard_changes;
387
388     my $marc_field = $item->as_marc_field;
389
390     is(
391         $marc_field->tag,
392         $itemtag,
393         'Generated field set the right tag number'
394     );
395
396     foreach my $column (@mapped_columns) {
397         my $tagsubfield = $mss->{ 'items.' . $column }[0]->{tagsubfield};
398         is( $marc_field->subfield($tagsubfield),
399             $item->$column, "Value is mapped correctly for column $column" );
400     }
401
402     my $unmapped_subfield = Koha::MarcSubfieldStructure->new(
403         {
404             frameworkcode => '',
405             tagfield      => $itemtag,
406             tagsubfield   => 'X',
407         }
408     )->store;
409     Koha::MarcSubfieldStructure->new(
410         {
411             frameworkcode => '',
412             tagfield      => $itemtag,
413             tagsubfield   => 'Y',
414             kohafield     => '',
415         }
416     )->store;
417
418     my @unlinked_subfields;
419     push @unlinked_subfields, X => 'Something weird', Y => 'Something else';
420     $item->more_subfields_xml( C4::Items::_get_unlinked_subfields_xml( \@unlinked_subfields ) )->store;
421
422     Koha::Caches->get_instance->clear_from_cache( "MarcStructure-1-" );
423     Koha::MarcSubfieldStructures->search(
424         { frameworkcode => '', tagfield => $itemtag } )
425       ->update( { display_order => \['FLOOR( 1 + RAND( ) * 10 )'] } );
426
427     $marc_field = $item->as_marc_field;
428
429     my $tagslib = C4::Biblio::GetMarcStructure(1, '');
430     my @subfields = $marc_field->subfields;
431     my $result = all { defined $_->[1] } @subfields;
432     ok( $result, 'There are no undef subfields' );
433     my @ordered_subfields = sort {
434             $tagslib->{$itemtag}->{ $a->[0] }->{display_order}
435         <=> $tagslib->{$itemtag}->{ $b->[0] }->{display_order}
436     } @subfields;
437     is_deeply(\@subfields, \@ordered_subfields);
438
439     is( scalar $marc_field->subfield('X'), 'Something weird', 'more_subfield_xml is considered when kohafield is NULL' );
440     is( scalar $marc_field->subfield('Y'), 'Something else', 'more_subfield_xml is considered when kohafield = ""' );
441
442     $schema->storage->txn_rollback;
443     Koha::Caches->get_instance->clear_from_cache( "MarcStructure-1-" );
444 };
445
446 subtest 'pickup_locations() tests' => sub {
447
448     plan tests => 68;
449
450     $schema->storage->txn_begin;
451
452     my $dbh = C4::Context->dbh;
453
454     my $root1 = $builder->build_object( { class => 'Koha::Library::Groups', value => { ft_local_hold_group => 1, branchcode => undef } } );
455     my $root2 = $builder->build_object( { class => 'Koha::Library::Groups', value => { ft_local_hold_group => 1, branchcode => undef } } );
456     my $library1 = $builder->build_object( { class => 'Koha::Libraries', value => { pickup_location => 1, } } );
457     my $library2 = $builder->build_object( { class => 'Koha::Libraries', value => { pickup_location => 1, } } );
458     my $library3 = $builder->build_object( { class => 'Koha::Libraries', value => { pickup_location => 0, } } );
459     my $library4 = $builder->build_object( { class => 'Koha::Libraries', value => { pickup_location => 1, } } );
460     my $group1_1 = $builder->build_object( { class => 'Koha::Library::Groups', value => { parent_id => $root1->id, branchcode => $library1->branchcode } } );
461     my $group1_2 = $builder->build_object( { class => 'Koha::Library::Groups', value => { parent_id => $root1->id, branchcode => $library2->branchcode } } );
462
463     my $group2_1 = $builder->build_object( { class => 'Koha::Library::Groups', value => { parent_id => $root2->id, branchcode => $library3->branchcode } } );
464     my $group2_2 = $builder->build_object( { class => 'Koha::Library::Groups', value => { parent_id => $root2->id, branchcode => $library4->branchcode } } );
465
466     our @branchcodes = (
467         $library1->branchcode, $library2->branchcode,
468         $library3->branchcode, $library4->branchcode
469     );
470
471     my $item1 = $builder->build_sample_item(
472         {
473             homebranch    => $library1->branchcode,
474             holdingbranch => $library2->branchcode,
475             copynumber    => 1,
476             ccode         => 'Gollum'
477         }
478     )->store;
479
480     my $item3 = $builder->build_sample_item(
481         {
482             homebranch    => $library3->branchcode,
483             holdingbranch => $library4->branchcode,
484             copynumber    => 3,
485             itype         => $item1->itype,
486         }
487     )->store;
488
489     Koha::CirculationRules->set_rules(
490         {
491             categorycode => undef,
492             itemtype     => $item1->itype,
493             branchcode   => undef,
494             rules        => {
495                 reservesallowed => 25,
496             }
497         }
498     );
499
500     throws_ok
501       { $item1->pickup_locations }
502       'Koha::Exceptions::MissingParameter',
503       'Exception thrown on missing parameter';
504
505     is( $@->parameter, 'patron', 'Exception param correctly set' );
506
507     my $patron1 = $builder->build_object( { class => 'Koha::Patrons', value => { branchcode => $library1->branchcode, firstname => '1' } } );
508     my $patron4 = $builder->build_object( { class => 'Koha::Patrons', value => { branchcode => $library4->branchcode, firstname => '4' } } );
509
510     my $results = {
511         "1-1-from_home_library-any"               => 3,
512         "1-1-from_home_library-holdgroup"         => 2,
513         "1-1-from_home_library-patrongroup"       => 2,
514         "1-1-from_home_library-homebranch"        => 1,
515         "1-1-from_home_library-holdingbranch"     => 1,
516         "1-1-from_any_library-any"                => 3,
517         "1-1-from_any_library-holdgroup"          => 2,
518         "1-1-from_any_library-patrongroup"        => 2,
519         "1-1-from_any_library-homebranch"         => 1,
520         "1-1-from_any_library-holdingbranch"      => 1,
521         "1-1-from_local_hold_group-any"           => 3,
522         "1-1-from_local_hold_group-holdgroup"     => 2,
523         "1-1-from_local_hold_group-patrongroup"   => 2,
524         "1-1-from_local_hold_group-homebranch"    => 1,
525         "1-1-from_local_hold_group-holdingbranch" => 1,
526         "1-4-from_home_library-any"               => 0,
527         "1-4-from_home_library-holdgroup"         => 0,
528         "1-4-from_home_library-patrongroup"       => 0,
529         "1-4-from_home_library-homebranch"        => 0,
530         "1-4-from_home_library-holdingbranch"     => 0,
531         "1-4-from_any_library-any"                => 3,
532         "1-4-from_any_library-holdgroup"          => 2,
533         "1-4-from_any_library-patrongroup"        => 1,
534         "1-4-from_any_library-homebranch"         => 1,
535         "1-4-from_any_library-holdingbranch"      => 1,
536         "1-4-from_local_hold_group-any"           => 0,
537         "1-4-from_local_hold_group-holdgroup"     => 0,
538         "1-4-from_local_hold_group-patrongroup"   => 0,
539         "1-4-from_local_hold_group-homebranch"    => 0,
540         "1-4-from_local_hold_group-holdingbranch" => 0,
541         "3-1-from_home_library-any"               => 0,
542         "3-1-from_home_library-holdgroup"         => 0,
543         "3-1-from_home_library-patrongroup"       => 0,
544         "3-1-from_home_library-homebranch"        => 0,
545         "3-1-from_home_library-holdingbranch"     => 0,
546         "3-1-from_any_library-any"                => 3,
547         "3-1-from_any_library-holdgroup"          => 1,
548         "3-1-from_any_library-patrongroup"        => 2,
549         "3-1-from_any_library-homebranch"         => 0,
550         "3-1-from_any_library-holdingbranch"      => 1,
551         "3-1-from_local_hold_group-any"           => 0,
552         "3-1-from_local_hold_group-holdgroup"     => 0,
553         "3-1-from_local_hold_group-patrongroup"   => 0,
554         "3-1-from_local_hold_group-homebranch"    => 0,
555         "3-1-from_local_hold_group-holdingbranch" => 0,
556         "3-4-from_home_library-any"               => 0,
557         "3-4-from_home_library-holdgroup"         => 0,
558         "3-4-from_home_library-patrongroup"       => 0,
559         "3-4-from_home_library-homebranch"        => 0,
560         "3-4-from_home_library-holdingbranch"     => 0,
561         "3-4-from_any_library-any"                => 3,
562         "3-4-from_any_library-holdgroup"          => 1,
563         "3-4-from_any_library-patrongroup"        => 1,
564         "3-4-from_any_library-homebranch"         => 0,
565         "3-4-from_any_library-holdingbranch"      => 1,
566         "3-4-from_local_hold_group-any"           => 3,
567         "3-4-from_local_hold_group-holdgroup"     => 1,
568         "3-4-from_local_hold_group-patrongroup"   => 1,
569         "3-4-from_local_hold_group-homebranch"    => 0,
570         "3-4-from_local_hold_group-holdingbranch" => 1
571     };
572
573     sub _doTest {
574         my ( $item, $patron, $ha, $hfp, $results ) = @_;
575
576         Koha::CirculationRules->set_rules(
577             {
578                 branchcode => undef,
579                 itemtype   => undef,
580                 rules => {
581                     holdallowed => $ha,
582                     hold_fulfillment_policy => $hfp,
583                     returnbranch => 'any'
584                 }
585             }
586         );
587         my $ha_value =
588           $ha eq 'from_local_hold_group' ? 'holdgroup'
589           : (
590             $ha eq 'from_any_library' ? 'any'
591             : 'homebranch'
592           );
593
594         my @pl = map {
595             my $pickup_location = $_;
596             grep { $pickup_location->branchcode eq $_ } @branchcodes
597         } $item->pickup_locations( { patron => $patron } )->as_list;
598
599         ok(
600             scalar(@pl) eq $results->{
601                     $item->copynumber . '-'
602                   . $patron->firstname . '-'
603                   . $ha . '-'
604                   . $hfp
605             },
606             'item'
607               . $item->copynumber
608               . ', patron'
609               . $patron->firstname
610               . ', holdallowed: '
611               . $ha_value
612               . ', hold_fulfillment_policy: '
613               . $hfp
614               . ' should return '
615               . $results->{
616                     $item->copynumber . '-'
617                   . $patron->firstname . '-'
618                   . $ha . '-'
619                   . $hfp
620               }
621               . ' and returns '
622               . scalar(@pl)
623         );
624
625     }
626
627
628     foreach my $item ($item1, $item3) {
629         foreach my $patron ($patron1, $patron4) {
630             #holdallowed 1: homebranch, 2: any, 3: holdgroup
631             foreach my $ha ('from_home_library', 'from_any_library', 'from_local_hold_group') {
632                 foreach my $hfp ('any', 'holdgroup', 'patrongroup', 'homebranch', 'holdingbranch') {
633                     _doTest($item, $patron, $ha, $hfp, $results);
634                 }
635             }
636         }
637     }
638
639     # Now test that branchtransferlimits will further filter the pickup locations
640
641     my $item_no_ccode = $builder->build_sample_item(
642         {
643             homebranch    => $library1->branchcode,
644             holdingbranch => $library2->branchcode,
645             itype         => $item1->itype,
646         }
647     )->store;
648
649     t::lib::Mocks::mock_preference('UseBranchTransferLimits', 1);
650     t::lib::Mocks::mock_preference('BranchTransferLimitsType', 'itemtype');
651     Koha::CirculationRules->set_rules(
652         {
653             branchcode => undef,
654             itemtype   => $item1->itype,
655             rules      => {
656                 holdallowed             => 'from_home_library',
657                 hold_fulfillment_policy => 1,
658                 returnbranch            => 'any'
659             }
660         }
661     );
662     $builder->build_object(
663         {
664             class => 'Koha::Item::Transfer::Limits',
665             value => {
666                 toBranch   => $library1->branchcode,
667                 fromBranch => $library2->branchcode,
668                 itemtype   => $item1->itype,
669                 ccode      => undef,
670             }
671         }
672     );
673
674     my @pickup_locations = map {
675         my $pickup_location = $_;
676         grep { $pickup_location->branchcode eq $_ } @branchcodes
677     } $item1->pickup_locations( { patron => $patron1 } )->as_list;
678
679     is( scalar @pickup_locations, 3 - 1, "With a transfer limits we get back the libraries that are pickup locations minus 1 limited library");
680
681     $builder->build_object(
682         {
683             class => 'Koha::Item::Transfer::Limits',
684             value => {
685                 toBranch   => $library4->branchcode,
686                 fromBranch => $library2->branchcode,
687                 itemtype   => $item1->itype,
688                 ccode      => undef,
689             }
690         }
691     );
692
693     @pickup_locations = map {
694         my $pickup_location = $_;
695         grep { $pickup_location->branchcode eq $_ } @branchcodes
696     } $item1->pickup_locations( { patron => $patron1 } )->as_list;
697
698     is( scalar @pickup_locations, 3 - 2, "With 2 transfer limits we get back the libraries that are pickup locations minus 2 limited libraries");
699
700     t::lib::Mocks::mock_preference('BranchTransferLimitsType', 'ccode');
701     @pickup_locations = map {
702         my $pickup_location = $_;
703         grep { $pickup_location->branchcode eq $_ } @branchcodes
704     } $item1->pickup_locations( { patron => $patron1 } )->as_list;
705     is( scalar @pickup_locations, 3, "With no transfer limits of type ccode we get back the libraries that are pickup locations");
706
707     @pickup_locations = map {
708         my $pickup_location = $_;
709         grep { $pickup_location->branchcode eq $_ } @branchcodes
710     } $item_no_ccode->pickup_locations( { patron => $patron1 } )->as_list;
711     is( scalar @pickup_locations, 3, "With no transfer limits of type ccode and an item with no ccode we get back the libraries that are pickup locations");
712
713     $builder->build_object(
714         {
715             class => 'Koha::Item::Transfer::Limits',
716             value => {
717                 toBranch   => $library2->branchcode,
718                 fromBranch => $library2->branchcode,
719                 itemtype   => undef,
720                 ccode      => $item1->ccode,
721             }
722         }
723     );
724
725     @pickup_locations = map {
726         my $pickup_location = $_;
727         grep { $pickup_location->branchcode eq $_ } @branchcodes
728     } $item1->pickup_locations( { patron => $patron1 } )->as_list;
729     is( scalar @pickup_locations, 3 - 1, "With a transfer limits we get back the libraries that are pickup locations minus 1 limited library");
730
731     $builder->build_object(
732         {
733             class => 'Koha::Item::Transfer::Limits',
734             value => {
735                 toBranch   => $library4->branchcode,
736                 fromBranch => $library2->branchcode,
737                 itemtype   => undef,
738                 ccode      => $item1->ccode,
739             }
740         }
741     );
742
743     @pickup_locations = map {
744         my $pickup_location = $_;
745         grep { $pickup_location->branchcode eq $_ } @branchcodes
746     } $item1->pickup_locations( { patron => $patron1 } )->as_list;
747     is( scalar @pickup_locations, 3 - 2, "With 2 transfer limits we get back the libraries that are pickup locations minus 2 limited libraries");
748
749     t::lib::Mocks::mock_preference('UseBranchTransferLimits', 0);
750
751     $schema->storage->txn_rollback;
752 };
753
754 subtest 'request_transfer' => sub {
755     plan tests => 13;
756     $schema->storage->txn_begin;
757
758     my $library1 = $builder->build_object( { class => 'Koha::Libraries' } );
759     my $library2 = $builder->build_object( { class => 'Koha::Libraries' } );
760     my $item     = $builder->build_sample_item(
761         {
762             homebranch    => $library1->branchcode,
763             holdingbranch => $library2->branchcode,
764         }
765     );
766
767     # Mandatory fields tests
768     throws_ok { $item->request_transfer( { to => $library1 } ) }
769     'Koha::Exceptions::MissingParameter',
770       'Exception thrown if `reason` parameter is missing';
771
772     throws_ok { $item->request_transfer( { reason => 'Manual' } ) }
773     'Koha::Exceptions::MissingParameter',
774       'Exception thrown if `to` parameter is missing';
775
776     # Successful request
777     my $transfer = $item->request_transfer({ to => $library1, reason => 'Manual' });
778     is( ref($transfer), 'Koha::Item::Transfer',
779         'Koha::Item->request_transfer should return a Koha::Item::Transfer object'
780     );
781     my $original_transfer = $transfer->get_from_storage;
782
783     # Transfer already in progress
784     throws_ok { $item->request_transfer( { to => $library2, reason => 'Manual' } ) }
785     'Koha::Exceptions::Item::Transfer::InQueue',
786       'Exception thrown if transfer is already in progress';
787
788     my $exception = $@;
789     is( ref( $exception->transfer ),
790         'Koha::Item::Transfer',
791         'The exception contains the found Koha::Item::Transfer' );
792
793     # Queue transfer
794     my $queued_transfer = $item->request_transfer(
795         { to => $library2, reason => 'Manual', enqueue => 1 } );
796     is( ref($queued_transfer), 'Koha::Item::Transfer',
797         'Koha::Item->request_transfer allowed when enqueue is set' );
798     my $transfers = $item->get_transfers;
799     is($transfers->count, 2, "There are now 2 live transfers in the queue");
800     $transfer = $transfer->get_from_storage;
801     is_deeply($transfer->unblessed, $original_transfer->unblessed, "Original transfer unchanged");
802     $queued_transfer->datearrived(dt_from_string)->store();
803
804     # Replace transfer
805     my $replaced_transfer = $item->request_transfer(
806         { to => $library2, reason => 'Manual', replace => 1 } );
807     is( ref($replaced_transfer), 'Koha::Item::Transfer',
808         'Koha::Item->request_transfer allowed when replace is set' );
809     $original_transfer->discard_changes;
810     ok($original_transfer->datecancelled, "Original transfer cancelled");
811     $transfers = $item->get_transfers;
812     is($transfers->count, 1, "There is only 1 live transfer in the queue");
813     $replaced_transfer->datearrived(dt_from_string)->store();
814
815     # BranchTransferLimits
816     t::lib::Mocks::mock_preference('UseBranchTransferLimits', 1);
817     t::lib::Mocks::mock_preference('BranchTransferLimitsType', 'itemtype');
818     my $limit = Koha::Item::Transfer::Limit->new({
819         fromBranch => $library2->branchcode,
820         toBranch => $library1->branchcode,
821         itemtype => $item->effective_itemtype,
822     })->store;
823
824     throws_ok { $item->request_transfer( { to => $library1, reason => 'Manual' } ) }
825     'Koha::Exceptions::Item::Transfer::Limit',
826       'Exception thrown if transfer is prevented by limits';
827
828     my $forced_transfer = $item->request_transfer( { to => $library1, reason => 'Manual', ignore_limits => 1 } );
829     is( ref($forced_transfer), 'Koha::Item::Transfer',
830         'Koha::Item->request_transfer allowed when ignore_limits is set'
831     );
832
833     $schema->storage->txn_rollback;
834 };
835
836 subtest 'deletion' => sub {
837     plan tests => 15;
838
839     $schema->storage->txn_begin;
840
841     my $biblio = $builder->build_sample_biblio();
842
843     my $item = $builder->build_sample_item(
844         {
845             biblionumber => $biblio->biblionumber,
846         }
847     );
848     is( $item->deleted_on, undef, 'deleted_on not set for new item' );
849
850     my $deleted_item = $item->move_to_deleted;
851     is( ref( $deleted_item ), 'Koha::Schema::Result::Deleteditem', 'Koha::Item->move_to_deleted should return the Deleted item' )
852       ;    # FIXME This should be Koha::Deleted::Item
853     is( t::lib::Dates::compare( $deleted_item->deleted_on, dt_from_string() ), 0 );
854
855     is( Koha::Old::Items->search({itemnumber => $item->itemnumber})->count, 1, '->move_to_deleted must have moved the item to deleteditem' );
856     $item = $builder->build_sample_item(
857         {
858             biblionumber => $biblio->biblionumber,
859         }
860     );
861     $item->delete;
862     is( Koha::Old::Items->search({itemnumber => $item->itemnumber})->count, 0, '->move_to_deleted must not have moved the item to deleteditem' );
863
864
865     my $library   = $builder->build_object({ class => 'Koha::Libraries' });
866     my $library_2 = $builder->build_object({ class => 'Koha::Libraries' });
867     t::lib::Mocks::mock_userenv({ branchcode => $library->branchcode });
868
869     my $patron = $builder->build_object({class => 'Koha::Patrons'});
870     $item = $builder->build_sample_item({ library => $library->branchcode });
871
872     # book_on_loan
873     C4::Circulation::AddIssue( $patron->unblessed, $item->barcode );
874
875     is(
876         @{$item->safe_to_delete->messages}[0]->message,
877         'book_on_loan',
878         'Koha::Item->safe_to_delete reports item on loan',
879     );
880
881     is(
882         @{$item->safe_to_delete->messages}[0]->message,
883         'book_on_loan',
884         'item that is on loan cannot be deleted',
885     );
886
887     ok(
888         ! $item->safe_to_delete,
889         'Koha::Item->safe_to_delete shows item NOT safe to delete'
890     );
891
892     AddReturn( $item->barcode, $library->branchcode );
893
894     # not_same_branch
895     t::lib::Mocks::mock_preference('IndependentBranches', 1);
896     my $item_2 = $builder->build_sample_item({ library => $library_2->branchcode });
897
898     is(
899         @{$item_2->safe_to_delete->messages}[0]->message,
900         'not_same_branch',
901         'Koha::Item->safe_to_delete reports IndependentBranches restriction',
902     );
903
904     is(
905         @{$item_2->safe_to_delete->messages}[0]->message,
906         'not_same_branch',
907         'IndependentBranches prevents deletion at another branch',
908     );
909
910     # linked_analytics
911
912     { # codeblock to limit scope of $module->mock
913
914         my $module = Test::MockModule->new('C4::Items');
915         $module->mock( GetAnalyticsCount => sub { return 1 } );
916
917         $item->discard_changes;
918         is(
919             @{$item->safe_to_delete->messages}[0]->message,
920             'linked_analytics',
921             'Koha::Item->safe_to_delete reports linked analytics',
922         );
923
924         is(
925             @{$item->safe_to_delete->messages}[0]->message,
926             'linked_analytics',
927             'Linked analytics prevents deletion of item',
928         );
929
930     }
931
932     ok(
933         $item->safe_to_delete,
934         'Koha::Item->safe_to_delete shows item safe to delete'
935     );
936
937     $item->safe_delete,
938
939     my $test_item = Koha::Items->find( $item->itemnumber );
940
941     is( $test_item, undef,
942         "Koha::Item->safe_delete should delete item if safe_to_delete returns true"
943     );
944
945     subtest 'holds tests' => sub {
946
947         plan tests => 9;
948
949         # to avoid noise
950         t::lib::Mocks::mock_preference( 'IndependentBranches', 0 );
951
952         $schema->storage->txn_begin;
953
954         my $item = $builder->build_sample_item;
955
956         my $processing     = $builder->build_object( { class => 'Koha::Holds', value => { itemnumber => $item->id, itemnumber => $item->id, found => 'P' } } );
957         my $safe_to_delete = $item->safe_to_delete;
958
959         ok( !$safe_to_delete, 'Cannot delete' );
960         is(
961             @{ $safe_to_delete->messages }[0]->message,
962             'book_reserved',
963             'Koha::Item->safe_to_delete reports a in processing hold blocks deletion'
964         );
965
966         $processing->delete;
967
968         my $in_transit = $builder->build_object( { class => 'Koha::Holds', value => { itemnumber => $item->id, itemnumber => $item->id, found => 'T' } } );
969         $safe_to_delete = $item->safe_to_delete;
970
971         ok( !$safe_to_delete, 'Cannot delete' );
972         is(
973             @{ $safe_to_delete->messages }[0]->message,
974             'book_reserved',
975             'Koha::Item->safe_to_delete reports a in transit hold blocks deletion'
976         );
977
978         $in_transit->delete;
979
980         my $waiting = $builder->build_object( { class => 'Koha::Holds', value => { itemnumber => $item->id, itemnumber => $item->id, found => 'W' } } );
981         $safe_to_delete = $item->safe_to_delete;
982
983         ok( !$safe_to_delete, 'Cannot delete' );
984         is(
985             @{ $safe_to_delete->messages }[0]->message,
986             'book_reserved',
987             'Koha::Item->safe_to_delete reports a waiting hold blocks deletion'
988         );
989
990         $waiting->delete;
991
992         # Add am unfilled biblio-level hold to catch the 'last_item_for_hold' use case
993         $builder->build_object( { class => 'Koha::Holds', value => { biblionumber => $item->biblionumber, itemnumber => undef, found => undef } } );
994
995         $safe_to_delete = $item->safe_to_delete;
996
997         ok( !$safe_to_delete );
998
999         is(
1000             @{ $safe_to_delete->messages}[0]->message,
1001             'last_item_for_hold',
1002             'Item cannot be deleted if a biblio-level is placed on the biblio and there is only 1 item attached to the biblio'
1003         );
1004
1005         my $extra_item = $builder->build_sample_item({ biblionumber => $item->biblionumber });
1006
1007         ok( $item->safe_to_delete );
1008
1009         $schema->storage->txn_rollback;
1010     };
1011
1012     $schema->storage->txn_rollback;
1013 };
1014
1015 subtest 'renewal_branchcode' => sub {
1016     plan tests => 13;
1017
1018     $schema->storage->txn_begin;
1019
1020     my $item = $builder->build_sample_item();
1021     my $branch = $builder->build_object({ class => 'Koha::Libraries' });
1022     my $checkout = $builder->build_object({
1023         class => 'Koha::Checkouts',
1024         value => {
1025             itemnumber => $item->itemnumber,
1026         }
1027     });
1028
1029
1030     C4::Context->interface( 'intranet' );
1031     t::lib::Mocks::mock_userenv({ branchcode => $branch->branchcode });
1032
1033     is( $item->renewal_branchcode, $branch->branchcode, "If interface not opac, we get the branch from context");
1034     is( $item->renewal_branchcode({ branch => "PANDA"}), $branch->branchcode, "If interface not opac, we get the branch from context even if we pass one in");
1035     C4::Context->set_userenv(51, 'userid4tests', undef, 'firstname', 'surname', undef, undef, 0, undef, undef, undef ); #mock userenv doesn't let us set null branch
1036     is( $item->renewal_branchcode({ branch => "PANDA"}), "PANDA", "If interface not opac, we get the branch we pass one in if context not set");
1037
1038     C4::Context->interface( 'opac' );
1039
1040     t::lib::Mocks::mock_preference('OpacRenewalBranch', undef);
1041     is( $item->renewal_branchcode, 'OPACRenew', "If interface opac and OpacRenewalBranch undef, we get OPACRenew");
1042     is( $item->renewal_branchcode({branch=>'COW'}), 'OPACRenew', "If interface opac and OpacRenewalBranch undef, we get OPACRenew even if branch passed");
1043
1044     t::lib::Mocks::mock_preference('OpacRenewalBranch', 'none');
1045     is( $item->renewal_branchcode, '', "If interface opac and OpacRenewalBranch is none, we get blank string");
1046     is( $item->renewal_branchcode({branch=>'COW'}), '', "If interface opac and OpacRenewalBranch is none, we get blank string even if branch passed");
1047
1048     t::lib::Mocks::mock_preference('OpacRenewalBranch', 'checkoutbranch');
1049     is( $item->renewal_branchcode, $checkout->branchcode, "If interface opac and OpacRenewalBranch set to checkoutbranch, we get branch of checkout");
1050     is( $item->renewal_branchcode({branch=>'MONKEY'}), $checkout->branchcode, "If interface opac and OpacRenewalBranch set to checkoutbranch, we get branch of checkout even if branch passed");
1051
1052     t::lib::Mocks::mock_preference('OpacRenewalBranch','patronhomebranch');
1053     is( $item->renewal_branchcode, $checkout->patron->branchcode, "If interface opac and OpacRenewalBranch set to patronbranch, we get branch of patron");
1054     is( $item->renewal_branchcode({branch=>'TURKEY'}), $checkout->patron->branchcode, "If interface opac and OpacRenewalBranch set to patronbranch, we get branch of patron even if branch passed");
1055
1056     t::lib::Mocks::mock_preference('OpacRenewalBranch','itemhomebranch');
1057     is( $item->renewal_branchcode, $item->homebranch, "If interface opac and OpacRenewalBranch set to itemhomebranch, we get homebranch of item");
1058     is( $item->renewal_branchcode({branch=>'MANATEE'}), $item->homebranch, "If interface opac and OpacRenewalBranch set to itemhomebranch, we get homebranch of item even if branch passed");
1059
1060     $schema->storage->txn_rollback;
1061 };
1062
1063 subtest 'Tests for itemtype' => sub {
1064     plan tests => 2;
1065     $schema->storage->txn_begin;
1066
1067     my $biblio = $builder->build_sample_biblio;
1068     my $itemtype = $builder->build_object({ class => 'Koha::ItemTypes' });
1069     my $item = $builder->build_sample_item({ biblionumber => $biblio->biblionumber, itype => $itemtype->itemtype });
1070
1071     t::lib::Mocks::mock_preference('item-level_itypes', 1);
1072     is( $item->itemtype->itemtype, $item->itype, 'Pref enabled' );
1073     t::lib::Mocks::mock_preference('item-level_itypes', 0);
1074     is( $item->itemtype->itemtype, $biblio->biblioitem->itemtype, 'Pref disabled' );
1075
1076     $schema->storage->txn_rollback;
1077 };
1078
1079 subtest 'get_transfers' => sub {
1080     plan tests => 16;
1081     $schema->storage->txn_begin;
1082
1083     my $item = $builder->build_sample_item();
1084
1085     my $transfers = $item->get_transfers();
1086     is(ref($transfers), 'Koha::Item::Transfers', 'Koha::Item->get_transfer should return a Koha::Item::Transfers object' );
1087     is($transfers->count, 0, 'When no transfers exist, the Koha::Item:Transfers object should be empty');
1088
1089     my $library_to = $builder->build_object( { class => 'Koha::Libraries' } );
1090
1091     my $transfer_1 = $builder->build_object(
1092         {
1093             class => 'Koha::Item::Transfers',
1094             value => {
1095                 itemnumber    => $item->itemnumber,
1096                 frombranch    => $item->holdingbranch,
1097                 tobranch      => $library_to->branchcode,
1098                 reason        => 'Manual',
1099                 datesent      => undef,
1100                 datearrived   => undef,
1101                 datecancelled => undef,
1102                 daterequested => \'NOW()'
1103             }
1104         }
1105     );
1106
1107     $transfers = $item->get_transfers();
1108     is($transfers->count, 1, 'When one transfer has been requested, the Koha::Item:Transfers object should contain one result');
1109
1110     my $transfer_2 = $builder->build_object(
1111         {
1112             class => 'Koha::Item::Transfers',
1113             value => {
1114                 itemnumber    => $item->itemnumber,
1115                 frombranch    => $item->holdingbranch,
1116                 tobranch      => $library_to->branchcode,
1117                 reason        => 'Manual',
1118                 datesent      => undef,
1119                 datearrived   => undef,
1120                 datecancelled => undef,
1121                 daterequested => \'NOW()'
1122             }
1123         }
1124     );
1125
1126     my $transfer_3 = $builder->build_object(
1127         {
1128             class => 'Koha::Item::Transfers',
1129             value => {
1130                 itemnumber    => $item->itemnumber,
1131                 frombranch    => $item->holdingbranch,
1132                 tobranch      => $library_to->branchcode,
1133                 reason        => 'Manual',
1134                 datesent      => undef,
1135                 datearrived   => undef,
1136                 datecancelled => undef,
1137                 daterequested => \'NOW()'
1138             }
1139         }
1140     );
1141
1142     $transfers = $item->get_transfers();
1143     is($transfers->count, 3, 'When there are multiple open transfer requests, the Koha::Item::Transfers object contains them all');
1144     my $result_1 = $transfers->next;
1145     my $result_2 = $transfers->next;
1146     my $result_3 = $transfers->next;
1147     is( $result_1->branchtransfer_id, $transfer_1->branchtransfer_id, 'Koha::Item->get_transfers returns the oldest transfer request first');
1148     is( $result_2->branchtransfer_id, $transfer_2->branchtransfer_id, 'Koha::Item->get_transfers returns the newer transfer request second');
1149     is( $result_3->branchtransfer_id, $transfer_3->branchtransfer_id, 'Koha::Item->get_transfers returns the newest transfer request last');
1150
1151     $transfer_2->datesent(\'NOW()')->store;
1152     $transfers = $item->get_transfers();
1153     is($transfers->count, 3, 'When one transfer is set to in_transit, the Koha::Item::Transfers object still contains them all');
1154     $result_1 = $transfers->next;
1155     $result_2 = $transfers->next;
1156     $result_3 = $transfers->next;
1157     is( $result_1->branchtransfer_id, $transfer_2->branchtransfer_id, 'Koha::Item->get_transfers returns the active transfer request first');
1158     is( $result_2->branchtransfer_id, $transfer_1->branchtransfer_id, 'Koha::Item->get_transfers returns the other transfers oldest to newest');
1159     is( $result_3->branchtransfer_id, $transfer_3->branchtransfer_id, 'Koha::Item->get_transfers returns the other transfers oldest to newest');
1160
1161     $transfer_2->datearrived(\'NOW()')->store;
1162     $transfers = $item->get_transfers();
1163     is($transfers->count, 2, 'Once a transfer is received, it no longer appears in the list from ->get_transfers()');
1164     $result_1 = $transfers->next;
1165     $result_2 = $transfers->next;
1166     is( $result_1->branchtransfer_id, $transfer_1->branchtransfer_id, 'Koha::Item->get_transfers returns the other transfers oldest to newest');
1167     is( $result_2->branchtransfer_id, $transfer_3->branchtransfer_id, 'Koha::Item->get_transfers returns the other transfers oldest to newest');
1168
1169     $transfer_1->datecancelled(\'NOW()')->store;
1170     $transfers = $item->get_transfers();
1171     is($transfers->count, 1, 'Once a transfer is cancelled, it no longer appears in the list from ->get_transfers()');
1172     $result_1 = $transfers->next;
1173     is( $result_1->branchtransfer_id, $transfer_3->branchtransfer_id, 'Koha::Item->get_transfers returns the only transfer that remains');
1174
1175     $schema->storage->txn_rollback;
1176 };
1177
1178 subtest 'Tests for relationship between item and item_orders via aqorders_item' => sub {
1179     plan tests => 3;
1180
1181     $schema->storage->txn_begin;
1182
1183     my $biblio = $builder->build_sample_biblio();
1184     my $item = $builder->build_sample_item({ biblionumber => $biblio->biblionumber });
1185
1186     my $orders = $item->orders;
1187     is ($orders->count, 0, 'No order on this item yet');
1188
1189     my $order_note = 'Order for ' . $item->itemnumber;
1190
1191     my $aq_order1 = $builder->build_object({
1192         class => 'Koha::Acquisition::Orders',
1193         value  => {
1194             biblionumber => $biblio->biblionumber,
1195             order_internalnote => $order_note,
1196         },
1197     });
1198     my $aq_order2 = $builder->build_object({
1199         class => 'Koha::Acquisition::Orders',
1200         value  => {
1201             biblionumber => $biblio->biblionumber,
1202         },
1203     });
1204     my $aq_order_item1 = $builder->build({
1205         source => 'AqordersItem',
1206         value  => {
1207             ordernumber => $aq_order1->ordernumber,
1208             itemnumber => $item->itemnumber,
1209         },
1210     });
1211
1212     $orders = $item->orders;
1213     is ($orders->count, 1, 'One order found by item with the relationship');
1214     is ($orders->next->order_internalnote, $order_note, 'Correct order found by item with the relationship');
1215 };
1216
1217 subtest 'move_to_biblio() tests' => sub {
1218     plan tests => 16;
1219
1220     $schema->storage->txn_begin;
1221
1222     my $dbh = C4::Context->dbh;
1223
1224     my $source_biblio = $builder->build_sample_biblio();
1225     my $target_biblio = $builder->build_sample_biblio();
1226
1227     my $source_biblionumber = $source_biblio->biblionumber;
1228     my $target_biblionumber = $target_biblio->biblionumber;
1229
1230     my $item1 = $builder->build_sample_item({ biblionumber => $source_biblionumber });
1231     my $item2 = $builder->build_sample_item({ biblionumber => $source_biblionumber });
1232     my $item3 = $builder->build_sample_item({ biblionumber => $source_biblionumber });
1233
1234     my $itemnumber1 = $item1->itemnumber;
1235     my $itemnumber2 = $item2->itemnumber;
1236
1237     my $library = $builder->build_object({ class => 'Koha::Libraries' });
1238
1239     my $patron = $builder->build_object({
1240         class => 'Koha::Patrons',
1241         value => { branchcode => $library->branchcode }
1242     });
1243     my $borrowernumber = $patron->borrowernumber;
1244
1245     my $aq_budget = $builder->build({
1246         source => 'Aqbudget',
1247         value  => {
1248             budget_notes => 'test',
1249         },
1250     });
1251
1252     my $aq_order1 = $builder->build_object({
1253         class => 'Koha::Acquisition::Orders',
1254         value  => {
1255             biblionumber => $source_biblionumber,
1256             budget_id => $aq_budget->{budget_id},
1257         },
1258     });
1259     my $aq_order_item1 = $builder->build({
1260         source => 'AqordersItem',
1261         value  => {
1262             ordernumber => $aq_order1->ordernumber,
1263             itemnumber => $itemnumber1,
1264         },
1265     });
1266     my $aq_order2 = $builder->build_object({
1267         class => 'Koha::Acquisition::Orders',
1268         value  => {
1269             biblionumber => $source_biblionumber,
1270             budget_id => $aq_budget->{budget_id},
1271         },
1272     });
1273     my $aq_order_item2 = $builder->build({
1274         source => 'AqordersItem',
1275         value  => {
1276             ordernumber => $aq_order2->ordernumber,
1277             itemnumber => $itemnumber2,
1278         },
1279     });
1280
1281     my $bib_level_hold = $builder->build_object({
1282         class => 'Koha::Holds',
1283         value  => {
1284             biblionumber => $source_biblionumber,
1285             itemnumber => undef,
1286         },
1287     });
1288     my $item_level_hold1 = $builder->build_object({
1289         class => 'Koha::Holds',
1290         value  => {
1291             biblionumber => $source_biblionumber,
1292             itemnumber => $itemnumber1,
1293         },
1294     });
1295     my $item_level_hold2 = $builder->build_object({
1296         class => 'Koha::Holds',
1297         value  => {
1298             biblionumber => $source_biblionumber,
1299             itemnumber => $itemnumber2,
1300         }
1301     });
1302
1303     my $tmp_holdsqueue1 = $builder->build({
1304         source => 'TmpHoldsqueue',
1305         value  => {
1306             borrowernumber => $borrowernumber,
1307             biblionumber   => $source_biblionumber,
1308             itemnumber     => $itemnumber1,
1309         }
1310     });
1311     my $tmp_holdsqueue2 = $builder->build({
1312         source => 'TmpHoldsqueue',
1313         value  => {
1314             borrowernumber => $borrowernumber,
1315             biblionumber   => $source_biblionumber,
1316             itemnumber     => $itemnumber2,
1317         }
1318     });
1319     my $hold_fill_target1 = $builder->build({
1320         source => 'HoldFillTarget',
1321         value  => {
1322             borrowernumber     => $borrowernumber,
1323             biblionumber       => $source_biblionumber,
1324             itemnumber         => $itemnumber1,
1325         }
1326     });
1327     my $hold_fill_target2 = $builder->build({
1328         source => 'HoldFillTarget',
1329         value  => {
1330             borrowernumber     => $borrowernumber,
1331             biblionumber       => $source_biblionumber,
1332             itemnumber         => $itemnumber2,
1333         }
1334     });
1335     my $linktracker1 = $builder->build({
1336         source => 'Linktracker',
1337         value  => {
1338             borrowernumber     => $borrowernumber,
1339             biblionumber       => $source_biblionumber,
1340             itemnumber         => $itemnumber1,
1341         }
1342     });
1343     my $linktracker2 = $builder->build({
1344         source => 'Linktracker',
1345         value  => {
1346             borrowernumber     => $borrowernumber,
1347             biblionumber       => $source_biblionumber,
1348             itemnumber         => $itemnumber2,
1349         }
1350     });
1351
1352     my $to_biblionumber_after_move = $item1->move_to_biblio($target_biblio);
1353     is($to_biblionumber_after_move, $target_biblionumber, 'move_to_biblio returns the target biblionumber if success');
1354
1355     $to_biblionumber_after_move = $item1->move_to_biblio($target_biblio);
1356     is($to_biblionumber_after_move, undef, 'move_to_biblio returns undef if the move has failed. If called twice, the item is not attached to the first biblio anymore');
1357
1358     my $get_item1 = Koha::Items->find( $item1->itemnumber );
1359     is($get_item1->biblionumber, $target_biblionumber, 'item1 is moved');
1360     my $get_item2 = Koha::Items->find( $item2->itemnumber );
1361     is($get_item2->biblionumber, $source_biblionumber, 'item2 is not moved');
1362     my $get_item3 = Koha::Items->find( $item3->itemnumber );
1363     is($get_item3->biblionumber, $source_biblionumber, 'item3 is not moved');
1364
1365     $aq_order1->discard_changes;
1366     $aq_order2->discard_changes;
1367     is($aq_order1->biblionumber, $target_biblionumber, 'move_to_biblio moves aq_orders for item 1');
1368     is($aq_order2->biblionumber, $source_biblionumber, 'move_to_biblio does not move aq_orders for item 2');
1369
1370     $bib_level_hold->discard_changes;
1371     $item_level_hold1->discard_changes;
1372     $item_level_hold2->discard_changes;
1373     is($bib_level_hold->biblionumber,   $source_biblionumber, 'move_to_biblio does not move the biblio-level hold');
1374     is($item_level_hold1->biblionumber, $target_biblionumber, 'move_to_biblio moves the item-level hold placed on item 1');
1375     is($item_level_hold2->biblionumber, $source_biblionumber, 'move_to_biblio does not move the item-level hold placed on item 2');
1376
1377     my $get_tmp_holdsqueue1 = $schema->resultset('TmpHoldsqueue')->search({ itemnumber => $tmp_holdsqueue1->{itemnumber} })->single;
1378     my $get_tmp_holdsqueue2 = $schema->resultset('TmpHoldsqueue')->search({ itemnumber => $tmp_holdsqueue2->{itemnumber} })->single;
1379     is($get_tmp_holdsqueue1->biblionumber->biblionumber, $target_biblionumber, 'move_to_biblio moves tmp_holdsqueue for item 1');
1380     is($get_tmp_holdsqueue2->biblionumber->biblionumber, $source_biblionumber, 'move_to_biblio does not move tmp_holdsqueue for item 2');
1381
1382     my $get_hold_fill_target1 = $schema->resultset('HoldFillTarget')->search({ itemnumber => $hold_fill_target1->{itemnumber} })->single;
1383     my $get_hold_fill_target2 = $schema->resultset('HoldFillTarget')->search({ itemnumber => $hold_fill_target2->{itemnumber} })->single;
1384     # Why does ->biblionumber return a Biblio object???
1385     is($get_hold_fill_target1->biblionumber->biblionumber, $target_biblionumber, 'move_to_biblio moves hold_fill_targets for item 1');
1386     is($get_hold_fill_target2->biblionumber->biblionumber, $source_biblionumber, 'move_to_biblio does not move hold_fill_targets for item 2');
1387
1388     my $get_linktracker1 = $schema->resultset('Linktracker')->search({ itemnumber => $linktracker1->{itemnumber} })->single;
1389     my $get_linktracker2 = $schema->resultset('Linktracker')->search({ itemnumber => $linktracker2->{itemnumber} })->single;
1390     is($get_linktracker1->biblionumber->biblionumber, $target_biblionumber, 'move_to_biblio moves linktracker for item 1');
1391     is($get_linktracker2->biblionumber->biblionumber, $source_biblionumber, 'move_to_biblio does not move linktracker for item 2');
1392
1393     $schema->storage->txn_rollback;
1394 };
1395
1396 subtest 'columns_to_str' => sub {
1397     plan tests => 4;
1398
1399     $schema->storage->txn_begin;
1400
1401     my ( $itemtag, $itemsubfield ) = C4::Biblio::GetMarcFromKohaField( "items.itemnumber" );
1402
1403     my $cache = Koha::Caches->get_instance();
1404     $cache->clear_from_cache("MarcStructure-0-");
1405     $cache->clear_from_cache("MarcStructure-1-");
1406     $cache->clear_from_cache("MarcSubfieldStructure-");
1407     $cache->clear_from_cache("libraries:name");
1408     $cache->clear_from_cache("itemtype:description:en");
1409     $cache->clear_from_cache("cn_sources:description");
1410     $cache->clear_from_cache("AV_descriptions:LOST");
1411
1412     # Creating subfields 'é', 'è' that are not linked with a kohafield
1413     Koha::MarcSubfieldStructures->search(
1414         {
1415             frameworkcode => '',
1416             tagfield => $itemtag,
1417             tagsubfield => ['é', 'è'],
1418         }
1419     )->delete;    # In case it exist already
1420
1421     # Ã© is not linked with a AV
1422     # Ã¨ is linked with AV branches
1423     Koha::MarcSubfieldStructure->new(
1424         {
1425             frameworkcode => '',
1426             tagfield      => $itemtag,
1427             tagsubfield   => 'é',
1428             kohafield     => undef,
1429             repeatable    => 1,
1430             defaultvalue  => 'ééé',
1431             tab           => 10,
1432         }
1433     )->store;
1434     Koha::MarcSubfieldStructure->new(
1435         {
1436             frameworkcode    => '',
1437             tagfield         => $itemtag,
1438             tagsubfield      => 'è',
1439             kohafield        => undef,
1440             repeatable       => 1,
1441             defaultvalue     => 'èèè',
1442             tab              => 10,
1443             authorised_value => 'branches',
1444         }
1445     )->store;
1446
1447     my $biblio = $builder->build_sample_biblio({ frameworkcode => '' });
1448     my $item = $builder->build_sample_item({ biblionumber => $biblio->biblionumber });
1449     my $lost_av = $builder->build_object({ class => 'Koha::AuthorisedValues', value => { category => 'LOST', authorised_value => '42' }});
1450     my $dateaccessioned = '2020-12-15';
1451     my $library = Koha::Libraries->search->next;
1452     my $branchcode = $library->branchcode;
1453
1454     my $some_marc_xml = qq{<?xml version="1.0" encoding="UTF-8"?>
1455 <collection
1456   xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
1457   xsi:schemaLocation="http://www.loc.gov/MARC21/slim http://www.loc.gov/standards/marcxml/schema/MARC21slim.xsd"
1458   xmlns="http://www.loc.gov/MARC21/slim">
1459
1460 <record>
1461   <leader>         a              </leader>
1462   <datafield tag="999" ind1=" " ind2=" ">
1463     <subfield code="é">value Ã©</subfield>
1464     <subfield code="è">$branchcode</subfield>
1465   </datafield>
1466 </record>
1467
1468 </collection>};
1469
1470     $item->update(
1471         {
1472             itemlost           => $lost_av->authorised_value,
1473             dateaccessioned    => $dateaccessioned,
1474             more_subfields_xml => $some_marc_xml,
1475         }
1476     );
1477
1478     Koha::Caches->get_instance->flush_all;
1479
1480     $item = $item->get_from_storage;
1481
1482     my $s = $item->columns_to_str;
1483     is( $s->{itemlost}, $lost_av->lib, 'Attributes linked with AV replaced with description' );
1484     is( $s->{dateaccessioned}, '2020-12-15', 'Date attributes iso formatted');
1485     is( $s->{'é'}, 'value Ã©', 'subfield ok with more than a-Z');
1486     is( $s->{'è'}, $library->branchname );
1487
1488     $cache->clear_from_cache("MarcStructure-0-");
1489     $cache->clear_from_cache("MarcStructure-1-");
1490     $cache->clear_from_cache("MarcSubfieldStructure-");
1491     $cache->clear_from_cache("libraries:name");
1492     $cache->clear_from_cache("itemtype:description:en");
1493     $cache->clear_from_cache("cn_sources:description");
1494     $cache->clear_from_cache("AV_descriptions:LOST");
1495
1496     $schema->storage->txn_rollback;
1497 };
1498
1499 subtest 'strings_map() tests' => sub {
1500
1501     plan tests => 6;
1502
1503     $schema->storage->txn_begin;
1504
1505     my ( $itemtag, $itemsubfield ) = C4::Biblio::GetMarcFromKohaField("items.itemnumber");
1506
1507     my $cache = Koha::Caches->get_instance();
1508     $cache->clear_from_cache("MarcStructure-0-");
1509     $cache->clear_from_cache("MarcStructure-1-");
1510     $cache->clear_from_cache("MarcSubfieldStructure-");
1511     $cache->clear_from_cache("libraries:name");
1512     $cache->clear_from_cache("itemtype:description:en");
1513     $cache->clear_from_cache("cn_sources:description");
1514     $cache->clear_from_cache("AV_descriptions:LOST");
1515
1516     # Recreating subfields just to be sure tests will be ok
1517     # 1 => av (LOST)
1518     # 3 => no link
1519     # a => branches
1520     # y => itemtypes
1521     Koha::MarcSubfieldStructures->search(
1522         {
1523             frameworkcode => '',
1524             tagfield      => $itemtag,
1525             tagsubfield   => [ '1', '2', '3', 'a', 'y' ],
1526         }
1527     )->delete;    # In case it exist already
1528
1529     Koha::MarcSubfieldStructure->new(
1530         {
1531             authorised_value => 'LOST',
1532             defaultvalue     => '',
1533             frameworkcode    => '',
1534             kohafield        => 'items.itemlost',
1535             repeatable       => 1,
1536             tab              => 10,
1537             tagfield         => $itemtag,
1538             tagsubfield      => '1',
1539         }
1540     )->store;
1541     Koha::MarcSubfieldStructure->new(
1542         {
1543             authorised_value => 'cn_source',
1544             defaultvalue     => '',
1545             frameworkcode    => '',
1546             kohafield        => 'items.cn_source',
1547             repeatable       => 1,
1548             tab              => 10,
1549             tagfield         => $itemtag,
1550             tagsubfield      => '2',
1551         }
1552     )->store;
1553     Koha::MarcSubfieldStructure->new(
1554         {
1555             authorised_value => '',
1556             defaultvalue     => '',
1557             frameworkcode    => '',
1558             kohafield        => 'items.materials',
1559             repeatable       => 1,
1560             tab              => 10,
1561             tagfield         => $itemtag,
1562             tagsubfield      => '3',
1563         }
1564     )->store;
1565     Koha::MarcSubfieldStructure->new(
1566         {
1567             authorised_value => 'branches',
1568             defaultvalue     => '',
1569             frameworkcode    => '',
1570             kohafield        => 'items.homebranch',
1571             repeatable       => 1,
1572             tab              => 10,
1573             tagfield         => $itemtag,
1574             tagsubfield      => 'a',
1575         }
1576     )->store;
1577     Koha::MarcSubfieldStructure->new(
1578         {
1579             authorised_value => 'itemtypes',
1580             defaultvalue     => '',
1581             frameworkcode    => '',
1582             kohafield        => 'items.itype',
1583             repeatable       => 1,
1584             tab              => 10,
1585             tagfield         => $itemtag,
1586             tagsubfield      => 'y',
1587         }
1588     )->store;
1589
1590     my $itype   = $builder->build_object( { class => 'Koha::ItemTypes' } );
1591     my $library = $builder->build_object( { class => 'Koha::Libraries' } );
1592     my $biblio  = $builder->build_sample_biblio( { frameworkcode => '' } );
1593     my $item    = $builder->build_sample_item(
1594         {
1595             biblionumber => $biblio->id,
1596             library      => $library->id
1597         }
1598     );
1599
1600     Koha::AuthorisedValues->search( { authorised_value => 3, category => 'LOST' } )->delete;
1601     my $lost_av = $builder->build_object(
1602         {
1603             class => 'Koha::AuthorisedValues',
1604             value => {
1605                 authorised_value => 3,
1606                 category         => 'LOST',
1607                 lib              => 'internal description',
1608                 lib_opac         => 'public description',
1609             }
1610         }
1611     );
1612
1613     my $class_sort_rule  = $builder->build_object( { class => 'Koha::ClassSortRules', value => { sort_routine => 'Generic' } } );
1614     my $class_split_rule = $builder->build_object( { class => 'Koha::ClassSplitRules' } );
1615     my $class_source     = $builder->build_object(
1616         {
1617             class => 'Koha::ClassSources',
1618             value => {
1619                 class_sort_rule  => $class_sort_rule->class_sort_rule,
1620                 class_split_rule => $class_split_rule->class_split_rule,
1621             }
1622         }
1623     )->store();
1624
1625     Koha::Caches->get_instance->flush_all;
1626
1627     $item->set(
1628         {
1629             cn_source => $class_source->id,
1630             itemlost  => $lost_av->authorised_value,
1631             itype     => $itype->itemtype,
1632             materials => 'Suff',
1633         }
1634     )->store->discard_changes;
1635
1636     my $strings = $item->strings_map;
1637
1638     subtest 'unmapped field tests' => sub {
1639
1640         plan tests => 1;
1641
1642         ok( !exists $strings->{materials}, "Unmapped field not present" );
1643     };
1644
1645     subtest 'av handling' => sub {
1646
1647         plan tests => 4;
1648
1649         ok( exists $strings->{itemlost}, "'itemlost' entry exists" );
1650         is( $strings->{itemlost}->{str},      $lost_av->lib, "'str' set to av->lib" );
1651         is( $strings->{itemlost}->{type},     'av',          "'type' is 'av'" );
1652         is( $strings->{itemlost}->{category}, 'LOST',        "'category' exists and set to 'LOST'" );
1653     };
1654
1655     subtest 'cn_source handling' => sub {
1656
1657         plan tests => 3;
1658
1659         ok( exists $strings->{cn_source}, "'cn_source' entry exists" );
1660         is( $strings->{cn_source}->{str},  $class_source->description,    "'str' set to \$class_source->description" );
1661         is( $strings->{cn_source}->{type}, 'call_number_source', "type is 'library'" );
1662     };
1663
1664     subtest 'branches handling' => sub {
1665
1666         plan tests => 3;
1667
1668         ok( exists $strings->{homebranch}, "'homebranch' entry exists" );
1669         is( $strings->{homebranch}->{str},  $library->branchname, "'str' set to 'branchname'" );
1670         is( $strings->{homebranch}->{type}, 'library',            "type is 'library'" );
1671     };
1672
1673     subtest 'itemtype handling' => sub {
1674
1675         plan tests => 3;
1676
1677         ok( exists $strings->{itype}, "'itype' entry exists" );
1678         is( $strings->{itype}->{str},  $itype->description, "'str' correctly set" );
1679         is( $strings->{itype}->{type}, 'item_type',         "'type' is 'item_type'" );
1680     };
1681
1682     subtest 'public flag tests' => sub {
1683
1684         plan tests => 4;
1685
1686         $strings = $item->strings_map( { public => 1 } );
1687
1688         ok( exists $strings->{itemlost}, "'itemlost' entry exists" );
1689         is( $strings->{itemlost}->{str},      $lost_av->lib_opac, "'str' set to av->lib" );
1690         is( $strings->{itemlost}->{type},     'av',               "'type' is 'av'" );
1691         is( $strings->{itemlost}->{category}, 'LOST',             "'category' exists and set to 'LOST'" );
1692     };
1693
1694     $cache->clear_from_cache("MarcStructure-0-");
1695     $cache->clear_from_cache("MarcStructure-1-");
1696     $cache->clear_from_cache("MarcSubfieldStructure-");
1697     $cache->clear_from_cache("libraries:name");
1698     $cache->clear_from_cache("itemtype:description:en");
1699     $cache->clear_from_cache("cn_sources:description");
1700
1701     $schema->storage->txn_rollback;
1702 };
1703
1704 subtest 'store() tests' => sub {
1705
1706     plan tests => 3;
1707
1708     subtest 'dateaccessioned handling' => sub {
1709
1710         plan tests => 3;
1711
1712         $schema->storage->txn_begin;
1713
1714         my $item = $builder->build_sample_item;
1715
1716         ok( defined $item->dateaccessioned, 'dateaccessioned is set' );
1717
1718         # reset dateaccessioned on the DB
1719         $schema->resultset('Item')->find({ itemnumber => $item->id })->update({ dateaccessioned => undef });
1720         $item->discard_changes;
1721
1722         ok( !defined $item->dateaccessioned );
1723
1724         # update something
1725         $item->replacementprice(100)->store->discard_changes;
1726
1727         ok( !defined $item->dateaccessioned, 'dateaccessioned not set on update if undefined' );
1728
1729         $schema->storage->txn_rollback;
1730     };
1731
1732     subtest '_set_found_trigger() tests' => sub {
1733
1734         plan tests => 9;
1735
1736         $schema->storage->txn_begin;
1737
1738         my $patron = $builder->build_object({ class => 'Koha::Patrons' });
1739         my $item   = $builder->build_sample_item({ itemlost => 1, itemlost_on => dt_from_string() });
1740
1741         # Add a lost item debit
1742         my $debit = $patron->account->add_debit(
1743             {
1744                 amount    => 10,
1745                 type      => 'LOST',
1746                 item_id   => $item->id,
1747                 interface => 'intranet',
1748             }
1749         );
1750
1751         # Add a lost item processing fee
1752         my $processing_debit = $patron->account->add_debit(
1753             {
1754                 amount    => 2,
1755                 type      => 'PROCESSING',
1756                 item_id   => $item->id,
1757                 interface => 'intranet',
1758             }
1759         );
1760
1761         my $lostreturn_policy = {
1762             lostreturn       => 'charge',
1763             processingreturn => 'refund'
1764         };
1765
1766         my $mocked_circ_rules = Test::MockModule->new('Koha::CirculationRules');
1767         $mocked_circ_rules->mock( 'get_lostreturn_policy', sub { return $lostreturn_policy; } );
1768
1769         # simulate it was found
1770         $item->set( { itemlost => 0 } )->store;
1771
1772         my $messages = $item->object_messages;
1773
1774         my $message_1 = $messages->[0];
1775
1776         is( $message_1->type,    'info',          'type is correct' );
1777         is( $message_1->message, 'lost_refunded', 'message is correct' );
1778
1779         # Find the refund credit
1780         my $credit = $debit->credits->next;
1781
1782         is_deeply(
1783             $message_1->payload,
1784             { credit_id => $credit->id },
1785             'type is correct'
1786         );
1787
1788         my $message_2 = $messages->[1];
1789
1790         is( $message_2->type,    'info',        'type is correct' );
1791         is( $message_2->message, 'lost_charge', 'message is correct' );
1792         is( $message_2->payload, undef,         'no payload' );
1793
1794         my $message_3 = $messages->[2];
1795         is( $message_3->message, 'processing_refunded', 'message is correct' );
1796
1797         my $processing_credit = $processing_debit->credits->next;
1798         is_deeply(
1799             $message_3->payload,
1800             { credit_id => $processing_credit->id },
1801             'type is correct'
1802         );
1803
1804         # Let's build a new item
1805         $item   = $builder->build_sample_item({ itemlost => 1, itemlost_on => dt_from_string() });
1806         $item->set( { itemlost => 0 } )->store;
1807
1808         $messages = $item->object_messages;
1809         is( scalar @{$messages}, 0, 'This item has no history, no associated lost fines, presumed not lost by patron, no messages returned');
1810
1811         $schema->storage->txn_rollback;
1812     };
1813
1814     subtest 'holds_queue update tests' => sub {
1815
1816         plan tests => 2;
1817
1818         $schema->storage->txn_begin;
1819
1820         my $biblio = $builder->build_sample_biblio;
1821
1822         my $mock = Test::MockModule->new('Koha::BackgroundJob::BatchUpdateBiblioHoldsQueue');
1823         $mock->mock( 'enqueue', sub {
1824             my ( $self, $args ) = @_;
1825             is_deeply(
1826                 $args->{biblio_ids},
1827                 [ $biblio->id ],
1828                 '->store triggers a holds queue update for the related biblio'
1829             );
1830         } );
1831
1832         t::lib::Mocks::mock_preference( 'RealTimeHoldsQueue', 1 );
1833
1834         # new item
1835         my $item = $builder->build_sample_item({ biblionumber => $biblio->id });
1836
1837         # updated item
1838         $item->set({ reserves => 1 })->store;
1839
1840         t::lib::Mocks::mock_preference( 'RealTimeHoldsQueue', 0 );
1841         # updated item
1842         $item->set({ reserves => 0 })->store;
1843
1844         $schema->storage->txn_rollback;
1845     };
1846 };
1847
1848 subtest 'Recalls tests' => sub {
1849
1850     plan tests => 22;
1851
1852     $schema->storage->txn_begin;
1853
1854     my $item1 = $builder->build_sample_item;
1855     my $biblio = $item1->biblio;
1856     my $branchcode = $item1->holdingbranch;
1857     my $patron1 = $builder->build_object({ class => 'Koha::Patrons', value => { branchcode => $branchcode } });
1858     my $patron2 = $builder->build_object({ class => 'Koha::Patrons', value => { branchcode => $branchcode } });
1859     my $patron3 = $builder->build_object({ class => 'Koha::Patrons', value => { branchcode => $branchcode } });
1860     my $item2 = $builder->build_object(
1861         {   class => 'Koha::Items',
1862             value => { holdingbranch => $branchcode, homebranch => $branchcode, biblionumber => $biblio->biblionumber, itype => $item1->effective_itemtype }
1863         }
1864     );
1865
1866     t::lib::Mocks::mock_userenv( { patron => $patron1 } );
1867     t::lib::Mocks::mock_preference('UseRecalls', 1);
1868
1869     my $recall1 = Koha::Recall->new(
1870         {   patron_id         => $patron1->borrowernumber,
1871             created_date      => \'NOW()',
1872             biblio_id         => $biblio->biblionumber,
1873             pickup_library_id => $branchcode,
1874             item_id           => $item1->itemnumber,
1875             expiration_date   => undef,
1876             item_level        => 1
1877         }
1878     )->store;
1879     my $recall2 = Koha::Recall->new(
1880         {   patron_id         => $patron2->borrowernumber,
1881             created_date      => \'NOW()',
1882             biblio_id         => $biblio->biblionumber,
1883             pickup_library_id => $branchcode,
1884             item_id           => $item1->itemnumber,
1885             expiration_date   => undef,
1886             item_level        => 1
1887         }
1888     )->store;
1889
1890     is( $item1->recall->patron_id, $patron1->borrowernumber, 'Correctly returns most relevant recall' );
1891
1892     $recall2->set_cancelled;
1893
1894     t::lib::Mocks::mock_preference('UseRecalls', 0);
1895     is( $item1->can_be_recalled({ patron => $patron1 }), 0, "Can't recall with UseRecalls disabled" );
1896
1897     t::lib::Mocks::mock_preference("UseRecalls", 1);
1898
1899     $item1->update({ notforloan => 1 });
1900     is( $item1->can_be_recalled({ patron => $patron1 }), 0, "Can't recall that is not for loan" );
1901     $item1->update({ notforloan => 0, itemlost => 1 });
1902     is( $item1->can_be_recalled({ patron => $patron1 }), 0, "Can't recall that is marked lost" );
1903     $item1->update({ itemlost => 0, withdrawn => 1 });
1904     is( $item1->can_be_recalled({ patron => $patron1 }), 0, "Can't recall that is withdrawn" );
1905     is( $item1->can_be_recalled({ patron => $patron1 }), 0, "Can't recall item if not checked out" );
1906
1907     $item1->update({ withdrawn => 0 });
1908     C4::Circulation::AddIssue( $patron2->unblessed, $item1->barcode );
1909
1910     Koha::CirculationRules->set_rules({
1911         branchcode => $branchcode,
1912         categorycode => $patron1->categorycode,
1913         itemtype => $item1->effective_itemtype,
1914         rules => {
1915             recalls_allowed => 0,
1916             recalls_per_record => 1,
1917             on_shelf_recalls => 'all',
1918         },
1919     });
1920     is( $item1->can_be_recalled({ patron => $patron1 }), 0, "Can't recall if recalls_allowed = 0" );
1921
1922     Koha::CirculationRules->set_rules({
1923         branchcode => $branchcode,
1924         categorycode => $patron1->categorycode,
1925         itemtype => $item1->effective_itemtype,
1926         rules => {
1927             recalls_allowed => 1,
1928             recalls_per_record => 1,
1929             on_shelf_recalls => 'all',
1930         },
1931     });
1932     is( $item1->can_be_recalled({ patron => $patron1 }), 0, "Can't recall if patron has more existing recall(s) than recalls_allowed" );
1933     is( $item1->can_be_recalled({ patron => $patron1 }), 0, "Can't recall if patron has more existing recall(s) than recalls_per_record" );
1934     is( $item1->can_be_recalled({ patron => $patron1 }), 0, "Can't recall if patron has already recalled this item" );
1935
1936     my $reserve_id = C4::Reserves::AddReserve({ branchcode => $branchcode, borrowernumber => $patron1->borrowernumber, biblionumber => $item1->biblionumber, itemnumber => $item1->itemnumber });
1937     is( $item1->can_be_recalled({ patron => $patron1 }), 0, "Can't recall item if patron has already reserved it" );
1938     C4::Reserves::ModReserve({ rank => 'del', reserve_id => $reserve_id, branchcode => $branchcode, itemnumber => $item1->itemnumber, borrowernumber => $patron1->borrowernumber, biblionumber => $item1->biblionumber });
1939
1940     $recall1->set_cancelled;
1941     is( $item1->can_be_recalled({ patron => $patron2 }), 0, "Can't recall if patron has already checked out an item attached to this biblio" );
1942
1943     is( $item1->can_be_recalled({ patron => $patron1 }), 0, "Can't recall if on_shelf_recalls = all and items are still available" );
1944
1945     Koha::CirculationRules->set_rules({
1946         branchcode => $branchcode,
1947         categorycode => $patron1->categorycode,
1948         itemtype => $item1->effective_itemtype,
1949         rules => {
1950             recalls_allowed => 1,
1951             recalls_per_record => 1,
1952             on_shelf_recalls => 'any',
1953         },
1954     });
1955     C4::Circulation::AddReturn( $item1->barcode, $branchcode );
1956     is( $item1->can_be_recalled({ patron => $patron1 }), 0, "Can't recall if no items are checked out" );
1957
1958     C4::Circulation::AddIssue( $patron2->unblessed, $item1->barcode );
1959     is( $item1->can_be_recalled({ patron => $patron1 }), 1, "Can recall item" );
1960
1961     $recall1 = Koha::Recall->new(
1962         {   patron_id         => $patron1->borrowernumber,
1963             created_date      => \'NOW()',
1964             biblio_id         => $biblio->biblionumber,
1965             pickup_library_id => $branchcode,
1966             item_id           => undef,
1967             expiration_date   => undef,
1968             item_level        => 0
1969         }
1970     )->store;
1971
1972     # Patron2 has Item1 checked out. Patron1 has placed a biblio-level recall on Biblio1, so check if Item1 can fulfill Patron1's recall.
1973
1974     Koha::CirculationRules->set_rules({
1975         branchcode => $branchcode,
1976         categorycode => $patron1->categorycode,
1977         itemtype => $item1->effective_itemtype,
1978         rules => {
1979             recalls_allowed => 0,
1980             recalls_per_record => 1,
1981             on_shelf_recalls => 'any',
1982         },
1983     });
1984     is( $item1->can_be_waiting_recall, 0, "Recalls not allowed for this itemtype" );
1985
1986     Koha::CirculationRules->set_rules({
1987         branchcode => $branchcode,
1988         categorycode => $patron1->categorycode,
1989         itemtype => $item1->effective_itemtype,
1990         rules => {
1991             recalls_allowed => 1,
1992             recalls_per_record => 1,
1993             on_shelf_recalls => 'any',
1994         },
1995     });
1996     is( $item1->can_be_waiting_recall, 1, "Recalls are allowed for this itemtype" );
1997
1998     # check_recalls tests
1999
2000     $recall1 = Koha::Recall->new(
2001         {   patron_id         => $patron2->borrowernumber,
2002             created_date      => \'NOW()',
2003             biblio_id         => $biblio->biblionumber,
2004             pickup_library_id => $branchcode,
2005             item_id           => $item1->itemnumber,
2006             expiration_date   => undef,
2007             item_level        => 1
2008         }
2009     )->store;
2010     $recall2 = Koha::Recall->new(
2011         {   patron_id         => $patron1->borrowernumber,
2012             created_date      => \'NOW()',
2013             biblio_id         => $biblio->biblionumber,
2014             pickup_library_id => $branchcode,
2015             item_id           => undef,
2016             expiration_date   => undef,
2017             item_level        => 0
2018         }
2019     )->store;
2020     $recall2->set_waiting( { item => $item1 } );
2021     is( $item1->has_pending_recall, 1, 'Item has pending recall' );
2022
2023     # return a waiting recall
2024     my $check_recall = $item1->check_recalls;
2025     is( $check_recall->patron_id, $patron1->borrowernumber, "Waiting recall is highest priority and returned" );
2026
2027     $recall2->revert_waiting;
2028
2029     is( $item1->has_pending_recall, 0, 'Item does not have pending recall' );
2030
2031     # return recall based on recalldate
2032     $check_recall = $item1->check_recalls;
2033     is( $check_recall->patron_id, $patron1->borrowernumber, "No waiting recall, so oldest recall is returned" );
2034
2035     $recall1->set_cancelled;
2036
2037     # return a biblio-level recall
2038     $check_recall = $item1->check_recalls;
2039     is( $check_recall->patron_id, $patron1->borrowernumber, "Only remaining recall is returned" );
2040
2041     $recall2->set_cancelled;
2042
2043     $schema->storage->txn_rollback;
2044 };
2045
2046 subtest 'Notforloan tests' => sub {
2047
2048     plan tests => 3;
2049
2050     $schema->storage->txn_begin;
2051
2052     my $item1 = $builder->build_sample_item;
2053     $item1->update({ notforloan => 0 });
2054     $item1->itemtype->notforloan(0);
2055     is ( $item1->is_notforloan, 0, 'Notforloan is correctly false by item status and item type');
2056     $item1->update({ notforloan => 1 });
2057     is ( $item1->is_notforloan, 1, 'Notforloan is correctly true by item status');
2058     $item1->update({ notforloan => 0 });
2059     $item1->itemtype->update({ notforloan => 1 });
2060     is ( $item1->is_notforloan, 1, 'Notforloan is correctly true by item type');
2061
2062     $schema->storage->txn_rollback;
2063 };
2064
2065 subtest 'item_group() tests' => sub {
2066
2067     plan tests => 4;
2068
2069     $schema->storage->txn_begin;
2070
2071     my $biblio = $builder->build_sample_biblio();
2072     my $item_1 = $builder->build_sample_item({ biblionumber => $biblio->biblionumber });
2073     my $item_2 = $builder->build_sample_item({ biblionumber => $biblio->biblionumber });
2074
2075     is( $item_1->item_group, undef, 'Item 1 has no item group');
2076     is( $item_2->item_group, undef, 'Item 2 has no item group');
2077
2078     my $item_group_1 = Koha::Biblio::ItemGroup->new( { biblio_id => $biblio->id } )->store();
2079     my $item_group_2 = Koha::Biblio::ItemGroup->new( { biblio_id => $biblio->id } )->store();
2080
2081     $item_group_1->add_item({ item_id => $item_1->id });
2082     $item_group_2->add_item({ item_id => $item_2->id });
2083
2084     is( $item_1->item_group->id, $item_group_1->id, 'Got item group 1 correctly' );
2085     is( $item_2->item_group->id, $item_group_2->id, 'Got item group 2 correctly' );
2086
2087     $schema->storage->txn_rollback;
2088 };
2089
2090 subtest 'has_pending_recall() tests' => sub {
2091
2092     plan tests => 2;
2093
2094     $schema->storage->txn_begin;
2095
2096     my $library = $builder->build_object({ class => 'Koha::Libraries' });
2097     my $item    = $builder->build_sample_item;
2098     my $patron  = $builder->build_object({ class => 'Koha::Patrons' });
2099
2100     t::lib::Mocks::mock_userenv({ branchcode => $library->branchcode });
2101     t::lib::Mocks::mock_preference( 'UseRecalls', 1 );
2102
2103     C4::Circulation::AddIssue( $patron->unblessed, $item->barcode );
2104
2105     my ($recall) = Koha::Recalls->add_recall({ biblio => $item->biblio, item => $item, patron => $patron });
2106
2107     ok( !$item->has_pending_recall, 'The item has no pending recalls' );
2108
2109     $recall->status('waiting')->store;
2110
2111     ok( $item->has_pending_recall, 'The item has a pending recall' );
2112
2113     $schema->storage->txn_rollback;
2114 };
2115
2116 subtest 'is_denied_renewal' => sub {
2117     plan tests => 11;
2118
2119     $schema->storage->txn_begin;
2120
2121     my $library = $builder->build_object({ class => 'Koha::Libraries'});
2122
2123     my $deny_book = $builder->build_object({ class => 'Koha::Items', value => {
2124         homebranch => $library->branchcode,
2125         withdrawn => 1,
2126         itype => 'HIDE',
2127         location => 'PROC',
2128         itemcallnumber => undef,
2129         itemnotes => "",
2130         }
2131     });
2132
2133     my $allow_book = $builder->build_object({ class => 'Koha::Items', value => {
2134         homebranch => $library->branchcode,
2135         withdrawn => 0,
2136         itype => 'NOHIDE',
2137         location => 'NOPROC'
2138         }
2139     });
2140
2141     my $idr_rules = "";
2142     C4::Context->set_preference('ItemsDeniedRenewal', $idr_rules);
2143     is( $deny_book->is_denied_renewal, 0, 'Renewal allowed when no rules' );
2144
2145     # The wrong column delete should be silently ignored and not trigger item delete
2146     $idr_rules="delete: [yes]\nwithdrawn: [1]";
2147     C4::Context->set_preference('ItemsDeniedRenewal', $idr_rules);
2148     is( $deny_book->is_denied_renewal, 1, 'Renewal blocked when 1 rules (withdrawn)' );
2149     is( $allow_book->is_denied_renewal, 0, 'Renewal allowed when 1 rules not matched (withdrawn)' );
2150
2151     $idr_rules="withdrawn: [1]\nitype: [HIDE,INVISIBLE]";
2152     is( $deny_book->is_denied_renewal, 1, 'Renewal blocked when 2 rules matched (withdrawn, itype)' );
2153     is( $allow_book->is_denied_renewal, 0, 'Renewal allowed when 2 rules not matched (withdrawn, itype)' );
2154
2155     $idr_rules="withdrawn: [1]\nitype: [HIDE,INVISIBLE]\nlocation: [PROC]";
2156     C4::Context->set_preference('ItemsDeniedRenewal', $idr_rules);
2157     is( $deny_book->is_denied_renewal, 1, 'Renewal blocked when 3 rules matched (withdrawn, itype, location)' );
2158     is( $allow_book->is_denied_renewal, 0, 'Renewal allowed when 3 rules not matched (withdrawn, itype, location)' );
2159
2160     $idr_rules="itemcallnumber: [null]";
2161     C4::Context->set_preference('ItemsDeniedRenewal', $idr_rules);
2162     is( $deny_book->is_denied_renewal, 1, 'Renewal blocked for undef when null in pref' );
2163
2164     $idr_rules="itemcallnumber: ['']";
2165     C4::Context->set_preference('ItemsDeniedRenewal', $idr_rules);
2166     is( $deny_book->is_denied_renewal, 0, 'Renewal not blocked for undef when "" in pref' );
2167
2168     $idr_rules="itemnotes: [null]";
2169     C4::Context->set_preference('ItemsDeniedRenewal', $idr_rules);
2170     is( $deny_book->is_denied_renewal, 0, 'Renewal not blocked for "" when null in pref' );
2171
2172     $idr_rules="itemnotes: ['']";
2173     C4::Context->set_preference('ItemsDeniedRenewal', $idr_rules);
2174     is( $deny_book->is_denied_renewal, 1, 'Renewal blocked for empty string when "" in pref' );
2175
2176     $schema->storage->txn_rollback;
2177 };
2178
2179 subtest 'current_branchtransfers relationship' => sub {
2180     plan tests => 3;
2181
2182     $schema->storage->txn_begin;
2183
2184     my $biblio = $builder->build_sample_biblio();
2185     my $item   = $builder->build_sample_item(
2186         {
2187             biblionumber => $biblio->biblionumber,
2188         }
2189     );
2190     my $transfers = $item->_result->current_branchtransfers;
2191     is( ref($transfers), 'DBIx::Class::ResultSet',
2192         'current_branchtransfers returns a DBIx::Class::ResultSet' );
2193     is( $transfers->count, 0,
2194         "Empty Koha::Item::Transfers set returned if no return_claims" );
2195     my $transfer1 = $builder->build(
2196         {
2197             source => 'Branchtransfer',
2198             value  => {
2199                 itemnumber  => $item->itemnumber,
2200                 datearrived => dt_from_string,
2201             }
2202         }
2203     );
2204     my $transfer2 = $builder->build(
2205         {
2206             source => 'Branchtransfer',
2207             value  => {
2208                 itemnumber    => $item->itemnumber,
2209                 datearrived   => undef,
2210                 datecancelled => dt_from_string,
2211             }
2212         }
2213     );
2214     my $transfer3 = $builder->build(
2215         {
2216             source => 'Branchtransfer',
2217             value  => {
2218                 itemnumber    => $item->itemnumber,
2219                 datearrived   => undef,
2220                 datecancelled => undef,
2221             }
2222         }
2223     );
2224
2225     is( $item->_result->current_branchtransfers()->count,
2226         1, "One transfer found for item" );
2227
2228     $schema->storage->txn_rollback;
2229 };