]> git.evergreen-ils.org Git - working/Evergreen.git/blob - Open-ILS/src/sql/Pg/002.functions.config.sql
serial.record_entry already had an owner column spelled "owning_lib"
[working/Evergreen.git] / Open-ILS / src / sql / Pg / 002.functions.config.sql
1 /*
2  * Copyright (C) 2004-2008  Georgia Public Library Service
3  * Copyright (C) 2008  Equinox Software, Inc.
4  * Mike Rylander <miker@esilibrary.com>
5  *
6  * This program is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU General Public License
8  * as published by the Free Software Foundation; either version 2
9  * of the License, or (at your option) any later version.
10  *
11  * This program is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  * GNU General Public License for more details.
15  *
16  */
17
18
19 BEGIN;
20
21 /*
22 CREATE OR REPLACE FUNCTION oils_xml_transform ( TEXT, TEXT ) RETURNS TEXT AS $_$
23         SELECT  CASE    WHEN (SELECT COUNT(*) FROM config.xml_transform WHERE name = $2 AND xslt = '---') > 0 THEN $1
24                         ELSE xslt_process($1, (SELECT xslt FROM config.xml_transform WHERE name = $2))
25                 END;
26 $_$ LANGUAGE SQL STRICT IMMUTABLE;
27
28 CREATE OR REPLACE FUNCTION public.extract_marc_field ( TEXT, BIGINT, TEXT, TEXT ) RETURNS TEXT AS $$
29     SELECT regexp_replace(array_to_string( array_accum( output ),' ' ),$4,'','g') FROM oils_xpath_table('id', 'marc', $1, $3, 'id='||$2)x(id INT, output TEXT);
30 $$ LANGUAGE SQL;
31
32 CREATE OR REPLACE FUNCTION oils_xml_uncache (xml TEXT) RETURNS BOOL AS $func$
33   delete $_SHARED{'_xslt_process'}{docs}{shift()};
34   return 1;
35 $func$ LANGUAGE PLPERLU;
36
37 CREATE OR REPLACE FUNCTION oils_xml_cache (xml TEXT) RETURNS BOOL AS $func$
38   use strict;
39   use XML::LibXML;
40
41   my $doc = shift;
42
43   # The following approach uses the older XML::LibXML 1.69 / XML::LibXSLT 1.68
44   # methods of parsing XML documents and stylesheets, in the hopes of broader
45   # compatibility with distributions
46   my $parser = $_SHARED{'_xslt_process'}{parsers}{xml} || XML::LibXML->new();
47
48   # Cache the XML parser, if we do not already have one
49   $_SHARED{'_xslt_process'}{parsers}{xml} = $parser
50     unless ($_SHARED{'_xslt_process'}{parsers}{xml});
51
52   # Parse and cache the doc
53   eval { $_SHARED{'_xslt_process'}{docs}{$doc} = $parser->parse_string($doc) };
54
55   return 0 if ($@);
56   return 1;
57 $func$ LANGUAGE PLPERLU;
58
59 -- if we use these, we need to ...
60 drop function oils_xpath(text, text, anyarray);
61
62 CREATE OR REPLACE FUNCTION oils_xpath (xpath TEXT, xml TEXT, ns TEXT[][]) RETURNS TEXT[] AS $func$
63   use strict;
64   use XML::LibXML;
65
66   my $xpath = shift;
67   my $doc = shift;
68   my $ns_string = shift || '';
69   #elog(NOTICE,"ns_string: $ns_string");
70
71   my %ns_list = $ns_string =~ m/\{([^{,]+),([^}]+)\}/g;
72   #elog(NOTICE,"NS Prefix $_: $ns_list{$_}") for (keys %ns_list);
73
74   # The following approach uses the older XML::LibXML 1.69 / XML::LibXSLT 1.68
75   # methods of parsing XML documents and stylesheets, in the hopes of broader
76   # compatibility with distributions
77   my $parser = eval { $_SHARED{'_xslt_process'}{parsers}{xml} || XML::LibXML->new() };
78
79   return undef if ($@);
80
81   # Cache the XML parser, if we do not already have one
82   $_SHARED{'_xslt_process'}{parsers}{xml} = $parser
83     unless ($_SHARED{'_xslt_process'}{parsers}{xml});
84
85   # Look for a cached version of the doc, or parse it if none
86   my $dom = eval { $_SHARED{'_xslt_process'}{docs}{$doc} || $parser->parse_string($doc) };
87
88   return undef if ($@);
89
90   # Cache the parsed XML doc, if already there
91   $_SHARED{'_xslt_process'}{docs}{$doc} = $dom
92     unless ($_SHARED{'_xslt_process'}{docs}{$doc});
93
94   # Register the requested namespaces
95   $dom->documentElement->setNamespace( $ns_list{$_} => $_ ) for ( keys %ns_list );
96
97   # Gather and return nodes
98   my @nodes = $dom->findnodes($xpath);
99   #elog(NOTICE,"nodes found by $xpath: ". scalar(@nodes));
100
101   return [ map { $_->toString } @nodes ];
102 $func$ LANGUAGE PLPERLU;
103
104 CREATE OR REPLACE FUNCTION oils_xpath ( TEXT, TEXT ) RETURNS TEXT[] AS $$SELECT oils_xpath( $1, $2, '{}'::TEXT[] );$$ LANGUAGE SQL IMMUTABLE;
105
106 */
107
108 CREATE FUNCTION version_specific_xpath () RETURNS TEXT AS $wrapper_function$
109 DECLARE
110     out_text TEXT;
111 BEGIN
112     
113     IF REGEXP_REPLACE(VERSION(),E'^.+?(\\d+\\.\\d+).*?$',E'\\1')::FLOAT < 8.3 THEN
114         out_text := 'Creating XPath functions that work like the native XPATH function in 8.3+';
115         
116         EXECUTE $create_82_funcs$
117                         
118 CREATE OR REPLACE FUNCTION oils_xpath ( xpath TEXT, xml TEXT, ns ANYARRAY ) RETURNS TEXT[] AS $func$
119 DECLARE
120     node_text   TEXT;
121     ns_regexp   TEXT;
122     munged_xpath    TEXT;
123 BEGIN
124
125     munged_xpath := xpath;
126
127     IF ns IS NOT NULL AND array_upper(ns, 1) IS NOT NULL THEN
128         FOR namespace IN 1 .. array_upper(ns, 1) LOOP
129             munged_xpath := REGEXP_REPLACE(
130                 munged_xpath,
131                 E'(' || ns[namespace][1] || E'):(\\w+)',
132                 E'*[local-name() = "\\2" and namespace-uri() = "' || ns[namespace][2] || E'"]',
133                 'g'
134             );
135         END LOOP;
136
137         munged_xpath := REGEXP_REPLACE( munged_xpath, E'\\]\\[(\\D)',E' and \\1', 'g');
138     END IF;
139
140     -- RAISE NOTICE 'munged xpath: %', munged_xpath;
141
142     node_text := xpath_nodeset(xml, munged_xpath, 'XXX_OILS_NODESET');
143     -- RAISE NOTICE 'node_text: %', node_text;
144
145     IF munged_xpath ~ $re$/[^/[]*@[^/]+$$re$ THEN
146         node_text := REGEXP_REPLACE(node_text,'<XXX_OILS_NODESET>[^"]+"', '<XXX_OILS_NODESET>', 'g');
147         node_text := REGEXP_REPLACE(node_text,'"</XXX_OILS_NODESET>', '</XXX_OILS_NODESET>', 'g');
148     END IF;
149
150     node_text := REGEXP_REPLACE(node_text,'^<XXX_OILS_NODESET>', '');
151     node_text := REGEXP_REPLACE(node_text,'</XXX_OILS_NODESET>$', '');
152
153     RETURN  STRING_TO_ARRAY(node_text, '</XXX_OILS_NODESET><XXX_OILS_NODESET>');
154 END;
155 $func$ LANGUAGE PLPGSQL IMMUTABLE;
156
157 CREATE OR REPLACE FUNCTION oils_xpath ( TEXT, TEXT ) RETURNS TEXT[] AS $$SELECT oils_xpath( $1, $2, '{}'::TEXT[] );$$ LANGUAGE SQL IMMUTABLE;
158
159 CREATE OR REPLACE FUNCTION oils_xslt_process(TEXT, TEXT) RETURNS TEXT AS $$
160     SELECT xslt_process( $1, $2 );
161 $$ LANGUAGE SQL IMMUTABLE;
162
163         $create_82_funcs$;
164     ELSIF REGEXP_REPLACE(VERSION(),E'^.+?(\\d+\\.\\d+).*?$',E'\\1')::FLOAT = 8.3 THEN
165         out_text := 'Creating XPath wrapper functions around the native XPATH function in 8.3.  contrib/xml2 still required!';
166
167         EXECUTE $create_83_funcs$
168 -- 8.3 or after
169 CREATE OR REPLACE FUNCTION oils_xpath ( TEXT, TEXT, ANYARRAY ) RETURNS TEXT[] AS 'SELECT XPATH( $1, $2::XML, $3 )::TEXT[];' LANGUAGE SQL IMMUTABLE;
170 CREATE OR REPLACE FUNCTION oils_xpath ( TEXT, TEXT ) RETURNS TEXT[] AS 'SELECT XPATH( $1, $2::XML )::TEXT[];' LANGUAGE SQL IMMUTABLE;
171
172 CREATE OR REPLACE FUNCTION oils_xslt_process(TEXT, TEXT) RETURNS TEXT AS $$
173     SELECT xslt_process( $1, $2 );
174 $$ LANGUAGE SQL IMMUTABLE;
175
176         $create_83_funcs$;
177
178     ELSE
179         out_text := 'Creating XPath wrapper functions around the native XPATH function in 8.4+, and plperlu-based xslt processor.  No contrib/xml2 needed!';
180
181         EXECUTE $create_84_funcs$
182 -- 8.4 or after
183 CREATE OR REPLACE FUNCTION oils_xpath ( TEXT, TEXT, ANYARRAY ) RETURNS TEXT[] AS 'SELECT XPATH( $1, $2::XML, $3 )::TEXT[];' LANGUAGE SQL IMMUTABLE;
184 CREATE OR REPLACE FUNCTION oils_xpath ( TEXT, TEXT ) RETURNS TEXT[] AS 'SELECT XPATH( $1, $2::XML )::TEXT[];' LANGUAGE SQL IMMUTABLE;
185
186 CREATE OR REPLACE FUNCTION oils_xslt_process(TEXT, TEXT) RETURNS TEXT AS $func$
187   use strict;
188
189   use XML::LibXSLT;
190   use XML::LibXML;
191
192   my $doc = shift;
193   my $xslt = shift;
194
195   # The following approach uses the older XML::LibXML 1.69 / XML::LibXSLT 1.68
196   # methods of parsing XML documents and stylesheets, in the hopes of broader
197   # compatibility with distributions
198   my $parser = $_SHARED{'_xslt_process'}{parsers}{xml} || XML::LibXML->new();
199
200   # Cache the XML parser, if we do not already have one
201   $_SHARED{'_xslt_process'}{parsers}{xml} = $parser
202     unless ($_SHARED{'_xslt_process'}{parsers}{xml});
203
204   my $xslt_parser = $_SHARED{'_xslt_process'}{parsers}{xslt} || XML::LibXSLT->new();
205
206   # Cache the XSLT processor, if we do not already have one
207   $_SHARED{'_xslt_process'}{parsers}{xslt} = $xslt_parser
208     unless ($_SHARED{'_xslt_process'}{parsers}{xslt});
209
210   my $stylesheet = $_SHARED{'_xslt_process'}{stylesheets}{$xslt} ||
211     $xslt_parser->parse_stylesheet( $parser->parse_string($xslt) );
212
213   $_SHARED{'_xslt_process'}{stylesheets}{$xslt} = $stylesheet
214     unless ($_SHARED{'_xslt_process'}{stylesheets}{$xslt});
215
216   return $stylesheet->output_string(
217     $stylesheet->transform(
218       $parser->parse_string($doc)
219     )
220   );
221
222 $func$ LANGUAGE 'plperlu' STRICT IMMUTABLE;
223
224         $create_84_funcs$;
225
226     END IF;
227
228     RETURN out_text;
229 END;
230 $wrapper_function$ LANGUAGE PLPGSQL;
231
232 SELECT version_specific_xpath();
233 DROP FUNCTION version_specific_xpath();
234
235
236 CREATE OR REPLACE FUNCTION oils_xpath_string ( TEXT, TEXT, TEXT, ANYARRAY ) RETURNS TEXT AS $func$
237     SELECT  ARRAY_TO_STRING(
238                 oils_xpath(
239                     $1 ||
240                         CASE WHEN $1 ~ $re$/[^/[]*@[^]]+$$re$ OR $1 ~ $re$text\(\)$$re$ THEN '' ELSE '//text()' END,
241                     $2,
242                     $4
243                 ),
244                 $3
245             );
246 $func$ LANGUAGE SQL IMMUTABLE;
247
248 CREATE OR REPLACE FUNCTION oils_xpath_string ( TEXT, TEXT, TEXT ) RETURNS TEXT AS $func$
249     SELECT oils_xpath_string( $1, $2, $3, '{}'::TEXT[] );
250 $func$ LANGUAGE SQL IMMUTABLE;
251
252 CREATE OR REPLACE FUNCTION oils_xpath_string ( TEXT, TEXT, ANYARRAY ) RETURNS TEXT AS $func$
253     SELECT oils_xpath_string( $1, $2, '', $3 );
254 $func$ LANGUAGE SQL IMMUTABLE;
255
256 CREATE OR REPLACE FUNCTION oils_xpath_string ( TEXT, TEXT ) RETURNS TEXT AS $func$
257     SELECT oils_xpath_string( $1, $2, '{}'::TEXT[] );
258 $func$ LANGUAGE SQL IMMUTABLE;
259
260
261 CREATE OR REPLACE FUNCTION oils_xpath_table ( key TEXT, document_field TEXT, relation_name TEXT, xpaths TEXT, criteria TEXT ) RETURNS SETOF RECORD AS $func$
262 DECLARE
263     xpath_list  TEXT[];
264     select_list TEXT[];
265     where_list  TEXT[];
266     q           TEXT;
267     out_record  RECORD;
268     empty_test  RECORD;
269 BEGIN
270     xpath_list := STRING_TO_ARRAY( xpaths, '|' );
271
272     select_list := ARRAY_APPEND( select_list, key || '::INT AS key' );
273
274     FOR i IN 1 .. ARRAY_UPPER(xpath_list,1) LOOP
275         select_list := ARRAY_APPEND(
276             select_list,
277             $sel$
278             EXPLODE_ARRAY(
279                 COALESCE(
280                     NULLIF(
281                         oils_xpath(
282                             $sel$ ||
283                                 quote_literal(
284                                     CASE
285                                         WHEN xpath_list[i] ~ $re$/[^/[]*@[^/]+$$re$ OR xpath_list[i] ~ $re$text\(\)$$re$ THEN xpath_list[i]
286                                         ELSE xpath_list[i] || '//text()'
287                                     END
288                                 ) ||
289                             $sel$,
290                             $sel$ || document_field || $sel$
291                         ),
292                        '{}'::TEXT[]
293                     ),
294                     '{NULL}'::TEXT[]
295                 )
296             ) AS c_$sel$ || i
297         );
298         where_list := ARRAY_APPEND(
299             where_list,
300             'c_' || i || ' IS NOT NULL'
301         );
302     END LOOP;
303
304     q := $q$
305 SELECT * FROM (
306     SELECT $q$ || ARRAY_TO_STRING( select_list, ', ' ) || $q$ FROM $q$ || relation_name || $q$ WHERE ($q$ || criteria || $q$)
307 )x WHERE $q$ || ARRAY_TO_STRING( where_list, ' AND ' );
308     -- RAISE NOTICE 'query: %', q;
309
310     FOR out_record IN EXECUTE q LOOP
311         RETURN NEXT out_record;
312     END LOOP;
313
314     RETURN;
315 END;
316 $func$ LANGUAGE PLPGSQL IMMUTABLE;
317
318
319 CREATE OR REPLACE FUNCTION extract_marc_field ( TEXT, BIGINT, TEXT, TEXT ) RETURNS TEXT AS $$
320 DECLARE
321     query TEXT;
322     output TEXT;
323 BEGIN
324     query := $q$
325         SELECT  regexp_replace(
326                     oils_xpath_string(
327                         $q$ || quote_literal($3) || $q$,
328                         marc,
329                         ' '
330                     ),
331                     $q$ || quote_literal($4) || $q$,
332                     '',
333                     'g')
334           FROM  $q$ || $1 || $q$
335           WHERE id = $q$ || $2;
336
337     EXECUTE query INTO output;
338
339     -- RAISE NOTICE 'query: %, output; %', query, output;
340
341     RETURN output;
342 END;
343 $$ LANGUAGE PLPGSQL IMMUTABLE;
344
345 CREATE OR REPLACE FUNCTION extract_marc_field ( TEXT, BIGINT, TEXT ) RETURNS TEXT AS $$
346     SELECT extract_marc_field($1,$2,$3,'');
347 $$ LANGUAGE SQL IMMUTABLE;
348
349
350
351 CREATE OR REPLACE FUNCTION oils_i18n_xlate ( keytable TEXT, keyclass TEXT, keycol TEXT, identcol TEXT, keyvalue TEXT, raw_locale TEXT ) RETURNS TEXT AS $func$
352 DECLARE
353     locale      TEXT := REGEXP_REPLACE( REGEXP_REPLACE( raw_locale, E'[;, ].+$', '' ), E'_', '-', 'g' );
354     language    TEXT := REGEXP_REPLACE( locale, E'-.+$', '' );
355     result      config.i18n_core%ROWTYPE;
356     fallback    TEXT;
357     keyfield    TEXT := keyclass || '.' || keycol;
358 BEGIN
359
360     -- Try the full locale
361     SELECT  * INTO result
362       FROM  config.i18n_core
363       WHERE fq_field = keyfield
364             AND identity_value = keyvalue
365             AND translation = locale;
366
367     -- Try just the language
368     IF NOT FOUND THEN
369         SELECT  * INTO result
370           FROM  config.i18n_core
371           WHERE fq_field = keyfield
372                 AND identity_value = keyvalue
373                 AND translation = language;
374     END IF;
375
376     -- Fall back to the string we passed in in the first place
377     IF NOT FOUND THEN
378         EXECUTE
379             'SELECT ' ||
380                 keycol ||
381             ' FROM ' || keytable ||
382             ' WHERE ' || identcol || ' = ' || quote_literal(keyvalue)
383                 INTO fallback;
384         RETURN fallback;
385     END IF;
386
387     RETURN result.string;
388 END;
389 $func$ LANGUAGE PLPGSQL STABLE;
390
391 -- Functions for marking translatable strings in SQL statements
392 -- Parameters are: primary key, string, class hint, property
393 CREATE OR REPLACE FUNCTION oils_i18n_gettext( INT, TEXT, TEXT, TEXT ) RETURNS TEXT AS $$
394     SELECT $2;
395 $$ LANGUAGE SQL;
396
397 CREATE OR REPLACE FUNCTION oils_i18n_gettext( TEXT, TEXT, TEXT, TEXT ) RETURNS TEXT AS $$
398     SELECT $2;
399 $$ LANGUAGE SQL;
400
401 CREATE OR REPLACE FUNCTION is_json( TEXT ) RETURNS BOOL AS $f$
402     use JSON::XS;
403     my $json = shift();
404     eval { JSON::XS->new->allow_nonref->decode( $json ) };
405     return $@ ? 0 : 1;
406 $f$ LANGUAGE PLPERLU;
407
408 -- turn a JSON scalar into an SQL TEXT value
409 CREATE OR REPLACE FUNCTION oils_json_to_text( TEXT ) RETURNS TEXT AS $f$
410     use JSON::XS;
411     my $json = shift();
412     my $txt;
413     eval { $txt = JSON::XS->new->allow_nonref->decode( $json ) };
414     return undef if ($@);
415     return $txt
416 $f$ LANGUAGE PLPERLU;
417
418 CREATE OR REPLACE FUNCTION maintain_901 () RETURNS TRIGGER AS $func$
419 BEGIN
420     -- Remove any existing 901 fields before we insert the authoritative one
421     NEW.marc := REGEXP_REPLACE(NEW.marc, E'<datafield\s*[^<>]*?\s*tag="901".+?</datafield>', '', 'g');
422     IF TG_TABLE_SCHEMA = 'biblio' THEN
423         NEW.marc := REGEXP_REPLACE(
424             NEW.marc,
425             E'(</(?:[^:]*?:)?record>)',
426             E'<datafield tag="901" ind1=" " ind2=" ">' ||
427                 '<subfield code="a">' || NEW.tcn_value || E'</subfield>' ||
428                 '<subfield code="b">' || NEW.tcn_source || E'</subfield>' ||
429                 '<subfield code="c">' || NEW.id || E'</subfield>' ||
430                 '<subfield code="t">' || TG_TABLE_SCHEMA || E'</subfield>' ||
431                 CASE WHEN NEW.owner IS NOT NULL THEN '<subfield code="o">' || NEW.owner || E'</subfield>' ELSE '' END ||
432                 CASE WHEN NEW.share_depth IS NOT NULL THEN '<subfield code="d">' || NEW.share_depth || E'</subfield>' ELSE '' END ||
433              E'</datafield>\\1'
434         );
435     ELSIF TG_TABLE_SCHEMA = 'authority' THEN
436         NEW.marc := REGEXP_REPLACE(
437             NEW.marc,
438             E'(</(?:[^:]*?:)?record>)',
439             E'<datafield tag="901" ind1=" " ind2=" ">' ||
440                 '<subfield code="a">' || NEW.arn_value || E'</subfield>' ||
441                 '<subfield code="b">' || NEW.arn_source || E'</subfield>' ||
442                 '<subfield code="c">' || NEW.id || E'</subfield>' ||
443                 '<subfield code="t">' || TG_TABLE_SCHEMA || E'</subfield>' ||
444              E'</datafield>\\1'
445         );
446     ELSIF TG_TABLE_SCHEMA = 'serial' THEN
447         NEW.marc := REGEXP_REPLACE(
448             NEW.marc,
449             E'(</(?:[^:]*?:)?record>)',
450             E'<datafield tag="901" ind1=" " ind2=" ">' ||
451                 '<subfield code="c">' || NEW.id || E'</subfield>' ||
452                 '<subfield code="t">' || TG_TABLE_SCHEMA || E'</subfield>' ||
453                 '<subfield code="o">' || NEW.owning_lib || E'</subfield>' ||
454                 CASE WHEN NEW.record IS NOT NULL THEN '<subfield code="r">' || NEW.record || E'</subfield>' ELSE '' END ||
455              E'</datafield>\\1'
456         );
457     ELSE
458         NEW.marc := REGEXP_REPLACE(
459             NEW.marc,
460             E'(</(?:[^:]*?:)?record>)',
461             E'<datafield tag="901" ind1=" " ind2=" ">' ||
462                 '<subfield code="c">' || NEW.id || E'</subfield>' ||
463                 '<subfield code="t">' || TG_TABLE_SCHEMA || E'</subfield>' ||
464              E'</datafield>\\1'
465         );
466     END IF;
467
468     RETURN NEW;
469 END;
470 $func$ LANGUAGE PLPGSQL;
471
472 CREATE OR REPLACE FUNCTION maintain_control_numbers() RETURNS TRIGGER AS $func$
473 use strict;
474 use MARC::Record;
475 use MARC::File::XML;
476 use Encode;
477 use Unicode::Normalize;
478
479 my $record = MARC::Record->new_from_xml($_TD->{new}{marc});
480 my $schema = $_TD->{table_schema};
481 my $rec_id = $_TD->{new}{id};
482
483 # Short-circuit if maintaining control numbers per MARC21 spec is not enabled
484 my $enable = spi_exec_query("SELECT enabled FROM config.global_flag WHERE name = 'cat.maintain_control_numbers'");
485 if (!($enable->{processed}) or $enable->{rows}[0]->{enabled} eq 'f') {
486     return;
487 }
488
489 # Get the control number identifier from an OU setting based on $_TD->{new}{owner}
490 my $ou_cni = 'EVRGRN';
491
492 my $owner;
493 if ($schema eq 'serial') {
494     $owner = $_TD->{new}{owning_lib};
495 } else {
496     # are.owner and bre.owner can be null, so fall back to the consortial setting
497     $owner = $_TD->{new}{owner} || 1;
498 }
499
500 my $ous_rv = spi_exec_query("SELECT value FROM actor.org_unit_ancestor_setting('cat.marc_control_number_identifier', $owner)");
501 if ($ous_rv->{processed}) {
502     $ou_cni = $ous_rv->{rows}[0]->{value};
503     $ou_cni =~ s/"//g; # Stupid VIM syntax highlighting"
504 } else {
505     # Fall back to the shortname of the OU if there was no OU setting
506     $ous_rv = spi_exec_query("SELECT shortname FROM actor.org_unit WHERE id = $owner");
507     if ($ous_rv->{processed}) {
508         $ou_cni = $ous_rv->{rows}[0]->{shortname};
509     }
510 }
511
512 my ($create, $munge) = (0, 0);
513 my ($orig_001, $orig_003) = ('', '');
514
515 # Incoming MARC records may have multiple 001s or 003s, despite the spec
516 my @control_ids = $record->field('003');
517 my @scns = $record->field('035');
518
519 foreach my $id_field ('001', '003') {
520     my $spec_value;
521     my @controls = $record->field($id_field);
522
523     if ($id_field eq '001') {
524         $spec_value = $rec_id;
525     } else {
526         $spec_value = $ou_cni;
527     }
528
529     # Create the 001/003 if none exist
530     if (scalar(@controls) == 0) {
531         $record->insert_fields_ordered(MARC::Field->new($id_field, $spec_value));
532         $create = 1;
533     } elsif (scalar(@controls) > 1) {
534         # Do we already have the right 001/003 value in the existing set?
535         unless (grep $_->data() eq $spec_value, @controls) {
536             $munge = 1;
537         }
538
539         # Delete the other fields, as with more than 1 001/003 we do not know which 003/001 to match
540         foreach my $control (@controls) {
541             unless ($control->data() eq $spec_value) {
542                 $record->delete_field($control);
543             }
544         }
545     } else {
546         # Only one field; check to see if we need to munge it
547         unless (grep $_->data() eq $spec_value, @controls) {
548             $munge = 1;
549         }
550     }
551 }
552
553 # Now, if we need to munge the 001, we will first push the existing 001/003 into the 035
554 if ($munge) {
555     my $scn = "(" . $record->field('003')->data() . ")" . $record->field('001')->data();
556
557     # Do not create duplicate 035 fields
558     unless (grep $_->subfield('a') eq $scn, @scns) {
559         $record->insert_fields_ordered(MARC::Field->new('035', '', '', 'a' => $scn));
560     }
561 }
562
563 # Set the 001/003 and update the MARC
564 if ($create or $munge) {
565     $record->field('001')->data($rec_id);
566     $record->field('003')->data($ou_cni);
567
568     my $xml = $record->as_xml_record();
569     $xml =~ s/\n//sgo;
570     $xml =~ s/^<\?xml.+\?\s*>//go;
571     $xml =~ s/>\s+</></go;
572     $xml =~ s/\p{Cc}//go;
573
574     # Embed a version of OpenILS::Application::AppUtils->entityize()
575     # to avoid having to set PERL5LIB for PostgreSQL as well
576
577     # If we are going to convert non-ASCII characters to XML entities,
578     # we had better be dealing with a UTF8 string to begin with
579     $xml = decode_utf8($xml);
580
581     $xml = NFC($xml);
582
583     # Convert raw ampersands to entities
584     $xml =~ s/&(?!\S+;)/&amp;/gso;
585
586     # Convert Unicode characters to entities
587     $xml =~ s/([\x{0080}-\x{fffd}])/sprintf('&#x%X;',ord($1))/sgoe;
588
589     $xml =~ s/[\x00-\x1f]//go;
590     $_TD->{new}{marc} = $xml;
591
592     return "MODIFY";
593 }
594
595 return;
596 $func$ LANGUAGE PLPERLU;
597
598 COMMIT;
599