da84a4d19e7606f667699f6d55d736e2b4794e77
[koha.git] / Koha / ExternalContent / OverDrive.pm
1 # Copyright 2014 Catalyst
2 #
3 # This file is part of Koha.
4 #
5 # Koha is free software; you can redistribute it and/or modify it under the
6 # terms of the GNU General Public License as published by the Free Software
7 # Foundation; either version 3 of the License, or (at your option) any later
8 # version.
9 #
10 # Koha is distributed in the hope that it will be useful, but WITHOUT ANY
11 # WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
12 # A PARTICULAR PURPOSE.  See the GNU General Public License for more details.
13 #
14 # You should have received a copy of the GNU General Public License along
15 # with Koha; if not, write to the Free Software Foundation, Inc.,
16 # 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
17
18 package Koha::ExternalContent::OverDrive;
19
20 use Modern::Perl;
21 use Carp;
22
23 use base qw(Koha::ExternalContent);
24 use WebService::ILS::OverDrive::Patron;
25 use C4::Context;
26 use Koha::Logger;
27
28 use constant logger => Koha::Logger->get();
29
30 =head1 NAME
31
32 Koha::ExternalContent::OverDrive
33
34 =head1 SYNOPSIS
35
36     Register return url with OverDrive:
37       base app url + /cgi-bin/koha/external/overdrive/auth.pl
38
39     use Koha::ExternalContent::OverDrive;
40     my $od_client = Koha::ExternalContent::OverDrive->new();
41     my $od_auth_url = $od_client->auth_url($return_page_url);
42
43 =head1 DESCRIPTION
44
45 A (very) thin wrapper around C<WebService::ILS::OverDrive::Patron>
46
47 Takes "OverDrive*" Koha preferences
48
49 =cut
50
51 sub new {
52     my $class  = shift;
53     my $params = shift || {};
54     $params->{koha_session_id} or croak "No koha_session_id";
55
56     my $self = $class->SUPER::new($params);
57     unless ($params->{client}) {
58         my $client_key     = C4::Context->preference('OverDriveClientKey')
59           or croak("OverDriveClientKey pref not set");
60         my $client_secret  = C4::Context->preference('OverDriveClientSecret')
61           or croak("OverDriveClientSecret pref not set");
62         my $library_id     = C4::Context->preference('OverDriveLibraryID')
63           or croak("OverDriveLibraryID pref not set");
64         my ($token, $token_type) = $self->get_token_from_koha_session();
65         $self->client( WebService::ILS::OverDrive::Patron->new(
66             client_id         => $client_key,
67             client_secret     => $client_secret,
68             library_id        => $library_id,
69             access_token      => $token,
70             access_token_type => $token_type,
71             user_agent_params => { agent => $class->agent_string }
72         ) );
73     }
74     return $self;
75 }
76
77 =head1 L<WebService::ILS::OverDrive::Patron> METHODS
78
79 Methods used without mods:
80
81 =over 4
82
83 =item C<error_message()>
84
85 =item C<patron()>
86
87 =item C<checkouts()>
88
89 =item C<holds()>
90
91 =item C<checkout($id, $format)>
92
93 =item C<checkout_download_url($id)>
94
95 =item C<return($id)>
96
97 =item C<place_hold($id)>
98
99 =item C<remove_hold($id)>
100
101 =back
102
103 Methods with slightly moded interfaces:
104
105 =head2 auth_url($page_url)
106
107   Input: url of the page from which OverDrive authentication was requested
108
109   Returns: Post OverDrive auth return handler url (see SYNOPSIS)
110
111 =cut
112
113 sub auth_url {
114     my $self = shift;
115     my $page_url = shift or croak "Page url not provided";
116
117     my ($return_url, $page) = $self->_return_url($page_url);
118     $self->set_return_page_in_koha_session($page);
119     return $self->client->auth_url($return_url);
120 }
121
122 =head2 auth_by_code($code, $base_url)
123
124   To be called in external/overdrive/auth.pl upon return from OverDrive auth
125
126 =cut
127
128 sub auth_by_code {
129     my $self = shift;
130     my $code = shift or croak "OverDrive auth code not provided";
131     my $base_url = shift or croak "App base url not provided";
132
133     my ($access_token, $access_token_type, $auth_token)
134       = $self->client->auth_by_code($code, $self->_return_url($base_url));
135     $access_token or die "Invalid OverDrive code returned";
136     $self->set_token_in_koha_session($access_token, $access_token_type);
137
138     if (my $koha_patron = $self->koha_patron) {
139         $koha_patron->set({overdrive_auth_token => $auth_token})->store;
140     }
141     return $self->get_return_page_from_koha_session;
142 }
143
144 use constant AUTH_RETURN_HANDLER => "/cgi-bin/koha/external/overdrive/auth.pl";
145 sub _return_url {
146     my $self = shift;
147     my $page_url = shift or croak "Page url not provided";
148
149     my ($base_url, $page) = ($page_url =~ m!^(https?://[^/]+)(.*)!);
150     my $return_url = $base_url.AUTH_RETURN_HANDLER;
151
152     return wantarray ? ($return_url, $page) : $return_url;
153 }
154
155 use constant RETURN_PAGE_SESSION_KEY => "overdrive.return_page";
156 sub get_return_page_from_koha_session {
157     my $self = shift;
158     my $return_page = $self->get_from_koha_session(RETURN_PAGE_SESSION_KEY) || "";
159     $self->logger->debug("get_return_page_from_koha_session: $return_page");
160     return $return_page;
161 }
162 sub set_return_page_in_koha_session {
163     my $self = shift;
164     my $return_page = shift || "";
165     $self->logger->debug("set_return_page_in_koha_session: $return_page");
166     return $self->set_in_koha_session( RETURN_PAGE_SESSION_KEY, $return_page );
167 }
168
169 use constant ACCESS_TOKEN_SESSION_KEY => "overdrive.access_token";
170 my $ACCESS_TOKEN_DELIMITER = ":";
171 sub get_token_from_koha_session {
172     my $self = shift;
173     my ($token, $token_type)
174       = split $ACCESS_TOKEN_DELIMITER, $self->get_from_koha_session(ACCESS_TOKEN_SESSION_KEY) || "";
175     $self->logger->debug("get_token_from_koha_session: ".($token || "(none)"));
176     return ($token, $token_type);
177 }
178 sub set_token_in_koha_session {
179     my $self = shift;
180     my $token = shift || "";
181     my $token_type = shift || "";
182     $self->logger->debug("set_token_in_koha_session: $token $token_type");
183     return $self->set_in_koha_session(
184         ACCESS_TOKEN_SESSION_KEY,
185         join($ACCESS_TOKEN_DELIMITER, $token, $token_type)
186     );
187 }
188
189 =head1 OTHER METHODS
190
191 =head2 is_logged_in()
192
193   Returns boolean
194
195 =cut
196
197 sub is_logged_in {
198     my $self = shift;
199     my ($token, $token_type) = $self->get_token_from_koha_session();
200     $token ||= $self->auth_by_saved_token;
201     return $token;
202 }
203
204 sub auth_by_saved_token {
205     my $self = shift;
206
207     my $koha_patron = $self->koha_patron or return;
208
209     if (my $auth_token = $koha_patron->overdrive_auth_token) {
210         my ($access_token, $access_token_type, $new_auth_token)
211           = $self->client->auth_by_token($auth_token);
212         $self->set_token_in_koha_session($access_token, $access_token_type);
213         $koha_patron->set({overdrive_auth_token => $new_auth_token})->store;
214         return $access_token;
215     }
216
217     return;
218 }
219
220 =head2 forget()
221
222   Removes stored OverDrive token
223
224 =cut
225
226 sub forget {
227     my $self = shift;
228
229     $self->set_token_in_koha_session("", "");
230     if (my $koha_patron = $self->koha_patron) {
231         $koha_patron->set({overdrive_auth_token => undef})->store;
232     }
233 }
234
235 use vars qw{$AUTOLOAD};
236 sub AUTOLOAD {
237     my $self = shift;
238     (my $method = $AUTOLOAD) =~ s/.*:://;
239     my $od = $self->client;
240     local $@;
241     my $ret = eval { $od->$method(@_) };
242     if ($@) {
243         if ( $od->is_access_token_error($@) && $self->auth_by_saved_token ) {
244             return $od->$method(@_);
245         }
246         die $@;
247     }
248     return $ret;
249 }
250 sub DESTROY { }
251
252 =head1 AUTHOR
253
254 CatalystIT
255
256 =cut
257
258 1;