use List::MoreUtils qw( any );
-use C4::Context;
+use C4::Context qw(preference);
use C4::Letters qw( GetPreparedLetter EnqueueLetter );
use C4::Log qw( logaction );
+use C4::Reserves;
use Koha::AuthorisedValues;
use Koha::DateUtils qw( dt_from_string );
use Koha::Patrons;
use Koha::Biblios;
+use Koha::Hold::CancellationRequests;
use Koha::Items;
use Koha::Libraries;
use Koha::Old::Holds;
use Koha::Calendar;
+use Koha::Plugins;
+use Koha::BackgroundJob::BatchUpdateBiblioHoldsQueue;
+
+use Koha::Exceptions;
use Koha::Exceptions::Hold;
use base qw(Koha::Object);
=head3 suspend_hold
-my $hold = $hold->suspend_hold( $suspend_until_dt );
+my $hold = $hold->suspend_hold( $suspend_until );
=cut
sub suspend_hold {
- my ( $self, $dt ) = @_;
+ my ( $self, $date ) = @_;
- my $date = $dt ? $dt->clone()->truncate( to => 'day' )->datetime : undef;
+ $date &&= dt_from_string($date)->truncate( to => 'day' )->datetime;
if ( $self->is_found ) { # We can't suspend found holds
if ( $self->is_waiting ) {
$self->suspend_until($date);
$self->store();
+ Koha::Plugins->call(
+ 'after_hold_action',
+ {
+ action => 'suspend',
+ payload => { hold => $self->get_from_storage }
+ }
+ );
+
logaction( 'HOLDS', 'SUSPEND', $self->reserve_id, $self )
if C4::Context->preference('HoldsLog');
+ Koha::BackgroundJob::BatchUpdateBiblioHoldsQueue->new->enqueue(
+ {
+ biblio_ids => [ $self->biblionumber ]
+ }
+ ) if C4::Context->preference('RealTimeHoldsQueue');
+
return $self;
}
$self->store();
+ Koha::Plugins->call(
+ 'after_hold_action',
+ {
+ action => 'resume',
+ payload => { hold => $self->get_from_storage }
+ }
+ );
+
logaction( 'HOLDS', 'RESUME', $self->reserve_id, $self )
if C4::Context->preference('HoldsLog');
+ Koha::BackgroundJob::BatchUpdateBiblioHoldsQueue->new->enqueue(
+ {
+ biblio_ids => [ $self->biblionumber ]
+ }
+ ) if C4::Context->preference('RealTimeHoldsQueue');
+
return $self;
}
$self->found('T');
$self->store();
+ Koha::Plugins->call(
+ 'after_hold_action',
+ {
+ action => 'transfer',
+ payload => { hold => $self->get_from_storage }
+ }
+ );
+
return $self;
}
$self->set($values)->store();
+ Koha::Plugins->call(
+ 'after_hold_action',
+ {
+ action => 'waiting',
+ payload => { hold => $self->get_from_storage }
+ }
+ );
+
return $self;
}
$self->found('P');
$self->store();
+ Koha::Plugins->call(
+ 'after_hold_action',
+ {
+ action => 'processing',
+ payload => { hold => $self->get_from_storage }
+ }
+ );
+
return $self;
}
return 0; # if ->is_in_transit or if ->is_waiting or ->is_in_processing
}
+=head3 cancellation_requestable_from_opac
+
+ if ( $hold->cancellation_requestable_from_opac ) { ... }
+
+Returns a I<boolean> representing if a cancellation request can be placed on the hold
+from the OPAC. It targets holds that cannot be cancelled from the OPAC (see the
+B<is_cancelable_from_opac> method above), but for which circulation rules allow
+requesting cancellation.
+
+Throws a B<Koha::Exceptions::InvalidStatus> exception with the following I<invalid_status>
+values:
+
+=over 4
+
+=item B<'hold_not_waiting'>: the hold is expected to be waiting and it is not.
+
+=item B<'no_item_linked'>: the waiting hold doesn't have an item properly linked.
+
+=back
+
+=cut
+
+sub cancellation_requestable_from_opac {
+ my ( $self ) = @_;
+
+ Koha::Exceptions::InvalidStatus->throw( invalid_status => 'hold_not_waiting' )
+ unless $self->is_waiting;
+
+ my $item = $self->item;
+
+ Koha::Exceptions::InvalidStatus->throw( invalid_status => 'no_item_linked' )
+ unless $item;
+
+ my $patron = $self->patron;
+
+ my $controlbranch = $patron->branchcode;
+
+ if ( C4::Context->preference('ReservesControlBranch') eq 'ItemHomeLibrary' ) {
+ $controlbranch = $item->homebranch;
+ }
+
+ return Koha::CirculationRules->get_effective_rule_value(
+ {
+ categorycode => $patron->categorycode,
+ itemtype => $item->itype,
+ branchcode => $controlbranch,
+ rule_name => 'waiting_hold_cancellation',
+ }
+ ) ? 1 : 0;
+}
+
=head3 is_at_destination
Returns true if hold is waiting
return $self->{_item};
}
+=head3 item_group
+
+Returns the related Koha::Biblio::ItemGroup object for this Hold
+
+=cut
+
+sub item_group {
+ my ($self) = @_;
+
+ my $item_group_rs = $self->_result->item_group;
+ return unless $item_group_rs;
+ return Koha::Biblio::ItemGroup->_new_from_dbic($item_group_rs);
+}
+
=head3 branch
Returns the related Koha::Library object for this Hold
return $self->suspend();
}
+=head3 add_cancellation_request
+
+ my $cancellation_request = $hold->add_cancellation_request({ [ creation_date => $creation_date ] });
+
+Adds a cancellation request to the hold. Returns the generated
+I<Koha::Hold::CancellationRequest> object.
+
+=cut
+
+sub add_cancellation_request {
+ my ( $self, $params ) = @_;
+
+ my $request = Koha::Hold::CancellationRequest->new(
+ { hold_id => $self->id,
+ ( $params->{creation_date} ? ( creation_date => $params->{creation_date} ) : () ),
+ }
+ )->store;
+
+ $request->discard_changes;
+
+ return $request;
+}
+
+=head3 cancellation_requests
+
+ my $cancellation_requests = $hold->cancellation_requests;
+
+Returns related a I<Koha::Hold::CancellationRequests> resultset.
+
+=cut
+
+sub cancellation_requests {
+ my ($self) = @_;
+
+ return Koha::Hold::CancellationRequests->search( { hold_id => $self->id } );
+}
=head3 cancel
my $cancel_hold = $hold->cancel(
{
- [ charge_cancel_fee => 1||0, ]
+ [ charge_cancel_fee => 1||0, ]
[ cancellation_reason => $cancellation_reason, ]
+ [ skip_holds_queue => 1||0 ]
}
);
sub cancel {
my ( $self, $params ) = @_;
+
+ my $autofill_next = $params->{autofill} && $self->itemnumber && $self->found && $self->found eq 'W';
+
$self->_result->result_source->schema->txn_do(
sub {
my $patron = $self->patron;
}
my $old_me = $self->_move_to_old;
+
+ Koha::Plugins->call(
+ 'after_hold_action',
+ {
+ action => 'cancel',
+ payload => { hold => $old_me->get_from_storage }
+ }
+ );
+
# anonymize if required
$old_me->anonymize
if $patron->privacy == 2;
$self->SUPER::delete(); # Do not add a DELETE log
-
# now fix the priority on the others....
C4::Reserves::_FixPriority({ biblionumber => $self->biblionumber });
C4::Log::logaction( 'HOLDS', 'CANCEL', $self->reserve_id, $self )
if C4::Context->preference('HoldsLog');
+
+ Koha::BackgroundJob::BatchUpdateBiblioHoldsQueue->new->enqueue(
+ {
+ biblio_ids => [ $old_me->biblionumber ]
+ }
+ ) unless $params->{skip_holds_queue} or !C4::Context->preference('RealTimeHoldsQueue');
}
);
+
+ if ($autofill_next) {
+ my ( undef, $next_hold ) = C4::Reserves::CheckReserves( $self->itemnumber );
+ if ($next_hold) {
+ my $is_transfer = $self->branchcode ne $next_hold->{branchcode};
+
+ C4::Reserves::ModReserveAffect( $self->itemnumber, $self->borrowernumber, $is_transfer, $next_hold->{reserve_id}, $self->desk_id, $autofill_next );
+ C4::Items::ModItemTransfer( $self->itemnumber, $self->branchcode, $next_hold->{branchcode}, "Reserve" ) if $is_transfer;
+ }
+ }
+
return $self;
}
}
);
- $self->_move_to_old;
+ my $old_me = $self->_move_to_old;
+
+ Koha::Plugins->call(
+ 'after_hold_action',
+ {
+ action => 'fill',
+ payload => { hold => $old_me->get_from_storage }
+ }
+ );
+
+ # anonymize if required
+ $old_me->anonymize
+ if $patron->privacy == 2;
+
$self->SUPER::delete(); # Do not add a DELETE log
# now fix the priority on the others....
C4::Log::logaction( 'HOLDS', 'FILL', $self->id, $self )
if C4::Context->preference('HoldsLog');
+
+ Koha::BackgroundJob::BatchUpdateBiblioHoldsQueue->new->enqueue(
+ {
+ biblio_ids => [ $old_me->biblionumber ]
+ }
+ ) if C4::Context->preference('RealTimeHoldsQueue');
}
);
return $self;
sub store {
my ($self) = @_;
+ Koha::Exceptions::Hold::MissingPickupLocation->throw() unless $self->branchcode;
+
if ( !$self->in_storage ) {
if ( ! $self->expirationdate && $self->patron_expiration_date ) {
$self->expirationdate($self->patron_expiration_date);
};
}
+=head3 can_update_pickup_location_opac
+
+ my $can_update_pickup_location_opac = $hold->can_update_pickup_location_opac;
+
+Returns if a hold can change pickup location from opac
+
+=cut
+
+sub can_update_pickup_location_opac {
+ my ($self) = @_;
+
+ my @statuses = split /,/, C4::Context->preference("OPACAllowUserToChangeBranch");
+ foreach my $status ( @statuses ){
+ return 1 if ($status eq 'pending' && !$self->is_found && !$self->is_suspended );
+ return 1 if ($status eq 'intransit' && $self->is_in_transit);
+ return 1 if ($status eq 'suspended' && $self->is_suspended);
+ }
+ return 0;
+}
+
=head2 Internal methods
=head3 _type