Bug 22284: Opac pickup_locations
authorAgustin Moyano <agustinmoyano@theke.io>
Fri, 12 Apr 2019 10:32:48 +0000 (10:32 +0000)
committerMartin Renvoize <martin.renvoize@ptfs-europe.com>
Fri, 3 Jan 2020 12:58:04 +0000 (12:58 +0000)
This patch modifies Koha::Libraries->pickup_location and moves most of the logic to
Koha::Item and Koha::Biblio in preparation for api endpoints in the future.

There where 2 methods added

1) Koha::Item->pickup_locations that given a patron, returns all pickup locations of
this item, considering hold fulfillment rules, and hold allowed rules.

2) Koha::Biblio->pickup_locations that given a patron, returns a distinct list of
libraries returned by each of this biblio items pickup location.

Koha::Libraries->pickup_location analyzes input param and calls Koha::Item->pickup_locations
or Koha::Biblio->pickup_locations as needed.

Also in opac-reserve.tt the way options where obtained to fill the pickup location select
was modified to pass the patron as a parameter.

To test:
1) opac: try to place hold on a item and check that all libraries are shown in the
pickup location select.
2) intranet: in Library groups, add 2 root groups marked as local hold group and
add different libraries to each.
3) opac: login as a user of a library belonging to one hold group, and search try to
place a hold on an item belongin to the other hold group.
4) intranet: in Circulation and fines rules, play with 'Hold policy' and 'Hold pickup
library match' rules.
5) opac: On each modification of the rules reload the page.
SUCCESS => Every time you reload the page, the number of pickup locations showed in
select varies.
6) prove t/db_dependent/Koha/Biblios.t t/db_dependent/Koha/Items.t
SUCCESS => Result: PASS
7) Sign off

Sponsored-by: VOKAL
Signed-off-by: Josef Moravec <josef.moravec@gmail.com>
Signed-off-by: Liz Rea <wizzyrea@gmail.com>
Signed-off-by: Tomas Cohen Arazi <tomascohen@theke.io>
Signed-off-by: Martin Renvoize <martin.renvoize@ptfs-europe.com>

Koha/Biblio.pm
Koha/Item.pm
Koha/Libraries.pm
koha-tmpl/opac-tmpl/bootstrap/en/modules/opac-reserve.tt
t/db_dependent/Koha/Biblios.t
t/db_dependent/Koha/Items.t

index b5ffe81..8fdc253 100644 (file)
@@ -169,6 +169,33 @@ sub can_be_transferred {
     return 0;
 }
 
+
+=head3 pickup_locations
+
+@pickup_locations = $biblio->pickup_locations( {patron => $patron } )
+
+Returns possible pickup locations for this biblio items, according to patron's home library (if patron is defined and holds are allowed only from hold groups)
+and if item can be transfered to each pickup location.
+
+=cut
+
+sub pickup_locations {
+    my ($self, $params) = @_;
+
+    my $patron = $params->{patron};
+
+    my @pickup_locations;
+    foreach my $item_of_bib ($self->items) {
+        push @pickup_locations, $item_of_bib->pickup_locations( {patron => $patron} );
+    }
+
+    my %seen;
+    @pickup_locations =
+      grep { !$seen{ $_->{branchcode} }++ } @pickup_locations;
+
+    return wantarray ? @pickup_locations : \@pickup_locations;
+}
+
 =head3 hidden_in_opac
 
 my $bool = $biblio->hidden_in_opac({ [ rules => $rules ] })
index b557c95..090826d 100644 (file)
@@ -26,7 +26,7 @@ use Koha::Database;
 use Koha::DateUtils qw( dt_from_string );
 
 use C4::Context;
-
+use C4::Circulation;
 use Koha::Checkouts;
 use Koha::IssuingRules;
 use Koha::Item::Transfer::Limits;
@@ -294,6 +294,59 @@ sub can_be_transferred {
     })->count ? 0 : 1;
 }
 
+=head3 pickup_locations
+
+@pickup_locations = $item->pickup_locations( {patron => $patron } )
+
+Returns possible pickup locations for this item, according to patron's home library (if patron is defined and holds are allowed only from hold groups)
+and if item can be transfered to each pickup location.
+
+=cut
+
+sub pickup_locations {
+    my ($self, $params) = @_;
+
+    my $patron = $params->{patron};
+
+    my $circ_control_branch =
+      C4::Circulation::_GetCircControlBranch( $self->unblessed(), $patron );
+    my $branchitemrule =
+      C4::Circulation::GetBranchItemRule( $circ_control_branch, $self->itype );
+
+    my $branch_control = C4::Context->preference('HomeOrHoldingBranch');
+    my $library = $branch_control eq 'holdingbranch' ? $self->holding_branch : $self->home_branch;
+
+    #warn $branch_control.' '.$branchitemrule->{holdallowed}.' '.$library->branchcode.' '.$patron->branchcode;
+
+    my @libs;
+    if(defined $patron) {
+        return @libs if $branchitemrule->{holdallowed} == 3 && !$library->validate_hold_sibling( {branchcode => $patron->branchcode} );
+        return @libs if $branchitemrule->{holdallowed} == 1 && $library->branchcode ne $patron->branchcode;
+    }
+
+    if ($branchitemrule->{hold_fulfillment_policy} eq 'holdgroup') {
+        @libs  = $library->get_hold_libraries;
+    } elsif ($branchitemrule->{hold_fulfillment_policy} eq 'homebranch') {
+        push @libs, $self->home_branch;
+    } elsif ($branchitemrule->{hold_fulfillment_policy} eq 'holdingbranch') {
+        push @libs, $self->holding_branch;
+    } else {
+        @libs = Koha::Libraries->search({
+            pickup_location => 1
+        }, {
+            order_by => ['branchname']
+        })->as_list;
+    }
+
+    my @pickup_locations;
+    foreach my $library (@libs) {
+        if ($library->pickup_location && $self->can_be_transferred({ to => $library })) {
+            push @pickup_locations, $library->unblessed;
+        }
+    }
+    return wantarray ? @pickup_locations : \@pickup_locations;
+}
+
 =head3 article_request_type
 
 my $type = $item->article_request_type( $borrower )
index 56eaf4f..bc92f0b 100644 (file)
@@ -66,12 +66,17 @@ sub pickup_locations {
 
     my $item = $params->{'item'};
     my $biblio = $params->{'biblio'};
+    my $patron = $params->{'patron'};
+
     if ($biblio && $item) {
         Koha::Exceptions::BadParameter->throw(
             error => "Koha::Libraries->pickup_locations takes either 'biblio' or "
             ." 'item' as parameter, but not both."
         );
     }
+    unless (! defined $patron || ref($patron) eq 'Koha::Patron') {
+        $patron = Koha::Items->find($patron);
+    }
 
     # Select libraries that are configured as pickup locations
     my $libraries = $self->search({
@@ -81,32 +86,19 @@ sub pickup_locations {
     });
 
     return $libraries->unblessed unless $item or $biblio;
-    return $libraries->unblessed
-        unless C4::Context->preference('UseBranchTransferLimits');
-    my $limittype = C4::Context->preference('BranchTransferLimitsType');
-
-    if ($item) {
+    if($item) {
         unless (ref($item) eq 'Koha::Item') {
             $item = Koha::Items->find($item);
             return $libraries->unblessed unless $item;
         }
+        return $item->pickup_locations( {patron => $patron} );
     } else {
         unless (ref($biblio) eq 'Koha::Biblio') {
             $biblio = Koha::Biblios->find($biblio);
             return $libraries->unblessed unless $biblio;
         }
+        return $biblio->pickup_locations( {patron => $patron} );
     }
-
-    my @pickup_locations;
-    foreach my $library ($libraries->as_list) {
-        if ($item && $item->can_be_transferred({ to => $library })) {
-            push @pickup_locations, $library->unblessed;
-        } elsif ($biblio && $biblio->can_be_transferred({ to => $library })) {
-            push @pickup_locations, $library->unblessed;
-        }
-    }
-
-    return wantarray ? @pickup_locations : \@pickup_locations;
 }
 
 =head3 search_filtered
index 17fca3c..6932d49 100644 (file)
                                                         <label for="branch_[% bibitemloo.biblionumber | html %]">Pick up location:</label>
                                                         [% UNLESS ( bibitemloo.holdable ) %]
                                                             <select name="branch" id="branch_[% bibitemloo.biblionumber | html %]" disabled="disabled">
-                                                                [% PROCESS options_for_libraries libraries => Branches.pickup_locations({ search_params => { biblio => bibitemloo.biblionumber }, selected => branch }) %]
+                                                                [% PROCESS options_for_libraries libraries => Branches.pickup_locations({ search_params => { biblio => bibitemloo.biblionumber, patron => logged_in_user }, selected => branch }) %]
                                                             </select>
                                                         [% ELSE %]
                                                             [% SET at_least_one_library_not_available_for_pickup = 0 %]
                                                             <select name="branch" id="branch_[% bibitemloo.biblionumber | html %]">
-                                                                [% FOREACH library IN Branches.all({ search_params => { pickup_location => 1 }, selected => branch }) %]
+                                                                [% FOREACH library IN Branches.pickup_locations({ search_params => { biblio => bibitemloo.biblionumber, patron => logged_in_user }, selected => branch }) %]
                                                                     [% SET pickup_available_at = bibitemloo.not_available_at.grep(library.branchcode).size ? 0 : 1 %]
                                                                     [% IF library.selected AND pickup_available_at %]
                                                                         <option value="[% library.branchcode | html %]" selected="selected" >[% library.branchname | html %]</option>
index 2f62524..0d5ac58 100644 (file)
@@ -37,6 +37,8 @@ use t::lib::Mocks;
 my $schema = Koha::Database->new->schema;
 $schema->storage->txn_begin;
 
+my $dbh     = C4::Context->dbh;
+
 my $builder = t::lib::TestBuilder->new;
 my $patron = $builder->build( { source => 'Borrower' } );
 $patron = Koha::Patrons->find( $patron->{borrowernumber} );
@@ -215,3 +217,343 @@ subtest 'custom_cover_image_url' => sub {
 };
 
 $schema->storage->txn_rollback;
+
+
+subtest 'pickup_locations' => sub {
+    plan tests => 25;
+
+    $schema->storage->txn_begin;
+
+    # Cleanup database
+    Koha::Holds->search->delete;
+    Koha::Patrons->search->delete;
+    Koha::Items->search->delete;
+    Koha::Libraries->search->delete;
+    $dbh->do('DELETE FROM issues');
+    $dbh->do('DELETE FROM issuingrules');
+    $dbh->do(
+        q{INSERT INTO issuingrules (categorycode, branchcode, itemtype, reservesallowed)
+        VALUES (?, ?, ?, ?)},
+        {},
+        '*', '*', '*', 25
+    );
+    $dbh->do('DELETE FROM branch_item_rules');
+    $dbh->do('DELETE FROM default_branch_circ_rules');
+    $dbh->do('DELETE FROM default_branch_item_rules');
+    $dbh->do('DELETE FROM default_circ_rules');
+
+    my $root1 = $builder->build_object( { class => 'Koha::Library::Groups', value => { ft_local_hold_group => 1 } } );
+    my $root2 = $builder->build_object( { class => 'Koha::Library::Groups', value => { ft_local_hold_group => 1 } } );
+    my $root3 = $builder->build_object( { class => 'Koha::Library::Groups', value => { ft_local_hold_group => 1 } } );
+
+    my $library1 = $builder->build_object( { class => 'Koha::Libraries', value => { pickup_location => 1 } } );
+    my $library2 = $builder->build_object( { class => 'Koha::Libraries', value => { pickup_location => 1 } } );
+    my $library3 = $builder->build_object( { class => 'Koha::Libraries', value => { pickup_location => 0 } } );
+    my $library4 = $builder->build_object( { class => 'Koha::Libraries', value => { pickup_location => 1 } } );
+    my $library5 = $builder->build_object( { class => 'Koha::Libraries', value => { pickup_location => 1 } } );
+    my $library6 = $builder->build_object( { class => 'Koha::Libraries', value => { pickup_location => 1 } } );
+
+    my $group1_1 = $builder->build_object( { class => 'Koha::Library::Groups', value => { parent_id => $root1->id, branchcode => $library1->branchcode } } );
+    my $group1_2 = $builder->build_object( { class => 'Koha::Library::Groups', value => { parent_id => $root1->id, branchcode => $library2->branchcode } } );
+
+    my $group2_3 = $builder->build_object( { class => 'Koha::Library::Groups', value => { parent_id => $root2->id, branchcode => $library3->branchcode } } );
+    my $group2_4 = $builder->build_object( { class => 'Koha::Library::Groups', value => { parent_id => $root2->id, branchcode => $library4->branchcode } } );
+
+    my $group3_5 = $builder->build_object( { class => 'Koha::Library::Groups', value => { parent_id => $root3->id, branchcode => $library5->branchcode } } );
+    my $group3_6 = $builder->build_object( { class => 'Koha::Library::Groups', value => { parent_id => $root3->id, branchcode => $library6->branchcode } } );
+
+    my $biblio1  = $builder->build_object( { class => 'Koha::Biblios' } );
+    my $biblioitem1 = $builder->build_object( { class => 'Koha::Biblioitems', value => { biblionumber => $biblio1->biblionumber } } );
+    my $biblio2  = $builder->build_object( { class => 'Koha::Biblios' } );
+    my $biblioitem2 = $builder->build_object( { class => 'Koha::Biblioitems', value => { biblionumber => $biblio2->biblionumber } } );
+
+    my $item1_1  = Koha::Item->new({
+        biblionumber     => $biblio1->biblionumber,
+        biblioitemnumber => $biblioitem1->biblioitemnumber,
+        homebranch       => $library1->branchcode,
+        holdingbranch    => $library2->branchcode,
+        itype            => 'test',
+        barcode          => "item11barcode",
+    })->store;
+
+    my $item1_3  = Koha::Item->new({
+        biblionumber     => $biblio1->biblionumber,
+        biblioitemnumber => $biblioitem1->biblioitemnumber,
+        homebranch       => $library3->branchcode,
+        holdingbranch    => $library4->branchcode,
+        itype            => 'test',
+        barcode          => "item13barcode",
+    })->store;
+
+    my $item2_2  = Koha::Item->new({
+        biblionumber     => $biblio2->biblionumber,
+        biblioitemnumber => $biblioitem2->biblioitemnumber,
+        homebranch       => $library2->branchcode,
+        holdingbranch    => $library1->branchcode,
+        itype            => 'test',
+        barcode          => "item22barcode",
+    })->store;
+
+    my $item2_3  = Koha::Item->new({
+        biblionumber     => $biblio2->biblionumber,
+        biblioitemnumber => $biblioitem2->biblioitemnumber,
+        homebranch       => $library3->branchcode,
+        holdingbranch    => $library3->branchcode,
+        itype            => 'test',
+        barcode          => "item23barcode",
+    })->store;
+
+    my $item2_4  = Koha::Item->new({
+        biblionumber     => $biblio2->biblionumber,
+        biblioitemnumber => $biblioitem2->biblioitemnumber,
+        homebranch       => $library4->branchcode,
+        holdingbranch    => $library4->branchcode,
+        itype            => 'test',
+        barcode          => "item24barcode",
+    })->store;
+
+    my $patron1 = $builder->build_object( { class => 'Koha::Patrons', value => { branchcode => $library1->branchcode } } );
+    my $patron4 = $builder->build_object( { class => 'Koha::Patrons', value => { branchcode => $library4->branchcode } } );
+
+    t::lib::Mocks::mock_preference('HomeOrHoldingBranch', 'homebranch');
+
+    #Case 1: holdallowed any, hold_fulfillment_policy any
+    $dbh->do(
+        q{INSERT INTO default_circ_rules (holdallowed, hold_fulfillment_policy, returnbranch)
+        VALUES (?,?,?)},
+        {},
+        2, 'any', 'any'
+    );
+
+    my @pl_1_1 = $biblio1->pickup_locations( { patron => $patron1 } );
+    my @pl_1_4 = $biblio1->pickup_locations( { patron => $patron4 } );
+    my @pl_2_1 = $biblio2->pickup_locations( { patron => $patron1 } );
+    my @pl_2_4 = $biblio2->pickup_locations( { patron => $patron4 } );
+
+
+    ok(scalar(@pl_1_1) == 5 && scalar(@pl_1_4) == 5 && scalar(@pl_2_1) == 5 && scalar(@pl_2_4) == 5, 'Returns all libraries that are pickup locations');
+
+    #Case 2: holdallowed homebranch, hold_fulfillment_policy any, HomeOrHoldingBranch 'homebranch'
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        1, 'any'
+    );
+
+    @pl_1_1 = $biblio1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $biblio1->pickup_locations( { patron => $patron4 } );
+    @pl_2_1 = $biblio2->pickup_locations( { patron => $patron1 } );
+    @pl_2_4 = $biblio2->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_1) == 5 && scalar(@pl_2_4) == 5, 'Returns all libraries that are pickup locations, when item\'s hombebranch equals patron\' homebranch');
+    ok(scalar(@pl_1_4) == 0 && scalar(@pl_2_1) == 0, 'Returns no pickup locations');
+
+    #Case 3: holdallowed holdgroup, hold_fulfillment_policy any
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        3, 'any'
+    );
+
+    @pl_1_1 = $biblio1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $biblio1->pickup_locations( { patron => $patron4 } );
+    @pl_2_1 = $biblio2->pickup_locations( { patron => $patron1 } );
+    @pl_2_4 = $biblio2->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_1) == 5 && scalar(@pl_2_4) == 5 && scalar(@pl_1_4) == 5 && scalar(@pl_2_1) == 5, 'Returns all libraries that are pickup_locations, when item\'s hombebranch is in patron\' holdgroup');
+
+    #Case 4: holdallowed any, hold_fulfillment_policy holdgroup
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        2, 'holdgroup'
+    );
+
+    @pl_1_1 = $biblio1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $biblio1->pickup_locations( { patron => $patron4 } );
+    @pl_2_1 = $biblio2->pickup_locations( { patron => $patron1 } );
+    @pl_2_4 = $biblio2->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_1) == 3 && scalar(@pl_2_4) == 3 && scalar(@pl_1_4) == 3 && scalar(@pl_2_1) == 3, 'Returns libraries in item\'s holdgroup, and that are pickup_locations');
+
+    #Case 5: holdallowed homebranch, hold_fulfillment_policy holdgroup, HomeOrHoldingBranch 'homebranch'
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        1, 'holdgroup'
+    );
+
+    @pl_1_1 = $biblio1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $biblio1->pickup_locations( { patron => $patron4 } );
+    @pl_2_1 = $biblio2->pickup_locations( { patron => $patron1 } );
+    @pl_2_4 = $biblio2->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_1) == 2 && scalar(@pl_2_4) == 1, 'Returns libraries in item\'s holdgroup whose homebranch equals patron\'s homebranch, and that are pickup_locations');
+    ok(scalar(@pl_1_4) == 0 && scalar(@pl_2_1) == 0, 'Returns no pickup locations');
+
+    #Case 6: holdallowed holdgroup, hold_fulfillment_policy holdgroup
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        3, 'holdgroup'
+    );
+
+    @pl_1_1 = $biblio1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $biblio1->pickup_locations( { patron => $patron4 } );
+    @pl_2_1 = $biblio2->pickup_locations( { patron => $patron1 } );
+    @pl_2_4 = $biblio2->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_1) == 2 && scalar(@pl_2_1) == 2 && scalar(@pl_2_4) == 1 && scalar(@pl_1_4) == 1, 'Returns libraries in item\'s holdgroup whose homebranch is included patron\'s holdgroup, and that are pickup_locations');
+
+    #Case 7: holdallowed any, hold_fulfillment_policy homebranch
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        2, 'homebranch'
+    );
+
+    @pl_1_1 = $biblio1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $biblio1->pickup_locations( { patron => $patron4 } );
+    @pl_2_1 = $biblio2->pickup_locations( { patron => $patron1 } );
+    @pl_2_4 = $biblio2->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_1) == 1 && scalar(@pl_1_4) == 1 && scalar(@pl_2_1) == 2 && scalar(@pl_2_4) == 2, 'Returns homebranch of items in biblio, that are pickup_locations');
+
+    #Case 8: holdallowed homebranch, hold_fulfillment_policy homebranch, HomeOrHoldingBranch 'homebranch'
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        1, 'homebranch'
+    );
+
+    @pl_1_1 = $biblio1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $biblio1->pickup_locations( { patron => $patron4 } );
+    @pl_2_1 = $biblio2->pickup_locations( { patron => $patron1 } );
+    @pl_2_4 = $biblio2->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_1) == 1 && scalar(@pl_2_4) == 1 && $pl_1_1[0]->{branchcode} eq $library1->branchcode && $pl_2_4[0]->{branchcode} eq $library4->branchcode, 'Returns homebranch of items in biblio that equals patron\'s homebranch, and that are pickup_locations');
+    ok(scalar(@pl_1_4) == 0 && scalar(@pl_2_1) == 0, 'No pickup locations returned');
+
+    #Case 9: holdallowed holdgroup, hold_fulfillment_policy homebranch
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        3, 'homebranch'
+    );
+
+    @pl_1_1 = $biblio1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $biblio1->pickup_locations( { patron => $patron4 } );
+    @pl_2_1 = $biblio2->pickup_locations( { patron => $patron1 } );
+    @pl_2_4 = $biblio2->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_1) == 1 && scalar(@pl_2_1) == 1 && scalar(@pl_2_4) == 1, 'Returns homebranch of items in biblio that are within patron\'s holdgroup, and that are pickup_locations');
+    ok(scalar(@pl_1_4) == 0, 'No pickup locations returned');
+
+    #Case 10: holdallowed any, hold_fulfillment_policy holdingbranch
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        2, 'holdingbranch'
+    );
+
+    @pl_1_1 = $biblio1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $biblio1->pickup_locations( { patron => $patron4 } );
+    @pl_2_1 = $biblio2->pickup_locations( { patron => $patron1 } );
+    @pl_2_4 = $biblio2->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_1) == 2 && scalar(@pl_1_4) == 2 && scalar(@pl_2_1) == 2 && scalar(@pl_2_4) == 2, 'Returns holdingbranch of items in biblio, that are pickup_locations');
+
+    #Case 11: holdallowed homebranch, hold_fulfillment_policy holdingbranch, HomeOrHoldingBranch 'homebranch'
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        1, 'holdingbranch'
+    );
+
+    @pl_1_1 = $biblio1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $biblio1->pickup_locations( { patron => $patron4 } );
+    @pl_2_1 = $biblio2->pickup_locations( { patron => $patron1 } );
+    @pl_2_4 = $biblio2->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_1) == 1 && scalar(@pl_2_4) == 1, 'Returns holdingbranch of items in biblio, whose homebranch equals patron\'s, and that are pickup_locations');
+    ok(scalar(@pl_1_4) == 0 && scalar(@pl_2_1) == 0, 'No pickup locations returned');
+
+    #Case 12: holdallowed holdgroup, hold_fulfillment_policy holdingbranch
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        3, 'holdingbranch'
+    );
+
+    @pl_1_1 = $biblio1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $biblio1->pickup_locations( { patron => $patron4 } );
+    @pl_2_1 = $biblio2->pickup_locations( { patron => $patron1 } );
+    @pl_2_4 = $biblio2->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_1) == 1 && scalar(@pl_2_4) == 1 && scalar(@pl_1_4) == 1 && scalar(@pl_2_1) == 1, 'Returns holdingbranch of items in biblio, whose homebranch are within patron\'s holdgroup, and that are pickup_locations');
+
+    t::lib::Mocks::mock_preference('HomeOrHoldingBranch', 'holdingbranch');
+
+    #Case 13: holdallowed homebranch, hold_fulfillment_policy any, HomeOrHoldingBranch 'holdingbranch'
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        1, 'any'
+    );
+
+    @pl_1_1 = $biblio1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $biblio1->pickup_locations( { patron => $patron4 } );
+    @pl_2_1 = $biblio2->pickup_locations( { patron => $patron1 } );
+    @pl_2_4 = $biblio2->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_4) == 5 && scalar(@pl_2_1) == 5 && scalar(@pl_2_4) == 5, 'Returns all libraries when item\'s holdingbranch equals patron\'s homebranch, and that are pickup_locations');
+    ok(scalar(@pl_1_1) == 0, 'No pickup locations returned');
+
+    #Case 14: holdallowed homebranch, hold_fulfillment_policy holdgroup, HomeOrHoldingBranch 'holdingbranch'
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        1, 'holdgroup'
+    );
+
+    @pl_1_1 = $biblio1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $biblio1->pickup_locations( { patron => $patron4 } );
+    @pl_2_1 = $biblio2->pickup_locations( { patron => $patron1 } );
+    @pl_2_4 = $biblio2->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_4) == 1 && scalar(@pl_2_1) == 2 && scalar(@pl_2_4) == 1, 'Returns libraries in item\'s holdgroup whose holdingbranch equals patron\'s homebranch, and that are pickup_locations');
+    ok(scalar(@pl_1_1) == 0, 'No pickup locations returned');
+
+    #Case 15: holdallowed homebranch, hold_fulfillment_policy homebranch, HomeOrHoldingBranch 'holdingbranch'
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        1, 'homebranch'
+    );
+
+    @pl_1_1 = $biblio1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $biblio1->pickup_locations( { patron => $patron4 } );
+    @pl_2_1 = $biblio2->pickup_locations( { patron => $patron1 } );
+    @pl_2_4 = $biblio2->pickup_locations( { patron => $patron4 } );
+
+    #ok(scalar(@pl_2_4) == 1 && $pl_2_4[0]->{branchcode} eq $library4->branchcode, 'Pickup location for patron 4 and item 3 renders item\'s holding branch');
+    ok(scalar(@pl_2_1) == 1 && scalar(@pl_2_4) == 1, 'Returns homebranch of items in biblio whose holdingbranch equals patron\'s homebranch, and that are pickup_locations');
+    ok(scalar(@pl_1_1) == 0 && scalar(@pl_1_4) == 0, 'No pickup locations returned');
+
+    #Case 16: holdallowed homebranch, hold_fulfillment_policy holdingbranch, HomeOrHoldingBranch 'holdingbranch'
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        1, 'holdingbranch'
+    );
+
+    @pl_1_1 = $biblio1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $biblio1->pickup_locations( { patron => $patron4 } );
+    @pl_2_1 = $biblio2->pickup_locations( { patron => $patron1 } );
+    @pl_2_4 = $biblio2->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_4) == 1 && scalar(@pl_2_1) == 1 && scalar(@pl_2_4) == 1, 'Returns holdingbranch of items in biblio that equals patron\'s homebranch, and that are pickup_locations');
+    ok(scalar(@pl_1_1) == 0, 'No pickup locations returned');
+
+    $schema->storage->txn_rollback;
+};
\ No newline at end of file
index 5f97d46..cf4cf46 100644 (file)
@@ -23,6 +23,7 @@ use Test::More tests => 10;
 use Test::Exception;
 
 use C4::Circulation;
+use C4::Context;
 use Koha::Item;
 use Koha::Item::Transfer::Limits;
 use Koha::Items;
@@ -34,6 +35,8 @@ use t::lib::Mocks;
 my $schema = Koha::Database->new->schema;
 $schema->storage->txn_begin;
 
+my $dbh     = C4::Context->dbh;
+
 my $builder     = t::lib::TestBuilder->new;
 my $library     = $builder->build( { source => 'Branch' } );
 my $nb_of_items = Koha::Items->search->count;
@@ -175,3 +178,313 @@ is( Koha::Items->search->count, $nb_of_items + 1, 'Delete should have deleted th
 
 $schema->storage->txn_rollback;
 
+subtest 'pickup_locations' => sub {
+    plan tests => 33;
+
+    $schema->storage->txn_begin;
+
+    # Cleanup database
+    Koha::Holds->search->delete;
+    Koha::Patrons->search->delete;
+    Koha::Items->search->delete;
+    Koha::Libraries->search->delete;
+    $dbh->do('DELETE FROM issues');
+    $dbh->do('DELETE FROM issuingrules');
+    $dbh->do(
+        q{INSERT INTO issuingrules (categorycode, branchcode, itemtype, reservesallowed)
+        VALUES (?, ?, ?, ?)},
+        {},
+        '*', '*', '*', 25
+    );
+    $dbh->do('DELETE FROM branch_item_rules');
+    $dbh->do('DELETE FROM default_branch_circ_rules');
+    $dbh->do('DELETE FROM default_branch_item_rules');
+    $dbh->do('DELETE FROM default_circ_rules');
+
+    my $root1 = $builder->build_object( { class => 'Koha::Library::Groups', value => { ft_local_hold_group => 1 } } );
+    my $root2 = $builder->build_object( { class => 'Koha::Library::Groups', value => { ft_local_hold_group => 1 } } );
+
+    my $library1 = $builder->build_object( { class => 'Koha::Libraries', value => { pickup_location => 1 } } );
+    my $library2 = $builder->build_object( { class => 'Koha::Libraries', value => { pickup_location => 1 } } );
+    my $library3 = $builder->build_object( { class => 'Koha::Libraries', value => { pickup_location => 0 } } );
+    my $library4 = $builder->build_object( { class => 'Koha::Libraries', value => { pickup_location => 1 } } );
+
+    my $group1_1 = $builder->build_object( { class => 'Koha::Library::Groups', value => { parent_id => $root1->id, branchcode => $library1->branchcode } } );
+    my $group1_2 = $builder->build_object( { class => 'Koha::Library::Groups', value => { parent_id => $root1->id, branchcode => $library2->branchcode } } );
+
+    my $group2_1 = $builder->build_object( { class => 'Koha::Library::Groups', value => { parent_id => $root2->id, branchcode => $library3->branchcode } } );
+    my $group2_2 = $builder->build_object( { class => 'Koha::Library::Groups', value => { parent_id => $root2->id, branchcode => $library4->branchcode } } );
+
+    my $biblioitem  = $builder->build( { source => 'Biblioitem' } );
+
+    my $item1  = Koha::Item->new({
+        biblionumber     => $biblioitem->{biblionumber},
+        biblioitemnumber => $biblioitem->{biblioitemnumber},
+        homebranch       => $library1->branchcode,
+        holdingbranch    => $library2->branchcode,
+        itype            => 'test',
+        barcode          => "item1barcode",
+    })->store;
+
+    my $item3  = Koha::Item->new({
+        biblionumber     => $biblioitem->{biblionumber},
+        biblioitemnumber => $biblioitem->{biblioitemnumber},
+        homebranch       => $library3->branchcode,
+        holdingbranch    => $library4->branchcode,
+        itype            => 'test',
+        barcode          => "item3barcode",
+    })->store;
+
+    my $patron1 = $builder->build_object( { class => 'Koha::Patrons', value => { branchcode => $library1->branchcode } } );
+    my $patron4 = $builder->build_object( { class => 'Koha::Patrons', value => { branchcode => $library4->branchcode } } );
+
+    t::lib::Mocks::mock_preference('HomeOrHoldingBranch', 'homebranch');
+
+    #Case 1: holdallowed any, hold_fulfillment_policy any
+    $dbh->do(
+        q{INSERT INTO default_circ_rules (holdallowed, hold_fulfillment_policy, returnbranch)
+        VALUES (?,?,?)},
+        {},
+        2, 'any', 'any'
+    );
+
+    my @pl_1_1 = $item1->pickup_locations( { patron => $patron1 } );
+    my @pl_1_4 = $item1->pickup_locations( { patron => $patron4 } );
+    my @pl_3_1 = $item3->pickup_locations( { patron => $patron1 } );
+    my @pl_3_4 = $item3->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_1) == scalar(@pl_1_4) && scalar(@pl_1_1) == scalar(@pl_3_1) && scalar(@pl_1_1) == scalar(@pl_3_4), 'All combinations of patron/item renders the same number of locations');
+
+    #Case 2: holdallowed homebranch, hold_fulfillment_policy any, HomeOrHoldingBranch 'homebranch'
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        1, 'any'
+    );
+
+    @pl_1_1 = $item1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $item1->pickup_locations( { patron => $patron4 } );
+    @pl_3_1 = $item3->pickup_locations( { patron => $patron1 } );
+    @pl_3_4 = $item3->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_1) == 3, 'Pickup location for patron 1 and item 1 renders all libraries that are pickup_locations');
+    ok(scalar(@pl_1_4) == 0 && scalar(@pl_3_1) == 0 && scalar(@pl_3_4) == 0, 'Any other combination renders no locations');
+
+    #Case 3: holdallowed holdgroup, hold_fulfillment_policy any
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        3, 'any'
+    );
+
+    @pl_1_1 = $item1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $item1->pickup_locations( { patron => $patron4 } );
+    @pl_3_1 = $item3->pickup_locations( { patron => $patron1 } );
+    @pl_3_4 = $item3->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_1) == 3, 'Pickup location for patron 1 and item 1 renders all libraries that are pickup_locations');
+    ok(scalar(@pl_3_4) == 3, 'Pickup location for patron 4 and item 3 renders all libraries that are pickup_locations');
+    ok(scalar(@pl_1_4) == 0 && scalar(@pl_3_1) == 0, 'Any other combination renders no locations');
+
+    #Case 4: holdallowed any, hold_fulfillment_policy holdgroup
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        2, 'holdgroup'
+    );
+
+    @pl_1_1 = $item1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $item1->pickup_locations( { patron => $patron4 } );
+    @pl_3_1 = $item3->pickup_locations( { patron => $patron1 } );
+    @pl_3_4 = $item3->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_1) == 2 && scalar(@pl_1_4) == 2, 'Pickup locations for item 1 renders all libraries in items\'s holdgroup that are pickup_locations');
+    ok(scalar(@pl_3_1) == 1 && scalar(@pl_3_4) == 1, 'Pickup locations for item 3 renders all libraries in items\'s holdgroup that are pickup_locations');
+
+    #Case 5: holdallowed homebranch, hold_fulfillment_policy holdgroup, HomeOrHoldingBranch 'homebranch'
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        1, 'holdgroup'
+    );
+
+    @pl_1_1 = $item1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $item1->pickup_locations( { patron => $patron4 } );
+    @pl_3_1 = $item3->pickup_locations( { patron => $patron1 } );
+    @pl_3_4 = $item3->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_1) == 2, 'Pickup location for patron 1 and item 1 renders all libraries in holdgroup that are pickup_locations');
+    ok(scalar(@pl_1_4) == 0 && scalar(@pl_3_1) == 0 && scalar(@pl_3_4) == 0, 'Any other combination renders no locations');
+
+    #Case 6: holdallowed holdgroup, hold_fulfillment_policy holdgroup
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        3, 'holdgroup'
+    );
+
+    @pl_1_1 = $item1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $item1->pickup_locations( { patron => $patron4 } );
+    @pl_3_1 = $item3->pickup_locations( { patron => $patron1 } );
+    @pl_3_4 = $item3->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_1) == 2, 'Pickup location for patron 1 and item 1 renders all libraries that are pickup_locations');
+    ok(scalar(@pl_3_4) == 1, 'Pickup location for patron 4 and item 3 renders all libraries that are pickup_locations');
+    ok(scalar(@pl_1_4) == 0 && scalar(@pl_3_1) == 0, 'Any other combination renders no locations');
+
+    #Case 7: holdallowed any, hold_fulfillment_policy homebranch
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        2, 'homebranch'
+    );
+
+    @pl_1_1 = $item1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $item1->pickup_locations( { patron => $patron4 } );
+    @pl_3_1 = $item3->pickup_locations( { patron => $patron1 } );
+    @pl_3_4 = $item3->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_1) == 1 && scalar(@pl_1_4) == 1 && $pl_1_1[0]->{branchcode} eq $library1->branchcode && $pl_1_4[0]->{branchcode} eq $library1->id, 'Pickup locations for item 1 renders item\'s homelibrary');
+    ok(scalar(@pl_3_1) == 0 && scalar(@pl_3_4) == 0, 'Any other combination renders no locations, because library3 is not pickup_location');
+
+    #Case 8: holdallowed homebranch, hold_fulfillment_policy homebranch, HomeOrHoldingBranch 'homebranch'
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        1, 'homebranch'
+    );
+
+    @pl_1_1 = $item1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $item1->pickup_locations( { patron => $patron4 } );
+    @pl_3_1 = $item3->pickup_locations( { patron => $patron1 } );
+    @pl_3_4 = $item3->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_1) == 1 && $pl_1_1[0]->{branchcode} eq $library1->branchcode, 'Pickup location for patron 1 and item 1 renders item\'s homebranch');
+    ok(scalar(@pl_1_4) == 0 && scalar(@pl_3_1) == 0 && scalar(@pl_3_4) == 0, 'Any other combination renders no locations');
+
+    #Case 9: holdallowed holdgroup, hold_fulfillment_policy homebranch
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        3, 'homebranch'
+    );
+
+    @pl_1_1 = $item1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $item1->pickup_locations( { patron => $patron4 } );
+    @pl_3_1 = $item3->pickup_locations( { patron => $patron1 } );
+    @pl_3_4 = $item3->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_1) == 1, 'Pickup location for patron 1 and item 1 renders item\'s homebranch');
+    ok(scalar(@pl_1_4) == 0 && scalar(@pl_3_1) == 0 && scalar(@pl_3_4) == 0, 'Any other combination renders no locations');
+
+    #Case 10: holdallowed any, hold_fulfillment_policy holdingbranch
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        2, 'holdingbranch'
+    );
+
+    @pl_1_1 = $item1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $item1->pickup_locations( { patron => $patron4 } );
+    @pl_3_1 = $item3->pickup_locations( { patron => $patron1 } );
+    @pl_3_4 = $item3->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_1) == 1 && scalar(@pl_1_4) == 1 && $pl_1_1[0]->{branchcode} eq $library2->branchcode && $pl_1_4[0]->{branchcode} eq $library2->branchcode, 'Pickup locations for item 1 renders item\'s holding branch');
+    ok(scalar(@pl_3_1) == 1 && scalar(@pl_3_4) == 1 && $pl_3_1[0]->{branchcode} eq $library4->branchcode && $pl_3_4[0]->{branchcode} eq $library4->branchcode, 'Pickup locations for item 3 renders item\'s holding branch');
+
+
+    #Case 11: holdallowed homebranch, hold_fulfillment_policy holdingbranch, HomeOrHoldingBranch 'homebranch'
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        1, 'holdingbranch'
+    );
+
+    @pl_1_1 = $item1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $item1->pickup_locations( { patron => $patron4 } );
+    @pl_3_1 = $item3->pickup_locations( { patron => $patron1 } );
+    @pl_3_4 = $item3->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_1) == 1 && $pl_1_1[0]->{branchcode} eq $library2->branchcode, 'Pickup location for patron 1 and item 1 renders item\'s holding branch');
+    ok(scalar(@pl_1_4) == 0 && scalar(@pl_3_1) == 0 && scalar(@pl_3_4) == 0, 'Any other combination renders no locations');
+
+    #Case 12: holdallowed holdgroup, hold_fulfillment_policy holdingbranch
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        3, 'holdingbranch'
+    );
+
+    @pl_1_1 = $item1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $item1->pickup_locations( { patron => $patron4 } );
+    @pl_3_1 = $item3->pickup_locations( { patron => $patron1 } );
+    @pl_3_4 = $item3->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_1_1) == 1 && $pl_1_1[0]->{branchcode} eq $library2->branchcode, 'Pickup location for patron 1 and item 1 renders item\'s holding branch');
+    ok(scalar(@pl_3_4) == 1 && $pl_3_4[0]->{branchcode} eq $library4->branchcode, 'Pickup location for patron 4 and item 3 renders item\'s holding branch');
+    ok(scalar(@pl_1_4) == 0 && scalar(@pl_3_1) == 0, 'Any other combination renders no locations');
+
+    t::lib::Mocks::mock_preference('HomeOrHoldingBranch', 'holdingbranch');
+
+    #Case 13: holdallowed homebranch, hold_fulfillment_policy any, HomeOrHoldingBranch 'holdingbranch'
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        1, 'any'
+    );
+
+    @pl_1_1 = $item1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $item1->pickup_locations( { patron => $patron4 } );
+    @pl_3_1 = $item3->pickup_locations( { patron => $patron1 } );
+    @pl_3_4 = $item3->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_3_4) == 3, 'Pickup location for patron 4 and item 3 renders all libraries that are pickup_locations');
+    ok(scalar(@pl_1_4) == 0 && scalar(@pl_3_1) == 0 && scalar(@pl_1_1) == 0, 'Any other combination renders no locations');
+
+    #Case 14: holdallowed homebranch, hold_fulfillment_policy holdgroup, HomeOrHoldingBranch 'holdingbranch'
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        1, 'holdgroup'
+    );
+
+    @pl_1_1 = $item1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $item1->pickup_locations( { patron => $patron4 } );
+    @pl_3_1 = $item3->pickup_locations( { patron => $patron1 } );
+    @pl_3_4 = $item3->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_3_4) == 1, 'Pickup location for patron 4 and item 3 renders all libraries in holdgroup that are pickup_locations');
+    ok(scalar(@pl_1_4) == 0 && scalar(@pl_3_1) == 0 && scalar(@pl_1_1) == 0, 'Any other combination renders no locations');
+
+    #Case 15: holdallowed homebranch, hold_fulfillment_policy homebranch, HomeOrHoldingBranch 'holdingbranch'
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        1, 'homebranch'
+    );
+
+    @pl_1_1 = $item1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $item1->pickup_locations( { patron => $patron4 } );
+    @pl_3_1 = $item3->pickup_locations( { patron => $patron1 } );
+    @pl_3_4 = $item3->pickup_locations( { patron => $patron4 } );
+
+    #ok(scalar(@pl_3_4) == 1 && $pl_3_4[0]->{branchcode} eq $library4->branchcode, 'Pickup location for patron 4 and item 3 renders item\'s holding branch');
+    ok(scalar(@pl_3_4) == 0 && scalar(@pl_1_4) == 0 && scalar(@pl_3_1) == 0 && scalar(@pl_1_1) == 0, 'Any combination of patron/item renders no locations');
+
+    #Case 16: holdallowed homebranch, hold_fulfillment_policy holdingbranch, HomeOrHoldingBranch 'holdingbranch'
+    $dbh->do(
+        q{UPDATE default_circ_rules set holdallowed = ?, hold_fulfillment_policy = ?},
+        {},
+        1, 'holdingbranch'
+    );
+
+    @pl_1_1 = $item1->pickup_locations( { patron => $patron1 } );
+    @pl_1_4 = $item1->pickup_locations( { patron => $patron4 } );
+    @pl_3_1 = $item3->pickup_locations( { patron => $patron1 } );
+    @pl_3_4 = $item3->pickup_locations( { patron => $patron4 } );
+
+    ok(scalar(@pl_3_4) == 1 && $pl_3_4[0]->{branchcode} eq $library4->branchcode, 'Pickup location for patron 1 and item 1 renders item\'s holding branch');
+    ok(scalar(@pl_1_4) == 0 && scalar(@pl_3_1) == 0 && scalar(@pl_1_1) == 0, 'Any other combination renders no locations');
+
+    $schema->storage->txn_rollback;
+};