3 # This file is part of Koha.
5 # Copyright (C) 2014 Hochschule für Gesundheit (hsg), Germany
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 automatic_renewals.pl - cron script to renew loans
26 ./automatic_renewals.pl [-c|--confirm] [-s|--send-notices] [-d|--digest] [-b|--digest-per-branch] [-v|--verbose]
29 # Once every day for digest messages
30 0 3 * * * automatic_renewals.pl -c -d
31 # Three times a day for non digest messages
32 0 0,8,16 * * * automatic_renewals.pl -c
36 This script searches for issues scheduled for automatic renewal
37 (issues.auto_renew). If there are still renews left (Renewals allowed)
38 and the renewal isn't premature (No Renewal before) the issue is renewed.
44 =item B<-s|--send-notices>
46 DEPRECATED: The system preference AutoRenewalNotices should be used to determine
47 whether notices are sent or not
48 Send AUTO_RENEWALS notices to patrons if the auto renewal has been done.
52 Print report to standard out.
56 Without this parameter no changes will be made
58 =item B<-b|--digest-per-branch>
60 Flag to indicate that generation of message digests should be
61 performed separately for each branch.
63 A patron could potentially have loans at several different branches
64 There is no natural branch to set as the sender on the aggregated
65 message in this situation so the default behavior is to use the
66 borrowers home branch. This could surprise to the borrower when
67 message sender is a library where they have not borrowed anything.
69 Enabling this flag ensures that the issuing library is the sender of
70 the digested message. It has no effect unless the borrower has
71 chosen 'Digests only' on the advance messages.
78 use Pod::Usage qw( pod2usage );
79 use Getopt::Long qw( GetOptions );
81 use Koha::Script -cron;
82 use C4::Circulation qw( CanBookBeRenewed AddRenewal );
84 use C4::Log qw( cronlogaction );
90 my $command_line_options = join(" ",@ARGV);
92 my ( $help, $send_notices, $verbose, $confirm, $digest_per_branch );
95 's|send-notices' => \$send_notices,
96 'v|verbose' => \$verbose,
97 'c|confirm' => \$confirm,
98 'b|digest-per-branch' => \$digest_per_branch,
101 pod2usage(0) if $help;
103 my $send_notices_pref = C4::Context->preference('AutoRenewalNotices');
104 if ( $send_notices_pref eq 'cron' ) {
107 The "AutoRenewalNotices" syspref is set to 'Follow the cron switch'.
108 The send_notices switch for this script is deprecated, you should either set the preference
109 to 'Never send emails' or 'Follow patron messaging preferences'
113 # If not following cron then:
114 # - we should not send if set to never
115 # - we will send any notice generated according to preferences if following those
116 $send_notices = $send_notices_pref eq 'never' ? 0 : 1;
119 # Since advance notice options are not visible in the web-interface
120 # unless EnhancedMessagingPreferences is on, let the user know that
121 # this script probably isn't going to do much
122 if ( ! C4::Context->preference('EnhancedMessagingPreferences') ) {
125 The "EnhancedMessagingPreferences" syspref is off.
126 Therefore, it is unlikely that this script will actually produce any messages to be sent.
127 To change this, edit the "EnhancedMessagingPreferences" syspref.
132 cronlogaction({ info => $command_line_options });
134 $verbose = 1 unless $verbose or $confirm;
135 print "Test run only\n" unless $confirm;
137 print "getting auto renewals\n" if $verbose;
138 my $auto_renews = Koha::Checkouts->search(
141 'patron.autorenew_checkouts' => 1,
144 join => ['patron','item']
147 print "found " . $auto_renews->count . " auto renewals\n" if $verbose;
149 my $renew_digest = {};
151 my @item_renewal_ids;
152 while ( my $auto_renew = $auto_renews->next ) {
153 print "examining item '" . $auto_renew->itemnumber . "' to auto renew\n" if $verbose;
155 my ( $borrower_preferences, $wants_messages, $wants_digest ) = ( undef, 0, 0 );
156 if ( $send_notices_pref eq 'preferences' ){
157 $borrower_preferences = C4::Members::Messaging::GetMessagingPreferences(
159 borrowernumber => $auto_renew->borrowernumber,
160 message_name => 'auto_renewals'
164 if $borrower_preferences
165 && $borrower_preferences->{transports};
168 && $borrower_preferences->{wants_digest};
169 } else { # Preference is never or cron
170 $wants_messages = $send_notices;
174 # CanBookBeRenewed returns 'auto_renew' when the renewal should be done by this script
175 my ( $ok, $error ) = CanBookBeRenewed( $auto_renew->patron, $auto_renew, undef, 1 );
177 if ( $error eq 'auto_renew' ) {
180 say sprintf "Issue id: %s for borrower: %s and item: %s %s be renewed.",
181 $auto_renew->issue_id, $auto_renew->borrowernumber, $auto_renew->itemnumber, $confirm ? 'will' : 'would';
184 my $date_due = AddRenewal(
186 borrowernumber => $auto_renew->borrowernumber,
187 itemnumber => $auto_renew->itemnumber,
188 branch => $auto_renew->branchcode,
193 push @item_renewal_ids, $auto_renew->itemnumber;
194 $auto_renew->auto_renew_error(undef)->store;
196 push @{ $report{ $auto_renew->borrowernumber } }, $auto_renew
197 if ( $wants_messages ) && !$wants_digest;
199 $error eq 'too_many' ||
200 $error eq 'on_reserve' ||
201 $error eq 'restriction' ||
202 $error eq 'overdue' ||
203 $error eq 'too_unseen' ||
204 $error eq 'auto_account_expired' ||
205 $error eq 'auto_too_late' ||
206 $error eq 'auto_too_much_oweing' ||
207 $error eq 'auto_too_soon' ||
208 $error eq 'item_denied_renewal' ||
209 $error eq 'item_issued_to_other_patron'
212 say sprintf "Issue id: %s for borrower: %s and item: %s %s not be renewed. (%s)",
213 $auto_renew->issue_id, $auto_renew->borrowernumber, $auto_renew->itemnumber, $confirm ? 'will' : 'would', $error;
215 $updated = 1 if (!$auto_renew->auto_renew_error || $error ne $auto_renew->auto_renew_error);
217 $auto_renew->auto_renew_error($error)->store if $confirm;
218 push @{ $report{ $auto_renew->borrowernumber } }, $auto_renew
219 if $error ne 'auto_too_soon' && ( $wants_messages && !$wants_digest ); # Do not notify if it's too soon
223 if ( $wants_digest ) {
224 # cache this one to process after we've run through all of the items.
225 if ($digest_per_branch) {
226 $renew_digest->{ $auto_renew->branchcode }->{ $auto_renew->borrowernumber }->{success}++ if $error eq 'auto_renew';
227 $renew_digest->{ $auto_renew->branchcode }->{ $auto_renew->borrowernumber }->{error}++ unless $error eq 'auto_renew' || $error eq 'auto_too_soon';
228 $renew_digest->{ $auto_renew->branchcode }->{ $auto_renew->borrowernumber }->{results}->{$error}++ ;
229 push @{$renew_digest->{ $auto_renew->branchcode }->{ $auto_renew->borrowernumber }->{issues}}, $auto_renew->itemnumber;
230 $renew_digest->{ $auto_renew->branchcode }->{ $auto_renew->borrowernumber }->{updated} = 1 if $updated && $error ne 'auto_too_soon';
232 $renew_digest->{ $auto_renew->borrowernumber }->{success} ++ if $error eq 'auto_renew';
233 $renew_digest->{ $auto_renew->borrowernumber }->{error}++ unless $error eq 'auto_renew' || $error eq 'auto_too_soon';
234 $renew_digest->{ $auto_renew->borrowernumber }->{results}->{$error}++ ;
235 $renew_digest->{ $auto_renew->borrowernumber }->{updated} = 1 if $updated && $error ne 'auto_too_soon';
236 push @{$renew_digest->{ $auto_renew->borrowernumber }->{issues}}, $auto_renew->itemnumber;
242 if( @item_renewal_ids ){
243 my $indexer = Koha::SearchEngine::Indexer->new({ index => $Koha::SearchEngine::BIBLIOS_INDEX });
244 $indexer->index_records( \@item_renewal_ids, "specialUpdate", "biblioserver" );
247 if ( $send_notices && $confirm ) {
248 for my $borrowernumber ( keys %report ) {
249 my $patron = Koha::Patrons->find($borrowernumber);
250 my $borrower_preferences =
251 C4::Members::Messaging::GetMessagingPreferences(
253 borrowernumber => $borrowernumber,
254 message_name => 'auto_renewals'
257 for my $issue ( @{ $report{$borrowernumber} } ) {
258 my $item = $issue->item;
259 # Force sending of email and only email if pref is set to "cron"
260 my @transports = $send_notices_pref eq 'preferences' ? keys %{ $borrower_preferences->{'transports'} } : 'email';
261 foreach my $transport ( @transports ) {
262 my $letter = C4::Letters::GetPreparedLetter (
263 module => 'circulation',
264 letter_code => 'AUTO_RENEWALS',
266 borrowers => $patron->borrowernumber,
267 issues => $issue->itemnumber,
268 items => $issue->itemnumber,
269 biblio => $item->biblionumber,
271 lang => $patron->lang,
272 message_transport_type => $transport,
276 my $library = $patron->library;
277 my $admin_email_address = $library->from_email_address;
279 C4::Letters::EnqueueLetter(
282 borrowernumber => $borrowernumber,
283 from_address => $admin_email_address,
284 message_transport_type => $transport
292 if ($digest_per_branch) {
293 while (my ($branchcode, $digests) = each %$renew_digest) {
296 branchcode => $branchcode,
297 letter_code => 'AUTO_RENEWALS_DGST',
302 digests => $renew_digest,
303 letter_code => 'AUTO_RENEWALS_DGST',
308 cronlogaction({ action => 'End', info => "COMPLETED" });
319 Enqueue digested letters.
327 Reference to the array of digested messages.
329 =item C<$letter_code>
331 String that denote the letter code.
340 PATRON: while ( my ( $borrowernumber, $digest ) = each %{$params->{digests}} ) {
341 next unless defined $digest->{updated} && $digest->{updated} == 1;
342 my $borrower_preferences =
343 C4::Members::Messaging::GetMessagingPreferences(
345 borrowernumber => $borrowernumber,
346 message_name => 'auto_renewals'
350 next PATRON unless $borrower_preferences; # how could this happen?
352 my $patron = Koha::Patrons->find( $borrowernumber );
354 if ( defined $params->{branchcode} ) {
355 $branchcode = $params->{branchcode};
357 $branchcode = $patron->branchcode;
359 my $library = Koha::Libraries->find( $branchcode );
360 my $from_address = $library->from_email_address;
362 foreach my $transport ( keys %{ $borrower_preferences->{'transports'} } ) {
363 my $letter = C4::Letters::GetPreparedLetter (
364 module => 'circulation',
365 letter_code => $params->{letter_code},
366 branchcode => $branchcode,
367 lang => $patron->lang,
369 error => $digest->{error}||0,
370 success => $digest->{success}||0,
371 results => $digest->{results},
373 loops => { issues => \@{$digest->{issues}} },
375 borrowers => $patron->borrowernumber,
377 message_transport_type => $transport,
381 C4::Letters::EnqueueLetter(
384 borrowernumber => $borrowernumber,
385 from_address => $from_address,
386 message_transport_type => $transport
392 "no letter of type '$params->{letter_code}' found for borrowernumber $borrowernumber. Please see sample_notices.sql";