]> git.evergreen-ils.org Git - working/Evergreen.git/blob - Open-ILS/src/sql/Pg/002.schema.config.sql
cc06eee875810191726906c1f0b3bd80c54b5245
[working/Evergreen.git] / Open-ILS / src / sql / Pg / 002.schema.config.sql
1 /*
2  * Copyright (C) 2004-2008  Georgia Public Library Service
3  * Copyright (C) 2008-2011  Equinox Software, Inc.
4  * Mike Rylander <miker@esilibrary.com>
5  * Copyright (C) 2010 Merrimack Valley Library Consortium
6  * Jason Stephenson <jstephenson@mvlc.org>
7  * Copyright (C) 2010 Laurentian University
8  * Dan Scott <dscott@laurentian.ca>
9  *
10  * This program is free software; you can redistribute it and/or
11  * modify it under the terms of the GNU General Public License
12  * as published by the Free Software Foundation; either version 2
13  * of the License, or (at your option) any later version.
14  *
15  * This program is distributed in the hope that it will be useful,
16  * but WITHOUT ANY WARRANTY; without even the implied warranty of
17  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18  * GNU General Public License for more details.
19  *
20  */
21
22
23
24 DROP SCHEMA IF EXISTS stats CASCADE;
25 DROP SCHEMA IF EXISTS config CASCADE;
26
27 BEGIN;
28 CREATE SCHEMA stats;
29
30 CREATE SCHEMA config;
31 COMMENT ON SCHEMA config IS $$
32 The config schema holds static configuration data for the
33 Evergreen installation.
34 $$;
35
36 CREATE TABLE config.internal_flag (
37     name    TEXT    PRIMARY KEY,
38     value   TEXT,
39     enabled BOOL    NOT NULL DEFAULT FALSE
40 );
41 INSERT INTO config.internal_flag (name) VALUES ('ingest.metarecord_mapping.skip_on_insert');
42 INSERT INTO config.internal_flag (name) VALUES ('ingest.metarecord_mapping.skip_on_update');
43 INSERT INTO config.internal_flag (name) VALUES ('ingest.reingest.force_on_same_marc');
44 INSERT INTO config.internal_flag (name) VALUES ('ingest.disable_located_uri');
45 INSERT INTO config.internal_flag (name) VALUES ('ingest.disable_metabib_full_rec');
46 INSERT INTO config.internal_flag (name) VALUES ('ingest.disable_metabib_rec_descriptor');
47 INSERT INTO config.internal_flag (name) VALUES ('ingest.disable_metabib_field_entry');
48 INSERT INTO config.internal_flag (name) VALUES ('ingest.assume_inserts_only');
49
50 CREATE TABLE config.global_flag (
51     label   TEXT    NOT NULL
52 ) INHERITS (config.internal_flag);
53 ALTER TABLE config.global_flag ADD PRIMARY KEY (name);
54
55 CREATE TABLE config.upgrade_log (
56     version         TEXT    PRIMARY KEY,
57     install_date    TIMESTAMP WITH TIME ZONE NOT NULL DEFAULT NOW(),
58     applied_to      TEXT
59 );
60
61 CREATE TABLE config.db_patch_dependencies (
62   db_patch      TEXT PRIMARY KEY,
63   supersedes    TEXT[],
64   deprecates    TEXT[]
65 );
66
67 CREATE OR REPLACE FUNCTION evergreen.array_overlap_check (/* field */) RETURNS TRIGGER AS $$
68 DECLARE
69     fld     TEXT;
70     cnt     INT;
71 BEGIN
72     fld := TG_ARGV[1];
73     EXECUTE 'SELECT COUNT(*) FROM '|| TG_TABLE_SCHEMA ||'.'|| TG_TABLE_NAME ||' WHERE '|| fld ||' && ($1).'|| fld INTO cnt USING NEW;
74     IF cnt > 0 THEN
75         RAISE EXCEPTION 'Cannot insert duplicate array into field % of table %', fld, TG_TABLE_SCHEMA ||'.'|| TG_TABLE_NAME;
76     END IF;
77     RETURN NEW;
78 END;
79 $$ LANGUAGE PLPGSQL;
80
81 CREATE TRIGGER no_overlapping_sups
82     BEFORE INSERT OR UPDATE ON config.db_patch_dependencies
83     FOR EACH ROW EXECUTE PROCEDURE evergreen.array_overlap_check ('supersedes');
84
85 CREATE TRIGGER no_overlapping_deps
86     BEFORE INSERT OR UPDATE ON config.db_patch_dependencies
87     FOR EACH ROW EXECUTE PROCEDURE evergreen.array_overlap_check ('deprecates');
88
89 INSERT INTO config.upgrade_log (version, applied_to) VALUES ('0693', :eg_version); -- denials/miker
90
91 CREATE TABLE config.bib_source (
92         id              SERIAL  PRIMARY KEY,
93         quality         INT     CHECK ( quality BETWEEN 0 AND 100 ),
94         source          TEXT    NOT NULL UNIQUE,
95         transcendant    BOOL    NOT NULL DEFAULT FALSE,
96         can_have_copies BOOL    NOT NULL DEFAULT TRUE
97 );
98 COMMENT ON TABLE config.bib_source IS $$
99 This is table is used to set up the relative "quality" of each
100 MARC source, such as OCLC.  Also identifies "transcendant" sources,
101 i.e., sources of bib records that should display in the OPAC
102 even if no copies or located URIs are attached. Also indicates if
103 the source is allowed to have actual copies on its bibs. Volumes
104 for targeted URIs are unaffected by this setting.
105 $$;
106
107 CREATE TABLE config.standing (
108         id              SERIAL  PRIMARY KEY,
109         value           TEXT    NOT NULL UNIQUE
110 );
111 COMMENT ON TABLE config.standing IS $$
112 Patron Standings
113
114 This table contains the values that can be applied to a patron
115 by a staff member.  These values should not be changed, other
116 than for translation, as the ID column is currently a "magic
117 number" in the source. :(
118 $$;
119
120 CREATE TABLE config.standing_penalty (
121         id                      SERIAL  PRIMARY KEY,
122         name            TEXT    NOT NULL UNIQUE,
123         label           TEXT    NOT NULL,
124         block_list      TEXT,
125         staff_alert     BOOL    NOT NULL DEFAULT FALSE,
126         org_depth       INTEGER
127 );
128
129 CREATE TABLE config.xml_transform (
130         name            TEXT    PRIMARY KEY,
131         namespace_uri   TEXT    NOT NULL,
132         prefix          TEXT    NOT NULL,
133         xslt            TEXT    NOT NULL
134 );
135
136 CREATE TABLE config.biblio_fingerprint (
137         id                      SERIAL  PRIMARY KEY,
138         name            TEXT    NOT NULL, 
139         xpath           TEXT    NOT NULL,
140     first_word  BOOL    NOT NULL DEFAULT FALSE,
141         format          TEXT    NOT NULL DEFAULT 'marcxml'
142 );
143
144 INSERT INTO config.biblio_fingerprint (name, xpath, format)
145     VALUES (
146         'Title',
147         '//marc:datafield[@tag="700"]/marc:subfield[@code="t"]|' ||
148             '//marc:datafield[@tag="240"]/marc:subfield[@code="a"]|' ||
149             '//marc:datafield[@tag="242"]/marc:subfield[@code="a"]|' ||
150             '//marc:datafield[@tag="246"]/marc:subfield[@code="a"]|' ||
151             '//marc:datafield[@tag="245"]/marc:subfield[@code="a"]',
152         'marcxml'
153     );
154
155 INSERT INTO config.biblio_fingerprint (name, xpath, format, first_word)
156     VALUES (
157         'Author',
158         '//marc:datafield[@tag="700" and ./*[@code="t"]]/marc:subfield[@code="a"]|'
159             '//marc:datafield[@tag="100"]/marc:subfield[@code="a"]|'
160             '//marc:datafield[@tag="110"]/marc:subfield[@code="a"]|'
161             '//marc:datafield[@tag="111"]/marc:subfield[@code="a"]|'
162             '//marc:datafield[@tag="260"]/marc:subfield[@code="b"]',
163         'marcxml',
164         TRUE
165     );
166
167 CREATE TABLE config.metabib_class (
168     name     TEXT    PRIMARY KEY,
169     label    TEXT    NOT NULL UNIQUE,
170     buoyant  BOOL    DEFAULT FALSE NOT NULL,
171     restrict BOOL    DEFAULT FALSE NOT NULL
172 );
173
174 CREATE TABLE config.metabib_field (
175         id              SERIAL  PRIMARY KEY,
176         field_class     TEXT    NOT NULL REFERENCES config.metabib_class (name),
177         name            TEXT    NOT NULL,
178         label           TEXT    NOT NULL,
179         xpath           TEXT    NOT NULL,
180         weight          INT     NOT NULL DEFAULT 1,
181         format          TEXT    NOT NULL REFERENCES config.xml_transform (name) DEFAULT 'mods33',
182         search_field    BOOL    NOT NULL DEFAULT TRUE,
183         facet_field     BOOL    NOT NULL DEFAULT FALSE,
184         browse_field    BOOL    NOT NULL DEFAULT TRUE,
185         browse_xpath   TEXT,
186         facet_xpath     TEXT,
187         restrict        BOOL    DEFAULT FALSE NOT NULL
188 );
189 COMMENT ON TABLE config.metabib_field IS $$
190 XPath used for record indexing ingest
191
192 This table contains the XPath used to chop up MODS into its
193 indexable parts.  Each XPath entry is named and assigned to
194 a "class" of either title, subject, author, keyword, series
195 or identifier.
196 $$;
197
198 CREATE UNIQUE INDEX config_metabib_field_class_name_idx ON config.metabib_field (field_class, name);
199
200 CREATE TABLE config.metabib_search_alias (
201     alias       TEXT    PRIMARY KEY,
202     field_class TEXT    NOT NULL REFERENCES config.metabib_class (name),
203     field       INT     REFERENCES config.metabib_field (id)
204 );
205
206 CREATE TABLE config.non_cataloged_type (
207         id              SERIAL          PRIMARY KEY,
208         owning_lib      INT             NOT NULL, -- REFERENCES actor.org_unit (id),
209         name            TEXT            NOT NULL,
210         circ_duration   INTERVAL        NOT NULL DEFAULT '14 days'::INTERVAL,
211         in_house        BOOL            NOT NULL DEFAULT FALSE,
212         CONSTRAINT noncat_once_per_lib UNIQUE (owning_lib,name)
213 );
214 COMMENT ON TABLE config.non_cataloged_type IS $$
215 Types of valid non-cataloged items.
216 $$;
217
218 CREATE TABLE config.identification_type (
219         id              SERIAL  PRIMARY KEY,
220         name            TEXT    NOT NULL UNIQUE
221 );
222 COMMENT ON TABLE config.identification_type IS $$
223 Types of valid patron identification.
224
225 Each patron must display at least one valid form of identification
226 in order to get a library card.  This table lists those forms.
227 $$;
228
229 CREATE TABLE config.rule_circ_duration (
230         id              SERIAL          PRIMARY KEY,
231         name            TEXT            NOT NULL UNIQUE CHECK ( name ~ E'^\\w+$' ),
232         extended        INTERVAL        NOT NULL,
233         normal          INTERVAL        NOT NULL,
234         shrt            INTERVAL        NOT NULL,
235         max_renewals    INT             NOT NULL
236 );
237 COMMENT ON TABLE config.rule_circ_duration IS $$
238 Circulation Duration rules
239
240 Each circulation is given a duration based on one of these rules.
241 $$;
242
243 CREATE TABLE config.hard_due_date (
244     id                  SERIAL      PRIMARY KEY,
245     name                TEXT        NOT NULL UNIQUE,
246     ceiling_date        TIMESTAMPTZ NOT NULL,
247     forceto             BOOL        NOT NULL,
248     owner               INT         NOT NULL   -- REFERENCES actor.org_unit (id) DEFERRABLE INITIALLY DEFERRED,
249 );
250
251 CREATE TABLE config.hard_due_date_values (
252     id                  SERIAL      PRIMARY KEY,
253     hard_due_date       INT         NOT NULL REFERENCES config.hard_due_date (id)
254                                     DEFERRABLE INITIALLY DEFERRED,
255     ceiling_date        TIMESTAMPTZ NOT NULL,
256     active_date         TIMESTAMPTZ NOT NULL
257 );
258
259 CREATE OR REPLACE FUNCTION config.update_hard_due_dates () RETURNS INT AS $func$
260 DECLARE
261     temp_value  config.hard_due_date_values%ROWTYPE;
262     updated     INT := 0;
263 BEGIN
264     FOR temp_value IN
265       SELECT  DISTINCT ON (hard_due_date) *
266         FROM  config.hard_due_date_values
267         WHERE active_date <= NOW() -- We've passed (or are at) the rollover time
268         ORDER BY hard_due_date, active_date DESC -- Latest (nearest to us) active time
269    LOOP
270         UPDATE  config.hard_due_date
271           SET   ceiling_date = temp_value.ceiling_date
272           WHERE id = temp_value.hard_due_date
273                 AND ceiling_date <> temp_value.ceiling_date; -- Time is equal if we've already updated the chdd
274
275         IF FOUND THEN
276             updated := updated + 1;
277         END IF;
278     END LOOP;
279
280     RETURN updated;
281 END;
282 $func$ LANGUAGE plpgsql;
283
284 CREATE TABLE config.rule_max_fine (
285     id          SERIAL          PRIMARY KEY,
286     name        TEXT            NOT NULL UNIQUE CHECK ( name ~ E'^\\w+$' ),
287     amount      NUMERIC(6,2)    NOT NULL,
288     is_percent  BOOL            NOT NULL DEFAULT FALSE
289 );
290 COMMENT ON TABLE config.rule_max_fine IS $$
291 Circulation Max Fine rules
292
293 Each circulation is given a maximum fine based on one of
294 these rules.
295 $$;
296
297 CREATE TABLE config.rule_recurring_fine (
298         id                      SERIAL          PRIMARY KEY,
299         name                    TEXT            NOT NULL UNIQUE CHECK ( name ~ E'^\\w+$' ),
300         high                    NUMERIC(6,2)    NOT NULL,
301         normal                  NUMERIC(6,2)    NOT NULL,
302         low                     NUMERIC(6,2)    NOT NULL,
303         recurrence_interval     INTERVAL        NOT NULL DEFAULT '1 day'::INTERVAL,
304     grace_period       INTERVAL         NOT NULL DEFAULT '1 day'::INTERVAL
305 );
306 COMMENT ON TABLE config.rule_recurring_fine IS $$
307 Circulation Recurring Fine rules
308
309 Each circulation is given a recurring fine amount based on one of
310 these rules.  Note that it is recommended to run the fine generator
311 (from cron) at least as frequently as the lowest recurrence interval
312 used by your circulation rules so that accrued fines will be up
313 to date.
314 $$;
315
316
317 CREATE TABLE config.rule_age_hold_protect (
318         id      SERIAL          PRIMARY KEY,
319         name    TEXT            NOT NULL UNIQUE CHECK ( name ~ E'^\\w+$' ),
320         age     INTERVAL        NOT NULL,
321         prox    INT             NOT NULL
322 );
323 COMMENT ON TABLE config.rule_age_hold_protect IS $$
324 Hold Item Age Protection rules
325
326 A hold request can only capture new(ish) items when they are
327 within a particular proximity of the pickup_lib of the request.
328 The proximity ('prox' column) is calculated by counting
329 the number of tree edges between the pickup_lib and either the
330 owning_lib or circ_lib of the copy that could fulfill the hold,
331 as determined by the distance_is_from_owner value of the hold matrix
332 rule controlling the hold request.
333 $$;
334
335 CREATE TABLE config.copy_status (
336         id              SERIAL  PRIMARY KEY,
337         name            TEXT    NOT NULL UNIQUE,
338         holdable        BOOL    NOT NULL DEFAULT FALSE,
339         opac_visible    BOOL    NOT NULL DEFAULT FALSE,
340     copy_active  BOOL    NOT NULL DEFAULT FALSE,
341         restrict_copy_delete BOOL         NOT NULL DEFAULT FALSE
342 );
343 COMMENT ON TABLE config.copy_status IS $$
344 Copy Statuses
345
346 The available copy statuses, and whether a copy in that
347 status is available for hold request capture.  0 (zero) is
348 the only special number in this set, meaning that the item
349 is available for immediate checkout, and is counted as available
350 in the OPAC.
351
352 Statuses with an ID below 100 are not removable, and have special
353 meaning in the code.  Do not change them except to translate the
354 textual name.
355
356 You may add and remove statuses above 100, and these can be used
357 to remove items from normal circulation without affecting the rest
358 of the copy's values or its location.
359 $$;
360
361 CREATE TABLE config.net_access_level (
362         id      SERIAL          PRIMARY KEY,
363         name    TEXT            NOT NULL UNIQUE
364 );
365 COMMENT ON TABLE config.net_access_level IS $$
366 Patron Network Access level
367
368 This will be used to inform the in-library firewall of how much
369 internet access the using patron should be allowed.
370 $$;
371
372
373 CREATE TABLE config.remote_account (
374     id          SERIAL  PRIMARY KEY,
375     label       TEXT    NOT NULL,
376     host        TEXT    NOT NULL,   -- name or IP, :port optional
377     username    TEXT,               -- optional, since we could default to $USER
378     password    TEXT,               -- optional, since we could use SSH keys, or anonymous login.
379     account     TEXT,               -- aka profile or FTP "account" command
380     path        TEXT,               -- aka directory
381     owner       INT     NOT NULL,   -- REFERENCES actor.org_unit (id) DEFERRABLE INITIALLY DEFERRED,
382     last_activity TIMESTAMP WITH TIME ZONE
383 );
384
385 CREATE TABLE config.marc21_rec_type_map (
386     code        TEXT    PRIMARY KEY,
387     type_val    TEXT    NOT NULL,
388     blvl_val    TEXT    NOT NULL
389 );
390
391 CREATE TABLE config.marc21_ff_pos_map (
392     id          SERIAL  PRIMARY KEY,
393     fixed_field TEXT    NOT NULL,
394     tag         TEXT    NOT NULL,
395     rec_type    TEXT    NOT NULL,
396     start_pos   INT     NOT NULL,
397     length      INT     NOT NULL,
398     default_val TEXT    NOT NULL DEFAULT ' '
399 );
400
401 CREATE TABLE config.marc21_physical_characteristic_type_map (
402     ptype_key   TEXT    PRIMARY KEY,
403     label       TEXT    NOT NULL -- I18N
404 );
405
406 CREATE TABLE config.marc21_physical_characteristic_subfield_map (
407     id          SERIAL  PRIMARY KEY,
408     ptype_key   TEXT    NOT NULL REFERENCES config.marc21_physical_characteristic_type_map (ptype_key) ON DELETE CASCADE ON UPDATE CASCADE,
409     subfield    TEXT    NOT NULL,
410     start_pos   INT     NOT NULL,
411     length      INT     NOT NULL,
412     label       TEXT    NOT NULL -- I18N
413 );
414
415 CREATE TABLE config.marc21_physical_characteristic_value_map (
416     id              SERIAL  PRIMARY KEY,
417     value           TEXT    NOT NULL,
418     ptype_subfield  INT     NOT NULL REFERENCES config.marc21_physical_characteristic_subfield_map (id),
419     label           TEXT    NOT NULL -- I18N
420 );
421
422
423 CREATE TABLE config.z3950_source (
424     name                TEXT    PRIMARY KEY,
425     label               TEXT    NOT NULL UNIQUE,
426     host                TEXT    NOT NULL,
427     port                INT     NOT NULL,
428     db                  TEXT    NOT NULL,
429     record_format       TEXT    NOT NULL DEFAULT 'FI',
430     transmission_format TEXT    NOT NULL DEFAULT 'usmarc',
431     auth                BOOL    NOT NULL DEFAULT TRUE,
432     use_perm            INT     -- REFERENCES permission.perm_list (id)
433 );
434
435 COMMENT ON TABLE config.z3950_source IS $$
436 Z39.50 Sources
437
438 Each row in this table represents a database searchable via Z39.50.
439 $$;
440
441 COMMENT ON COLUMN config.z3950_source.record_format IS $$
442 Z39.50 element set.
443 $$;
444
445 COMMENT ON COLUMN config.z3950_source.transmission_format IS $$
446 Z39.50 preferred record syntax..
447 $$;
448
449 COMMENT ON COLUMN config.z3950_source.use_perm IS $$
450 If set, this permission is required for the source to be listed in the staff
451 client Z39.50 interface.  Similar to permission.grp_tree.application_perm.
452 $$;
453
454 CREATE TABLE config.z3950_attr (
455     id          SERIAL  PRIMARY KEY,
456     source      TEXT    NOT NULL REFERENCES config.z3950_source (name) DEFERRABLE INITIALLY DEFERRED,
457     name        TEXT    NOT NULL,
458     label       TEXT    NOT NULL,
459     code        INT     NOT NULL,
460     format      INT     NOT NULL,
461     truncation  INT     NOT NULL DEFAULT 0,
462     CONSTRAINT z_code_format_once_per_source UNIQUE (code,format,source)
463 );
464
465 CREATE TABLE config.i18n_locale (
466     code        TEXT    PRIMARY KEY,
467     marc_code   TEXT    NOT NULL, -- should exist in config.coded_value_map WHERE ctype = 'item_lang'
468     name        TEXT    UNIQUE NOT NULL,
469     description TEXT
470 );
471
472 CREATE TABLE config.i18n_core (
473     id              BIGSERIAL   PRIMARY KEY,
474     fq_field        TEXT        NOT NULL,
475     identity_value  TEXT        NOT NULL,
476     translation     TEXT        NOT NULL    REFERENCES config.i18n_locale (code) ON UPDATE CASCADE ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED,
477     string          TEXT        NOT NULL
478 );
479
480 CREATE UNIQUE INDEX i18n_identity ON config.i18n_core (fq_field,identity_value,translation);
481
482 CREATE OR REPLACE FUNCTION oils_i18n_update_apply(old_ident TEXT, new_ident TEXT, hint TEXT) RETURNS VOID AS $_$
483 BEGIN
484
485     EXECUTE $$
486         UPDATE  config.i18n_core
487           SET   identity_value = $$ || quote_literal(new_ident) || $$ 
488           WHERE fq_field LIKE '$$ || hint || $$.%' 
489                 AND identity_value = $$ || quote_literal(old_ident) || $$::TEXT;$$;
490
491     RETURN;
492
493 END;
494 $_$ LANGUAGE PLPGSQL;
495
496 CREATE OR REPLACE FUNCTION oils_i18n_id_tracking(/* hint */) RETURNS TRIGGER AS $_$
497 BEGIN
498     PERFORM oils_i18n_update_apply( OLD.id::TEXT, NEW.id::TEXT, TG_ARGV[0]::TEXT );
499     RETURN NEW;
500 END;
501 $_$ LANGUAGE PLPGSQL;
502
503 CREATE OR REPLACE FUNCTION oils_i18n_code_tracking(/* hint */) RETURNS TRIGGER AS $_$
504 BEGIN
505     PERFORM oils_i18n_update_apply( OLD.code::TEXT, NEW.code::TEXT, TG_ARGV[0]::TEXT );
506     RETURN NEW;
507 END;
508 $_$ LANGUAGE PLPGSQL;
509
510 CREATE TABLE config.billing_type (
511     id              SERIAL  PRIMARY KEY,
512     name            TEXT    NOT NULL,
513     owner           INT     NOT NULL, -- REFERENCES actor.org_unit (id)
514     default_price   NUMERIC(6,2),
515     CONSTRAINT billing_type_once_per_lib UNIQUE (name, owner)
516 );
517
518 CREATE TABLE config.settings_group (
519     name    TEXT PRIMARY KEY,
520     label   TEXT UNIQUE NOT NULL -- I18N
521 );
522
523 CREATE TABLE config.org_unit_setting_type (
524     name            TEXT    PRIMARY KEY,
525     label           TEXT    UNIQUE NOT NULL,
526     grp             TEXT    REFERENCES config.settings_group (name),
527     description     TEXT,
528     datatype        TEXT    NOT NULL DEFAULT 'string',
529     fm_class        TEXT,
530     view_perm       INT,
531     update_perm     INT,
532     --
533     -- define valid datatypes
534     --
535     CONSTRAINT coust_valid_datatype CHECK ( datatype IN
536     ( 'bool', 'integer', 'float', 'currency', 'interval',
537       'date', 'string', 'object', 'array', 'link' ) ),
538     --
539     -- fm_class is meaningful only for 'link' datatype
540     --
541     CONSTRAINT coust_no_empty_link CHECK
542     ( ( datatype =  'link' AND fm_class IS NOT NULL ) OR
543       ( datatype <> 'link' AND fm_class IS NULL ) )
544 );
545
546 CREATE TABLE config.usr_setting_type (
547
548     name TEXT PRIMARY KEY,
549     opac_visible BOOL NOT NULL DEFAULT FALSE,
550     label TEXT UNIQUE NOT NULL,
551     description TEXT,
552     grp             TEXT    REFERENCES config.settings_group (name),
553     datatype TEXT NOT NULL DEFAULT 'string',
554     fm_class TEXT,
555
556     --
557     -- define valid datatypes
558     --
559     CONSTRAINT coust_valid_datatype CHECK ( datatype IN
560     ( 'bool', 'integer', 'float', 'currency', 'interval',
561         'date', 'string', 'object', 'array', 'link' ) ),
562
563     --
564     -- fm_class is meaningful only for 'link' datatype
565     --
566     CONSTRAINT coust_no_empty_link CHECK
567     ( ( datatype = 'link' AND fm_class IS NOT NULL ) OR
568         ( datatype <> 'link' AND fm_class IS NULL ) )
569
570 );
571
572 -- Some handy functions, based on existing ones, to provide optional ingest normalization
573
574 CREATE OR REPLACE FUNCTION public.left_trunc( TEXT, INT ) RETURNS TEXT AS $func$
575         SELECT SUBSTRING($1,$2);
576 $func$ LANGUAGE SQL STRICT IMMUTABLE;
577
578 CREATE OR REPLACE FUNCTION public.right_trunc( TEXT, INT ) RETURNS TEXT AS $func$
579         SELECT SUBSTRING($1,1,$2);
580 $func$ LANGUAGE SQL STRICT IMMUTABLE;
581
582 CREATE OR REPLACE FUNCTION public.split_date_range( TEXT ) RETURNS TEXT AS $func$
583         SELECT REGEXP_REPLACE( $1, E'(\\d{4})-(\\d{4})', E'\\1 \\2', 'g' );
584 $func$ LANGUAGE SQL STRICT IMMUTABLE;
585
586 CREATE OR REPLACE FUNCTION public.approximate_date( TEXT, TEXT ) RETURNS TEXT AS $func$
587         SELECT REGEXP_REPLACE( $1, E'\\D', $2, 'g' );
588 $func$ LANGUAGE SQL STRICT IMMUTABLE;
589
590 CREATE OR REPLACE FUNCTION public.approximate_low_date( TEXT ) RETURNS TEXT AS $func$
591         SELECT approximate_date( $1, '0');
592 $func$ LANGUAGE SQL STRICT IMMUTABLE;
593
594 CREATE OR REPLACE FUNCTION public.approximate_high_date( TEXT ) RETURNS TEXT AS $func$
595         SELECT approximate_date( $1, '9');
596 $func$ LANGUAGE SQL STRICT IMMUTABLE;
597
598 CREATE OR REPLACE FUNCTION public.content_or_null( TEXT ) RETURNS TEXT AS $func$
599         SELECT CASE WHEN $1 ~ E'^\\s*$' THEN NULL ELSE $1 END
600 $func$ LANGUAGE SQL STRICT IMMUTABLE;
601
602 CREATE OR REPLACE FUNCTION public.integer_or_null( TEXT ) RETURNS TEXT AS $func$
603         SELECT CASE WHEN $1 ~ E'^\\d+$' THEN $1 ELSE NULL END
604 $func$ LANGUAGE SQL STRICT IMMUTABLE;
605
606 CREATE OR REPLACE FUNCTION public.force_to_isbn13( TEXT ) RETURNS TEXT AS $func$
607     use Business::ISBN;
608     use strict;
609     use warnings;
610
611     # Find the first ISBN, force it to ISBN13 and return it
612
613     my $input = shift;
614
615     foreach my $word (split(/\s/, $input)) {
616         my $isbn = Business::ISBN->new($word);
617
618         # First check the checksum; if it is not valid, fix it and add the original
619         # bad-checksum ISBN to the output
620         if ($isbn && $isbn->is_valid_checksum() == Business::ISBN::BAD_CHECKSUM) {
621             $isbn->fix_checksum();
622         }
623
624         # If we now have a valid ISBN, force it to ISBN13 and return it
625         return $isbn->as_isbn13->isbn if ($isbn && $isbn->is_valid());
626     }
627     return undef;
628 $func$ LANGUAGE PLPERLU;
629
630 COMMENT ON FUNCTION public.force_to_isbn13(TEXT) IS $$
631 Inspired by translate_isbn1013
632
633 The force_to_isbn13 function takes an input ISBN and returns the ISBN13
634 version without hypens and with a repaired checksum if the checksum was bad
635 $$;
636
637
638 CREATE OR REPLACE FUNCTION public.translate_isbn1013( TEXT ) RETURNS TEXT AS $func$
639     use Business::ISBN;
640     use strict;
641     use warnings;
642
643     # For each ISBN found in a single string containing a set of ISBNs:
644     #   * Normalize an incoming ISBN to have the correct checksum and no hyphens
645     #   * Convert an incoming ISBN10 or ISBN13 to its counterpart and return
646
647     my $input = shift;
648     my $output = '';
649
650     foreach my $word (split(/\s/, $input)) {
651         my $isbn = Business::ISBN->new($word);
652
653         # First check the checksum; if it is not valid, fix it and add the original
654         # bad-checksum ISBN to the output
655         if ($isbn && $isbn->is_valid_checksum() == Business::ISBN::BAD_CHECKSUM) {
656             $output .= $isbn->isbn() . " ";
657             $isbn->fix_checksum();
658         }
659
660         # If we now have a valid ISBN, convert it to its counterpart ISBN10/ISBN13
661         # and add the normalized original ISBN to the output
662         if ($isbn && $isbn->is_valid()) {
663             my $isbn_xlated = ($isbn->type eq "ISBN13") ? $isbn->as_isbn10 : $isbn->as_isbn13;
664             $output .= $isbn->isbn . " ";
665
666             # If we successfully converted the ISBN to its counterpart, add the
667             # converted ISBN to the output as well
668             $output .= ($isbn_xlated->isbn . " ") if ($isbn_xlated);
669         }
670     }
671     return $output if $output;
672
673     # If there were no valid ISBNs, just return the raw input
674     return $input;
675 $func$ LANGUAGE PLPERLU;
676
677 COMMENT ON FUNCTION public.translate_isbn1013(TEXT) IS $$
678 The translate_isbn1013 function takes an input ISBN and returns the
679 following in a single space-delimited string if the input ISBN is valid:
680   - The normalized input ISBN (hyphens stripped)
681   - The normalized input ISBN with a fixed checksum if the checksum was bad
682   - The ISBN converted to its ISBN10 or ISBN13 counterpart, if possible
683 $$;
684
685 -- And ... a table in which to register them
686
687 CREATE TABLE config.index_normalizer (
688         id              SERIAL  PRIMARY KEY,
689         name            TEXT    UNIQUE NOT NULL,
690         description     TEXT,
691         func            TEXT    NOT NULL,
692         param_count     INT     NOT NULL DEFAULT 0
693 );
694
695 CREATE TABLE config.metabib_field_index_norm_map (
696         id      SERIAL  PRIMARY KEY,
697         field   INT     NOT NULL REFERENCES config.metabib_field (id) ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED,
698         norm    INT     NOT NULL REFERENCES config.index_normalizer (id) ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED,
699         params  TEXT,
700         pos     INT     NOT NULL DEFAULT 0
701 );
702
703 CREATE TABLE config.record_attr_definition (
704     name        TEXT    PRIMARY KEY,
705     label       TEXT    NOT NULL, -- I18N
706     description TEXT,
707     filter      BOOL    NOT NULL DEFAULT TRUE,  -- becomes QP filter if true
708     sorter      BOOL    NOT NULL DEFAULT FALSE, -- becomes QP sort() axis if true
709
710 -- For pre-extracted fields. Takes the first occurance, uses naive subfield ordering
711     tag         TEXT, -- LIKE format
712     sf_list     TEXT, -- pile-o-values, like 'abcd' for a and b and c and d
713
714 -- This is used for both tag/sf and xpath entries
715     joiner      TEXT,
716
717 -- For xpath-extracted attrs
718     xpath       TEXT,
719     format      TEXT    REFERENCES config.xml_transform (name) ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED,
720     start_pos   INT,
721     string_len  INT,
722
723 -- For fixed fields
724     fixed_field TEXT, -- should exist in config.marc21_ff_pos_map.fixed_field
725
726 -- For phys-char fields
727     phys_char_sf    INT REFERENCES config.marc21_physical_characteristic_subfield_map (id)
728 );
729
730 CREATE TABLE config.record_attr_index_norm_map (
731     id      SERIAL  PRIMARY KEY,
732     attr    TEXT    NOT NULL REFERENCES config.record_attr_definition (name) ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED,
733     norm    INT     NOT NULL REFERENCES config.index_normalizer (id) ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED,
734     params  TEXT,
735     pos     INT     NOT NULL DEFAULT 0
736 );
737
738 CREATE TABLE config.coded_value_map (
739     id          SERIAL  PRIMARY KEY,
740     ctype       TEXT    NOT NULL REFERENCES config.record_attr_definition (name) ON DELETE CASCADE DEFERRABLE INITIALLY DEFERRED,
741     code        TEXT    NOT NULL,
742     value       TEXT    NOT NULL,
743     description TEXT
744 );
745
746 CREATE VIEW config.language_map AS SELECT code, value FROM config.coded_value_map WHERE ctype = 'item_lang';
747 CREATE VIEW config.bib_level_map AS SELECT code, value FROM config.coded_value_map WHERE ctype = 'bib_level';
748 CREATE VIEW config.item_form_map AS SELECT code, value FROM config.coded_value_map WHERE ctype = 'item_form';
749 CREATE VIEW config.item_type_map AS SELECT code, value FROM config.coded_value_map WHERE ctype = 'item_type';
750 CREATE VIEW config.lit_form_map AS SELECT code, value, description FROM config.coded_value_map WHERE ctype = 'lit_form';
751 CREATE VIEW config.audience_map AS SELECT code, value, description FROM config.coded_value_map WHERE ctype = 'audience';
752 CREATE VIEW config.videorecording_format_map AS SELECT code, value FROM config.coded_value_map WHERE ctype = 'vr_format';
753
754 CREATE OR REPLACE FUNCTION oils_tsearch2 () RETURNS TRIGGER AS $$
755 DECLARE
756     normalizer      RECORD;
757     value           TEXT := '';
758 BEGIN
759
760     value := NEW.value;
761
762     IF TG_TABLE_NAME::TEXT ~ 'field_entry$' THEN
763         FOR normalizer IN
764             SELECT  n.func AS func,
765                     n.param_count AS param_count,
766                     m.params AS params
767               FROM  config.index_normalizer n
768                     JOIN config.metabib_field_index_norm_map m ON (m.norm = n.id)
769               WHERE field = NEW.field AND m.pos < 0
770               ORDER BY m.pos LOOP
771                 EXECUTE 'SELECT ' || normalizer.func || '(' ||
772                     quote_literal( value ) ||
773                     CASE
774                         WHEN normalizer.param_count > 0
775                             THEN ',' || REPLACE(REPLACE(BTRIM(normalizer.params,'[]'),E'\'',E'\\\''),E'"',E'\'')
776                             ELSE ''
777                         END ||
778                     ')' INTO value;
779
780         END LOOP;
781
782         NEW.value := value;
783     END IF;
784
785     IF NEW.index_vector = ''::tsvector THEN
786         RETURN NEW;
787     END IF;
788
789     IF TG_TABLE_NAME::TEXT ~ 'field_entry$' THEN
790         FOR normalizer IN
791             SELECT  n.func AS func,
792                     n.param_count AS param_count,
793                     m.params AS params
794               FROM  config.index_normalizer n
795                     JOIN config.metabib_field_index_norm_map m ON (m.norm = n.id)
796               WHERE field = NEW.field AND m.pos >= 0
797               ORDER BY m.pos LOOP
798                 EXECUTE 'SELECT ' || normalizer.func || '(' ||
799                     quote_literal( value ) ||
800                     CASE
801                         WHEN normalizer.param_count > 0
802                             THEN ',' || REPLACE(REPLACE(BTRIM(normalizer.params,'[]'),E'\'',E'\\\''),E'"',E'\'')
803                             ELSE ''
804                         END ||
805                     ')' INTO value;
806
807         END LOOP;
808     END IF;
809
810     IF TG_TABLE_NAME::TEXT ~ 'browse_entry$' THEN
811         value :=  ARRAY_TO_STRING(
812             evergreen.regexp_split_to_array(value, E'\\W+'), ' '
813         );
814         value := public.search_normalize(value);
815     END IF;
816
817     NEW.index_vector = to_tsvector((TG_ARGV[0])::regconfig, value);
818
819     RETURN NEW;
820 END;
821 $$ LANGUAGE PLPGSQL;
822
823 -- List applied db patches that are deprecated by (and block the application of) my_db_patch
824 CREATE OR REPLACE FUNCTION evergreen.upgrade_list_applied_deprecates ( my_db_patch TEXT ) RETURNS SETOF evergreen.patch AS $$
825     SELECT  DISTINCT l.version
826       FROM  config.upgrade_log l
827             JOIN config.db_patch_dependencies d ON (l.version::TEXT[] && d.deprecates)
828       WHERE d.db_patch = $1
829 $$ LANGUAGE SQL;
830
831 -- List applied db patches that are superseded by (and block the application of) my_db_patch
832 CREATE OR REPLACE FUNCTION evergreen.upgrade_list_applied_supersedes ( my_db_patch TEXT ) RETURNS SETOF evergreen.patch AS $$
833     SELECT  DISTINCT l.version
834       FROM  config.upgrade_log l
835             JOIN config.db_patch_dependencies d ON (l.version::TEXT[] && d.supersedes)
836       WHERE d.db_patch = $1
837 $$ LANGUAGE SQL;
838
839 -- List applied db patches that deprecates (and block the application of) my_db_patch
840 CREATE FUNCTION evergreen.upgrade_list_applied_deprecated ( my_db_patch TEXT ) RETURNS SETOF TEXT AS $$
841     SELECT  db_patch
842       FROM  config.db_patch_dependencies
843       WHERE ARRAY[$1]::TEXT[] && deprecates
844 $$ LANGUAGE SQL;
845
846 -- List applied db patches that supersedes (and block the application of) my_db_patch
847 CREATE FUNCTION evergreen.upgrade_list_applied_superseded ( my_db_patch TEXT ) RETURNS SETOF TEXT AS $$
848     SELECT  db_patch
849       FROM  config.db_patch_dependencies
850       WHERE ARRAY[$1]::TEXT[] && supersedes
851 $$ LANGUAGE SQL;
852
853 -- Make sure that no deprecated or superseded db patches are currently applied
854 CREATE OR REPLACE FUNCTION evergreen.upgrade_verify_no_dep_conflicts ( my_db_patch TEXT ) RETURNS BOOL AS $$
855     SELECT  COUNT(*) = 0
856       FROM  (SELECT * FROM evergreen.upgrade_list_applied_deprecates( $1 )
857                 UNION
858              SELECT * FROM evergreen.upgrade_list_applied_supersedes( $1 )
859                 UNION
860              SELECT * FROM evergreen.upgrade_list_applied_deprecated( $1 )
861                 UNION
862              SELECT * FROM evergreen.upgrade_list_applied_superseded( $1 ))x
863 $$ LANGUAGE SQL;
864
865 -- Raise an exception if there are, in fact, dep/sup conflict
866 CREATE OR REPLACE FUNCTION evergreen.upgrade_deps_block_check ( my_db_patch TEXT, my_applied_to TEXT ) RETURNS BOOL AS $$
867 DECLARE 
868     deprecates TEXT;
869     supersedes TEXT;
870 BEGIN
871     IF NOT evergreen.upgrade_verify_no_dep_conflicts( my_db_patch ) THEN
872         SELECT  STRING_AGG(patch, ', ') INTO deprecates FROM evergreen.upgrade_list_applied_deprecates(my_db_patch);
873         SELECT  STRING_AGG(patch, ', ') INTO supersedes FROM evergreen.upgrade_list_applied_supersedes(my_db_patch);
874         RAISE EXCEPTION '
875 Upgrade script % can not be applied:
876   applied deprecated scripts %
877   applied superseded scripts %
878   deprecated by %
879   superseded by %',
880             my_db_patch,
881             ARRAY_AGG(evergreen.upgrade_list_applied_deprecates(my_db_patch)),
882             ARRAY_AGG(evergreen.upgrade_list_applied_supersedes(my_db_patch)),
883             evergreen.upgrade_list_applied_deprecated(my_db_patch),
884             evergreen.upgrade_list_applied_superseded(my_db_patch);
885     END IF;
886
887     INSERT INTO config.upgrade_log (version, applied_to) VALUES (my_db_patch, my_applied_to);
888     RETURN TRUE;
889 END;
890 $$ LANGUAGE PLPGSQL;
891
892 CREATE TABLE config.barcode_completion (
893     id          SERIAL PRIMARY KEY,
894     active      BOOL NOT NULL DEFAULT true,
895     org_unit    INT NOT NULL, -- REFERENCES actor.org_unit(id) DEFERRABLE INITIALLY DEFERRED,
896     prefix      TEXT,
897     suffix      TEXT,
898     length      INT NOT NULL DEFAULT 0,
899     padding     TEXT,
900     padding_end BOOL NOT NULL DEFAULT false,
901     asset       BOOL NOT NULL DEFAULT true,
902     actor       BOOL NOT NULL DEFAULT true
903 );
904
905 CREATE TYPE evergreen.barcode_set AS (type TEXT, id BIGINT, barcode TEXT);
906
907 -- Add support for logging, only keep the most recent five rows for each category. 
908
909
910 CREATE TABLE config.org_unit_setting_type_log (
911     id              BIGSERIAL   PRIMARY KEY,
912     date_applied    TIMESTAMP WITH TIME ZONE NOT NULL DEFAULT NOW(),
913     org             INT,   --REFERENCES actor.org_unit (id),
914     original_value  TEXT,
915     new_value       TEXT,
916     field_name      TEXT      REFERENCES config.org_unit_setting_type (name)
917 );
918
919 COMMENT ON TABLE config.org_unit_setting_type_log IS $$
920 Org Unit setting Logs
921
922 This table contains the most recent changes to each setting 
923 in actor.org_unit_setting, allowing for mistakes to be undone.
924 This is NOT meant to be an auditor, but rather an undo/redo.
925 $$;
926
927 CREATE OR REPLACE FUNCTION limit_oustl() RETURNS TRIGGER AS $oustl_limit$
928     BEGIN
929         -- Only keeps the most recent five settings changes.
930         DELETE FROM config.org_unit_setting_type_log WHERE field_name = NEW.field_name AND date_applied NOT IN 
931         (SELECT date_applied FROM config.org_unit_setting_type_log WHERE field_name = NEW.field_name ORDER BY date_applied DESC LIMIT 4);
932         
933         IF (TG_OP = 'UPDATE') THEN
934             RETURN NEW;
935         ELSIF (TG_OP = 'INSERT') THEN
936             RETURN NEW;
937         END IF;
938         RETURN NULL;
939     END;
940 $oustl_limit$ LANGUAGE plpgsql;
941
942 CREATE TRIGGER limit_logs_oust
943     BEFORE INSERT OR UPDATE ON config.org_unit_setting_type_log
944     FOR EACH ROW EXECUTE PROCEDURE limit_oustl();
945
946 CREATE TABLE config.sms_carrier (
947     id              SERIAL PRIMARY KEY,
948     region          TEXT,
949     name            TEXT,
950     email_gateway   TEXT,
951     active          BOOLEAN DEFAULT TRUE
952 );
953
954 CREATE TYPE config.usr_activity_group AS ENUM ('authen','authz','circ','hold','search');
955
956 CREATE TABLE config.usr_activity_type (
957     id          SERIAL                      PRIMARY KEY, 
958     ewho        TEXT,
959     ewhat       TEXT,
960     ehow        TEXT,
961     label       TEXT                        NOT NULL, -- i18n
962     egroup      config.usr_activity_group   NOT NULL,
963     enabled     BOOL                        NOT NULL DEFAULT TRUE,
964     transient   BOOL                        NOT NULL DEFAULT FALSE,
965     CONSTRAINT  one_of_wwh CHECK (COALESCE(ewho,ewhat,ehow) IS NOT NULL)
966 );
967
968 CREATE UNIQUE INDEX unique_wwh ON config.usr_activity_type 
969     (COALESCE(ewho,''), COALESCE (ewhat,''), COALESCE(ehow,''));
970
971
972 COMMIT;