-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathworldcat_code.pl
executable file
·412 lines (336 loc) · 13.4 KB
/
worldcat_code.pl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
#!/usr/bin/perl
# Author: Tom Laudeman
# The Institute for Advanced Technology in the Humanities
# Copyright 2013 University of Virginia. Licensed under the Educational Community License, Version 2.0 (the
# "License"); you may not use this file except in compliance with the License. You may obtain a copy of the
# License at
# http://www.osedu.org/licenses/ECL-2.0
# Unless required by applicable law or agreed to in writing, software distributed under the License is
# distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See
# the License for the specific language governing permissions and limitations under the License.
# See readme.md
# ./exec_record.pl config=agency.cfg &
# cat agency_code.log | perl -ne 'if ($_ =~ m/040\$a: (.*)/) { print "$1\n";} ' | sort -fu > agency_unique.txt
# worldcat_code.pl also creates a directory of cached results ./wc_data/ and the command line argument
# supplied via file=foo.txt and writes worldcat_code.xml. If the file foo.txt exists (often agency_unique.txt), you simply run
# worldcat_code.pl file=agency_unique.txt > tmp.log
# Reminder: scripts must be executable, so if you haven't already done it: chmod +x worldcat_code.pl
# ./worldcat_code.pl file=agency_test.txt > tmp.log
use strict;
use CGI;
use Time::HiRes qw(usleep nanosleep);
my $ac_file = ""; # We require a command line arg now. Old default was "agency_unique.txt";
# my $ac_file = "agency_test.txt";
my $data_dir = "wc_data";
my $output_file = "worldcat_code.xml";
my $ofs; # output file stream
main();
exit();
sub main
{
$| = 1; # unbuffer stdout
my $qq = new CGI();
my %ch = $qq->Vars();
if (! exists($ch{file}))
{
print "Error: missing file command line argument.\n";
print "Usage: $0 file=agency_unique_filename\n";
exit(1);
}
$ac_file = $ch{file};
if (! -e $ac_file)
{
print "Error: cannot find agency unique filename: $ac_file.\n";
print "Usage: $0 file=agency_unique_filename\n";
exit(1);
}
if (! -e $data_dir)
{
print "Created dir $data_dir.\n";
mkdir $data_dir
}
print "Using cache dir $data_dir\n";
open($ofs, ">", $output_file) || die "Cannot open $output_file for write.\n";
print "Ouput file is $output_file\n";
print $ofs '<?xml version="1.0" encoding="UTF-8"?>';
print $ofs "\n<all xmlns=\"http://socialarchive.iath.virginia.edu/worldcat\">\n";
# Read in a line of input
my $xx = 0;
my $good_hit = 0;
while (defined(my $marc_code = get_marc()))
{
# Get the first query back from worldcat. Sleep 250 milliseconds aka 1/4 second aka 250000
# microseconds between queries in order to be polite to the server.
my $first;
my @rid_list;
if ($marc_code)
{
$first = first_query($marc_code, 'local.marcOrgCode');
@rid_list = get_rid($first);
if (@rid_list == 0)
{
$first = first_query($marc_code, 'local.oclcSymbol');
@rid_list = get_rid($first);
}
# if (@rid_list == 0)
# {
# $first = first_query($marc_code, 'srw.serverChoice');
# @rid_list = get_rid($first);
# }
}
else
{
next;
}
# If we have a resourceID then get a second query. Sleep 250 milliseconds aka 1/4 second between
# queries to be polite to the server.
# Code ALM does not return a record with an exact match for ALM. In fact, that exact match is the
# third record, so we have to check all the returned records.
# Cache second queries. These seem more likely to repeat than first queries since each first is based
# on a unique marc code, and first returns multiple potiential hits.
my $second = "";
$good_hit = 0;
if (@rid_list > 0)
{
if (@rid_list > 1)
{
print "multi mc: $marc_code\n";
}
foreach my $rid (@rid_list)
{
# Older versions wrongly just converted rid incompatible characters to _ which could lead to
# filename conflicts. Better to encode the rid in a form guaranteed to be ascii
# alphanum. We only use it one way: we get an rid and encode then use as a basename either to
# write or read.
# print "trying: ($marc_code) $rid\n";
my $safe_data = sprintf("$data_dir/%s.xml", fencode("$rid"));
if (-e $safe_data)
{
print "Using cache for rid: $rid safe_data: $safe_data\n";
$second = read_file($safe_data);
# Reading from disk, no need to sleep here.
}
else
{
print "Sending requst for rid: $rid\n";
$second = second_query($rid);
# Write the data to the cache
open(my $dfs, ">", $safe_data) || die "Cannot open $safe_data for write (rid: $rid)\n";
print $dfs $second;
close($dfs);
}
if (parse_second($marc_code, $second))
{
# Note that we don't stop at the first good hit. Some codes
# such as "C" produce multiple answers. The code that
# consumes this data must choose the best, which is probably
# the shortest.
$good_hit = 1;
}
}
}
else
{
# print "no rid, skipped: $marc_code\n";
}
if (! $good_hit)
{
# If we didn't get a hit, for any reason, output an empty record.
print $ofs " <container>\n";
print $ofs " <orig_query>$marc_code</orig_query>\n";
print $ofs " <marc_code/>\n";
print $ofs " <name/>\n";
print $ofs " <isil/>\n";
print $ofs " <matching_element/>\n";
print $ofs " </container>\n";
}
$xx++;
# if ($xx > 10)
# {
# exit();
# }
}
print $ofs "</all>\n";
close($ofs);
}
# Just use a global for our single file handle.
my $fh;
sub get_marc
{
if (!$fh)
{
open($fh, "<", $ac_file) || die "Cannot open $ac_file for read.\n";
}
if (my $in = <$fh>)
{
chomp($in);
return $in;
}
else
{
return undef;
}
}
sub first_query
{
my $orig_code = $_[0];
my $qfield = $_[1];
# worldcat.org might return a zillion records, but I think the default is
# 10. In practice it seems to return no more than 10, and if there are more
# than 10 records, we probably don't want to check each one for our hit.
# my $safe_code = $marc_code;
# $safe_code =~ s/[^A-Za-z0-9]/_/g; # All non-alphanum to underscore to be safe in a file name.
# my $data = "$data_dir/mc_$safe_code\_$qfield.html";
# Simply converting non-alphanum could result in duplicate conflicts. Need to encode the complete original
# query in a safe manner to preserver all its unique goodness.
my $data = sprintf("$data_dir/mc_%s\_$qfield.html", fencode($orig_code));
my $first = '';
if (-e $data)
{
# print "First uses $data\n";
$first = read_file($data);
}
else
{
# local.marcOrgCode or local.oclcSymbol. We are not using
# srw.serverChoice, although it produces a result.
# query=local.marcOrgCode="Y%24E"+not+local.logicalDelete%3D%221%22
# Perl string interpolation understands $uri_query. The char "&" is not
# special for interpolation. Use single quote ' around the URL arg to
# curl because double quote " interpolates in the shell, and the shell
# gets upset about things like ( which are fine (apparently) in a URL.
my $uri_query = "$qfield=\"" . CGI::Util::escape($orig_code) . '"+not+local.logicalDelete="1"';
use URI::Escape;
my $test_uri_query = "$qfield=\"" . URI::Escape::uri_escape($orig_code) . '"+not+local.logicalDelete="1"';
# print " uri: $uri_query\n";
# print "test: $test_uri_query\n";
my $cmd = "curl -s \'http://worldcat.org/webservices/registry/search/Institutions?query=$uri_query&operation=searchRetrieve&recordSchema=info:rfa/rfaRegistry/schemaInfos/adminData&recordPacking=xml\'";
$first = `$cmd`;
usleep(250000);
open(my $dfs, ">", $data) || die "Cannot open $data (orig_code: $orig_code) for write\n";
print $dfs $first;
close($dfs);
}
# print "first: $first\n";
return $first;
}
sub get_rid
{
my $first = $_[0];
# Initially I thought that the exact match was the first record, but I found cases where that is not
# true. Use a /g regex so we check all the possible matches. I think later code checks for exact matches
# as the "best" choice.
# <adminData:resourceID>info:rfa/localhost/Institutions/50042</adminData:resourceID>
my @inst_list;
while($first =~ m/Institutions\/(\d+)<\/adminData/g)
{
push(@inst_list, $1);
}
return @inst_list;
}
sub second_query
{
my $rid = $_[0];
my $second = "";
if ($rid)
{
$second = `curl -s "http://worldcat.org/webservices/registry/enhancedContent/Institutions/$rid"`;
usleep(250000);
}
return $second;
}
sub parse_second
{
my $orig_query = $_[0];
my $second = $_[1];
# <institutionName>Huntington Library, Art Collections & Botanical Gardens</institutionName>
# <identifiers xmlns="info:rfa/rfaRegistry/xmlSchemas/institutions/identifiers"
# xsi:schemaLocation="info:rfa/rfaRegistry/xmlSchemas/institutions/identifiers http://worldcat.org/registry/xsd/collections/Institutions/identifiers.xsd">
# <lastUpdated>2012-02-16</lastUpdated>
# <ISIL>Oclc-HUV</ISIL>
# <OCLCInstitutionNumber>8385</OCLCInstitutionNumber>
# <oclcSymbol>HUV</oclcSymbol>
# <oclcAccountName>HUNTINGTON LIBR ART GALLERY & GARDENS</oclcAccountName>
# <oclcAccountName>HUNTINGTON LIBR ART & BOTANICAL GARDEN</oclcAccountName>
# <marcOrgCode>CSmH</marcOrgCode>
# </identifiers>
my %res;
my $exact = 0;
# <oclcSymbol status="inactive">AEM</oclcSymbol>
# Look for an exact match as any element value, case insensitive. In addition to the example above, a
# match could (theoretically) be <cobissLettersCode>BMSNS</cobissLettersCode> although that seems to
# stretch credibility. One record I checked seemed rational in that 040$a BMSNS was with a record that
# could be Serbian, and 040$b is srp.
if ($second =~ m/(<marcOrgCode>\Q$orig_query\E<\/marcOrgCode>)/i ||
$second =~ m/(<oclcSymbol.*?>\Q$orig_query\E<\/oclcSymbol>)/i)
{
$exact = 1;
my $matching_element = $1;
$res{iname} = '';
if ($second =~ m/<institutionName>(.+?)<\/institutionName>/i)
{
# <institutionName> with no attributes only occurs once per file.
$res{iname} = $1;
}
$res{isil} = '';
if ($second =~ m/<ISIL>(.*?\Q$orig_query\E.*?)<\/ISIL>/i ||
$second =~ m/<ISIL>(.+?)<\/ISIL>/i)
{
# May have more than one isil, so get the one that matches the marc
# query. Else try to find any isil since they often do not contain
# a string that matches the original query.
# <ISIL>OCLC-AEM</ISIL>
# <ISIL>OCLC-AFM</ISIL>
$res{isil} = $1;
}
$res{marc} = "";
if ($second =~ m/<marcOrgCode>(.+?)<\/marcOrgCode>/i)
{
$res{marc} = $1;
}
# The initial value for this is different because it is a copy-of rather than value-of. We are saving
# this for possible later processing. Some bad matches don't include a string match for the original
# query, and also have status="inactive". Search RHI.
# If there are multiple <oclcSymbol> we will on get the first.
$res{oclc_symbol} = '<oclcSymbol/>';
if ($second =~ m/(<oclcSymbol.*?>.*?<\/oclcSymbol>)/i)
{
$res{oclc_symbol} = $1;
}
print $ofs " <container>\n";
print $ofs " <orig_query>$orig_query</orig_query>\n";
print $ofs " <marc_code>$res{marc}</marc_code>\n";
print $ofs " <name>$res{iname}</name>\n";
print $ofs " <isil>$res{isil}</isil>\n";
print $ofs " $res{oclc_symbol}\n";
print $ofs " <matching_element>$matching_element</matching_element>\n";
print $ofs " </container>\n";
}
return $exact;
}
sub read_file
{
my @stat_array = stat($_[0]);
if ($#stat_array < 7)
{
die "read_file: File $_[0] not found\n";
}
my $temp;
# It is possible that someone will ask us to open a file with a leading space.
# That requires separate args for the < and for the file name.
# It also works for files with trailing space.
if (! open(IN, "<", $_[0]))
{
die "Could not open $_[0] $!\n";
}
sysread(IN, $temp, 100000); # $stat_array[7]);
close(IN);
return $temp;
}
sub fencode
{
# Turn a string into a new string that is a list of 2 character hex digits. ^I becomes 09 and so on.
my $var = $_[0];
$var =~ s/(.)/sprintf("%2.2x", ord($1))/eg;
return $var;
}