45a25236bf0ce7d73cb49e127498de56166ee65c
[Evergreen.git] / Open-ILS / src / perlmods / OpenILS / WWW / SuperCat.pm
1 package OpenILS::WWW::SuperCat;
2 use strict; use warnings;
3
4 use Apache2 ();
5 use Apache2::Log;
6 use Apache2::Const -compile => qw(OK REDIRECT DECLINED NOT_FOUND :log);
7 use APR::Const    -compile => qw(:error SUCCESS);
8 use Apache2::RequestRec ();
9 use Apache2::RequestIO ();
10 use Apache2::RequestUtil;
11 use CGI;
12 use Data::Dumper;
13
14 use OpenSRF::EX qw(:try);
15 use OpenSRF::Utils qw/:datetime/;
16 use OpenSRF::System;
17 use OpenSRF::AppSession;
18 use XML::LibXML;
19
20 use Unicode::Normalize;
21 use OpenILS::Utils::Fieldmapper;
22 use OpenILS::WWW::SuperCat::Feed;
23
24
25 # set the bootstrap config when this module is loaded
26 my ($bootstrap, $supercat, $actor, $parser, $search);
27
28 sub import {
29         my $self = shift;
30         $bootstrap = shift;
31 }
32
33
34 sub child_init {
35         OpenSRF::System->bootstrap_client( config_file => $bootstrap );
36         $supercat = OpenSRF::AppSession->create('open-ils.supercat');
37         $actor = OpenSRF::AppSession->create('open-ils.actor');
38         $search = OpenSRF::AppSession->create('open-ils.search');
39         $parser = new XML::LibXML;
40 }
41
42 sub oisbn {
43
44         my $apache = shift;
45         return Apache2::Const::DECLINED if (-e $apache->filename);
46
47         (my $isbn = $apache->path_info) =~ s{^.*?([^/]+)$}{$1}o;
48
49         my $list = $supercat
50                 ->request("open-ils.supercat.oisbn", $isbn)
51                 ->gather(1);
52
53         print "Content-type: application/xml; charset=utf-8\n\n";
54         print "<?xml version='1.0' encoding='UTF-8' ?>\n";
55
56         unless (exists $$list{metarecord}) {
57                 print '<idlist/>';
58                 return Apache2::Const::OK;
59         }
60
61         print "<idlist metarecord='$$list{metarecord}'>\n";
62
63         for ( keys %{ $$list{record_list} } ) {
64                 (my $o = $$list{record_list}{$_}) =~s/^(\S+).*?$/$1/o;
65                 print "  <isbn record='$_'>$o</isbn>\n"
66         }
67
68         print "</idlist>\n";
69
70         return Apache2::Const::OK;
71 }
72
73 sub unapi {
74
75         my $apache = shift;
76         return Apache2::Const::DECLINED if (-e $apache->filename);
77
78         print "Content-type: application/xml; charset=utf-8\n";
79         
80         my $cgi = new CGI;
81
82         my $uri = $cgi->param('uri') || '';
83         my $base = $cgi->url;
84         my $host = $cgi->virtual_host || $cgi->server_name;
85
86         my $format = $cgi->param('format');
87         my ($id,$type,$command) = ('','','');
88
89         if (!$format) {
90                 if ($uri =~ m{^tag:[^:]+:([^\/]+)/(\d+)}o) {
91                         $id = $2;
92                         $type = 'record';
93                         $type = 'metarecord' if ($1 =~ /^m/o);
94
95                         my $list = $supercat
96                         ->request("open-ils.supercat.$type.formats")
97                                 ->gather(1);
98
99                         print "\n";
100
101                         my $body =
102                                 "<formats>
103                                  <uri>$uri</uri>
104                                    <format>
105                                      <name>opac</name>
106                                      <type>text/html</type>
107                                    </format>";
108
109                         for my $h (@$list) {
110                                 my ($type) = keys %$h;
111                                 $body .= "<format><name>$type</name><type>application/$type+xml</type>";
112
113                                 for my $part ( qw/namespace_uri docs schema_location/ ) {
114                                         $body .= "<$part>$$h{$type}{$part}</$part>"
115                                                 if ($$h{$type}{$part});
116                                 }
117                                 
118                                 $body .= '</format>';
119                         }
120
121                         $body .= "</formats>\n";
122
123                         $apache->custom_response( 300, $body);
124                         return 300;
125                 } else {
126                         my $list = $supercat
127                                 ->request("open-ils.supercat.record.formats")
128                                 ->gather(1);
129                                 
130                         push @$list,
131                                 @{ $supercat
132                                         ->request("open-ils.supercat.metarecord.formats")
133                                         ->gather(1);
134                                 };
135
136                         my %hash = map { ( (keys %$_)[0] => (values %$_)[0] ) } @$list;
137                         $list = [ map { { $_ => $hash{$_} } } sort keys %hash ];
138
139                         print "\n<formats>
140                                    <format>
141                                      <name>opac</name>
142                                      <type>text/html</type>
143                                    </format>";
144
145                         for my $h (@$list) {
146                                 my ($type) = keys %$h;
147                                 print "<format><name>$type</name><type>application/x-$type+xml</type>";
148
149                                 for my $part ( qw/namespace_uri docs schema_location/ ) {
150                                         print "<$part>$$h{$type}{$part}</$part>"
151                                                 if ($$h{$type}{$part});
152                                 }
153                                 
154                                 print '</format>';
155                         }
156
157                         print "</formats>\n";
158
159
160                         return Apache2::Const::OK;
161                 }
162         }
163
164                 
165         if ($uri =~ m{^tag:[^:]+:([^\/]+)/(\d+)}o) {
166                 $id = $2;
167                 $type = 'record';
168                 $type = 'metarecord' if ($1 =~ /^m/o);
169                 $command = 'retrieve';
170         }
171
172         if ($format eq 'opac') {
173                 print "Location: $base/../../en-US/skin/default/xml/rresult.xml?m=$id\n\n"
174                         if ($type eq 'metarecord');
175                 print "Location: $base/../../en-US/skin/default/xml/rdetail.xml?r=$id\n\n"
176                         if ($type eq 'record');
177                 return 302;
178         }
179
180         print "\n" . $supercat->request("open-ils.supercat.$type.$format.$command",$id)->gather(1);
181
182         return Apache2::Const::OK;
183 }
184
185 sub supercat {
186
187         my $apache = shift;
188         return Apache2::Const::DECLINED if (-e $apache->filename);
189
190         my $path = $apache->path_info;
191
192         my $cgi = new CGI;
193         my $base = $cgi->url;
194
195         my ($id,$type,$format,$command) = reverse split '/', $path;
196
197         print "Content-type: application/xml; charset=utf-8\n";
198         
199         if ( $path =~ m{^/formats(?:/([^\/]+))?$}o ) {
200                 if ($1) {
201                         my $list = $supercat
202                                 ->request("open-ils.supercat.$1.formats")
203                                 ->gather(1);
204
205                         print "\n";
206
207                         print "<formats>
208                                    <format>
209                                      <name>opac</name>
210                                      <type>text/html</type>
211                                    </format>";
212
213                         for my $h (@$list) {
214                                 my ($type) = keys %$h;
215                                 print "<format><name>$type</name><type>application/$type+xml</type>";
216
217                                 for my $part ( qw/namespace_uri docs schema_location/ ) {
218                                         print "<$part>$$h{$type}{$part}</$part>"
219                                                 if ($$h{$type}{$part});
220                                 }
221                                 
222                                 print '</format>';
223                         }
224
225                         print "</formats>\n";
226
227                         return Apache2::Const::OK;
228                 }
229
230                 my $list = $supercat
231                         ->request("open-ils.supercat.record.formats")
232                         ->gather(1);
233                                 
234                 push @$list,
235                         @{ $supercat
236                                 ->request("open-ils.supercat.metarecord.formats")
237                                 ->gather(1);
238                         };
239
240                 my %hash = map { ( (keys %$_)[0] => (values %$_)[0] ) } @$list;
241                 $list = [ map { { $_ => $hash{$_} } } sort keys %hash ];
242
243                 print "\n<formats>
244                            <format>
245                              <name>opac</name>
246                              <type>text/html</type>
247                            </format>";
248
249                 for my $h (@$list) {
250                         my ($type) = keys %$h;
251                         print "<format><name>$type</name><type>application/$type+xml</type>";
252
253                         for my $part ( qw/namespace_uri docs schema_location/ ) {
254                                 print "<$part>$$h{$type}{$part}</$part>"
255                                         if ($$h{$type}{$part});
256                         }
257                         
258                         print '</format>';
259                 }
260
261                 print "</formats>\n";
262
263
264                 return Apache2::Const::OK;
265         }
266
267         if ($format eq 'opac') {
268                 print "Location: $base/../../en-US/skin/default/xml/rresult.xml?m=$id\n\n"
269                         if ($type eq 'metarecord');
270                 print "Location: $base/../../en-US/skin/default/xml/rdetail.xml?r=$id\n\n"
271                         if ($type eq 'record');
272                 return 302;
273         }
274
275         print "\n" . $supercat->request("open-ils.supercat.$type.$format.$command",$id)->gather(1);
276
277         return Apache2::Const::OK;
278 }
279
280
281 sub bookbag_feed {
282         my $apache = shift;
283         return Apache2::Const::DECLINED if (-e $apache->filename);
284
285         my $cgi = new CGI;
286
287         my $year = (gmtime())[5] + 1900;
288         my $host = $cgi->virtual_host || $cgi->server_name;
289
290         my $rel_name = quotemeta($cgi->url(-relative=>1));
291
292         my $add_path = 1;
293         $add_path = 0 if ($cgi->url(-path_info=>1) =~ /$rel_name$/);
294
295         my $url = $cgi->url(-path_info=>$add_path);
296         my $root = (split 'feed', $url)[0];
297         my $base = (split 'bookbag', $url)[0] . 'bookbag';
298         my $path = (split 'bookbag', $url)[1];
299         my $unapi = (split 'feed', $url)[0] . 'unapi';
300
301
302         #warn "URL breakdown: $url ($rel_name) -> $root -> $base -> $path -> $unapi";
303
304         my ($id,$type) = reverse split '/', $path;
305
306         my $bucket = $actor->request("open-ils.actor.container.public.flesh", 'biblio', $id)->gather(1);
307         return Apache2::Const::NOT_FOUND unless($bucket);
308
309         my $bucket_tag = "tag:$host,$year:record_bucket/$id";
310         if ($type eq 'opac') {
311                 print "Location: $base/../../../opac/en-US/skin/default/xml/rresult.xml?rt=list&" .
312                         join('&', map { "rl=" . $_->target_biblio_record_entry } @{ $bucket->items }) .
313                         "\n\n";
314                 return Apache2::Const::OK;
315         }
316
317         my $feed = create_record_feed(
318                 $type,
319                 [ map { $_->target_biblio_record_entry } @{ $bucket->items } ],
320                 $unapi,
321         );
322         $feed->root($root);
323
324         $feed->title("Items in Book Bag #".$bucket->id);
325         $feed->creator($host);
326         $feed->update_ts(gmtime_ISO8601());
327
328         $feed->link(atom => $base . "/bookbag/atom/$id" => 'application/atom+xml');
329         $feed->link(rss2 => $base . "/bookbag/rss2/$id");
330         $feed->link(html => $base . "/bookbag/html/$id" => 'text/html');
331
332         $feed->link(
333                 OPAC =>
334                 $root . '../en-US/skin/default/xml/rresult.xml?rt=list&' .
335                         join('&', map { 'rl=' . $_->target_biblio_record_entry } @{$bucket->items} ),
336                 'text/xhtml'
337         );
338
339
340         print "Content-type: ". $feed->type ."; charset=utf-8\n\n";
341         print entityize($feed->toString) . "\n";
342
343         return Apache2::Const::OK;
344 }
345
346 sub opensearch_osd {
347         my $version = shift;
348         my $lib = shift;
349         my $class = shift;
350         my $base = shift;
351
352         if ($version eq '1.0') {
353                 print <<OSD;
354 Content-type: application/opensearchdescription+xml; charset=utf-8
355
356 <?xml version="1.0" encoding="UTF-8"?>
357 <OpenSearchDescription xmlns="http://a9.com/-/spec/opensearchdescription/1.0/">
358   <Url>$base/1.0/$lib/-/$class/{searchTerms}?startPage={startPage}&amp;startIndex={startIndex}&amp;count={count}</Url>
359   <Format>http://a9.com/-/spec/opensearchrss/1.0/</Format>
360   <ShortName>$lib</ShortName>
361   <LongName>Search $lib</LongName>
362   <Description>Search the $lib OPAC by $class.</Description>
363   <Tags>$lib book library</Tags>
364   <SampleSearch>harry+potter</SampleSearch>
365   <Developer>Mike Rylander for GPLS/PINES</Developer>
366   <Contact>feedback\@open-ils.org</Contact>
367   <SyndicationRight>open</SyndicationRight>
368   <AdultContent>false</AdultContent>
369 </OpenSearchDescription>
370 OSD
371         } else {
372                 print <<OSD;
373 Content-type: application/opensearchdescription+xml; charset=utf-8
374
375 <?xml version="1.0" encoding="UTF-8"?>
376 <OpenSearchDescription xmlns="http://a9.com/-/spec/opensearch/1.1/">
377   <ShortName>$lib</ShortName>
378   <Description>Search the $lib OPAC by $class.</Description>
379   <Tags>$lib book library</Tags>
380   <Url type="application/atom+xml"
381        template="$base/1.1/$lib/atom/$class/{searchTerms}?startPage={startPage?}&amp;startIndex={startIndex?}&amp;count={count?}&amp;language={language?}"/>
382   <Url type="application/x-rss+xml"
383        template="$base/1.1/$lib/rss2/$class/{searchTerms}?startPage={startPage?}&amp;startIndex={startIndex?}&amp;count={count?}&amp;language={language?}"/>
384   <Url type="application/x-mods3+xml"
385        template="$base/1.1/$lib/mods3/$class/{searchTerms}?startPage={startPage?}&amp;startIndex={startIndex?}&amp;count={count?}&amp;language={language?}"/>
386   <Url type="application/x-mods+xml"
387        template="$base/1.1/$lib/mods/$class/{searchTerms}?startPage={startPage?}&amp;startIndex={startIndex?}&amp;count={count?}&amp;language={language?}"/>
388   <Url type="application/x-marcxml+xml"
389        template="$base/1.1/$lib/marcxml/$class/{searchTerms}?startPage={startPage?}&amp;startIndex={startIndex?}&amp;count={count?}&amp;language={language?}"/>
390   <LongName>Search $lib</LongName>
391   <Query role="example" searchTerms="harry+potter" />
392   <Developer>Mike Rylander for GPLS/PINES</Developer>
393   <SyndicationRight>open</SyndicationRight>
394   <AdultContent>false</AdultContent>
395   <Language>en-US</Language>
396   <OutputEncoding>UTF-8</OutputEncoding>
397   <InputEncoding>UTF-8</InputEncoding>
398 </OpenSearchDescription>
399 OSD
400         }
401
402         return Apache2::Const::OK;
403 }
404
405 sub opensearch_feed {
406         my $apache = shift;
407         return Apache2::Const::DECLINED if (-e $apache->filename);
408
409         my $cgi = new CGI;
410         my $year = (gmtime())[5] + 1900;
411
412         my $host = $cgi->virtual_host || $cgi->server_name;
413
414         my $rel_name = quotemeta($cgi->url(-relative=>1));
415
416         my $add_path = 1;
417         $add_path = 0 if ($cgi->url(-path_info=>1) =~ /$rel_name$/);
418
419         my $url = $cgi->url(-path_info=>$add_path);
420         my $root = (split 'opensearch', $url)[0];
421         my $base = (split 'opensearch', $url)[0] . 'opensearch';
422         my $unapi = (split 'opensearch', $url)[0] . 'unapi';
423
424         my $path = (split 'opensearch', $url)[1];
425
426         #warn "URL breakdown: $url ($rel_name) -> $root -> $base -> $path -> $unapi";
427
428         if ($path =~ m{^/?(1\.\d{1})/(?:([^/]+)/)?([^/]+)/osd.xml}o) {
429                 
430                 my $version = $1;
431                 my $lib = $2;
432                 my $class = $3;
433
434                 if (!$lib) {
435                         $lib = $actor->request(
436                                 'open-ils.actor.org_unit_list.search' => parent_ou => undef
437                         )->gather(1)->[0]->shortname;
438                 }
439
440                 if ($class eq '-') {
441                         $class = 'keyword';
442                 }
443
444                 return opensearch_osd($version, $lib, $class, $base);
445         }
446
447
448         my $page = $cgi->param('startPage') || 1;
449         my $offset = $cgi->param('startIndex') || 1;
450         my $limit = $cgi->param('count') || 10;
451         my $lang = $cgi->param('language') || 'en-US';
452
453         $page = 1 if ($page =~ /^{/);
454         $offset = 1 if ($offset =~ /^{/);
455         $limit = 10 if ($limit =~ /^{/);
456         $lang = 'en-US' if ($lang =~ /^{/);
457
458         if ($page > 1) {
459                 $offset = ($page - 1) * $limit;
460         } else {
461                 $offset -= 1;
462         }
463
464         my ($terms,$class,$type,$org,$version) = reverse split '/', $path;
465
466         if ($version eq '1.0') {
467                 $type = 'rss2';
468         } elsif ($type eq '-') {
469                 $type = 'atom';
470         }
471
472         $class = 'keyword' if ($class eq '-');
473         $terms =~ s/\+/ /go;
474
475         #warn "searching for $class -> [$terms] via OS $version, response type $type";
476
477         my $org_unit;
478         if ($org eq '-') {
479                 $org_unit = $actor->request(
480                         'open-ils.actor.org_unit_list.search' => parent_ou => undef
481                 )->gather(1);
482         } else {
483                 $org_unit = $actor->request(
484                         'open-ils.actor.org_unit_list.search' => shortname => $org
485                 )->gather(1);
486         }
487
488         my $recs = $search->request(
489                 'open-ils.search.biblio.record.class.search' => $class,
490                 { term          => $terms,
491                   org_unit      => $org_unit->[0]->id,
492                   limit         => $limit,
493                   offset        => $offset,
494                 }
495         )->gather(1);
496
497         my $feed = create_record_feed(
498                 $type,
499                 [ map { $_->[0] } @{$recs->{ids}} ],
500                 $unapi,
501         );
502         $feed->root($root);
503
504         $feed->title("Search results for [$class => $terms] at ".$org_unit->[0]->name);
505         $feed->creator($host);
506         $feed->update_ts(gmtime_ISO8601());
507
508         $feed->_create_node(
509                 $feed->{item_xpath},
510                 'http://a9.com/-/spec/opensearch/1.1/',
511                 'totalResults',
512                 $recs->{count},
513         );
514
515         $feed->_create_node(
516                 $feed->{item_xpath},
517                 'http://a9.com/-/spec/opensearch/1.1/',
518                 'startIndex',
519                 $offset + 1,
520         );
521
522         $feed->_create_node(
523                 $feed->{item_xpath},
524                 'http://a9.com/-/spec/opensearch/1.1/',
525                 'itemsPerPage',
526                 $limit,
527         );
528
529         $feed->link(
530                 next =>
531                 $base . $path . "?startIndex=" . int($offset + $limit + 1) . "&count=" . $limit =>
532                 'application/opensearch+xml'
533         ) if ($offset + $limit < $recs->{count});
534
535         $feed->link(
536                 previous =>
537                 $base . $path . "?startIndex=" . int(($offset - $limit) + 1) . "&count=" . $limit =>
538                 'application/opensearch+xml'
539         ) if ($offset);
540
541         $feed->link(
542                 self =>
543                 $base . $path =>
544                 'application/opensearch+xml'
545         );
546
547         $feed->link(
548                 unapi =>
549                 $unapi
550         );
551
552         $feed->link(
553                 alternate =>
554                 $root . "../$lang/skin/default/xml/rresult.xml?rt=list&" .
555                         join('&', map { 'rl=' . $_->[0] } @{$recs->{ids}} ),
556                 'text/html'
557         );
558
559         $feed->link(
560                 opac =>
561                 $root . "../$lang/skin/default/xml/rresult.xml?rt=list&" .
562                         join('&', map { 'rl=' . $_->[0] } @{$recs->{ids}} ),
563                 'text/html'
564         );
565
566         print "Content-type: ". $feed->type ."; charset=utf-8\n\n";
567         print entityize($feed->toString) . "\n";
568
569         return Apache2::Const::OK;
570 }
571
572 sub create_record_feed {
573         my $type = shift;
574         my $records = shift;
575         my $unapi = shift;
576
577         my $cgi = new CGI;
578         my $base = $cgi->url;
579         my $host = $cgi->virtual_host || $cgi->server_name;
580
581         my $year = (gmtime())[5] + 1900;
582
583         my $feed = new OpenILS::WWW::SuperCat::Feed ($type);
584         $feed->base($base);
585         $feed->unapi($unapi);
586
587         $type = 'atom' if ($type eq 'html');
588
589         for my $rec (@$records) {
590                 my $item_tag = "tag:$host,$year:biblio-record_entry/" . $rec;
591
592
593                 my $xml = $supercat->request(
594                         "open-ils.supercat.record.$type.retrieve",
595                         $rec
596                 )->gather(1);
597
598                 my $node = $feed->add_item($xml);
599
600                 $node->id($item_tag);
601                 $node->link(alternate => $feed->unapi . "?uri=$item_tag&format=opac" => 'text/html');
602                 $node->link(opac => $feed->unapi . "?uri=$item_tag&format=opac");
603                 $node->link(unapi => $feed->unapi . "?uri=$item_tag");
604                 $node->link('unapi-uri' => $item_tag);
605         }
606
607         return $feed;
608 }
609
610 sub entityize {
611         my $stuff = NFC(shift());
612         $stuff =~ s/&(?!\S+;)/&amp;/gso;
613         $stuff =~ s/([\x{0080}-\x{fffd}])/sprintf('&#x%X;',ord($1))/sgoe;
614         return $stuff;
615 }
616
617 1;