fixing html output (server side now)
[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 :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/$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");
329         $feed->link(rss2 => $base . "/bookbag/rss2/$id");
330         $feed->link(html => $base . "/bookbag/html/$id");
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: application/xml; 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>$class</ShortName>
361   <LongName>Search by $class</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>$class</ShortName>
378   <Description>Search the $lib OPAC by $class.</Description>
379   <Tags>$lib book library</Tags>
380   <Url type="application/atom+xml"
381        method="post"
382        template="$base/1.1/$lib/$class/atom/{searchTerms}">
383     <Param name="startPage" value="{startPage?}"/>
384     <Param name="startIndex" value="{startIndex?}"/>
385     <Param name="count" value="{count?}"/>
386     <Param name="language" value="{language?}"/>
387   </Url>
388   <Url type="application/rss+xml"
389        method="post"
390        template="$base/1.1/$lib/$class/rss2/{searchTerms}">
391     <Param name="startPage" value="{startPage?}"/>
392     <Param name="startIndex" value="{startIndex?}"/>
393     <Param name="count" value="{count?}"/>
394     <Param name="language" value="{language?}"/>
395   </Url>
396   <Url type="application/mods+xml"
397        method="post"
398        template="$base/1.1/$lib/$class/mods/{searchTerms}">
399     <Param name="startPage" value="{startPage?}"/>
400     <Param name="startIndex" value="{startIndex?}"/>
401     <Param name="count" value="{count?}"/>
402     <Param name="language" value="{language?}"/>
403   </Url>
404   <Url type="application/marcxml+xml"
405        method="post"
406        template="$base/1.1/$lib/$class/marcxml/{searchTerms}">
407     <Param name="startPage" value="{startPage?}"/>
408     <Param name="startIndex" value="{startIndex?}"/>
409     <Param name="count" value="{count?}"/>
410     <Param name="language" value="{language?}"/>
411   </Url>
412   <LongName>Search by $class</LongName>
413   <Query role="example" searchTerms="harry+potter" />
414   <Developer>Mike Rylander for GPLS/PINES</Developer>
415   <SyndicationRight>open</SyndicationRight>
416   <AdultContent>false</AdultContent>
417   <Language>en-US</Language>
418   <OutputEncoding>UTF-8</OutputEncoding>
419   <InputEncoding>UTF-8</InputEncoding>
420 </OpenSearchDescription>
421 OSD
422         }
423
424         return Apache2::Const::OK;
425 }
426
427 sub opensearch_feed {
428         my $apache = shift;
429         return Apache2::Const::DECLINED if (-e $apache->filename);
430
431         my $cgi = new CGI;
432         my $year = (gmtime())[5] + 1900;
433
434         my $host = $cgi->virtual_host || $cgi->server_name;
435
436         my $rel_name = quotemeta($cgi->url(-relative=>1));
437
438         my $add_path = 1;
439         $add_path = 0 if ($cgi->url(-path_info=>1) =~ /$rel_name$/);
440
441         my $url = $cgi->url(-path_info=>$add_path);
442         my $root = (split 'opensearch', $url)[0];
443         my $base = (split 'opensearch', $url)[0] . 'opensearch';
444         my $unapi = (split 'opensearch', $url)[0] . 'unapi';
445
446         my $path = (split 'opensearch', $url)[1];
447
448         #warn "URL breakdown: $url ($rel_name) -> $root -> $base -> $path -> $unapi";
449
450         if ($path =~ m{^/?(1\.\d{1})/(?:([^/]+)/)?([^/]+)/osd.xml}o) {
451                 
452                 my $version = $1;
453                 my $lib = $2;
454                 my $class = $3;
455
456                 if (!$lib) {
457                         $lib = 'PINES';
458                 }
459
460                 if ($class eq '-') {
461                         $class = 'keyword';
462                 }
463
464                 return opensearch_osd($version, $lib, $class, $base);
465         }
466
467
468         my $page = $cgi->param('startPage') || 1;
469         my $offset = $cgi->param('startIndex') || 1;
470         my $limit = $cgi->param('count') || 10;
471         my $lang = $cgi->param('language') || 'en-US';
472
473         if ($page > 1) {
474                 $offset = ($page - 1) * $limit;
475         } else {
476                 $offset -= 1;
477         }
478
479         my ($terms,$class,$type,$org,$version) = reverse split '/', $path;
480
481         if ($version eq '1.0') {
482                 $type = 'rss2';
483         } elsif ($type eq '-') {
484                 $type = 'atom';
485         }
486
487         $class = 'keyword' if ($class eq '-');
488         $terms =~ s/\+/ /go;
489
490         #warn "searching for $class -> [$terms] via OS $version, response type $type";
491
492         my $org_unit;
493         if ($org eq '-') {
494                 $org_unit = $actor->request(
495                         'open-ils.actor.org_unit_list.search' => parent_ou => undef
496                 )->gather(1);
497         } else {
498                 $org_unit = $actor->request(
499                         'open-ils.actor.org_unit_list.search' => shortname => $org
500                 )->gather(1);
501         }
502
503         my $recs = $search->request(
504                 'open-ils.search.biblio.record.class.search' => $class,
505                 { term          => $terms,
506                   org_unit      => $org_unit->[0]->id,
507                   limit         => $limit,
508                   offset        => $offset,
509                 }
510         )->gather(1);
511
512         my $feed = create_record_feed(
513                 $type,
514                 [ map { $_->[0] } @{$recs->{ids}} ],
515                 $unapi,
516         );
517         $feed->root($root);
518
519         $feed->title("$class search results for [$terms] at ".$org_unit->[0]->name);
520         $feed->creator($host);
521         $feed->update_ts(gmtime_ISO8601());
522
523         $feed->_create_node(
524                 $feed->{item_xpath},
525                 'http://a9.com/-/spec/opensearch/1.1/',
526                 'totalResults',
527                 $recs->{count},
528         );
529
530         $feed->_create_node(
531                 $feed->{item_xpath},
532                 'http://a9.com/-/spec/opensearch/1.1/',
533                 'startIndex',
534                 $offset + 1,
535         );
536
537         $feed->_create_node(
538                 $feed->{item_xpath},
539                 'http://a9.com/-/spec/opensearch/1.1/',
540                 'itemsPerPage',
541                 $limit,
542         );
543
544         $feed->link(
545                 next =>
546                 $base . $path . "?startIndex=" . int($offset + $limit + 1) . "&count=" . $limit =>
547                 'application/opensearch+xml'
548         ) if ($offset + $limit < $recs->{count});
549
550         $feed->link(
551                 previous =>
552                 $base . $path . "?startIndex=" . int(($offset - $limit) + 1) . "&count=" . $limit =>
553                 'application/opensearch+xml'
554         ) if ($offset);
555
556         $feed->link(
557                 self =>
558                 $base . $path =>
559                 'application/opensearch+xml'
560         );
561
562         $feed->link(
563                 alternate =>
564                 $root . "../$lang/skin/default/xml/rresult.xml?rt=list&" .
565                         join('&', map { 'rl=' . $_->[0] } @{$recs->{ids}} ),
566                 'text/xhtml'
567         );
568
569         $feed->link(
570                 opac =>
571                 $root . "../$lang/skin/default/xml/rresult.xml?rt=list&" .
572                         join('&', map { 'rl=' . $_->[0] } @{$recs->{ids}} ),
573                 'text/xhtml'
574         );
575
576         print "Content-type: ". $feed->type ."; charset=utf-8\n\n";
577         print entityize($feed->toString) . "\n";
578
579         return Apache2::Const::OK;
580 }
581
582 sub create_record_feed {
583         my $type = shift;
584         my $records = shift;
585         my $unapi = shift;
586
587         my $cgi = new CGI;
588         my $base = $cgi->url;
589         my $host = $cgi->virtual_host || $cgi->server_name;
590
591         my $year = (gmtime())[5] + 1900;
592
593         my $feed = new OpenILS::WWW::SuperCat::Feed ($type);
594         $feed->base($base);
595         $feed->unapi($unapi);
596
597         $type = 'atom' if ($type eq 'html');
598
599         for my $rec (@$records) {
600                 my $item_tag = "tag:$host,$year:biblio-record_entry/" . $rec;
601
602
603                 my $xml = $supercat->request(
604                         "open-ils.supercat.record.$type.retrieve",
605                         $rec
606                 )->gather(1);
607
608                 my $node = $feed->add_item($xml);
609
610                 $node->id($item_tag);
611                 $node->link(alternate => $feed->unapi . "?uri=$item_tag&format=opac" => 'application/xml');
612                 $node->link(opac => $feed->unapi . "?uri=$item_tag&format=opac");
613                 $node->link(unapi => $feed->unapi . "?uri=$item_tag");
614         }
615
616         return $feed;
617 }
618
619 sub entityize {
620         my $stuff = NFC(shift());
621         $stuff =~ s/&(?!#.{2,4};|amp;)/&amp;/gso;
622         $stuff =~ s/([\x{0080}-\x{fffd}])/sprintf('&#x%X;',ord($1))/sgoe;
623         return $stuff;
624 }
625
626 1;