From 9adf2354d7b976205a4165677ae59929700cd0a9 Mon Sep 17 00:00:00 2001 From: Jason Stephenson <jason@sigio.com> Date: Fri, 29 Jan 2021 19:36:19 -0500 Subject: [PATCH] Lp 1861319: Add Release Notes and Perl Live Tests Signed-off-by: Jason Stephenson <jason@sigio.com> Signed-off-by: Galen Charlton <gmc@equinoxinitiative.org> --- .../32-lp1861319-circ-renewal-expired-patrons.t | 368 +++++++++++++++++++++ .../Circulation/renewal_for_expired_patrons.adoc | 15 + 2 files changed, 383 insertions(+) create mode 100755 Open-ILS/src/perlmods/live_t/32-lp1861319-circ-renewal-expired-patrons.t create mode 100644 docs/RELEASE_NOTES_NEXT/Circulation/renewal_for_expired_patrons.adoc diff --git a/Open-ILS/src/perlmods/live_t/32-lp1861319-circ-renewal-expired-patrons.t b/Open-ILS/src/perlmods/live_t/32-lp1861319-circ-renewal-expired-patrons.t new file mode 100755 index 0000000000..64e6b188bd --- /dev/null +++ b/Open-ILS/src/perlmods/live_t/32-lp1861319-circ-renewal-expired-patrons.t @@ -0,0 +1,368 @@ +#!perl +use strict; use warnings; +use Test::More tests => 18; +use OpenILS::Utils::TestUtils; +use OpenILS::Const qw(:const); +use OpenILS::Utils::DateTime; +use DateTime; +use DateTime::TimeZone; +use DateTime::Format::ISO8601; + +# We need the local timezone for later. +my $localTZ = DateTime::TimeZone->new(name => 'local'); + +my $script = OpenILS::Utils::TestUtils->new(); +my $U = 'OpenILS::Application::AppUtils'; + +diag("Test LP 1861319 Allow Circulation Renewal for Expired Patrons"); + +use constant { + BR1_ID => 4, + BR2_ID => 6, + BR1_WORKSTATION => 'BR1-lp1861319-test-renewal', + BR2_WORKSTATION => 'BR2-lp1861319-test renewal', + ADMIN_USER => 'admin', + ADMIN_PASS => 'demo123' +}; + +# Login +$script->authenticate({ + username => ADMIN_USER, + password => ADMIN_PASS, + type => 'staff' +}); +BAIL_OUT('Failed to Login') unless ($script->authtoken); + +# Register BR1 workstation. +my $ws = $script->find_or_register_workstation(BR1_WORKSTATION, BR1_ID); +BAIL_OUT("Failed to register " . BR1_WORKSTATION) unless($ws); + +# Logout +$script->logout(); + +# Login with BR1 Workstation +$script->authenticate({ + username => ADMIN_USER, + password => ADMIN_PASS, + type => 'staff', + workstation => BR1_WORKSTATION +}); +BAIL_OUT('Failed to login with ' . BR1_WORKSTATION) unless($script->authtoken); + +# Get a cstore editor for later use. +my $editor = $script->editor(authtoken=>$script->authtoken); + +# Check that the circ.renew.expired_patron_allow setting constant is defined +ok(defined(OILS_SETTING_ALLOW_RENEW_FOR_EXPIRED_PATRON), + 'OILS_SETTING_ALLOW_RENEW_FOR_EXPIRED_PATRON constant exists'); + +# Check that the circ.renew.expired_patron_allow setting exists in the database +my $setting = $editor->search_config_org_unit_setting_type( + {name => OILS_SETTING_ALLOW_RENEW_FOR_EXPIRED_PATRON} +); +# Get the first/only one: +$setting = (defined($setting)) ? $setting->[0] : $setting; +ok(defined($setting), OILS_SETTING_ALLOW_RENEW_FOR_EXPIRED_PATRON . ' setting exists in database'); + +# Find a circulation with renewals remaining +my $circ = $editor->search_action_circulation([ + { + circ_lib => BR1_ID, + checkin_time => undef, + renewal_remaining => {'>' => 0} + }, + {limit => 1} +]); +$circ = defined($circ) ? $circ->[0] : $circ; +isa_ok($circ, 'Fieldmapper::action::circulation', 'Found open circulation at BR1'); + +# Get the circ patron information. +my $patron = $editor->retrieve_actor_user($circ->usr); +isa_ok($patron, 'Fieldmapper::actor::user', 'Found circulation user'); + +# Expire the patron if they are not already expired. +my ($saved_expire_date); +SKIP: { + if (check_usr_expired($patron)) { + skip 'Patron already expired', 1; + } else { + $saved_expire_date = $patron->expire_date; + $patron->expire_date(DateTime->now()->set_time_zone($localTZ)->subtract(days => 1)->strftime('%FT%T%z')); + $editor->xact_begin; + if ($editor->update_actor_user($patron)) { + $editor->xact_commit; + } else { + $editor->xact_rollback; + BAIL_OUT("Patron update failed"); + } + $patron = $editor->retrieve_actor_user($patron->id); + ok(check_usr_expired($patron), 'Patron set to expired.'); + } +} + +# Attempt a renewal. It should fail. +my $renew_params = { + copy_id => $circ->target_copy, + patron_id => $circ->usr, + desk_renewal => 1 +}; +my $result = $U->simplereq( + 'open-ils.circ', + 'open-ils.circ.renew.override', + $script->authtoken, + $renew_params +); +if (ref($result) eq 'ARRAY') { + $result = $result->[0]; +} +is($result->{textcode}, 'PATRON_ACCOUNT_EXPIRED', 'Renewal failed: ' . $result->{textcode}); + +# Set the circ.renew.expired_patron_allow setting at BR1 +$setting = Fieldmapper::actor::org_unit_setting->new; +$setting->name(OILS_SETTING_ALLOW_RENEW_FOR_EXPIRED_PATRON); +$setting->org_unit(BR1_ID); +$setting->value('true'); +$editor->xact_begin; +$result = $editor->create_actor_org_unit_setting($setting); +if ($result) { + $editor->xact_commit; + ok($result, 'Set ' . OILS_SETTING_ALLOW_RENEW_FOR_EXPIRED_PATRON . ' for BR1'); +} else { + $editor->xact_rollback; + BAIL_OUT("Failed to set " . OILS_SETTING_ALLOW_RENEW_FOR_EXPIRED_PATRON . ' for BR1'); +} + +# Attempt the renewal again, expect success. +$result = $U->simplereq( + 'open-ils.circ', + 'open-ils.circ.renew.override', + $script->authtoken, + $renew_params +); +if (ref($result) eq 'ARRAY') { + $result = $result->[0]; +} +isnt($result->{textcode}, 'PATRON_ACCOUNT_EXPIRED', 'Renewal Result: ' . $result->{textcode}); + +# Find a circulating copy at BR1 that is not checked out. +my $copy = $editor->search_asset_copy([ + { + circ_lib => BR1_ID, + status => OILS_COPY_STATUS_AVAILABLE, + circulate => 't' + }, + {limit => 1} +]); +$copy = defined($copy) ? $copy->[0] : $copy; +isa_ok($copy, 'Fieldmapper::asset::copy', 'Found copy at BR1'); + +# Check it out, expect failure +my $checkout_params = { + copy_id => $copy->id, + patron_id => $patron->id +}; +$result = $U->simplereq( + 'open-ils.circ', + 'open-ils.circ.checkout.full.override', + $script->authtoken, + $checkout_params +); +if (ref($result) eq 'ARRAY') { + $result = $result->[0]; +} +is($result->{textcode}, 'PATRON_ACCOUNT_EXPIRED', 'Checkout failed: ' . $result->{textcode}); + +# Reset the patron expire_date +if (defined($saved_expire_date)) { + $patron->expire_date($saved_expire_date); + $editor->xact_begin; + if ($editor->update_actor_user($patron)) { + $editor->xact_commit; + } else { + $editor->xact_rollback; + } + undef($saved_expire_date); +} + +# Logout +$script->logout(); + +# Destroy our editor. +undef($editor); + +# Login +$script->authenticate({ + username => ADMIN_USER, + password => ADMIN_PASS, + type => 'staff' +}); +BAIL_OUT('Failed to Login') unless ($script->authtoken); + +# Register BR2 workstation. +$ws = $script->find_or_register_workstation(BR2_WORKSTATION, BR2_ID); +BAIL_OUT("Failed to register " . BR2_WORKSTATION) unless($ws); + +# Logout +$script->logout(); + +# Login with BR2 Workstation +$script->authenticate({ + username => ADMIN_USER, + password => ADMIN_PASS, + type => 'staff', + workstation => BR2_WORKSTATION +}); +BAIL_OUT('Failed to login with ' . BR2_WORKSTATION) unless($script->authtoken); + +# Get a new editor with our authtoken. +$editor = $script->editor(authtoken=>$script->authtoken); + +# Find a circulation with renewals remaining at BR2 +$circ = $editor->search_action_circulation([ + { + circ_lib => BR2_ID, + checkin_time => undef, + renewal_remaining => {'>' => 0} + }, + {limit => 1} +]); +$circ = defined($circ) ? $circ->[0] : $circ; +isa_ok($circ, 'Fieldmapper::action::circulation', 'Found open circulation at BR2'); + +# Get the circ patron information. +$patron = $editor->retrieve_actor_user($circ->usr); +isa_ok($patron, 'Fieldmapper::actor::user', 'Found circulation user'); + +# Expire the patron if they are not already expired. +SKIP: { + if (check_usr_expired($patron)) { + skip 'Patron already expired', 1; + } else { + $saved_expire_date = $patron->expire_date; + $patron->expire_date(DateTime->now()->set_time_zone($localTZ)->subtract(days => 1)->strftime('%FT%T%z')); + $editor->xact_begin; + if ($editor->update_actor_user($patron)) { + $editor->xact_commit; + $patron = $editor->retrieve_actor_user($patron->id); + ok(check_usr_expired($patron), 'Patron set to expired.'); + } else { + $editor->xact_rollback; + BAIL_OUT("Patron update failed"); + } + } +} + +# Attempt a renewal. It should fail. +$renew_params = { + copy_id => $circ->target_copy, + patron_id => $circ->usr, + desk_renewal => 1 +}; +$result = $U->simplereq( + 'open-ils.circ', + 'open-ils.circ.renew.override', + $script->authtoken, + $renew_params +); +if (ref($result) eq 'ARRAY') { + $result = $result->[0]; +} +is($result->{textcode}, 'PATRON_ACCOUNT_EXPIRED', 'Renewal failed: ' . $result->{textcode}); + +# Reset the patron expire_date +if (defined($saved_expire_date)) { + $patron->expire_date($saved_expire_date); + $editor->xact_begin; + if ($editor->update_actor_user($patron)) { + $editor->xact_commit; + ok(!check_usr_expired($patron), 'Patron set to not expired'); + } else { + $editor->xact_rollback; + BAIL_OUT('Patron expire date reset failed'); + } + undef($saved_expire_date); +} else { + # Set patron expire date to 30 days in the future. + $saved_expire_date = $patron->expire_date; + $patron->expire_date(DateTime->now()->set_time_zone($localTZ)->add(days => 30)->strftime('%FT%T%z')); + $editor->xact_begin; + if ($editor->update_actor_user($patron)) { + $editor->xact_commit; + $patron = $editor->retrieve_actor_user($patron->id); + ok(!check_usr_expired($patron), 'Patron is not expired.'); + } else { + $editor->xact_rollback; + BAIL_OUT("Patron update failed"); + } +} + +# Attempt renewal, expect success. +$result = $U->simplereq( + 'open-ils.circ', + 'open-ils.circ.renew.override', + $script->authtoken, + $renew_params +); +if (ref($result) eq 'ARRAY') { + $result = $result->[0]; +} +isnt($result->{textcode}, 'PATRON_ACCOUNT_EXPIRED', 'Renewal Result: ' . $result->{textcode}); + +# Find a circulating copy at BR2 that is not checked out. +$copy = $editor->search_asset_copy([ + { + circ_lib => BR2_ID, + status => OILS_COPY_STATUS_AVAILABLE, + circulate => 't' + }, + {limit => 1} +]); +$copy = defined($copy) ? $copy->[0] : $copy; +isa_ok($copy, 'Fieldmapper::asset::copy', 'Found copy at BR2'); + +# Check it out, expect failure +$checkout_params = { + copy_id => $copy->id, + patron_id => $patron->id +}; +$result = $U->simplereq( + 'open-ils.circ', + 'open-ils.circ.checkout.full.override', + $script->authtoken, + $checkout_params +); +if (ref($result) eq 'ARRAY') { + $result = $result->[0]; +} +isnt($result->{textcode}, 'PATRON_ACCOUNT_EXPIRED', 'Checkout result: ' . $result->{textcode}); + +# Reset the patron expire_date if necessary +if (defined($saved_expire_date)) { + $patron->expire_date($saved_expire_date); + $editor->xact_begin; + if ($editor->update_actor_user($patron)) { + $editor->xact_commit; + } else { + $editor->xact_rollback; + } + undef($saved_expire_date); +} + +# Delete the setting from actor.org_unit_setting +$editor->xact_begin; +if ($editor->delete_actor_org_unit_setting($setting)) { + $editor->commit; # Commit so that we disconnect. +} else { + $editor->rollback; +} + +# Logout +$script->logout(); # Not a test, just to be pedantic. + +# Utiity functions +sub check_usr_expired { + my ($user) = @_; + my $expire = DateTime::Format::ISO8601->new->parse_datetime( + OpenILS::Utils::DateTime->clean_ISO8601($user->expire_date)); + return (time > $expire->epoch); +} diff --git a/docs/RELEASE_NOTES_NEXT/Circulation/renewal_for_expired_patrons.adoc b/docs/RELEASE_NOTES_NEXT/Circulation/renewal_for_expired_patrons.adoc new file mode 100644 index 0000000000..8281b2e097 --- /dev/null +++ b/docs/RELEASE_NOTES_NEXT/Circulation/renewal_for_expired_patrons.adoc @@ -0,0 +1,15 @@ +Allow Circulation Renewal for Expired Patrons +^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ +The "Allow renewal request if renewal recipient privileges have +expired" organizational unit setting can be set to true to permit +expired patrons to renew circulations. Allowing renewals for expired +patrons reduces the number of auto-renewal failures and assumes that a +patron with items out eligible for renewals has not been expired for +very long and that such patrons are likely to renew their privileges +in a timely manner. + +The setting is referenced based on the current circulation library for +the renewal. It takes into account the global flags for "Circ: Use +original circulation library on desk renewal instead of the +workstation library" and "Circ: Use original circulation library on +opac renewal instead of user home library." -- 2.11.0