line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
package WebService::ILS::RecordedBooks; |
2
|
|
|
|
|
|
|
|
3
|
1
|
|
|
1
|
|
371
|
use Modern::Perl; |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
5
|
|
4
|
|
|
|
|
|
|
|
5
|
|
|
|
|
|
|
=encoding utf-8 |
6
|
|
|
|
|
|
|
|
7
|
|
|
|
|
|
|
=head1 NAME |
8
|
|
|
|
|
|
|
|
9
|
|
|
|
|
|
|
WebService::ILS::RecordedBooks - WebService::ILS module for RecordedBooks services |
10
|
|
|
|
|
|
|
|
11
|
|
|
|
|
|
|
=head1 SYNOPSIS |
12
|
|
|
|
|
|
|
|
13
|
|
|
|
|
|
|
use WebService::ILS::RecordedBooks::Partner; |
14
|
|
|
|
|
|
|
or |
15
|
|
|
|
|
|
|
use WebService::ILS::RecordedBooks::Patron; |
16
|
|
|
|
|
|
|
|
17
|
|
|
|
|
|
|
=head1 DESCRIPTION |
18
|
|
|
|
|
|
|
|
19
|
|
|
|
|
|
|
L - services |
20
|
|
|
|
|
|
|
that use partner credentials, for any patron |
21
|
|
|
|
|
|
|
|
22
|
|
|
|
|
|
|
L - same as above, |
23
|
|
|
|
|
|
|
except it operates on a single patron account |
24
|
|
|
|
|
|
|
|
25
|
|
|
|
|
|
|
L - services |
26
|
|
|
|
|
|
|
that use individual patron credentials, in addition to partner credentials |
27
|
|
|
|
|
|
|
|
28
|
|
|
|
|
|
|
L is preferred over |
29
|
|
|
|
|
|
|
L because the later requires patron |
30
|
|
|
|
|
|
|
credentials - username and password. However, if you do not know patron's |
31
|
|
|
|
|
|
|
email or RecordedBooks id (barcode) you are stuck with Patron interface. |
32
|
|
|
|
|
|
|
|
33
|
|
|
|
|
|
|
See L |
34
|
|
|
|
|
|
|
|
35
|
|
|
|
|
|
|
=cut |
36
|
|
|
|
|
|
|
|
37
|
1
|
|
|
1
|
|
92
|
use Carp; |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
45
|
|
38
|
1
|
|
|
1
|
|
7
|
use HTTP::Request::Common; |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
72
|
|
39
|
1
|
|
|
1
|
|
6
|
use URI::Escape; |
|
1
|
|
|
|
|
1
|
|
|
1
|
|
|
|
|
46
|
|
40
|
1
|
|
|
1
|
|
395
|
use JSON qw(to_json); |
|
1
|
|
|
|
|
7073
|
|
|
1
|
|
|
|
|
7
|
|
41
|
|
|
|
|
|
|
|
42
|
1
|
|
|
1
|
|
188
|
use parent qw(WebService::ILS::JSON); |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
9
|
|
43
|
|
|
|
|
|
|
|
44
|
1
|
|
|
1
|
|
50
|
use constant API_VERSION => "v1"; |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
48
|
|
45
|
1
|
|
|
1
|
|
5
|
use constant BASE_DOMAIN => "rbdigital.com"; |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
44
|
|
46
|
|
|
|
|
|
|
|
47
|
|
|
|
|
|
|
=head1 CONSTRUCTOR |
48
|
|
|
|
|
|
|
|
49
|
|
|
|
|
|
|
=head2 new (%params_hash or $params_hashref) |
50
|
|
|
|
|
|
|
|
51
|
|
|
|
|
|
|
=head3 Additional constructor params: |
52
|
|
|
|
|
|
|
|
53
|
|
|
|
|
|
|
=over 12 |
54
|
|
|
|
|
|
|
|
55
|
|
|
|
|
|
|
=item C => if set to true use https |
56
|
|
|
|
|
|
|
|
57
|
|
|
|
|
|
|
=item C => RecordedBooks domain for title url |
58
|
|
|
|
|
|
|
|
59
|
|
|
|
|
|
|
=back |
60
|
|
|
|
|
|
|
|
61
|
|
|
|
|
|
|
C is either RecordedBooks id (barcode) or email |
62
|
|
|
|
|
|
|
|
63
|
|
|
|
|
|
|
C if set is either "whatever.rbdigital.com" or "whatever", |
64
|
|
|
|
|
|
|
in which case rbdigital.com is appended. |
65
|
|
|
|
|
|
|
|
66
|
|
|
|
|
|
|
=cut |
67
|
|
|
|
|
|
|
|
68
|
1
|
|
|
|
|
4
|
use Class::Tiny qw( |
69
|
|
|
|
|
|
|
ssl |
70
|
|
|
|
|
|
|
domain |
71
|
|
|
|
|
|
|
_api_base_url |
72
|
1
|
|
|
1
|
|
6
|
); |
|
1
|
|
|
|
|
1
|
|
73
|
|
|
|
|
|
|
|
74
|
|
|
|
|
|
|
__PACKAGE__->_set_param_spec({ |
75
|
|
|
|
|
|
|
client_id => { required => 0 }, |
76
|
|
|
|
|
|
|
library_id => { required => 1 }, |
77
|
|
|
|
|
|
|
domain => { required => 0 }, |
78
|
|
|
|
|
|
|
ssl => { required => 0, default => 1 }, |
79
|
|
|
|
|
|
|
}); |
80
|
|
|
|
|
|
|
|
81
|
|
|
|
|
|
|
sub BUILD { |
82
|
0
|
|
|
0
|
0
|
|
my $self = shift; |
83
|
0
|
|
|
|
|
|
my $params = shift; |
84
|
|
|
|
|
|
|
|
85
|
0
|
0
|
|
|
|
|
if (my $domain = $self->domain) { |
86
|
0
|
0
|
|
|
|
|
$self->domain("$domain.".BASE_DOMAIN) unless $domain =~ m/\./; |
87
|
|
|
|
|
|
|
} |
88
|
|
|
|
|
|
|
|
89
|
0
|
|
|
|
|
|
my $ssl = $self->ssl; |
90
|
0
|
|
|
|
|
|
my $ua = $self->user_agent; |
91
|
0
|
0
|
|
|
|
|
$ua->ssl_opts( verify_hostname => 0 ) if $ssl; |
92
|
|
|
|
|
|
|
|
93
|
0
|
0
|
|
|
|
|
my $api_url = sprintf "%s://api.%s", $ssl ? "https" : "http", BASE_DOMAIN; |
94
|
0
|
|
|
|
|
|
$self->_api_base_url($api_url); |
95
|
|
|
|
|
|
|
} |
96
|
|
|
|
|
|
|
|
97
|
|
|
|
|
|
|
sub api_url { |
98
|
0
|
|
|
0
|
0
|
|
my $self = shift; |
99
|
0
|
0
|
|
|
|
|
my $action = shift or croak "No action"; |
100
|
|
|
|
|
|
|
|
101
|
0
|
|
|
|
|
|
return sprintf "%s/%s%s", $self->_api_base_url, API_VERSION, $action; |
102
|
|
|
|
|
|
|
} |
103
|
|
|
|
|
|
|
|
104
|
|
|
|
|
|
|
sub library_action_base_url { |
105
|
0
|
|
|
0
|
0
|
|
my $self = shift; |
106
|
|
|
|
|
|
|
|
107
|
0
|
|
|
|
|
|
return $self->api_url("/libraries/".$self->library_id); |
108
|
|
|
|
|
|
|
} |
109
|
|
|
|
|
|
|
|
110
|
|
|
|
|
|
|
sub products_url { |
111
|
0
|
|
|
0
|
0
|
|
my $self = shift; |
112
|
0
|
|
|
|
|
|
return $self->library_action_base_url."/search"; |
113
|
|
|
|
|
|
|
} |
114
|
|
|
|
|
|
|
|
115
|
|
|
|
|
|
|
sub circulation_action_url { |
116
|
0
|
|
|
0
|
0
|
|
my $self = shift; |
117
|
0
|
0
|
|
|
|
|
my $action = shift or croak "No action"; |
118
|
|
|
|
|
|
|
|
119
|
0
|
|
|
|
|
|
return $self->circulation_action_base_url(@_).$action; |
120
|
|
|
|
|
|
|
} |
121
|
|
|
|
|
|
|
|
122
|
|
|
|
|
|
|
sub _access_auth_string { |
123
|
0
|
|
|
0
|
|
|
my $self = shift; |
124
|
0
|
|
|
|
|
|
return $self->client_secret; |
125
|
|
|
|
|
|
|
} |
126
|
|
|
|
|
|
|
|
127
|
|
|
|
|
|
|
sub native_countries { |
128
|
0
|
|
|
0
|
0
|
|
my $self = shift; |
129
|
|
|
|
|
|
|
|
130
|
0
|
|
|
|
|
|
my $url = $self->api_url("/countries"); |
131
|
0
|
|
|
|
|
|
return $self->get_without_auth($url); |
132
|
|
|
|
|
|
|
} |
133
|
|
|
|
|
|
|
|
134
|
|
|
|
|
|
|
sub native_facets { |
135
|
0
|
|
|
0
|
0
|
|
my $self = shift; |
136
|
|
|
|
|
|
|
|
137
|
0
|
|
|
|
|
|
my $url = $self->api_url("/facets"); |
138
|
0
|
|
|
|
|
|
return $self->get_response($url); |
139
|
|
|
|
|
|
|
} |
140
|
|
|
|
|
|
|
|
141
|
|
|
|
|
|
|
|
142
|
|
|
|
|
|
|
sub native_facet_values { |
143
|
0
|
|
|
0
|
0
|
|
my $self = shift; |
144
|
0
|
0
|
|
|
|
|
my $facet = shift or croak "No facet"; |
145
|
|
|
|
|
|
|
|
146
|
0
|
|
|
|
|
|
my $url = $self->api_url("/facets/$facet"); |
147
|
0
|
|
|
|
|
|
return $self->get_without_auth($url); |
148
|
|
|
|
|
|
|
} |
149
|
|
|
|
|
|
|
|
150
|
|
|
|
|
|
|
sub native_libraries_search { |
151
|
0
|
|
|
0
|
0
|
|
my $self = shift; |
152
|
0
|
0
|
|
|
|
|
my $query = shift or croak "No query"; |
153
|
0
|
|
|
|
|
|
my $region = shift; |
154
|
|
|
|
|
|
|
|
155
|
0
|
|
|
|
|
|
my %search_params = ( term => $query ); |
156
|
0
|
0
|
|
|
|
|
$search_params{ar} = $region if $region; |
157
|
0
|
|
|
|
|
|
my $url = $self->api_url("/suggestive/libraries"); |
158
|
0
|
|
|
|
|
|
return $self->get_without_auth($url, \%search_params); |
159
|
|
|
|
|
|
|
} |
160
|
|
|
|
|
|
|
|
161
|
|
|
|
|
|
|
sub get_without_auth { |
162
|
0
|
|
|
0
|
0
|
|
my $self = shift; |
163
|
0
|
0
|
|
|
|
|
my $url = shift or croak "No url"; |
164
|
0
|
|
|
|
|
|
my $get_params = shift; # hash ref |
165
|
|
|
|
|
|
|
|
166
|
0
|
|
|
|
|
|
my $uri = URI->new($url); |
167
|
0
|
0
|
|
|
|
|
$uri->query_form($get_params) if $get_params; |
168
|
0
|
|
|
|
|
|
my $request = HTTP::Request::Common::GET( $uri ); |
169
|
0
|
|
|
|
|
|
my $response = $self->user_agent->request( $request ); |
170
|
0
|
|
|
|
|
|
$self->check_response($response); |
171
|
|
|
|
|
|
|
|
172
|
|
|
|
|
|
|
return $self->process_json_response($response, sub { |
173
|
0
|
|
|
0
|
|
|
my ($data) = @_; |
174
|
0
|
0
|
|
|
|
|
die "No data\n" unless $data; |
175
|
0
|
|
|
|
|
|
return $data; |
176
|
0
|
|
|
|
|
|
}); |
177
|
|
|
|
|
|
|
} |
178
|
|
|
|
|
|
|
|
179
|
|
|
|
|
|
|
=head1 DISCOVERY METHODS |
180
|
|
|
|
|
|
|
|
181
|
|
|
|
|
|
|
=head2 facets () |
182
|
|
|
|
|
|
|
|
183
|
|
|
|
|
|
|
=head3 Returns a hashref of facet => [values] |
184
|
|
|
|
|
|
|
|
185
|
|
|
|
|
|
|
=cut |
186
|
|
|
|
|
|
|
|
187
|
|
|
|
|
|
|
sub facets { |
188
|
0
|
|
|
0
|
1
|
|
my $self = shift; |
189
|
|
|
|
|
|
|
|
190
|
0
|
|
|
|
|
|
my $facets = $self->native_facets; |
191
|
0
|
|
|
|
|
|
my %facet_values; |
192
|
0
|
|
|
|
|
|
foreach (@$facets) { |
193
|
0
|
|
|
|
|
|
my $f = $_->{facetToken}; |
194
|
0
|
|
|
|
|
|
$facet_values{$f} = [map $_->{valueToken}, @{ $self->native_facet_values($f) }]; |
|
0
|
|
|
|
|
|
|
195
|
|
|
|
|
|
|
} |
196
|
0
|
|
|
|
|
|
return \%facet_values; |
197
|
|
|
|
|
|
|
} |
198
|
|
|
|
|
|
|
|
199
|
|
|
|
|
|
|
=head2 search ($params_hashref) |
200
|
|
|
|
|
|
|
|
201
|
|
|
|
|
|
|
=head3 Additional input params: |
202
|
|
|
|
|
|
|
|
203
|
|
|
|
|
|
|
=over 12 |
204
|
|
|
|
|
|
|
|
205
|
|
|
|
|
|
|
=item C => a hashref of facet values |
206
|
|
|
|
|
|
|
|
207
|
|
|
|
|
|
|
=back |
208
|
|
|
|
|
|
|
|
209
|
|
|
|
|
|
|
=cut |
210
|
|
|
|
|
|
|
|
211
|
|
|
|
|
|
|
my %SORT_XLATE = ( |
212
|
|
|
|
|
|
|
rating => undef, |
213
|
|
|
|
|
|
|
publication_date => undef, # not available |
214
|
|
|
|
|
|
|
); |
215
|
|
|
|
|
|
|
sub search { |
216
|
0
|
|
|
0
|
1
|
|
my $self = shift; |
217
|
0
|
|
0
|
|
|
|
my $params = shift || {}; |
218
|
|
|
|
|
|
|
|
219
|
0
|
|
|
|
|
|
my $url = $self->products_url; |
220
|
|
|
|
|
|
|
|
221
|
0
|
0
|
|
|
|
|
if (my $query = delete $params->{query}) { |
222
|
0
|
0
|
|
|
|
|
$query = join " ", @$query if ref $query; |
223
|
0
|
|
|
|
|
|
$params->{all} = $query; |
224
|
|
|
|
|
|
|
} |
225
|
0
|
0
|
|
|
|
|
if (my $page_size = delete $params->{page_size}) { |
226
|
0
|
|
|
|
|
|
$params->{'page-size'} = $page_size; |
227
|
|
|
|
|
|
|
} |
228
|
0
|
0
|
|
|
|
|
if (my $page_number = delete $params->{page}) { |
229
|
0
|
0
|
|
|
|
|
die "page_size must be specified for paging" unless $params->{'page-size'}; |
230
|
0
|
|
|
|
|
|
$params->{'page-index'} = $page_number - 1; |
231
|
|
|
|
|
|
|
} |
232
|
0
|
0
|
|
|
|
|
if (my $sort = delete $params->{sort}) { |
233
|
0
|
|
|
|
|
|
my $sa = $self->_parse_sort_string($sort, \%SORT_XLATE); |
234
|
0
|
0
|
|
|
|
|
if (@$sa) { |
235
|
0
|
|
|
|
|
|
my @params = %$params; |
236
|
0
|
|
|
|
|
|
foreach (@$sa) { |
237
|
0
|
|
|
|
|
|
my ($s, $d) = split ':'; |
238
|
0
|
|
|
|
|
|
push @params, "sort-by", $s; |
239
|
0
|
0
|
|
|
|
|
push @params, "sort-order", $d if $d; |
240
|
|
|
|
|
|
|
} |
241
|
0
|
|
|
|
|
|
return $self->_search_result_xlate( $self->get_response($url, \@params) ); |
242
|
|
|
|
|
|
|
} |
243
|
|
|
|
|
|
|
} |
244
|
|
|
|
|
|
|
|
245
|
0
|
|
|
|
|
|
return $self->_search_result_xlate( $self->get_response($url, $params) ); |
246
|
|
|
|
|
|
|
} |
247
|
|
|
|
|
|
|
|
248
|
|
|
|
|
|
|
sub _search_result_xlate { |
249
|
0
|
|
|
0
|
|
|
my $self = shift; |
250
|
0
|
0
|
|
|
|
|
my $res = shift or return; |
251
|
|
|
|
|
|
|
|
252
|
0
|
|
|
|
|
|
my $domain = $self->domain; |
253
|
|
|
|
|
|
|
return { |
254
|
|
|
|
|
|
|
items => [ map { |
255
|
0
|
|
|
|
|
|
my $i = $self->_item_xlate($_->{item}); |
256
|
0
|
0
|
0
|
|
|
|
$i->{url} ||= "https://$domain/#titles/$i->{isbn}" if $domain; |
257
|
0
|
|
|
|
|
|
$i; |
258
|
0
|
0
|
|
|
|
|
} @{$res->{items} || []} ], |
259
|
|
|
|
|
|
|
page_size => $res->{pageSize}, |
260
|
|
|
|
|
|
|
page => $res->{pageIndex} + 1, |
261
|
|
|
|
|
|
|
pages => $res->{pageCount}, |
262
|
0
|
|
|
|
|
|
}; |
263
|
|
|
|
|
|
|
} |
264
|
|
|
|
|
|
|
|
265
|
|
|
|
|
|
|
my %SEARCH_RESULT_ITEM_XLATE = ( |
266
|
|
|
|
|
|
|
id => "id", |
267
|
|
|
|
|
|
|
title => "title", |
268
|
|
|
|
|
|
|
subtitle => "subtitle", |
269
|
|
|
|
|
|
|
shortDescription => "description", |
270
|
|
|
|
|
|
|
mediaType => "media", |
271
|
|
|
|
|
|
|
downloadUrl => "url", |
272
|
|
|
|
|
|
|
encryptionKey => "encryption_key", |
273
|
|
|
|
|
|
|
isbn => "isbn", |
274
|
|
|
|
|
|
|
hasDrm => "drm", |
275
|
|
|
|
|
|
|
releasedDate => "publication_date", |
276
|
|
|
|
|
|
|
size => "size", |
277
|
|
|
|
|
|
|
language => "language", |
278
|
|
|
|
|
|
|
expiration => "expires", |
279
|
|
|
|
|
|
|
); |
280
|
|
|
|
|
|
|
my %ITEM_FILES_XLATE = ( |
281
|
|
|
|
|
|
|
id => "id", |
282
|
|
|
|
|
|
|
filename => "filename", |
283
|
|
|
|
|
|
|
display => "title", |
284
|
|
|
|
|
|
|
downloadUrl => "url", |
285
|
|
|
|
|
|
|
size => "size", |
286
|
|
|
|
|
|
|
); |
287
|
|
|
|
|
|
|
sub _item_xlate { |
288
|
0
|
|
|
0
|
|
|
my $self = shift; |
289
|
0
|
|
|
|
|
|
my $item = shift; |
290
|
|
|
|
|
|
|
|
291
|
0
|
|
|
|
|
|
my $std_item = $self->_result_xlate($item, \%SEARCH_RESULT_ITEM_XLATE); |
292
|
|
|
|
|
|
|
|
293
|
0
|
0
|
|
|
|
|
if (my $images = delete $item->{images}) { # XXX let's say that caller wouldn't mind |
294
|
0
|
|
|
|
|
|
$std_item->{images} = {map { $_->{name} => $_->{url} } @$images}; |
|
0
|
|
|
|
|
|
|
295
|
|
|
|
|
|
|
} |
296
|
|
|
|
|
|
|
|
297
|
0
|
0
|
|
|
|
|
if (my $files = delete $item->{files}) { |
298
|
0
|
|
|
|
|
|
$std_item->{files} = [ map $self->_result_xlate($_, \%ITEM_FILES_XLATE), @$files ]; |
299
|
|
|
|
|
|
|
} |
300
|
|
|
|
|
|
|
|
301
|
0
|
|
|
|
|
|
my %facets; |
302
|
0
|
0
|
|
|
|
|
if (my $publisher = delete $item->{publisher}) { |
303
|
0
|
0
|
|
|
|
|
if (ref $publisher) { |
304
|
0
|
0
|
|
|
|
|
if (my $f = $publisher->{facet}) { |
305
|
0
|
|
|
|
|
|
$facets{$f} = [$publisher->{token}]; |
306
|
|
|
|
|
|
|
} |
307
|
0
|
|
|
|
|
|
$publisher = $publisher->{text}; |
308
|
|
|
|
|
|
|
} |
309
|
0
|
|
|
|
|
|
$std_item->{publisher} = $publisher; |
310
|
|
|
|
|
|
|
} |
311
|
0
|
0
|
|
|
|
|
if (my $authors = delete $item->{authors}) { |
312
|
0
|
|
|
|
|
|
my @a; |
313
|
0
|
0
|
|
|
|
|
if (ref $authors) { |
314
|
0
|
|
|
|
|
|
foreach (@$authors) { |
315
|
0
|
0
|
|
|
|
|
push @a, $_->{text} if $_->{text}; |
316
|
0
|
0
|
|
|
|
|
if (my $f = $_->{facet}) { |
317
|
0
|
|
0
|
|
|
|
my $f_a = $facets{$f} ||= []; |
318
|
0
|
|
|
|
|
|
push @$f_a, $_->{token}; |
319
|
|
|
|
|
|
|
} |
320
|
|
|
|
|
|
|
} |
321
|
|
|
|
|
|
|
} |
322
|
|
|
|
|
|
|
else { |
323
|
0
|
|
|
|
|
|
push @a, $authors; |
324
|
|
|
|
|
|
|
} |
325
|
0
|
|
|
|
|
|
$std_item->{author} = join ", ", @a; |
326
|
|
|
|
|
|
|
} |
327
|
0
|
|
|
|
|
|
foreach my $v (values %$item) { |
328
|
0
|
0
|
|
|
|
|
my $ref = ref $v or next; |
329
|
0
|
0
|
|
|
|
|
$v = [$v] if $ref eq "HASH"; |
330
|
0
|
0
|
|
|
|
|
next unless ref($v) eq "ARRAY"; |
331
|
0
|
|
|
|
|
|
foreach (@$v) { |
332
|
0
|
0
|
|
|
|
|
if (my $f = $_->{facet}) { |
333
|
0
|
|
0
|
|
|
|
my $f_a = $facets{$f} ||= []; |
334
|
0
|
|
|
|
|
|
push @$f_a, $_->{token}; |
335
|
|
|
|
|
|
|
} |
336
|
|
|
|
|
|
|
} |
337
|
|
|
|
|
|
|
} |
338
|
0
|
0
|
|
|
|
|
$std_item->{facets} = \%facets if keys %facets; |
339
|
|
|
|
|
|
|
|
340
|
0
|
|
|
|
|
|
return $std_item; |
341
|
|
|
|
|
|
|
} |
342
|
|
|
|
|
|
|
|
343
|
|
|
|
|
|
|
=head2 named_query_search ($query, $media) |
344
|
|
|
|
|
|
|
|
345
|
|
|
|
|
|
|
See C below for $query, $media |
346
|
|
|
|
|
|
|
|
347
|
|
|
|
|
|
|
=cut |
348
|
|
|
|
|
|
|
|
349
|
|
|
|
|
|
|
sub named_query_search { |
350
|
0
|
|
|
0
|
1
|
|
my $self = shift; |
351
|
0
|
|
|
|
|
|
return $self->_search_result_xlate( $self->native_named_query_search(@_) ); |
352
|
|
|
|
|
|
|
} |
353
|
|
|
|
|
|
|
|
354
|
|
|
|
|
|
|
=head2 facet_search ($facets) |
355
|
|
|
|
|
|
|
|
356
|
|
|
|
|
|
|
See C below for $facets |
357
|
|
|
|
|
|
|
|
358
|
|
|
|
|
|
|
=cut |
359
|
|
|
|
|
|
|
|
360
|
|
|
|
|
|
|
sub facet_search { |
361
|
0
|
|
|
0
|
1
|
|
my $self = shift; |
362
|
0
|
|
|
|
|
|
return $self->_search_result_xlate( $self->native_facet_search(@_) ); |
363
|
|
|
|
|
|
|
} |
364
|
|
|
|
|
|
|
|
365
|
|
|
|
|
|
|
sub item_metadata { |
366
|
0
|
|
|
0
|
1
|
|
my $self = shift; |
367
|
0
|
0
|
|
|
|
|
my $ni = $self->native_item(@_) or return; |
368
|
0
|
|
|
|
|
|
return $self->_item_xlate( $ni->{item} ); |
369
|
|
|
|
|
|
|
} |
370
|
|
|
|
|
|
|
|
371
|
|
|
|
|
|
|
=head1 CIRCULATION METHOD SPECIFICS |
372
|
|
|
|
|
|
|
|
373
|
|
|
|
|
|
|
Differences to general L interface |
374
|
|
|
|
|
|
|
|
375
|
|
|
|
|
|
|
=cut |
376
|
|
|
|
|
|
|
|
377
|
|
|
|
|
|
|
=head2 holds () |
378
|
|
|
|
|
|
|
|
379
|
|
|
|
|
|
|
=head2 place_hold ($isbn) |
380
|
|
|
|
|
|
|
|
381
|
|
|
|
|
|
|
=head2 remove_hold ($isbn) |
382
|
|
|
|
|
|
|
|
383
|
|
|
|
|
|
|
=cut |
384
|
|
|
|
|
|
|
|
385
|
|
|
|
|
|
|
sub holds { |
386
|
0
|
|
|
0
|
1
|
|
my $self = shift; |
387
|
|
|
|
|
|
|
|
388
|
0
|
|
|
|
|
|
my $items = $self->native_holds(@_); |
389
|
|
|
|
|
|
|
return { |
390
|
|
|
|
|
|
|
total => scalar @$items, |
391
|
|
|
|
|
|
|
items => [ map { |
392
|
0
|
|
|
|
|
|
my $i = $self->_item_xlate($_); |
|
0
|
|
|
|
|
|
|
393
|
0
|
|
|
|
|
|
$i->{hold_id} = $_->{transactionId}; |
394
|
0
|
|
|
|
|
|
$i; |
395
|
|
|
|
|
|
|
} @$items ], |
396
|
|
|
|
|
|
|
}; |
397
|
|
|
|
|
|
|
} |
398
|
|
|
|
|
|
|
|
399
|
|
|
|
|
|
|
sub place_hold { |
400
|
0
|
|
|
0
|
1
|
|
my $self = shift; |
401
|
0
|
0
|
|
|
|
|
my $isbn = shift or croak "No isbn"; |
402
|
|
|
|
|
|
|
|
403
|
0
|
|
|
|
|
|
my $url = $self->circulation_action_url("/holds/$isbn", @_); |
404
|
0
|
|
|
|
|
|
my $request = HTTP::Request::Common::POST( $url ); |
405
|
0
|
|
|
|
|
|
my $response = $self->_request_with_auth($request); |
406
|
0
|
0
|
|
|
|
|
unless ($response->is_success) { |
407
|
|
|
|
|
|
|
$self->process_json_error_response($response, sub { |
408
|
0
|
|
|
0
|
|
|
my ($data) = @_; |
409
|
0
|
0
|
|
|
|
|
if (my $message = $data->{message}) { |
410
|
0
|
0
|
|
|
|
|
return 1 if $message =~ m/already exists/i; |
411
|
0
|
|
|
|
|
|
die $message; |
412
|
|
|
|
|
|
|
} |
413
|
0
|
|
0
|
|
|
|
die $self->_error_from_json($data) || "Cannot place hold: ".to_json($data); |
414
|
0
|
|
|
|
|
|
}); |
415
|
|
|
|
|
|
|
} |
416
|
|
|
|
|
|
|
|
417
|
0
|
0
|
|
|
|
|
if (my $holds = $self->holds(@_)) { |
418
|
0
|
|
|
|
|
|
foreach my $i (@{ $holds->{items} }) { |
|
0
|
|
|
|
|
|
|
419
|
0
|
0
|
|
|
|
|
if ($i->{isbn} eq $isbn) { |
420
|
0
|
|
|
|
|
|
$i->{total} = $holds->{total}; |
421
|
0
|
|
|
|
|
|
return $i; |
422
|
|
|
|
|
|
|
} |
423
|
|
|
|
|
|
|
} |
424
|
|
|
|
|
|
|
} |
425
|
|
|
|
|
|
|
|
426
|
0
|
|
|
|
|
|
my $content = $response->decoded_content; |
427
|
0
|
|
|
|
|
|
my $content_type = $response->header('Content-Type'); |
428
|
0
|
|
|
|
|
|
my $error; |
429
|
0
|
0
|
0
|
|
|
|
if ($content_type && $content_type =~ m!application/json!) { |
430
|
0
|
0
|
|
|
|
|
if (my $data = eval { from_json( $content ) }) { |
|
0
|
|
|
|
|
|
|
431
|
0
|
|
|
|
|
|
$error = $self->_error_from_json($data); |
432
|
|
|
|
|
|
|
} |
433
|
|
|
|
|
|
|
} |
434
|
|
|
|
|
|
|
|
435
|
0
|
|
0
|
|
|
|
die $error || "Cannot place hold:\n$content"; |
436
|
|
|
|
|
|
|
} |
437
|
|
|
|
|
|
|
|
438
|
|
|
|
|
|
|
sub remove_hold { |
439
|
0
|
|
|
0
|
1
|
|
my $self = shift; |
440
|
0
|
0
|
|
|
|
|
my $isbn = shift or croak "No isbn"; |
441
|
|
|
|
|
|
|
|
442
|
0
|
|
|
|
|
|
my $url = $self->circulation_action_url("/holds/$isbn", @_); |
443
|
0
|
|
|
|
|
|
my $request = HTTP::Request::Common::DELETE( $url ); |
444
|
0
|
|
|
|
|
|
my $response = $self->_request_with_auth($request); |
445
|
0
|
0
|
|
|
|
|
unless ($response->is_success) { |
446
|
|
|
|
|
|
|
return $self->process_json_error_response($response, sub { |
447
|
0
|
|
|
0
|
|
|
my ($data) = @_; |
448
|
0
|
0
|
|
|
|
|
if (my $message = $data->{message}) { |
449
|
0
|
0
|
|
|
|
|
return 1 if $message =~ m/not exists|expired/i; |
450
|
0
|
|
|
|
|
|
die $message; |
451
|
|
|
|
|
|
|
} |
452
|
0
|
|
0
|
|
|
|
die $self->_error_from_json($data) || "Cannot remove hold: ".to_json($data); |
453
|
0
|
|
|
|
|
|
}); |
454
|
|
|
|
|
|
|
} |
455
|
0
|
|
|
|
|
|
return 1; |
456
|
|
|
|
|
|
|
} |
457
|
|
|
|
|
|
|
|
458
|
|
|
|
|
|
|
=head2 checkouts () |
459
|
|
|
|
|
|
|
|
460
|
|
|
|
|
|
|
=head2 checkout ($isbn, $days) |
461
|
|
|
|
|
|
|
|
462
|
|
|
|
|
|
|
=head2 renew ($isbn) |
463
|
|
|
|
|
|
|
|
464
|
|
|
|
|
|
|
=head2 return ($isbn) |
465
|
|
|
|
|
|
|
|
466
|
|
|
|
|
|
|
=cut |
467
|
|
|
|
|
|
|
|
468
|
|
|
|
|
|
|
sub checkouts { |
469
|
0
|
|
|
0
|
1
|
|
my $self = shift; |
470
|
|
|
|
|
|
|
|
471
|
0
|
|
|
|
|
|
my $items = $self->native_checkouts(@_); |
472
|
|
|
|
|
|
|
return { |
473
|
|
|
|
|
|
|
total => scalar @$items, |
474
|
|
|
|
|
|
|
items => [ map { |
475
|
0
|
|
|
|
|
|
my $i = $self->_item_xlate($_); |
|
0
|
|
|
|
|
|
|
476
|
0
|
|
|
|
|
|
$i->{checkout_id} = $_->{transactionId}; |
477
|
0
|
|
|
|
|
|
$i; |
478
|
|
|
|
|
|
|
} @$items ], |
479
|
|
|
|
|
|
|
}; |
480
|
|
|
|
|
|
|
} |
481
|
|
|
|
|
|
|
|
482
|
|
|
|
|
|
|
sub checkout { |
483
|
0
|
|
|
0
|
1
|
|
my $self = shift; |
484
|
0
|
0
|
|
|
|
|
my $isbn = shift or croak "No isbn"; |
485
|
0
|
|
|
|
|
|
my $days = shift; |
486
|
|
|
|
|
|
|
|
487
|
0
|
0
|
|
|
|
|
if (my $checkouts = $self->checkouts(@_)) { |
488
|
0
|
|
|
|
|
|
foreach my $i (@{ $checkouts->{items} }) { |
|
0
|
|
|
|
|
|
|
489
|
0
|
0
|
|
|
|
|
if ( $i->{isbn} eq $isbn ) { |
490
|
0
|
|
|
|
|
|
$i->{total} = scalar @{ $checkouts->{items} }; |
|
0
|
|
|
|
|
|
|
491
|
0
|
|
|
|
|
|
return $i; |
492
|
|
|
|
|
|
|
} |
493
|
|
|
|
|
|
|
} |
494
|
|
|
|
|
|
|
} |
495
|
|
|
|
|
|
|
|
496
|
0
|
|
|
|
|
|
my $url = $self->circulation_action_url("/checkouts/$isbn", @_); |
497
|
0
|
0
|
|
|
|
|
$url .= "?days=$days" if $days; |
498
|
0
|
|
|
|
|
|
my $res = $self->with_post_request( |
499
|
|
|
|
|
|
|
\&_basic_callback, |
500
|
|
|
|
|
|
|
$url |
501
|
|
|
|
|
|
|
); |
502
|
|
|
|
|
|
|
|
503
|
0
|
0
|
|
|
|
|
my $checkouts = $self->checkouts(@_) or die "Cannot checkout, unknown error"; |
504
|
0
|
|
|
|
|
|
foreach my $i (@{ $checkouts->{items} }) { |
|
0
|
|
|
|
|
|
|
505
|
0
|
0
|
|
|
|
|
if ($i->{isbn} eq $isbn) { |
506
|
0
|
|
|
|
|
|
$i->{total} = scalar @{ $checkouts->{items} }; |
|
0
|
|
|
|
|
|
|
507
|
0
|
|
|
|
|
|
return $i; |
508
|
|
|
|
|
|
|
} |
509
|
|
|
|
|
|
|
} |
510
|
0
|
|
0
|
|
|
|
die $res->{message} || "Cannot checkout, unknown error"; |
511
|
|
|
|
|
|
|
} |
512
|
|
|
|
|
|
|
|
513
|
|
|
|
|
|
|
sub renew { |
514
|
0
|
|
|
0
|
1
|
|
my $self = shift; |
515
|
0
|
0
|
|
|
|
|
my $isbn = shift or croak "No isbn"; |
516
|
|
|
|
|
|
|
|
517
|
0
|
|
|
|
|
|
my $url = $self->circulation_action_url("/checkouts/$isbn", @_); |
518
|
0
|
|
|
|
|
|
my $res = $self->with_put_request( |
519
|
|
|
|
|
|
|
\&_basic_callback, |
520
|
|
|
|
|
|
|
$url |
521
|
|
|
|
|
|
|
); |
522
|
|
|
|
|
|
|
|
523
|
0
|
0
|
|
|
|
|
my $checkouts = $self->checkouts(@_) or die "Cannot renew, unkmown error"; |
524
|
0
|
|
|
|
|
|
foreach my $i (@{ $checkouts->{items} }) { |
|
0
|
|
|
|
|
|
|
525
|
0
|
0
|
|
|
|
|
if ($i->{isbn} eq $isbn) { |
526
|
0
|
|
|
|
|
|
$i->{total} = scalar @{ $checkouts->{items} }; |
|
0
|
|
|
|
|
|
|
527
|
0
|
|
|
|
|
|
return $i; |
528
|
|
|
|
|
|
|
} |
529
|
|
|
|
|
|
|
} |
530
|
0
|
|
0
|
|
|
|
die $res->{output} || "Cannot renew, unknown error"; |
531
|
|
|
|
|
|
|
} |
532
|
|
|
|
|
|
|
|
533
|
|
|
|
|
|
|
sub return { |
534
|
0
|
|
|
0
|
1
|
|
my $self = shift; |
535
|
0
|
0
|
|
|
|
|
my $isbn = shift or croak "No isbn"; |
536
|
|
|
|
|
|
|
|
537
|
0
|
|
|
|
|
|
my $url = $self->circulation_action_url("/checkouts/$isbn", @_); |
538
|
0
|
|
|
|
|
|
my $request = HTTP::Request::Common::DELETE( $url ); |
539
|
0
|
|
|
|
|
|
my $response = $self->_request_with_auth($request); |
540
|
0
|
0
|
|
|
|
|
unless ($response->is_success) { |
541
|
|
|
|
|
|
|
return $self->process_json_error_response($response, sub { |
542
|
0
|
|
|
0
|
|
|
my ($data) = @_; |
543
|
0
|
0
|
|
|
|
|
if (my $message = $data->{message}) { |
544
|
0
|
0
|
|
|
|
|
return 1 if $message =~ m/not exists|expired/i; |
545
|
0
|
|
|
|
|
|
die $message; |
546
|
|
|
|
|
|
|
} |
547
|
0
|
|
|
|
|
|
die "Cannot return: ".to_json($data); |
548
|
0
|
|
|
|
|
|
}); |
549
|
|
|
|
|
|
|
} |
550
|
0
|
|
|
|
|
|
return 1; |
551
|
|
|
|
|
|
|
} |
552
|
|
|
|
|
|
|
|
553
|
|
|
|
|
|
|
=head1 NATIVE METHODS |
554
|
|
|
|
|
|
|
|
555
|
|
|
|
|
|
|
=head2 native_search ($params_hashref) |
556
|
|
|
|
|
|
|
|
557
|
|
|
|
|
|
|
See L |
558
|
|
|
|
|
|
|
|
559
|
|
|
|
|
|
|
=cut |
560
|
|
|
|
|
|
|
|
561
|
|
|
|
|
|
|
sub native_search { |
562
|
0
|
|
|
0
|
1
|
|
my $self = shift; |
563
|
0
|
|
|
|
|
|
my $search_params = shift; |
564
|
|
|
|
|
|
|
|
565
|
0
|
|
|
|
|
|
return $self->get_response($self->products_url, $search_params); |
566
|
|
|
|
|
|
|
} |
567
|
|
|
|
|
|
|
|
568
|
|
|
|
|
|
|
=head2 native_named_query_search ($query, $media) |
569
|
|
|
|
|
|
|
|
570
|
|
|
|
|
|
|
$query can be one of 'bestsellers', 'most-popular', 'newly-added' |
571
|
|
|
|
|
|
|
$media can be 'eaudio' or 'ebook' |
572
|
|
|
|
|
|
|
|
573
|
|
|
|
|
|
|
=cut |
574
|
|
|
|
|
|
|
|
575
|
|
|
|
|
|
|
my @MEDIA = qw( eaudio ebook ); |
576
|
|
|
|
|
|
|
my @NAMED_QUERY = ( 'bestsellers', 'most-popular', 'newly-added' ); |
577
|
|
|
|
|
|
|
sub native_named_query_search { |
578
|
0
|
|
|
0
|
1
|
|
my $self = shift; |
579
|
0
|
0
|
|
|
|
|
my $query = shift or croak "No query"; |
580
|
0
|
0
|
|
|
|
|
my $media = shift or croak "No media"; |
581
|
|
|
|
|
|
|
|
582
|
|
|
|
|
|
|
croak "Invalid media $media - should be one of ".join(", ", @MEDIA) |
583
|
0
|
0
|
|
|
|
|
unless grep { $_ eq $media } @MEDIA; |
|
0
|
|
|
|
|
|
|
584
|
|
|
|
|
|
|
croak "Invalid named query $query - should be one of ".join(", ", @NAMED_QUERY) |
585
|
0
|
0
|
|
|
|
|
unless grep { $_ eq $query } @NAMED_QUERY; |
|
0
|
|
|
|
|
|
|
586
|
|
|
|
|
|
|
|
587
|
0
|
|
|
|
|
|
my $url = $self->products_url."/$media/$query"; |
588
|
0
|
|
|
|
|
|
return $self->get_response($url); |
589
|
|
|
|
|
|
|
} |
590
|
|
|
|
|
|
|
|
591
|
|
|
|
|
|
|
=head2 native_facet_search ($facets) |
592
|
|
|
|
|
|
|
|
593
|
|
|
|
|
|
|
$facets can be either: |
594
|
|
|
|
|
|
|
* a hashref of facet => [values], |
595
|
|
|
|
|
|
|
* an arrayref of values |
596
|
|
|
|
|
|
|
* a single value |
597
|
|
|
|
|
|
|
|
598
|
|
|
|
|
|
|
=cut |
599
|
|
|
|
|
|
|
|
600
|
|
|
|
|
|
|
sub native_facet_search { |
601
|
0
|
|
|
0
|
1
|
|
my $self = shift; |
602
|
0
|
0
|
|
|
|
|
my $facets = shift or croak "No facets"; |
603
|
0
|
0
|
|
|
|
|
$facets = [$facets] unless ref $facets; |
604
|
|
|
|
|
|
|
|
605
|
0
|
|
|
|
|
|
my $url = $self->products_url; |
606
|
0
|
0
|
|
|
|
|
if (ref ($facets) eq "ARRAY") { |
607
|
0
|
|
|
|
|
|
$url = join "/", $url, @$facets; |
608
|
0
|
|
|
|
|
|
undef $facets; |
609
|
|
|
|
|
|
|
} |
610
|
0
|
|
|
|
|
|
return $self->get_response($url, $facets); |
611
|
|
|
|
|
|
|
} |
612
|
|
|
|
|
|
|
|
613
|
|
|
|
|
|
|
# Item API |
614
|
|
|
|
|
|
|
|
615
|
|
|
|
|
|
|
=head2 native_item ($isbn) |
616
|
|
|
|
|
|
|
|
617
|
|
|
|
|
|
|
=head2 native_item_summary ($isbn) |
618
|
|
|
|
|
|
|
|
619
|
|
|
|
|
|
|
=head3 Returns subset of item fields, with addition of summary field |
620
|
|
|
|
|
|
|
|
621
|
|
|
|
|
|
|
=cut |
622
|
|
|
|
|
|
|
|
623
|
|
|
|
|
|
|
sub native_item { |
624
|
0
|
|
|
0
|
1
|
|
my $self = shift; |
625
|
0
|
0
|
|
|
|
|
my $isbn = shift or croak "No isbn"; |
626
|
|
|
|
|
|
|
|
627
|
0
|
|
|
|
|
|
my $url = $self->title_url($isbn); |
628
|
0
|
|
|
|
|
|
return $self->get_response($url); |
629
|
|
|
|
|
|
|
} |
630
|
|
|
|
|
|
|
|
631
|
|
|
|
|
|
|
sub native_item_summary { |
632
|
0
|
|
|
0
|
1
|
|
my $self = shift; |
633
|
0
|
0
|
|
|
|
|
my $isbn = shift or croak "No isbn"; |
634
|
|
|
|
|
|
|
|
635
|
0
|
|
|
|
|
|
my $url = $self->title_url("$isbn/summary"); |
636
|
0
|
|
|
|
|
|
return $self->get_response($url); |
637
|
|
|
|
|
|
|
} |
638
|
|
|
|
|
|
|
|
639
|
|
|
|
|
|
|
=head2 native_holds () |
640
|
|
|
|
|
|
|
|
641
|
|
|
|
|
|
|
See L |
642
|
|
|
|
|
|
|
|
643
|
|
|
|
|
|
|
=cut |
644
|
|
|
|
|
|
|
|
645
|
|
|
|
|
|
|
sub native_holds { |
646
|
0
|
|
|
0
|
1
|
|
my $self = shift; |
647
|
|
|
|
|
|
|
|
648
|
0
|
|
|
|
|
|
my $url = $self->circulation_action_url("/holds/all", @_); |
649
|
0
|
|
|
|
|
|
return $self->get_response($url); |
650
|
|
|
|
|
|
|
} |
651
|
|
|
|
|
|
|
|
652
|
|
|
|
|
|
|
=head2 native_checkouts () |
653
|
|
|
|
|
|
|
|
654
|
|
|
|
|
|
|
=cut |
655
|
|
|
|
|
|
|
|
656
|
|
|
|
|
|
|
sub native_checkouts { |
657
|
0
|
|
|
0
|
1
|
|
my $self = shift; |
658
|
|
|
|
|
|
|
|
659
|
0
|
|
|
|
|
|
my $url = $self->circulation_action_url("/checkouts/all", @_); |
660
|
0
|
|
|
|
|
|
return $self->get_response($url); |
661
|
|
|
|
|
|
|
} |
662
|
|
|
|
|
|
|
|
663
|
|
|
|
|
|
|
# Utility methods |
664
|
|
|
|
|
|
|
|
665
|
0
|
|
|
0
|
|
|
sub _basic_callback { return $_[0]; } |
666
|
|
|
|
|
|
|
|
667
|
|
|
|
|
|
|
sub get_response { |
668
|
0
|
|
|
0
|
0
|
|
my $self = shift; |
669
|
0
|
0
|
|
|
|
|
my $url = shift or croak "No url"; |
670
|
0
|
|
|
|
|
|
my $get_params = shift; # hash ref |
671
|
|
|
|
|
|
|
|
672
|
0
|
|
|
|
|
|
return $self->with_get_request(\&_basic_callback, $url, $get_params); |
673
|
|
|
|
|
|
|
} |
674
|
|
|
|
|
|
|
|
675
|
|
|
|
|
|
|
sub _error_from_json { |
676
|
0
|
|
|
0
|
|
|
my $self = shift; |
677
|
0
|
0
|
|
|
|
|
my $data = shift or croak "No json data"; |
678
|
0
|
|
|
|
|
|
return join " ", grep defined, $data->{errorCode}, $data->{message}; |
679
|
|
|
|
|
|
|
} |
680
|
|
|
|
|
|
|
|
681
|
|
|
|
|
|
|
1; |
682
|
|
|
|
|
|
|
|
683
|
|
|
|
|
|
|
__END__ |