3 # Copyright 2009 SARL Biblibre
5 # This file is part of Koha.
7 # Koha is free software; you can redistribute it and/or modify it
8 # under the terms of the GNU General Public License as published by
9 # the Free Software Foundation; either version 3 of the License, or
10 # (at your option) any later version.
12 # Koha is distributed in the hope that it will be useful, but
13 # WITHOUT ANY WARRANTY; without even the implied warranty of
14 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 # GNU General Public License for more details.
17 # You should have received a copy of the GNU General Public License
18 # along with Koha; if not, see <http://www.gnu.org/licenses>.
22 use C4::ILSDI::Services;
27 use List::MoreUtils qw(any);
31 =head1 DLF ILS-DI for Koha
33 This script is a basic implementation of ILS-DI protocol for Koha.
34 It acts like a dispatcher, that get the CGI request, check required and
35 optionals arguments, call a function from C4::ILS-DI, and finally
36 outputs the returned hashref as XML.
40 # Instanciate the CGI request
43 # List of available services, sorted by level
45 'Describe', # Not part of ILS-DI, online API doc
47 # Level 1: Basic Discovery Interfaces
48 # 'HarvestBibliographicRecords', # OAI-PMH
49 # 'HarvestExpandedRecords', # OAI-PMH
50 'GetAvailability', # FIXME Add bibliographic level
52 # 'GoToBibliographicRequestPage' # I don't understant this one
53 # Level 2: Elementary OPAC supplement
54 # 'HarvestAuthorityRecords', # OAI-PMH
55 # 'HarvestHoldingsRecords', # OAI-PMH
56 'GetRecords', # Note that we can use OAI-PMH for this too
60 'GetAuthorityRecords',
62 # 'OutputRewritablePage', # I don't understant this one
63 # 'OutputIntermediateFormat', # I don't understant this one
64 # Level 3: Elementary OPAC alternative
69 'GetServices', # FIXME Loans
71 'HoldTitle', # FIXME Add dates support
72 'HoldItem', # FIXME Add dates support
75 # 'RecallItem', # Not supported by Koha
76 # 'CancelRecall', # Not supported by Koha
77 # Level 4: Robust/domain specific discovery platforms
78 # 'SearchCourseReserves', # TODO
82 # List of required arguments
84 'Describe' => ['verb'],
85 'GetAvailability' => [ 'id', 'id_type' ],
86 'GetRecords' => ['id'],
87 'GetAuthorityRecords' => ['id'],
88 'LookupPatron' => ['id'],
89 'AuthenticatePatron' => [ 'username', 'password' ],
90 'GetPatronInfo' => ['patron_id'],
91 'GetPatronStatus' => ['patron_id'],
92 'GetServices' => [ 'patron_id', 'item_id' ],
93 'RenewLoan' => [ 'patron_id', 'item_id' ],
94 'HoldTitle' => [ 'patron_id', 'bib_id', 'request_location' ],
95 'HoldItem' => [ 'patron_id', 'bib_id', 'item_id' ],
96 'CancelHold' => [ 'patron_id', 'item_id' ],
99 # List of optional arguments
102 'GetAvailability' => [ 'return_type', 'return_fmt' ],
103 'GetRecords' => ['schema'],
104 'GetAuthorityRecords' => ['schema'],
105 'LookupPatron' => ['id_type'],
106 'AuthenticatePatron' => [],
107 'GetPatronInfo' => [ 'show_contact', 'show_fines', 'show_holds', 'show_loans', 'show_attributes' ],
108 'GetPatronStatus' => [],
110 'RenewLoan' => ['desired_due_date'],
111 'HoldTitle' => [ 'pickup_location', 'needed_before_date', 'pickup_expiry_date' ],
112 'HoldItem' => [ 'pickup_location', 'needed_before_date', 'pickup_expiry_date' ],
116 # If no service is requested, display the online documentation
117 unless ( $cgi->param('service') ) {
118 my ( $template, $loggedinuser, $cookie ) = get_template_and_user(
119 { template_name => "ilsdi.tt",
122 authnotrequired => 1,
126 output_html_with_http_headers $cgi, $cookie, $template->output;
130 # If user requested a service description, then display it
131 if ( scalar $cgi->param('service') eq "Describe" and any { scalar $cgi->param('verb') eq $_ } @services ) {
132 my ( $template, $loggedinuser, $cookie ) = get_template_and_user(
133 { template_name => "ilsdi.tt",
136 authnotrequired => 1,
140 $template->param( scalar $cgi->param('verb') => 1 );
141 output_html_with_http_headers $cgi, $cookie, $template->output;
145 # any output after this point will be UTF-8 XML
146 binmode STDOUT, ':encoding(UTF-8)';
147 print CGI::header('-type'=>'text/xml', '-charset'=>'utf-8');
151 # If ILS-DI module is disabled in System->Preferences, redirect to 404
152 unless ( C4::Context->preference('ILS-DI') ) {
153 $out->{'code'} = "NotAllowed";
154 $out->{'message'} = "ILS-DI is disabled.";
157 # If the remote address is not allowed, redirect to 403
158 my @AuthorizedIPs = split(/,/, C4::Context->preference('ILS-DI:AuthorizedIPs'));
159 if ( @AuthorizedIPs # If no filter set, allow access to everybody
160 and not any { $ENV{'REMOTE_ADDR'} eq $_ } @AuthorizedIPs # IP Check
162 $out->{'code'} = "NotAllowed";
163 $out->{'message'} = "Unauthorized IP address: ".$ENV{'REMOTE_ADDR'}.".";
166 my $service = $cgi->param('service') || "ilsdi";
168 # Check if the requested service is in the list
169 if ( $service and any { $service eq $_ } @services ) {
171 my @parmsrequired = @{ $required{$service} };
172 my @parmsoptional = @{ $optional{$service} };
173 my @parmsall = ( @parmsrequired, @parmsoptional );
174 my @names = $cgi->multi_param;
176 $paramhash{$_} = 1 for @names;
178 # check for missing parameters
179 for ( @parmsrequired ) {
180 unless ( exists $paramhash{$_} ) {
181 $out->{'code'} = "MissingParameter";
182 $out->{'message'} = "The required parameter ".$_." is missing.";
186 # check for illegal parameters
187 for my $name ( @names ) {
189 for my $name2 (@parmsall) {
190 if ( $name eq $name2 ) {
194 if ( $found == 0 && $name ne 'service' ) {
195 $out->{'code'} = "IllegalParameter";
196 $out->{'message'} = "The parameter ".$name." is illegal.";
200 # check for multiple parameters
202 my @values = $cgi->multi_param($_);
203 if ( $#values != 0 ) {
204 $out->{'code'} = "MultipleValuesNotAllowed";
205 $out->{'message'} = "Multiple values not allowed for the parameter ".$_.".";
209 if ( !$out->{'message'} ) {
211 # GetAvailability is a special case, as it cannot use XML::Simple
212 if ( $service eq "GetAvailability" ) {
213 print C4::ILSDI::Services::GetAvailability($cgi);
220 my $symbol = 'C4::ILSDI::Services::' . $service;
224 # Call the requested service, and get its return value
229 $out->{'message'} = "NotSupported";
232 # Output XML by passing the hashref to XMLOut
237 xmldecl => '<?xml version="1.0" encoding="UTF-8" ?>',
238 RootName => $service,