Skip to content

Commit

Permalink
circulation: fix check is loan overdue method
Browse files Browse the repository at this point in the history
Closes #2014.

Co-Authored-by: Renaud Michotte <renaud.michotte@gmail.com>
  • Loading branch information
zannkukai committed Jul 2, 2021
1 parent 61cd8ba commit 09c9660
Show file tree
Hide file tree
Showing 2 changed files with 62 additions and 31 deletions.
4 changes: 2 additions & 2 deletions rero_ils/modules/loans/api.py
Original file line number Diff line number Diff line change
Expand Up @@ -847,7 +847,7 @@ def get_overdue_loan_pids(patron_pid=None, tstamp=None):
:return a generator of loan pid
"""
end_date = tstamp or datetime.now()
end_date = end_date.strftime('%Y-%m-%d')
end_date = end_date.strftime('%Y-%m-%dT%H:%M:%S.000Z')
query = current_circulation.loan_search_cls() \
.filter('term', state=LoanState.ITEM_ON_LOAN) \
.filter('range', end_date={'lte': end_date})
Expand All @@ -856,7 +856,7 @@ def get_overdue_loan_pids(patron_pid=None, tstamp=None):
results = query\
.params(preserve_order=True) \
.sort({'_created': {'order': 'asc'}}) \
.source(['pid']).scan()
.source(['pid', 'end_date']).scan()
for hit in results:
yield hit.pid

Expand Down
89 changes: 60 additions & 29 deletions tests/api/circulation/test_borrow_limits.py
Original file line number Diff line number Diff line change
Expand Up @@ -17,9 +17,10 @@

"""Borrow limits."""
from copy import deepcopy
from datetime import datetime, timedelta, timezone
from datetime import datetime, timedelta

from flask import url_for
from freezegun import freeze_time
from invenio_accounts.testutils import login_user_via_session
from invenio_circulation.search.api import LoansSearch
from utils import flush_index, get_json, postdata
Expand Down Expand Up @@ -191,6 +192,7 @@ def test_checkout_library_limit(
item3.update(item3_original_data, dbcommit=True, reindex=True)


@freeze_time("2021-06-15")
def test_overdue_limit(
client, app, librarian_martigny, lib_martigny, item_lib_martigny,
item2_lib_martigny, patron_type_children_martigny,
Expand All @@ -202,10 +204,32 @@ def test_overdue_limit(
item = item_lib_martigny
item_pid = item.pid
patron_pid = patron_martigny.pid
date_format = '%Y/%m/%dT%H:%M:%S.000Z'
today = datetime.utcnow()
eod = today.replace(hour=23, minute=59, second=0, microsecond=0,
tzinfo=lib_martigny.get_timezone())

# STEP 0 :: Prepare data for test
# * Update the patron_type to set a overdue_items_limit rule.
# We define than only 1 overdue items are allowed. Trying a second
# checkout is disallowed if patron has an overdue item
patron_type = patron_type_children_martigny
patron_type \
.setdefault('limits', {}) \
.setdefault('overdue_items_limits', {}) \
.setdefault('default_value', 1)
patron_type.update(patron_type, dbcommit=True, reindex=True)
patron_type = PatronType.get_record_by_pid(patron_type.pid)
assert patron_type\
.get('limits', {})\
.get('overdue_items_limits', {}) \
.get('default_value') == 1

# [0] prepare overdue transaction
# STEP 1 :: Create an checkout with a end_date at the current date
# * Create a checkout and set end_date to a fixed_date equals to
# current tested date. The loan should not be considered as overdue
# and a second checkout should be possible
login_user_via_session(client, librarian_martigny.user)
# checkout
res, data = postdata(
client,
'api_item.checkout',
Expand All @@ -214,20 +238,43 @@ def test_overdue_limit(
patron_pid=patron_pid,
transaction_location_pid=loc_public_martigny.pid,
transaction_user_pid=librarian_martigny.pid,
end_date=eod.strftime(date_format)
)
)
assert res.status_code == 200

loan_pid = data.get('action_applied')[LoanAction.CHECKOUT].get('pid')
loan = Loan.get_record_by_pid(loan_pid)
assert not loan.is_loan_overdue()
end_date = datetime.now(timezone.utc) - timedelta(days=7)
loan['end_date'] = end_date.isoformat()
loan.update(
loan,
dbcommit=True,
reindex=True

res, data = postdata(
client,
'api_item.checkout',
dict(
item_pid=item2_lib_martigny.pid,
patron_pid=patron_pid,
transaction_location_pid=loc_public_martigny.pid,
transaction_user_pid=librarian_martigny.pid
)
)
assert res.status_code == 200
res, _ = postdata(
client,
'api_item.checkin',
dict(
item_pid=item2_lib_martigny.pid,
pid=data.get('action_applied')[LoanAction.CHECKOUT].get('pid'),
transaction_location_pid=loc_public_martigny.pid,
transaction_user_pid=librarian_martigny.pid,
)
)
assert res.status_code == 200

# STEP 2 :: Set the loan as overdue and test a new checkout
# Now there is one loan in overdue, then the limit is reached and a new
# checkout shouldn't be possible
end_date = eod - timedelta(days=7)
loan['end_date'] = end_date.isoformat()
loan.update(loan, dbcommit=True, reindex=True)

overdue_loans = list(get_overdue_loans(patron_pid=patron_pid))
assert loan.is_loan_overdue()
Expand All @@ -242,20 +289,7 @@ def test_overdue_limit(
flush_index(LoansSearch.Meta.index)
assert number_of_reminders_sent(loan) == 1

# [1] test overdue items limit

# Update the patron_type to set a overdue_items_limit rule
patron_type = patron_type_children_martigny
patron_type \
.setdefault('limits', {}) \
.setdefault('overdue_items_limits', {}) \
.setdefault('default_value', 1)
patron_type.update(patron_type, dbcommit=True, reindex=True)
patron_type = PatronType.get_record_by_pid(patron_type.pid)
assert patron_type.get('limits', {}).get('overdue_items_limits', {})\
.get('default_value') == 1

# [1.1] test overdue items limit when we try to checkout a second item
# Try a second checkout - limit should be reached
res, data = postdata(
client,
'api_item.checkout',
Expand All @@ -268,8 +302,7 @@ def test_overdue_limit(
)
assert res.status_code == 403
assert 'Checkout denied' in data['message']

# [1.2] test overdue items limit when we try to request another item
# Try a request - limit should be reached
res, data = postdata(
client,
'api_item.librarian_request',
Expand All @@ -283,8 +316,7 @@ def test_overdue_limit(
)
assert res.status_code == 403
assert 'maximal number of overdue items is reached' in data['message']

# [1.3] test overdue items limit when we try to extend loan
# Try to extend - limit should be reached
res, _ = postdata(
client,
'api_item.extend_loan',
Expand All @@ -294,7 +326,6 @@ def test_overdue_limit(
transaction_location_pid=loc_public_martigny.pid
)
)

assert res.status_code == 403
assert 'maximal number of overdue items is reached' in data['message']

Expand Down

0 comments on commit 09c9660

Please sign in to comment.