Bug 14187 - DBRev 16.12.00.002
[koha.git] / Koha / Patron.pm
blob3790884c975190533f9391ac18dc958408f61050
1 package Koha::Patron;
3 # Copyright ByWater Solutions 2014
4 # Copyright PTFS Europe 2016
6 # This file is part of Koha.
8 # Koha is free software; you can redistribute it and/or modify it under the
9 # terms of the GNU General Public License as published by the Free Software
10 # Foundation; either version 3 of the License, or (at your option) any later
11 # version.
13 # Koha is distributed in the hope that it will be useful, but WITHOUT ANY
14 # WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
15 # A PARTICULAR PURPOSE. See the GNU General Public License for more details.
17 # You should have received a copy of the GNU General Public License along
18 # with Koha; if not, write to the Free Software Foundation, Inc.,
19 # 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
21 use Modern::Perl;
23 use Carp;
25 use C4::Context;
26 use C4::Log;
27 use Koha::Checkouts;
28 use Koha::Database;
29 use Koha::DateUtils;
30 use Koha::Holds;
31 use Koha::Old::Checkouts;
32 use Koha::Patron::Categories;
33 use Koha::Patron::HouseboundProfile;
34 use Koha::Patron::HouseboundRole;
35 use Koha::Patron::Images;
36 use Koha::Patrons;
37 use Koha::Virtualshelves;
39 use base qw(Koha::Object);
41 =head1 NAME
43 Koha::Patron - Koha Patron Object class
45 =head1 API
47 =head2 Class Methods
49 =cut
51 =head3 delete
53 $patron->delete
55 Delete patron's holds, lists and finally the patron.
57 Lists owned by the borrower are deleted, but entries from the borrower to
58 other lists are kept.
60 =cut
62 sub delete {
63 my ($self) = @_;
65 my $deleted;
66 $self->_result->result_source->schema->txn_do(
67 sub {
68 # Delete Patron's holds
69 $self->holds->delete;
71 # Delete all lists and all shares of this borrower
72 # Consistent with the approach Koha uses on deleting individual lists
73 # Note that entries in virtualshelfcontents added by this borrower to
74 # lists of others will be handled by a table constraint: the borrower
75 # is set to NULL in those entries.
76 # NOTE:
77 # We could handle the above deletes via a constraint too.
78 # But a new BZ report 11889 has been opened to discuss another approach.
79 # Instead of deleting we could also disown lists (based on a pref).
80 # In that way we could save shared and public lists.
81 # The current table constraints support that idea now.
82 # This pref should then govern the results of other routines/methods such as
83 # Koha::Virtualshelf->new->delete too.
84 # FIXME Could be $patron->get_lists
85 $_->delete for Koha::Virtualshelves->search( { owner => $self->borrowernumber } );
87 $deleted = $self->SUPER::delete;
89 logaction( "MEMBERS", "DELETE", $self->borrowernumber, "" ) if C4::Context->preference("BorrowersLog");
92 return $deleted;
96 =head3 category
98 my $patron_category = $patron->category
100 Return the patron category for this patron
102 =cut
104 sub category {
105 my ( $self ) = @_;
106 return Koha::Patron::Category->_new_from_dbic( $self->_result->categorycode );
109 =head3 guarantor
111 Returns a Koha::Patron object for this patron's guarantor
113 =cut
115 sub guarantor {
116 my ( $self ) = @_;
118 return unless $self->guarantorid();
120 return Koha::Patrons->find( $self->guarantorid() );
123 sub image {
124 my ( $self ) = @_;
126 return Koha::Patron::Images->find( $self->borrowernumber );
129 sub library {
130 my ( $self ) = @_;
131 return Koha::Library->_new_from_dbic($self->_result->branchcode);
134 =head3 guarantees
136 Returns the guarantees (list of Koha::Patron) of this patron
138 =cut
140 sub guarantees {
141 my ( $self ) = @_;
143 return Koha::Patrons->search( { guarantorid => $self->borrowernumber } );
146 =head3 housebound_profile
148 Returns the HouseboundProfile associated with this patron.
150 =cut
152 sub housebound_profile {
153 my ( $self ) = @_;
154 my $profile = $self->_result->housebound_profile;
155 return Koha::Patron::HouseboundProfile->_new_from_dbic($profile)
156 if ( $profile );
157 return;
160 =head3 housebound_role
162 Returns the HouseboundRole associated with this patron.
164 =cut
166 sub housebound_role {
167 my ( $self ) = @_;
169 my $role = $self->_result->housebound_role;
170 return Koha::Patron::HouseboundRole->_new_from_dbic($role) if ( $role );
171 return;
174 =head3 siblings
176 Returns the siblings of this patron.
178 =cut
180 sub siblings {
181 my ( $self ) = @_;
183 my $guarantor = $self->guarantor;
185 return unless $guarantor;
187 return Koha::Patrons->search(
189 guarantorid => {
190 '!=' => undef,
191 '=' => $guarantor->id,
193 borrowernumber => {
194 '!=' => $self->borrowernumber,
200 =head3 wants_check_for_previous_checkout
202 $wants_check = $patron->wants_check_for_previous_checkout;
204 Return 1 if Koha needs to perform PrevIssue checking, else 0.
206 =cut
208 sub wants_check_for_previous_checkout {
209 my ( $self ) = @_;
210 my $syspref = C4::Context->preference("checkPrevCheckout");
212 # Simple cases
213 ## Hard syspref trumps all
214 return 1 if ($syspref eq 'hardyes');
215 return 0 if ($syspref eq 'hardno');
216 ## Now, patron pref trumps all
217 return 1 if ($self->checkprevcheckout eq 'yes');
218 return 0 if ($self->checkprevcheckout eq 'no');
220 # More complex: patron inherits -> determine category preference
221 my $checkPrevCheckoutByCat = $self->category->checkprevcheckout;
222 return 1 if ($checkPrevCheckoutByCat eq 'yes');
223 return 0 if ($checkPrevCheckoutByCat eq 'no');
225 # Finally: category preference is inherit, default to 0
226 if ($syspref eq 'softyes') {
227 return 1;
228 } else {
229 return 0;
233 =head3 do_check_for_previous_checkout
235 $do_check = $patron->do_check_for_previous_checkout($item);
237 Return 1 if the bib associated with $ITEM has previously been checked out to
238 $PATRON, 0 otherwise.
240 =cut
242 sub do_check_for_previous_checkout {
243 my ( $self, $item ) = @_;
245 # Find all items for bib and extract item numbers.
246 my @items = Koha::Items->search({biblionumber => $item->{biblionumber}});
247 my @item_nos;
248 foreach my $item (@items) {
249 push @item_nos, $item->itemnumber;
252 # Create (old)issues search criteria
253 my $criteria = {
254 borrowernumber => $self->borrowernumber,
255 itemnumber => \@item_nos,
258 # Check current issues table
259 my $issues = Koha::Checkouts->search($criteria);
260 return 1 if $issues->count; # 0 || N
262 # Check old issues table
263 my $old_issues = Koha::Old::Checkouts->search($criteria);
264 return $old_issues->count; # 0 || N
267 =head2 is_debarred
269 my $debarment_expiration = $patron->is_debarred;
271 Returns the date a patron debarment will expire, or undef if the patron is not
272 debarred
274 =cut
276 sub is_debarred {
277 my ($self) = @_;
279 return unless $self->debarred;
280 return $self->debarred
281 if $self->debarred =~ '^9999'
282 or dt_from_string( $self->debarred ) > dt_from_string;
283 return;
286 =head2 is_expired
288 my $is_expired = $patron->is_expired;
290 Returns 1 if the patron is expired or 0;
292 =cut
294 sub is_expired {
295 my ($self) = @_;
296 return 0 unless $self->dateexpiry;
297 return 0 if $self->dateexpiry eq '0000-00-00';
298 return 1 if dt_from_string( $self->dateexpiry ) < dt_from_string->truncate( to => 'day' );
299 return 0;
302 =head2 is_going_to_expire
304 my $is_going_to_expire = $patron->is_going_to_expire;
306 Returns 1 if the patron is going to expired, depending on the NotifyBorrowerDeparture pref or 0
308 =cut
310 sub is_going_to_expire {
311 my ($self) = @_;
313 my $delay = C4::Context->preference('NotifyBorrowerDeparture') || 0;
315 return 0 unless $delay;
316 return 0 unless $self->dateexpiry;
317 return 0 if $self->dateexpiry eq '0000-00-00';
318 return 1 if dt_from_string( $self->dateexpiry )->subtract( days => $delay ) < dt_from_string->truncate( to => 'day' );
319 return 0;
322 =head2 update_password
324 my $updated = $patron->update_password( $userid, $password );
326 Update the userid and the password of a patron.
327 If the userid already exists, returns and let DBIx::Class warns
328 This will add an entry to action_logs if BorrowersLog is set.
330 =cut
332 sub update_password {
333 my ( $self, $userid, $password ) = @_;
334 eval { $self->userid($userid)->store; };
335 return if $@; # Make sure the userid is not already in used by another patron
336 $self->password($password)->store;
337 logaction( "MEMBERS", "CHANGE PASS", $self->borrowernumber, "" ) if C4::Context->preference("BorrowersLog");
338 return 1;
341 =head3 renew_account
343 my $new_expiry_date = $patron->renew_account
345 Extending the subscription to the expiry date.
347 =cut
349 sub renew_account {
350 my ($self) = @_;
351 my $date;
352 if ( C4::Context->preference('BorrowerRenewalPeriodBase') eq 'combination' ) {
353 $date = ( dt_from_string gt dt_from_string( $self->dateexpiry ) ) ? dt_from_string : dt_from_string( $self->dateexpiry );
354 } else {
355 $date =
356 C4::Context->preference('BorrowerRenewalPeriodBase') eq 'dateexpiry'
357 ? dt_from_string( $self->dateexpiry )
358 : dt_from_string;
360 my $expiry_date = $self->category->get_expiry_date($date);
362 $self->dateexpiry($expiry_date)->store;
364 $self->add_enrolment_fee_if_needed;
366 logaction( "MEMBERS", "RENEW", $self->borrowernumber, "Membership renewed" ) if C4::Context->preference("BorrowersLog");
367 return dt_from_string( $expiry_date )->truncate( to => 'day' );
370 =head2 has_overdues
372 my $has_overdues = $patron->has_overdues;
374 Returns the number of patron's overdues
376 =cut
378 sub has_overdues {
379 my ($self) = @_;
380 my $dtf = Koha::Database->new->schema->storage->datetime_parser;
381 return $self->_result->issues->search({ date_due => { '<' => $dtf->format_datetime( dt_from_string() ) } })->count;
384 =head2 track_login
386 $patron->track_login;
387 $patron->track_login({ force => 1 });
389 Tracks a (successful) login attempt.
390 The preference TrackLastPatronActivity must be enabled. Or you
391 should pass the force parameter.
393 =cut
395 sub track_login {
396 my ( $self, $params ) = @_;
397 return if
398 !$params->{force} &&
399 !C4::Context->preference('TrackLastPatronActivity');
400 $self->lastseen( dt_from_string() )->store;
403 =head2 move_to_deleted
405 my $is_moved = $patron->move_to_deleted;
407 Move a patron to the deletedborrowers table.
408 This can be done before deleting a patron, to make sure the data are not completely deleted.
410 =cut
412 sub move_to_deleted {
413 my ($self) = @_;
414 my $patron_infos = $self->unblessed;
415 return Koha::Database->new->schema->resultset('Deletedborrower')->create($patron_infos);
418 =head3 article_requests
420 my @requests = $borrower->article_requests();
421 my $requests = $borrower->article_requests();
423 Returns either a list of ArticleRequests objects,
424 or an ArtitleRequests object, depending on the
425 calling context.
427 =cut
429 sub article_requests {
430 my ( $self ) = @_;
432 $self->{_article_requests} ||= Koha::ArticleRequests->search({ borrowernumber => $self->borrowernumber() });
434 return $self->{_article_requests};
437 =head3 article_requests_current
439 my @requests = $patron->article_requests_current
441 Returns the article requests associated with this patron that are incomplete
443 =cut
445 sub article_requests_current {
446 my ( $self ) = @_;
448 $self->{_article_requests_current} ||= Koha::ArticleRequests->search(
450 borrowernumber => $self->id(),
451 -or => [
452 { status => Koha::ArticleRequest::Status::Pending },
453 { status => Koha::ArticleRequest::Status::Processing }
458 return $self->{_article_requests_current};
461 =head3 article_requests_finished
463 my @requests = $biblio->article_requests_finished
465 Returns the article requests associated with this patron that are completed
467 =cut
469 sub article_requests_finished {
470 my ( $self, $borrower ) = @_;
472 $self->{_article_requests_finished} ||= Koha::ArticleRequests->search(
474 borrowernumber => $self->id(),
475 -or => [
476 { status => Koha::ArticleRequest::Status::Completed },
477 { status => Koha::ArticleRequest::Status::Canceled }
482 return $self->{_article_requests_finished};
485 =head3 add_enrolment_fee_if_needed
487 my $enrolment_fee = $patron->add_enrolment_fee_if_needed;
489 Add enrolment fee for a patron if needed.
491 =cut
493 sub add_enrolment_fee_if_needed {
494 my ($self) = @_;
495 my $enrolment_fee = $self->category->enrolmentfee;
496 if ( $enrolment_fee && $enrolment_fee > 0 ) {
497 # insert fee in patron debts
498 C4::Accounts::manualinvoice( $self->borrowernumber, '', '', 'A', $enrolment_fee );
500 return $enrolment_fee || 0;
503 =head3 checkouts
505 my $issues = $patron->checkouts
507 =cut
509 sub checkouts {
510 my ($self) = @_;
511 my $issues = $self->_result->issues;
512 return Koha::Checkouts->_new_from_dbic( $issues );
515 =head3 get_overdues
517 my $overdue_items = $patron->get_overdues
519 Return the overdued items
521 =cut
523 sub get_overdues {
524 my ($self) = @_;
525 my $dtf = Koha::Database->new->schema->storage->datetime_parser;
526 return $self->checkouts->search(
528 'me.date_due' => { '<' => $dtf->format_datetime(dt_from_string) },
531 prefetch => { item => { biblio => 'biblioitems' } },
536 =head3 get_age
538 my $age = $patron->get_age
540 Return the age of the patron
542 =cut
544 sub get_age {
545 my ($self) = @_;
546 my $today_str = dt_from_string->strftime("%Y-%m-%d");
547 my $dob_str = dt_from_string( $self->dateofbirth )->strftime("%Y-%m-%d");
549 my ( $dob_y, $dob_m, $dob_d ) = split /-/, $dob_str;
550 my ( $today_y, $today_m, $today_d ) = split /-/, $today_str;
552 my $age = $today_y - $dob_y;
553 if ( $dob_m . $dob_d > $today_m . $today_d ) {
554 $age--;
557 return $age;
560 =head3 account
562 my $account = $patron->account
564 =cut
566 sub account {
567 my ($self) = @_;
568 return Koha::Account->new( { patron_id => $self->borrowernumber } );
571 =head3 holds
573 my $holds = $patron->holds
575 Return all the holds placed by this patron
577 =cut
579 sub holds {
580 my ($self) = @_;
581 my $holds_rs = $self->_result->reserves->search( {}, { order_by => 'reservedate' } );
582 return Koha::Holds->_new_from_dbic($holds_rs);
585 =head3 type
587 =cut
589 sub _type {
590 return 'Borrower';
593 =head1 AUTHOR
595 Kyle M Hall <kyle@bywatersolutions.com>
596 Alex Sassmannshausen <alex.sassmannshausen@ptfs-europe.com>
598 =cut