From 251ad10db75f72336a282fef2dc955aa03b48ef9 Mon Sep 17 00:00:00 2001 From: Jason Etheridge Date: Thu, 12 Jan 2023 01:16:29 -0500 Subject: [PATCH] LP1965446 Option to Disable Title-Level Holds on Bib Records with Parts This feature adds one global flag and one library setting, respectively: * circ.holds.api_require_monographic_part_when_present Holds: Require Monographic Part When Present for hold check. * circ.holds.ui_require_monographic_part_when_present Require Monographic Part when Present Normally the selection of a monographic part during hold placement is optional if there is at least one copy on the bib without a monographic part. A true value for this setting for any involved owning library for the bib or for the global flag will require part selection even under this condition. This essentially removes the All/Any Parts option from the part selection drop-down, for both versions of the public catalog (TPAC and BOOPAC), and for the Angular staff catalog interface. At the API level, we consider just the global flag and will throw a TITLE_HOLD_WHEN_MONOGRAPHIC_PART_REQUIRED event for a title hold request when there are items with monographic parts on the bib. It is possible for the library settings and the global flag to differ, but the global flag will catch every instance of hold placement including those by third party callers, SIP, etc. Signed-off-by: Jason Etheridge Signed-off-by: Ruth Frasur Signed-off-by: Jennifer Weston Signed-off-by: Michele Morgan --- .../src/app/staff/catalog/hold/hold.component.html | 4 +- .../src/app/staff/catalog/hold/hold.component.ts | 7 +- .../eg2/src/app/staff/share/holds/holds.service.ts | 1 + .../perlmods/lib/OpenILS/Application/Circ/Holds.pm | 91 ++++++++++++++++++++++ .../src/perlmods/lib/OpenILS/WWW/EGCatLoader.pm | 3 + .../lib/OpenILS/WWW/EGCatLoader/Account.pm | 56 ++++++++++--- Open-ILS/src/sql/Pg/950.data.seed-values.sql | 27 +++++++ Open-ILS/src/sql/Pg/upgrade/XXXX.schema.eparts.sql | 33 ++++++++ .../opac/parts/hold_error_messages.tt2 | 1 + .../templates/opac/parts/hold_error_messages.tt2 | 1 + Open-ILS/web/js/dojo/openils/circ/nls/selfcheck.js | 1 + docs/RELEASE_NOTES_NEXT/miscellaneous.adoc | 20 +++++ 12 files changed, 230 insertions(+), 15 deletions(-) create mode 100644 Open-ILS/src/sql/Pg/upgrade/XXXX.schema.eparts.sql diff --git a/Open-ILS/src/eg2/src/app/staff/catalog/hold/hold.component.html b/Open-ILS/src/eg2/src/app/staff/catalog/hold/hold.component.html index a367287c03..ce18269e1b 100644 --- a/Open-ILS/src/eg2/src/app/staff/catalog/hold/hold.component.html +++ b/Open-ILS/src/eg2/src/app/staff/catalog/hold/hold.component.html @@ -310,8 +310,8 @@
diff --git a/Open-ILS/src/eg2/src/app/staff/catalog/hold/hold.component.ts b/Open-ILS/src/eg2/src/app/staff/catalog/hold/hold.component.ts index a74bc37354..fc4c83690f 100644 --- a/Open-ILS/src/eg2/src/app/staff/catalog/hold/hold.component.ts +++ b/Open-ILS/src/eg2/src/app/staff/catalog/hold/hold.component.ts @@ -245,7 +245,7 @@ export class HoldComponent implements OnInit { getTargetMeta(): Promise { return new Promise(resolve => { - this.holds.getHoldTargetMeta(this.holdType, this.holdTargets) + this.holds.getHoldTargetMeta(this.holdType, this.holdTargets, this.auth.user().ws_ou()) .subscribe( meta => { this.holdContexts.filter(ctx => ctx.holdTarget === meta.target) @@ -612,6 +612,11 @@ export class HoldComponent implements OnInit { let hType = this.holdType; let hTarget = ctx.holdTarget; + + if (ctx.holdMeta.parts && !ctx.holdMeta.part) { + ctx.holdMeta.part = (ctx.holdMeta.part_required ? ctx.holdMeta.parts[0] : null); + } + if (hType === 'T' && ctx.holdMeta.part) { // A Title hold morphs into a Part hold at hold placement time // if a part is selected. This can happen on a per-hold basis diff --git a/Open-ILS/src/eg2/src/app/staff/share/holds/holds.service.ts b/Open-ILS/src/eg2/src/app/staff/share/holds/holds.service.ts index f1c6d0b106..b5d5531fcb 100644 --- a/Open-ILS/src/eg2/src/app/staff/share/holds/holds.service.ts +++ b/Open-ILS/src/eg2/src/app/staff/share/holds/holds.service.ts @@ -55,6 +55,7 @@ export interface HoldRequestTarget { copy?: IdlObject; issuance?: IdlObject; metarecord_filters?: any; + part_required?: boolean; } /** Service for performing various hold-related actions */ diff --git a/Open-ILS/src/perlmods/lib/OpenILS/Application/Circ/Holds.pm b/Open-ILS/src/perlmods/lib/OpenILS/Application/Circ/Holds.pm index 6e9878be21..d295fa2e72 100644 --- a/Open-ILS/src/perlmods/lib/OpenILS/Application/Circ/Holds.pm +++ b/Open-ILS/src/perlmods/lib/OpenILS/Application/Circ/Holds.pm @@ -2877,6 +2877,46 @@ sub _check_title_hold_is_possible { # $holdable_formats is now unused. We pre-filter the MR's records. my $e = new_editor(); + + # T holds on records that have parts are normally OK, but if the record has + # no non-part copies, the hold will ultimately fail, so let's test for that. + # + # If the global flag circ.holds.api_require_monographic_part_when_present is + # enabled, then any configured parts for the bib is enough to disallow title holds. + my $part_required = 0; + my $parts = $e->search_biblio_monograph_part( + { + record => $titleid + }, {idlist=>1} ); + + if (@$parts) { + my $part_required_flag = $e->retrieve_config_global_flag('circ.holds.api_require_monographic_part_when_present'); + $part_required = ($part_required_flag and $U->is_true($part_required_flag->enabled)); + if (!$part_required) { + my $np_copies = $e->json_query({ + select => { acp => [{column => 'id', transform => 'count', alias => 'count'}]}, + from => {acp => {acn => {}, acpm => {type => 'left'}}}, + where => { + '+acp' => {deleted => 'f'}, + '+acn' => {deleted => 'f', record => $titleid}, + '+acpm' => {id => undef} + } + }); + $part_required = 1 if $np_copies->[0]->{count} == 0; + } + } + if ($part_required) { + $logger->info("title hold when monographic part required"); + return ( + 0, 0, [ + new OpenILS::Event( + "TITLE_HOLD_WHEN_MONOGRAPHIC_PART_REQUIRED", + "payload" => {"fail_part" => "monographic_part_required"} + ) + ] + ); + } + my %org_filter = create_ranged_org_filter($e, $selection_ou, $depth); # this monster will grab the id and circ_lib of all of the "holdable" copies for the given record @@ -5153,6 +5193,7 @@ sub hold_metadata { issuance => $issuance, part => $part, parts => [], + part_required => 'f', bibrecord => $bre, metarecord => $metarecord, metarecord_filters => {} @@ -5178,6 +5219,56 @@ sub hold_metadata { {order_by => {bmp => 'label_sortkey'}} ] ); + + # T holds on records that have parts are normally OK, but if the record has + # no non-part copies, the hold will ultimately fail. When that happens, + # require the user to select a part. + # + # If the global flag circ.holds.api_require_monographic_part_when_present is + # enabled, or the library setting circ.holds.ui_require_monographic_part_when_present + # is true for any involved owning_library, then also require part selection. + my $part_required = 0; + if ($meta->{parts}) { + my $part_required_flag = $e->retrieve_config_global_flag('circ.holds.api_require_monographic_part_when_present'); + $part_required = ($part_required_flag and $U->is_true($part_required_flag->enabled)); + if (!$part_required) { + my $resp = $e->json_query({ + select => { + acn => ['owning_lib'] + }, + from => {acn => {acp => {type => 'left'}}}, + where => { + '+acp' => { + '-or' => [ + {deleted => 'f'}, + {id => undef} # left join + ] + }, + '+acn' => {deleted => 'f', record => $bre->id} + }, + distinct => 't' + }); + my $org_ids = [map {$_->{owning_lib}} @$resp]; + foreach my $org (@$org_ids) { # FIXME: worth shortcutting/optimizing? + if ($U->ou_ancestor_setting_value($org, 'circ.holds.ui_require_monographic_part_when_present')) { + $part_required = 1; + } + } + } + if (!$part_required) { + my $np_copies = $e->json_query({ + select => { acp => [{column => 'id', transform => 'count', alias => 'count'}]}, + from => {acp => {acn => {}, acpm => {type => 'left'}}}, + where => { + '+acp' => {deleted => 'f'}, + '+acn' => {deleted => 'f', record => $bre->id}, + '+acpm' => {id => undef} + } + }); + $part_required = 1 if $np_copies->[0]->{count} == 0; + } + } + $meta->{part_required} = $part_required; } if ($meta->{metarecord}) { diff --git a/Open-ILS/src/perlmods/lib/OpenILS/WWW/EGCatLoader.pm b/Open-ILS/src/perlmods/lib/OpenILS/WWW/EGCatLoader.pm index fab9dd6a6e..f5c59a968c 100644 --- a/Open-ILS/src/perlmods/lib/OpenILS/WWW/EGCatLoader.pm +++ b/Open-ILS/src/perlmods/lib/OpenILS/WWW/EGCatLoader.pm @@ -374,6 +374,9 @@ sub load_common { my $geo_org = $ctx->{physical_loc} || $self->cgi->param('loc') || $ctx->{aou_tree}->()->id; my $geo_sort_for_org = $ctx->{get_org_setting}->($geo_org, 'opac.holdings_sort_by_geographic_proximity'); $ctx->{geo_sort} = $geo_sort && $U->is_true($geo_sort_for_org); + my $part_required_flag = $e->retrieve_config_global_flag('circ.holds.api_require_monographic_part_when_present'); + $part_required_flag = ($part_required_flag and $U->is_true($part_required_flag->enabled)); + $ctx->{part_required_when_present_global_flag} = $part_required_flag; # capture some commonly accessed pages $ctx->{home_page} = $ctx->{proto} . '://' . $ctx->{hostname} . $self->ctx->{opac_root} . "/home"; diff --git a/Open-ILS/src/perlmods/lib/OpenILS/WWW/EGCatLoader/Account.pm b/Open-ILS/src/perlmods/lib/OpenILS/WWW/EGCatLoader/Account.pm index 6dacc2d20b..a98d789e79 100644 --- a/Open-ILS/src/perlmods/lib/OpenILS/WWW/EGCatLoader/Account.pm +++ b/Open-ILS/src/perlmods/lib/OpenILS/WWW/EGCatLoader/Account.pm @@ -1622,21 +1622,53 @@ sub load_place_hold { {record => $rec->id} ); - # T holds on records that have parts are OK, but if the record has - # no non-part copies, the hold will ultimately fail. When that - # happens, require the user to select a part. + # T holds on records that have parts are normally OK, but if the record has + # no non-part copies, the hold will ultimately fail. When that happens, + # require the user to select a part. + # + # If the global flag circ.holds.api_require_monographic_part_when_present is + # enabled, or the library setting circ.holds.ui_require_monographic_part_when_present + # is active for any item owning library associated with the bib, then any configured + # parts for the bib is enough to disallow title holds (aka require part selection). my $part_required = 0; if (@$parts) { - my $np_copies = $e->json_query({ - select => { acp => [{column => 'id', transform => 'count', alias => 'count'}]}, - from => {acp => {acn => {}, acpm => {type => 'left'}}}, - where => { - '+acp' => {deleted => 'f'}, - '+acn' => {deleted => 'f', record => $rec->id}, - '+acpm' => {id => undef} + $part_required = $ctx->{part_required_when_present_global_flag}; + if (!$part_required) { + my $resp = $e->json_query({ + select => { + acn => ['owning_lib'] + }, + from => {acn => {acp => {type => 'left'}}}, + where => { + '+acp' => { + '-or' => [ + {deleted => 'f'}, + {id => undef} # left join + ] + }, + '+acn' => {deleted => 'f', record => $rec->id} + }, + distinct => 't' + }); + my $org_ids = [map {$_->{owning_lib}} @$resp]; + foreach my $org (@$org_ids) { # FIXME: worth shortcutting/optimizing? + if ($self->ctx->{get_org_setting}->($org, 'circ.holds.ui_require_monographic_part_when_present')) { + $part_required = 1; + } } - }); - $part_required = 1 if $np_copies->[0]->{count} == 0; + } + if (!$part_required) { + my $np_copies = $e->json_query({ + select => { acp => [{column => 'id', transform => 'count', alias => 'count'}]}, + from => {acp => {acn => {}, acpm => {type => 'left'}}}, + where => { + '+acp' => {deleted => 'f'}, + '+acn' => {deleted => 'f', record => $rec->id}, + '+acpm' => {id => undef} + } + }); + $part_required = 1 if $np_copies->[0]->{count} == 0; + } } push(@hold_data, $data_filler->({ diff --git a/Open-ILS/src/sql/Pg/950.data.seed-values.sql b/Open-ILS/src/sql/Pg/950.data.seed-values.sql index 2867f6ff88..3d123adffc 100644 --- a/Open-ILS/src/sql/Pg/950.data.seed-values.sql +++ b/Open-ILS/src/sql/Pg/950.data.seed-values.sql @@ -21608,6 +21608,33 @@ VALUES ( 'integer' ); +-- eparts + +INSERT INTO config.global_flag (name, value, enabled, label) +VALUES ( + 'circ.holds.api_require_monographic_part_when_present', + NULL, + FALSE, + oils_i18n_gettext( + 'circ.holds.api_require_monographic_part_when_present', + 'Holds: Require Monographic Part When Present for hold check.', + 'cgf', 'label' + ) +); + +INSERT INTO config.org_unit_setting_type (name, label, grp, description, datatype) +VALUES ( + 'circ.holds.ui_require_monographic_part_when_present', + oils_i18n_gettext('circ.holds.ui_require_monographic_part_when_present', + 'Require Monographic Part when Present', + 'coust', 'label'), + 'circ', + oils_i18n_gettext('circ.holds.ui_require_monographic_part_when_present', + 'Normally the selection of a monographic part during hold placement is optional if there is at least one copy on the bib without a monographic part. A true value for this setting will require part selection even under this condition.', + 'coust', 'description'), + 'bool' +); + ------------------- Disabled example A/T defintions ------------------------------ -- Create a "dummy" slot when applicable, and trigger the "offer curbside" events diff --git a/Open-ILS/src/sql/Pg/upgrade/XXXX.schema.eparts.sql b/Open-ILS/src/sql/Pg/upgrade/XXXX.schema.eparts.sql new file mode 100644 index 0000000000..f384372b53 --- /dev/null +++ b/Open-ILS/src/sql/Pg/upgrade/XXXX.schema.eparts.sql @@ -0,0 +1,33 @@ +BEGIN; + +-- check whether patch can be applied +SELECT evergreen.upgrade_deps_block_check('XXXX', :eg_version); + +-- 950.data.seed-values.sql + +INSERT INTO config.global_flag (name, value, enabled, label) +VALUES ( + 'circ.holds.api_require_monographic_part_when_present', + NULL, + FALSE, + oils_i18n_gettext( + 'circ.holds.api_require_monographic_part_when_present', + 'Holds: Require Monographic Part When Present for hold check.', + 'cgf', 'label' + ) +); + +INSERT INTO config.org_unit_setting_type (name, label, grp, description, datatype) +VALUES ( + 'circ.holds.ui_require_monographic_part_when_present', + oils_i18n_gettext('circ.holds.ui_require_monographic_part_when_present', + 'Require Monographic Part when Present', + 'coust', 'label'), + 'circ', + oils_i18n_gettext('circ.holds.ui_require_monographic_part_when_present', + 'Normally the selection of a monographic part during hold placement is optional if there is at least one copy on the bib without a monographic part. A true value for this setting will require part selection even under this condition.', + 'coust', 'description'), + 'bool' +); + +COMMIT; diff --git a/Open-ILS/src/templates-bootstrap/opac/parts/hold_error_messages.tt2 b/Open-ILS/src/templates-bootstrap/opac/parts/hold_error_messages.tt2 index 407ea52daf..f36a0ac433 100755 --- a/Open-ILS/src/templates-bootstrap/opac/parts/hold_error_messages.tt2 +++ b/Open-ILS/src/templates-bootstrap/opac/parts/hold_error_messages.tt2 @@ -23,6 +23,7 @@ "status.holdable" => l("The item is not in a holdable status"), "no_item" => l("The system could not find this item"), "no_ultimate_items" => l("The system could not find any items to match this hold request"), + "monographic_part_required" => l("Title hold request invalid when monographic part required"), "no_matchpoint" => l("System rules do not define how to handle this item"), "no_user" => l("The system could not find this patron"), "transit_range" => l("The item cannot transit this far") diff --git a/Open-ILS/src/templates/opac/parts/hold_error_messages.tt2 b/Open-ILS/src/templates/opac/parts/hold_error_messages.tt2 index 407ea52daf..f36a0ac433 100644 --- a/Open-ILS/src/templates/opac/parts/hold_error_messages.tt2 +++ b/Open-ILS/src/templates/opac/parts/hold_error_messages.tt2 @@ -23,6 +23,7 @@ "status.holdable" => l("The item is not in a holdable status"), "no_item" => l("The system could not find this item"), "no_ultimate_items" => l("The system could not find any items to match this hold request"), + "monographic_part_required" => l("Title hold request invalid when monographic part required"), "no_matchpoint" => l("System rules do not define how to handle this item"), "no_user" => l("The system could not find this patron"), "transit_range" => l("The item cannot transit this far") diff --git a/Open-ILS/web/js/dojo/openils/circ/nls/selfcheck.js b/Open-ILS/web/js/dojo/openils/circ/nls/selfcheck.js index 438d55678b..e9e8e60da6 100644 --- a/Open-ILS/web/js/dojo/openils/circ/nls/selfcheck.js +++ b/Open-ILS/web/js/dojo/openils/circ/nls/selfcheck.js @@ -41,6 +41,7 @@ "FAIL_PART_config_rule_age_hold_protect_prox": "The item is too new to transit this far", "FAIL_PART_no_item": "The system could not find this item", "FAIL_PART_no_ultimate_items": "The system could not find any items to match this hold request", + "FAIL_PART_monographic_part_required": "Title hold request invalid when monographic part required", "FAIL_PART_no_matchpoint": "System rules do not define how to handle this item", "FAIL_PART_no_user": "The system could not find this patron", "FAIL_PART_transit_range": "The item cannot transit this far", diff --git a/docs/RELEASE_NOTES_NEXT/miscellaneous.adoc b/docs/RELEASE_NOTES_NEXT/miscellaneous.adoc index a17df9a7b8..dc245056ce 100644 --- a/docs/RELEASE_NOTES_NEXT/miscellaneous.adoc +++ b/docs/RELEASE_NOTES_NEXT/miscellaneous.adoc @@ -32,3 +32,23 @@ * Adds new Local Administration entries for Item Statistical Categories Editor and Patron Statistical Categories Editor, which are angularized interfaces. * Tweaks eg-grids to underline hyperlinks within cells. This potentially affects multiple interfaces. * eg-org-family-select now supports persistKey +* LP1965446 Option to Disable Title-Level Holds on Bib Records with Parts + + This feature adds one global flag and one library setting, respectively: + + * circ.holds.api_require_monographic_part_when_present + Holds: Require Monographic Part When Present for hold check. + * circ.holds.ui_require_monographic_part_when_present + Require Monographic Part when Present + + Normally the selection of a monographic part during hold placement is optional if there is at least one copy + on the bib without a monographic part. A true value for this setting for any involved owning library for the + bib or for the global flag will require part selection even under this condition. This essentially removes + the All/Any Parts option from the part selection drop-down, for both versions of the public catalog (TPAC and + BOOPAC), and for the Angular staff catalog interface. It should be noted that if the library setting is set + below the consortium level, Title level holds may be allowed for some libraries and not others. + + At the API level, we consider just the global flag and will throw a TITLE_HOLD_WHEN_MONOGRAPHIC_PART_REQUIRED + event for a title hold request when there are items with monographic parts on the bib. It is possible for + the library settings and the global flag to differ, but the global flag will catch every instance of hold + placement including those by third party callers, SIP, etc. -- 2.11.0