1 # Copyright 2014 Catalyst
3 # This file is part of Koha.
5 # Koha is free software; you can redistribute it and/or modify it
6 # under the terms of the GNU General Public License as published by
7 # the Free Software Foundation; either version 3 of the License, or
8 # (at your option) any later version.
10 # Koha is distributed in the hope that it will be useful, but
11 # WITHOUT ANY WARRANTY; without even the implied warranty of
12 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 # GNU General Public License for more details.
15 # You should have received a copy of the GNU General Public License
16 # along with Koha; if not, see <http://www.gnu.org/licenses>.
18 package Koha::ExternalContent::OverDrive;
23 use base qw(Koha::ExternalContent);
24 use WebService::ILS::OverDrive::Patron;
30 Koha::ExternalContent::OverDrive
34 Register return url with OverDrive:
35 base app url + /cgi-bin/koha/external/overdrive/auth.pl
37 use Koha::ExternalContent::OverDrive;
38 my $od_client = Koha::ExternalContent::OverDrive->new();
39 my $od_auth_url = $od_client->auth_url($return_page_url);
43 A (very) thin wrapper around C<WebService::ILS::OverDrive::Patron>
45 Takes "OverDrive*" Koha preferences
51 my $params = shift || {};
52 $params->{koha_session_id} or croak "No koha_session_id";
54 my $self = $class->SUPER::new($params);
55 unless ($params->{client}) {
56 my $client_key = C4::Context->preference('OverDriveClientKey')
57 or croak("OverDriveClientKey pref not set");
58 my $client_secret = C4::Context->preference('OverDriveClientSecret')
59 or croak("OverDriveClientSecret pref not set");
60 my $library_id = C4::Context->preference('OverDriveLibraryID')
61 or croak("OverDriveLibraryID pref not set");
62 my ($token, $token_type) = $self->get_token_from_koha_session();
63 $self->client( WebService::ILS::OverDrive::Patron->new(
64 client_id => $client_key,
65 client_secret => $client_secret,
66 library_id => $library_id,
67 access_token => $token,
68 access_token_type => $token_type,
69 user_agent_params => { agent => $class->agent_string }
76 =head1 L<WebService::ILS::OverDrive::Patron> METHODS
78 Methods used without mods:
82 =item C<error_message()>
90 =item C<checkout($id, $format)>
92 =item C<checkout_download_url($id)>
96 =item C<place_hold($id)>
98 =item C<remove_hold($id)>
102 Methods with slightly moded interfaces:
104 =head2 auth_url($page_url)
106 Input: url of the page from which OverDrive authentication was requested
108 Returns: Post OverDrive auth return handler url (see SYNOPSIS)
114 my $page_url = shift or croak "Page url not provided";
116 my ($return_url, $page) = $self->_return_url($page_url);
117 $self->set_return_page_in_koha_session($page);
118 return $self->client->auth_url($return_url);
121 =head2 auth_by_code($code, $base_url)
123 To be called in external/overdrive/auth.pl upon return from OverDrive Granted auth
129 my $code = shift or croak "OverDrive auth code not provided";
130 my $base_url = shift or croak "App base url not provided";
132 my ($access_token, $access_token_type, $auth_token)
133 = $self->client->auth_by_code($code, $self->_return_url($base_url));
134 $access_token or die "Invalid OverDrive code returned";
135 $self->set_token_in_koha_session($access_token, $access_token_type);
137 if (my $koha_patron = $self->koha_patron) {
138 $koha_patron->set({overdrive_auth_token => $auth_token})->store;
140 return $self->get_return_page_from_koha_session;
143 =head2 auth_by_userid($userid, $password, $website_id, $authorization_name)
145 To be called to check auth of patron using OverDrive Patron Authentication method
146 This requires a SIP connection configured with OverDrive
152 my $userid = shift or croak "No user provided";
153 my $password = shift;
154 croak "No password provided" unless ($password || !C4::Context->preference("OverDrivePasswordRequired"));
155 my $website_id = shift or croak "OverDrive Library ID not provided";
156 my $authorization_name = shift or croak "OverDrive Authname not provided";
158 my ($access_token, $access_token_type, $auth_token)
159 = $self->client->auth_by_user_id($userid, $password, $website_id, $authorization_name);
160 $access_token or die "Invalid OverDrive code returned";
161 $self->set_token_in_koha_session($access_token, $access_token_type);
163 $self->koha_patron->set({overdrive_auth_token => $auth_token})->store;
164 return $self->get_return_page_from_koha_session;
167 use constant AUTH_RETURN_HANDLER => "/cgi-bin/koha/external/overdrive/auth.pl";
170 my $page_url = shift or croak "Page url not provided";
172 my ($base_url, $page) = ($page_url =~ m!^(https?://[^/]+)(.*)!);
173 my $return_url = $base_url.AUTH_RETURN_HANDLER;
175 return wantarray ? ($return_url, $page) : $return_url;
178 use constant RETURN_PAGE_SESSION_KEY => "overdrive.return_page";
179 sub get_return_page_from_koha_session {
181 my $return_page = $self->get_from_koha_session(RETURN_PAGE_SESSION_KEY) || "";
182 $self->logger->debug("get_return_page_from_koha_session: $return_page");
185 sub set_return_page_in_koha_session {
187 my $return_page = shift || "";
188 $self->logger->debug("set_return_page_in_koha_session: $return_page");
189 return $self->set_in_koha_session( RETURN_PAGE_SESSION_KEY, $return_page );
192 use constant ACCESS_TOKEN_SESSION_KEY => "overdrive.access_token";
193 my $ACCESS_TOKEN_DELIMITER = ":";
194 sub get_token_from_koha_session {
196 my ($token, $token_type)
197 = split $ACCESS_TOKEN_DELIMITER, $self->get_from_koha_session(ACCESS_TOKEN_SESSION_KEY) || "";
198 $self->logger->debug("get_token_from_koha_session: ".($token || "(none)"));
199 return ($token, $token_type);
201 sub set_token_in_koha_session {
203 my $token = shift || "";
204 my $token_type = shift || "";
205 $self->logger->debug("set_token_in_koha_session: $token $token_type");
206 return $self->set_in_koha_session(
207 ACCESS_TOKEN_SESSION_KEY,
208 join($ACCESS_TOKEN_DELIMITER, $token, $token_type)
212 =head2 checkout_download_url($item_id)
214 Input: id of the item to download
216 Returns: Fulfillment URL for reidrection
220 sub checkout_download_url {
222 my $item_id = shift or croak "Item ID not specified";
224 my $ua = LWP::UserAgent->new;
225 $ua->max_redirect(0);
226 my $response = $ua->get(
227 "https://patron.api.overdrive.com/v1/patrons/me/checkouts/".$item_id."/formats/downloadredirect",
228 'Authorization' => "Bearer ".$self->client->access_token,
231 my $redirect = { redirect => $response->{_headers}->{location} };
237 =head2 is_logged_in()
245 my ($token, $token_type) = $self->get_token_from_koha_session();
246 $token ||= $self->auth_by_saved_token;
250 sub auth_by_saved_token {
253 my $koha_patron = $self->koha_patron or return;
255 if (my $auth_token = $koha_patron->overdrive_auth_token) {
256 my ($access_token, $access_token_type, $new_auth_token)
257 = $self->client->make_access_token_request();
258 $self->set_token_in_koha_session($access_token, $access_token_type);
259 $koha_patron->set({overdrive_auth_token => $new_auth_token})->store;
260 return $access_token;
268 Removes stored OverDrive token
275 $self->set_token_in_koha_session("", "");
276 if (my $koha_patron = $self->koha_patron) {
277 $koha_patron->set({overdrive_auth_token => undef})->store;
281 use vars qw{$AUTOLOAD};
284 (my $method = $AUTOLOAD) =~ s/.*:://;
285 my $od = $self->client;
287 my $ret = eval { $od->$method(@_) };
289 if ( $od->is_access_token_error($@) && $self->auth_by_saved_token ) {
290 return $od->$method(@_);