1 package OpenILS::Application::Storage::Publisher::container;
2 use base qw/OpenILS::Application::Storage/;
4 use OpenSRF::EX qw/:try/;
5 use OpenSRF::Utils::Logger qw/:level :logger/;
6 use OpenILS::Utils::CStoreEditor;
7 #use OpenILS::Application::Storage::CDBI::config;
9 my $new_items_query = q(
10 WITH c_attr AS (SELECT c_attrs::query_int AS vis_test FROM asset.patron_default_visibility_mask() x)
11 SELECT acn.record AS bib
12 FROM asset.call_number acn
13 JOIN asset.copy acp ON (acp.call_number = acn.id)
14 JOIN asset.copy_location acpl ON (acp.location = acpl.id)
15 JOIN config.copy_status ccs ON (acp.status = ccs.id)
17 WHERE acn.owning_lib IN (ORG_LIST)
18 AND acp.circ_lib IN (ORG_LIST)
24 AND acp.active_date > NOW() - ?::INTERVAL
25 AND (EXISTS (SELECT 1 FROM asset.copy_vis_attr_cache WHERE record = acn.record AND vis_attr_vector @@ c_attr.vis_test))
26 AND (NOT EXISTS (SELECT 1 FROM metabib.record_attr_vector_list WHERE source = acn.record AND vlist @@ metabib.compile_composite_attr(' {"1":[{"_val":"s","_attr":"bib_level"}]}')::query_int))
28 ORDER BY MIN(AGE(acp.active_date))
31 my $recently_returned_query = q(
32 WITH c_attr AS (SELECT c_attrs::query_int AS vis_test FROM asset.patron_default_visibility_mask() x)
33 SELECT acn.record AS bib
34 FROM asset.call_number acn
35 JOIN asset.copy acp ON (acp.call_number = acn.id)
36 JOIN asset.copy_location acpl ON (acp.location = acpl.id)
37 JOIN config.copy_status ccs ON (acp.status = ccs.id)
38 JOIN action.circulation circ ON (circ.target_copy = acp.id)
40 WHERE acn.owning_lib IN (ORG_LIST)
41 AND acp.circ_lib IN (ORG_LIST)
47 AND circ.checkin_time > NOW() - ?::INTERVAL
48 AND circ.checkin_time IS NOT NULL
49 AND (EXISTS (SELECT 1 FROM asset.copy_vis_attr_cache WHERE record = acn.record AND vis_attr_vector @@ c_attr.vis_test))
50 AND (NOT EXISTS (SELECT 1 FROM metabib.record_attr_vector_list WHERE source = acn.record AND vlist @@ metabib.compile_composite_attr(' {"1":[{"_val":"s","_attr":"bib_level"}]}')::query_int))
52 ORDER BY MIN(AGE(circ.checkin_time))
55 my $top_circs_query = q(
56 WITH c_attr AS (SELECT c_attrs::query_int AS vis_test FROM asset.patron_default_visibility_mask() x)
57 SELECT acn.record AS bib
58 FROM asset.call_number acn
59 JOIN asset.copy acp ON (acp.call_number = acn.id)
60 JOIN asset.copy_location acpl ON (acp.location = acpl.id)
61 JOIN config.copy_status ccs ON (acp.status = ccs.id)
62 JOIN action.circulation circ ON (circ.target_copy = acp.id)
64 WHERE acn.owning_lib IN (ORG_LIST)
65 AND acp.circ_lib IN (ORG_LIST)
71 AND circ.xact_start > NOW() - ?::INTERVAL
72 AND (EXISTS (SELECT 1 FROM asset.copy_vis_attr_cache WHERE record = acn.record AND vis_attr_vector @@ c_attr.vis_test))
73 AND (NOT EXISTS (SELECT 1 FROM metabib.record_attr_vector_list WHERE source = acn.record AND vlist @@ metabib.compile_composite_attr(' {"1":[{"_val":"s","_attr":"bib_level"}]}')::query_int))
75 ORDER BY COUNT(circ.id) DESC
78 my $new_by_loc_query = q(
79 WITH c_attr AS (SELECT c_attrs::query_int AS vis_test FROM asset.patron_default_visibility_mask() x)
80 SELECT acn.record AS bib
81 FROM asset.call_number acn
82 JOIN asset.copy acp ON (acp.call_number = acn.id)
83 JOIN asset.copy_location acpl ON (acp.location = acpl.id)
84 JOIN config.copy_status ccs ON (acp.status = ccs.id)
86 WHERE acn.owning_lib IN (ORG_LIST)
87 AND acp.circ_lib IN (ORG_LIST)
88 AND acp.active_date > NOW() - ?::INTERVAL
89 AND acp.location IN (LOC_LIST)
95 AND (EXISTS (SELECT 1 FROM asset.copy_vis_attr_cache WHERE record = acn.record AND vis_attr_vector @@ c_attr.vis_test))
96 AND (NOT EXISTS (SELECT 1 FROM metabib.record_attr_vector_list WHERE source = acn.record AND vlist @@ metabib.compile_composite_attr(' {"1":[{"_val":"s","_attr":"bib_level"}]}')::query_int))
98 ORDER BY MIN(AGE(acp.active_date))
102 my %TYPE_QUERY_MAP = (
103 2 => $new_items_query,
104 3 => $recently_returned_query,
105 4 => $top_circs_query,
106 5 => $new_by_loc_query,
109 sub refresh_container_from_carousel_definition {
113 my $carousel_type = shift;
114 my $age = shift // '15 days';
115 my $libs = shift // [];
116 my $locs = shift // [];
117 my $limit = shift // 50;
119 my $e = OpenILS::Utils::CStoreEditor->new;
120 my $ctype = $e->retrieve_config_carousel_type($carousel_type) or return $e->die_event;
123 unless (exists($TYPE_QUERY_MAP{$carousel_type})) {
124 $logger->error("Carousel for bucket $bucket is misconfigured; type $carousel_type is not recognized");
128 my $query = $TYPE_QUERY_MAP{$carousel_type};
130 if ($ctype->filter_by_copy_owning_lib eq 't') {
131 if (scalar(@$libs) < 1) {
132 $logger->error("Carousel for bucket $bucket is misconfigured; owning library filter expected but none specified");
135 my $org_placeholders = join(',', map { '?' } @$libs);
136 $query =~ s/ORG_LIST/$org_placeholders/g;
138 $libs = []; # we'll ignore any superflous supplied values
141 if ($ctype->filter_by_copy_location eq 't') {
142 if (scalar(@$locs) < 1) {
143 $logger->error("Carousel for bucket $bucket is misconfigured; copy location filter expected but none specified");
146 my $loc_placeholders = join(',', map { '?' } @$locs);
147 $query =~ s/LOC_LIST/$loc_placeholders/g;
149 $locs = []; # we'll ignore any superflous supplied values
152 my $sth = container::biblio_record_entry_bucket_item->db_Main->prepare_cached($query);
154 $sth->execute(@$libs, @$libs, $age, @$locs, $limit);
156 while (my $row = $sth->fetchrow_hashref ) {
157 push @bibs, $row->{bib};
159 container::biblio_record_entry_bucket_item->search( bucket => $bucket )->delete_all;
161 foreach my $bib (@bibs) {
162 container::biblio_record_entry_bucket_item->create({ bucket => $bucket, target_biblio_record_entry => $bib, pos => $i++ });
164 return scalar(@bibs);
167 __PACKAGE__->register_method(
168 api_name => 'open-ils.storage.container.refresh_from_carousel',
169 method => 'refresh_container_from_carousel_definition',
174 sub refresh_all_carousels {
178 my $e = OpenILS::Utils::CStoreEditor->new;
180 my $automatic_types = $e->search_config_carousel_type({ automatic => 't' });
181 my $carousels = $e->search_container_carousel({ type => [ map { $_->id } @$automatic_types ], active => 't' });
183 my $meth = $self->method_lookup('open-ils.storage.container.refresh_from_carousel');
185 foreach my $carousel (@$carousels) {
189 if (defined($carousel->owning_lib_filter)) {
190 my $ou_filter = $carousel->owning_lib_filter;
191 $ou_filter =~ s/[{}]//g;
192 @$orgs = split /,/, $ou_filter;
194 if (defined($carousel->copy_location_filter)) {
195 my $loc_filter = $carousel->copy_location_filter;
196 $loc_filter =~ s/[{}]//g;
197 @$locs = split /,/, $loc_filter;
200 my @res = $meth->run($carousel->bucket, $carousel->type, $carousel->age_filter, $orgs, $locs, $carousel->max_items);
201 my $ct = scalar(@res) ? $res[0] : 0;
204 $carousel->last_refresh_time('now');
205 $e->update_container_carousel($carousel);
209 carousel => $carousel->id,
210 bucket => $carousel->bucket,
219 __PACKAGE__->register_method(
220 api_name => 'open-ils.storage.carousel.refresh_all',
221 method => 'refresh_all_carousels',