3 # Copyright 2017 Koha Development team
5 # This file is part of Koha
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.
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.
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>.
22 use Test
::More tests
=> 1;
30 use t
::lib
::TestBuilder
;
32 my $schema = Koha
::Database
->new->schema;
33 $schema->storage->txn_begin;
35 my $builder = t
::lib
::TestBuilder
->new;
37 subtest
'cancel' => sub {
39 my $biblioitem = $builder->build_object( { class => 'Koha::Biblioitems' } );
40 my $library = $builder->build_object( { class => 'Koha::Libraries' } );
41 my $itemtype = $builder->build_object( { class => 'Koha::ItemTypes', value
=> { rentalcharge
=> 0 } } );
43 biblionumber
=> $biblioitem->biblionumber,
44 biblioitemnumber
=> $biblioitem->biblioitemnumber,
45 homebranch
=> $library->branchcode,
46 holdingbranch
=> $library->branchcode,
47 itype
=> $itemtype->itemtype,
49 my $item = $builder->build_object( { class => 'Koha::Items', value
=> $item_info } );
50 my $manager = $builder->build_object({ class => "Koha::Patrons" });
51 t
::lib
::Mocks
::mock_userenv
({ patron
=> $manager,branchcode
=> $manager->branchcode });
53 my ( @patrons, @holds );
54 for my $i ( 0 .. 2 ) {
55 my $priority = $i + 1;
56 my $patron = $builder->build_object(
58 class => 'Koha::Patrons',
59 value
=> { branchcode
=> $library->branchcode, }
62 my $reserve_id = C4
::Reserves
::AddReserve
(
63 $library->branchcode, $patron->borrowernumber,
64 $item->biblionumber, '',
67 "title for fee", $item->itemnumber,
69 my $hold = Koha
::Holds
->find($reserve_id);
70 push @patrons, $patron;
74 # There are 3 holds on this records
76 Koha
::Holds
->search( { biblionumber
=> $item->biblionumber } )->count;
78 'There should have 3 holds placed on this biblio record' );
79 my $first_hold = $holds[0];
80 my $second_hold = $holds[1];
81 my $third_hold = $holds[2];
82 is
( ref($second_hold), 'Koha::Hold',
83 'We should play with Koha::Hold objects' );
84 is
( $second_hold->priority, 2,
85 'Second hold should have a priority set to 3' );
87 # Remove the second hold, only 2 should still exist in DB and priorities must have been updated
88 my $is_cancelled = $second_hold->cancel;
89 is
( ref($is_cancelled), 'Koha::Hold',
90 'Koha::Hold->cancel should return the Koha::Hold (?)' )
91 ; # This is can reconsidered
92 is
( $second_hold->in_storage, 0,
93 'The hold has been cancelled and does not longer exist in DB' );
95 Koha
::Holds
->search( { biblionumber
=> $item->biblionumber } )->count;
97 'a hold has been cancelled, there should have only 2 holds placed on this biblio record'
100 # discard_changes to refetch
101 is
( $first_hold->discard_changes->priority, 1, 'First hold should still be first' );
102 is
( $third_hold->discard_changes->priority, 2, 'Third hold should now be second' );
104 subtest
'charge_cancel_fee parameter' => sub {
106 my $patron_category = $builder->build_object({ class => 'Koha::Patron::Categories', value
=> { reservefee
=> 0 } } );
107 my $patron = $builder->build_object({ class => 'Koha::Patrons', value
=> { categorycode
=> $patron_category->categorycode } });
108 is
( $patron->account->balance, 0, 'A new patron does not have any charges' );
111 $library->branchcode, $patron->borrowernumber,
112 $item->biblionumber, '',
115 "title for fee", $item->itemnumber,
118 # First, test cancelling a reserve when there's no charge configured.
119 t
::lib
::Mocks
::mock_preference
('ExpireReservesMaxPickUpDelayCharge', 0);
120 my $reserve_id = C4
::Reserves
::AddReserve
( @hold_info );
121 Koha
::Holds
->find( $reserve_id )->cancel( { charge_cancel_fee
=> 1 } );
122 is
( $patron->account->balance, 0, 'ExpireReservesMaxPickUpDelayCharge=0 - The patron should not have been charged' );
124 # Then, test cancelling a reserve when there's no charge desired.
125 t
::lib
::Mocks
::mock_preference
('ExpireReservesMaxPickUpDelayCharge', 42);
126 $reserve_id = C4
::Reserves
::AddReserve
( @hold_info );
127 Koha
::Holds
->find( $reserve_id )->cancel(); # charge_cancel_fee => 0
128 is
( $patron->account->balance, 0, 'ExpireReservesMaxPickUpDelayCharge=42, but charge_cancel_fee => 0, The patron should not have been charged' );
131 # Finally, test cancelling a reserve when there's a charge desired and configured.
132 t
::lib
::Mocks
::mock_preference
('ExpireReservesMaxPickUpDelayCharge', 42);
133 $reserve_id = C4
::Reserves
::AddReserve
( @hold_info );
134 Koha
::Holds
->find( $reserve_id )->cancel( { charge_cancel_fee
=> 1 } );
135 is
( int($patron->account->balance), 42, 'ExpireReservesMaxPickUpDelayCharge=42 and charge_cancel_fee => 1, The patron should have been charged!' );
138 subtest
'waiting hold' => sub {
140 my $patron = $builder->build_object({ class => 'Koha::Patrons' });
141 my $reserve_id = C4
::Reserves
::AddReserve
(
142 $library->branchcode, $patron->borrowernumber,
143 $item->biblionumber, '',
146 "title for fee", $item->itemnumber,
149 Koha
::Holds
->find( $reserve_id )->cancel;
150 my $hold_old = Koha
::Old
::Holds
->find( $reserve_id );
151 is
( $hold_old->found, 'W', 'The found column should have been kept and a hold is cancelled' );
154 subtest
'HoldsLog' => sub {
156 my $patron = $builder->build_object({ class => 'Koha::Patrons' });
158 $library->branchcode, $patron->borrowernumber,
159 $item->biblionumber, '',
162 "title for fee", $item->itemnumber,
165 t
::lib
::Mocks
::mock_preference
('HoldsLog', 0);
166 my $reserve_id = C4
::Reserves
::AddReserve
(@hold_info);
167 Koha
::Holds
->find( $reserve_id )->cancel;
168 my $number_of_logs = $schema->resultset('ActionLog')->search( { module
=> 'HOLDS', action
=> 'CANCEL', object
=> $reserve_id } )->count;
169 is
( $number_of_logs, 0, 'Without HoldsLog, Koha::Hold->cancel should not have logged' );
171 t
::lib
::Mocks
::mock_preference
('HoldsLog', 1);
172 $reserve_id = C4
::Reserves
::AddReserve
(@hold_info);
173 Koha
::Holds
->find( $reserve_id )->cancel;
174 $number_of_logs = $schema->resultset('ActionLog')->search( { module
=> 'HOLDS', action
=> 'CANCEL', object
=> $reserve_id } )->count;
175 is
( $number_of_logs, 1, 'With HoldsLog, Koha::Hold->cancel should have logged' );
178 subtest
'rollback' => sub {
180 my $patron_category = $builder->build_object(
182 class => 'Koha::Patron::Categories',
183 value
=> { reservefee
=> 0 }
186 my $patron = $builder->build_object(
188 class => 'Koha::Patrons',
189 value
=> { categorycode
=> $patron_category->categorycode }
193 $library->branchcode, $patron->borrowernumber,
194 $item->biblionumber, '',
197 "title for fee", $item->itemnumber,
200 t
::lib
::Mocks
::mock_preference
( 'ExpireReservesMaxPickUpDelayCharge',42 );
201 my $reserve_id = C4
::Reserves
::AddReserve
(@hold_info);
202 my $hold = Koha
::Holds
->find($reserve_id);
204 # Add a row with the same id to make the cancel fails
205 Koha
::Old
::Hold
->new( $hold->unblessed )->store;
208 eval { $hold->cancel( { charge_cancel_fee
=> 1 } ) };
210 qr{.*DBD::mysql::st execute failed: Duplicate entry.*},
211 'DBD should have raised an error about dup primary key';
213 $hold = Koha
::Holds
->find($reserve_id);
214 is
( ref($hold), 'Koha::Hold', 'The hold should not have been deleted' );
215 is
( $patron->account->balance, 0,
216 'If the hold has not been cancelled, the patron should not have been charged'
222 $schema->storage->txn_rollback;