Project

General

Profile

1
--
2
-- PostgreSQL database dump
3
--
4

    
5
SET statement_timeout = 0;
6
SET lock_timeout = 0;
7
SET client_encoding = 'UTF8';
8
SET standard_conforming_strings = on;
9
SET check_function_bodies = false;
10
SET client_min_messages = warning;
11

    
12
--
13
-- Name: util; Type: SCHEMA; Schema: -; Owner: -
14
--
15

    
16
CREATE SCHEMA util;
17

    
18

    
19
--
20
-- Name: SCHEMA util; Type: COMMENT; Schema: -; Owner: -
21
--
22

    
23
COMMENT ON SCHEMA util IS '
24
IMPORTANT: Functions must always return NULL in place of '''' (the empty string). This ensures that empty strings do not find their way into VegBIEN.
25

    
26
NOTE: SQL-language functions should never be declared STRICT, because this prevents them from being inlined. inlining can create a significant speed improvement (7x+), by avoiding function calls and enabling additional constant folding. avoiding use of STRICT also makes functions *much* easier to troubleshoot, because they won''t mysteriously do nothing if called with only NULL arguments, even when you have added debug-print statements.
27
';
28

    
29

    
30
SET search_path = util, pg_catalog;
31

    
32
--
33
-- Name: col_cast; Type: TYPE; Schema: util; Owner: -
34
--
35

    
36
CREATE TYPE col_cast AS (
37
	col_name text,
38
	type regtype
39
);
40

    
41

    
42
--
43
-- Name: col_ref; Type: TYPE; Schema: util; Owner: -
44
--
45

    
46
CREATE TYPE col_ref AS (
47
	table_ regclass,
48
	name text
49
);
50

    
51

    
52
--
53
-- Name: compass_dir; Type: TYPE; Schema: util; Owner: -
54
--
55

    
56
CREATE TYPE compass_dir AS ENUM (
57
    'N',
58
    'E',
59
    'S',
60
    'W'
61
);
62

    
63

    
64
--
65
-- Name: datatype; Type: TYPE; Schema: util; Owner: -
66
--
67

    
68
CREATE TYPE datatype AS ENUM (
69
    'str',
70
    'float'
71
);
72

    
73

    
74
--
75
-- Name: %==(anyelement, anyelement); Type: FUNCTION; Schema: util; Owner: -
76
--
77

    
78
CREATE FUNCTION "%=="(left_ anyelement, right_ anyelement) RETURNS boolean
79
    LANGUAGE sql STABLE
80
    AS $_$
81
SELECT keys($1) = keys($2)
82
$_$;
83

    
84

    
85
--
86
-- Name: FUNCTION "%=="(left_ anyelement, right_ anyelement); Type: COMMENT; Schema: util; Owner: -
87
--
88

    
89
COMMENT ON FUNCTION "%=="(left_ anyelement, right_ anyelement) IS '
90
needs to be declared STABLE instead of IMMUTABLE because it depends on the search_path (as described at http://vegpath.org/links/#PostgreSQL:%20Documentation:%209.3:%20Function%20Volatility%20Categories%20**)
91
';
92

    
93

    
94
--
95
-- Name: _alt(anyelement, anyelement, anyelement, anyelement, anyelement, anyelement, anyelement, anyelement, anyelement, anyelement, anyelement, anyelement, anyelement); Type: FUNCTION; Schema: util; Owner: -
96
--
97

    
98
CREATE FUNCTION _alt("0" anyelement DEFAULT NULL::unknown, "1" anyelement DEFAULT NULL::unknown, "2" anyelement DEFAULT NULL::unknown, "3" anyelement DEFAULT NULL::unknown, "4" anyelement DEFAULT NULL::unknown, "5" anyelement DEFAULT NULL::unknown, "6" anyelement DEFAULT NULL::unknown, "7" anyelement DEFAULT NULL::unknown, "8" anyelement DEFAULT NULL::unknown, "9" anyelement DEFAULT NULL::unknown, "10" anyelement DEFAULT NULL::unknown, "11" anyelement DEFAULT NULL::unknown, "12" anyelement DEFAULT NULL::unknown) RETURNS anyelement
99
    LANGUAGE sql IMMUTABLE
100
    AS $_$
101
SELECT coalesce($1, $2, $3, $4, $5, $6, $7, $8, $9, $10, $11, $12, $13)
102
$_$;
103

    
104

    
105
--
106
-- Name: _and(boolean, boolean, boolean, boolean, boolean); Type: FUNCTION; Schema: util; Owner: -
107
--
108

    
109
CREATE FUNCTION _and("0" boolean DEFAULT NULL::boolean, "1" boolean DEFAULT NULL::boolean, "2" boolean DEFAULT NULL::boolean, "3" boolean DEFAULT NULL::boolean, "4" boolean DEFAULT NULL::boolean) RETURNS boolean
110
    LANGUAGE sql IMMUTABLE
111
    AS $_$
112
SELECT bool_and(value)
113
FROM
114
(VALUES
115
      ($1)
116
    , ($2)
117
    , ($3)
118
    , ($4)
119
    , ($5)
120
)
121
AS v (value)
122
$_$;
123

    
124

    
125
--
126
-- Name: FUNCTION _and("0" boolean, "1" boolean, "2" boolean, "3" boolean, "4" boolean); Type: COMMENT; Schema: util; Owner: -
127
--
128

    
129
COMMENT ON FUNCTION _and("0" boolean, "1" boolean, "2" boolean, "3" boolean, "4" boolean) IS '
130
_and() ignores NULL values, while AND combines them with the other values to potentially convert true to NULL. AND should be used with required fields, and _and() with optional fields.
131
';
132

    
133

    
134
--
135
-- Name: _avg(double precision, double precision, double precision, double precision, double precision); Type: FUNCTION; Schema: util; Owner: -
136
--
137

    
138
CREATE FUNCTION _avg("0" double precision DEFAULT NULL::double precision, "1" double precision DEFAULT NULL::double precision, "2" double precision DEFAULT NULL::double precision, "3" double precision DEFAULT NULL::double precision, "4" double precision DEFAULT NULL::double precision) RETURNS double precision
139
    LANGUAGE sql IMMUTABLE
140
    AS $_$
141
SELECT avg(value)
142
FROM
143
(VALUES
144
      ($1)
145
    , ($2)
146
    , ($3)
147
    , ($4)
148
    , ($5)
149
)
150
AS v (value)
151
$_$;
152

    
153

    
154
--
155
-- Name: _dms_to_dd(text); Type: FUNCTION; Schema: util; Owner: -
156
--
157

    
158
CREATE FUNCTION _dms_to_dd(value text DEFAULT NULL::text) RETURNS double precision
159
    LANGUAGE sql IMMUTABLE
160
    AS $_$
161
SELECT (g[1]||'1')::integer*util._dms_to_dd(deg := g[2]::double precision, min := g[3]::double precision, sec := g[4]::double precision, dir := g[5]::util.compass_dir)
162
FROM 
163
(
164
    SELECT regexp_matches($1, '^ *(-?)(\d{1,3}(?:\.\d*)?)(?:(?:deg|[°º])(?: *([\d.]+)(?:min|[''’]))?(?: *([\d.]+)(?:sec|["”]))?)? *([NESW])? *$')
165
    UNION ALL
166
    SELECT ARRAY[g[1], g[2], g[3]||'.'||g[4], NULL, g[5]]
167
    FROM regexp_matches($1, '^ *(-?)(\d{2,3})(\d{2})(\d{3}) *([NESW])? *$') matches (g) -- [D]DDMMmmm, where MMmmm = MM.mmm
168
)
169
matches (g)
170
$_$;
171

    
172

    
173
--
174
-- Name: _dms_to_dd(double precision, double precision, double precision, compass_dir); Type: FUNCTION; Schema: util; Owner: -
175
--
176

    
177
CREATE FUNCTION _dms_to_dd(deg double precision DEFAULT NULL::double precision, min double precision DEFAULT NULL::double precision, sec double precision DEFAULT NULL::double precision, dir compass_dir DEFAULT NULL::compass_dir) RETURNS double precision
178
    LANGUAGE sql IMMUTABLE
179
    AS $_$
180
SELECT sum(value)*COALESCE(util._map('N=>1,E=>1,S=>-1,W=>-1', $4::text)::integer, 1)
181
FROM
182
(VALUES
183
      ($1)
184
    , ($2/60)
185
    , ($3/60/60)
186
)
187
AS v (value)
188
$_$;
189

    
190

    
191
--
192
-- Name: _dms_to_dd(text, text, text, text); Type: FUNCTION; Schema: util; Owner: -
193
--
194

    
195
CREATE FUNCTION _dms_to_dd(deg text DEFAULT NULL::text, min text DEFAULT NULL::text, sec text DEFAULT NULL::text, dir text DEFAULT NULL::text) RETURNS double precision
196
    LANGUAGE sql IMMUTABLE
197
    AS $_$
198
SELECT util._dms_to_dd($1::double precision, $2::double precision, $3::double precision, $4::util.compass_dir)
199
$_$;
200

    
201

    
202
--
203
-- Name: _eq(anyelement, anyelement); Type: FUNCTION; Schema: util; Owner: -
204
--
205

    
206
CREATE FUNCTION _eq("left" anyelement DEFAULT NULL::unknown, "right" anyelement DEFAULT NULL::unknown) RETURNS boolean
207
    LANGUAGE sql IMMUTABLE
208
    AS $_$
209
SELECT $1 = $2
210
$_$;
211

    
212

    
213
--
214
-- Name: _fix_date(date, date); Type: FUNCTION; Schema: util; Owner: -
215
--
216

    
217
CREATE FUNCTION _fix_date(value date DEFAULT NULL::date, threshold date DEFAULT NULL::date) RETURNS date
218
    LANGUAGE sql IMMUTABLE
219
    AS $_$
220
-- Fix dates after threshold date
221
-- This fixes e.g. 2-digit years before 1970 misinterpreted as 20xx
222
SELECT (CASE WHEN $1 > COALESCE($2, now()) THEN ($1 - '100 years'::interval)::date ELSE $1 END)
223
$_$;
224

    
225

    
226
--
227
-- Name: _if(boolean, anyelement, anyelement); Type: FUNCTION; Schema: util; Owner: -
228
--
229

    
230
CREATE FUNCTION _if(cond boolean DEFAULT NULL::boolean, "then" anyelement DEFAULT NULL::unknown, "else" anyelement DEFAULT NULL::unknown) RETURNS anyelement
231
    LANGUAGE sql IMMUTABLE
232
    AS $_$
233
SELECT (CASE WHEN $1 THEN $2 ELSE $3 END)
234
$_$;
235

    
236

    
237
--
238
-- Name: _if(text, anyelement, anyelement); Type: FUNCTION; Schema: util; Owner: -
239
--
240

    
241
CREATE FUNCTION _if(cond text DEFAULT NULL::text, "then" anyelement DEFAULT NULL::unknown, "else" anyelement DEFAULT NULL::unknown) RETURNS anyelement
242
    LANGUAGE sql IMMUTABLE
243
    AS $_$
244
SELECT util._if($1 != '', $2, $3)
245
$_$;
246

    
247

    
248
--
249
-- Name: _join(text, text, text, text, text, text, text, text, text, text); Type: FUNCTION; Schema: util; Owner: -
250
--
251

    
252
CREATE FUNCTION _join("0" text DEFAULT NULL::text, "1" text DEFAULT NULL::text, "2" text DEFAULT NULL::text, "3" text DEFAULT NULL::text, "4" text DEFAULT NULL::text, "5" text DEFAULT NULL::text, "6" text DEFAULT NULL::text, "7" text DEFAULT NULL::text, "8" text DEFAULT NULL::text, "9" text DEFAULT NULL::text) RETURNS text
253
    LANGUAGE sql IMMUTABLE
254
    AS $_$
255
SELECT NULLIF(concat_ws('; ', $1, $2, $3, $4, $5, $6, $7, $8, $9, $10), '')
256
$_$;
257

    
258

    
259
--
260
-- Name: _join_words(text, text, text, text, text, text, text, text, text, text); Type: FUNCTION; Schema: util; Owner: -
261
--
262

    
263
CREATE FUNCTION _join_words("0" text DEFAULT NULL::text, "1" text DEFAULT NULL::text, "2" text DEFAULT NULL::text, "3" text DEFAULT NULL::text, "4" text DEFAULT NULL::text, "5" text DEFAULT NULL::text, "6" text DEFAULT NULL::text, "7" text DEFAULT NULL::text, "8" text DEFAULT NULL::text, "9" text DEFAULT NULL::text) RETURNS text
264
    LANGUAGE sql IMMUTABLE
265
    AS $_$
266
SELECT NULLIF(concat_ws(' ', $1, $2, $3, $4, $5, $6, $7, $8, $9, $10), '')
267
$_$;
268

    
269

    
270
--
271
-- Name: _km_to_m(double precision); Type: FUNCTION; Schema: util; Owner: -
272
--
273

    
274
CREATE FUNCTION _km_to_m(value double precision) RETURNS double precision
275
    LANGUAGE sql IMMUTABLE
276
    AS $_$
277
SELECT $1*1000.
278
$_$;
279

    
280

    
281
--
282
-- Name: _label(text, text); Type: FUNCTION; Schema: util; Owner: -
283
--
284

    
285
CREATE FUNCTION _label(label text, value text) RETURNS text
286
    LANGUAGE sql IMMUTABLE
287
    AS $_$
288
SELECT coalesce($1 || ': ', '') || $2
289
$_$;
290

    
291

    
292
--
293
-- Name: _lowercase(text); Type: FUNCTION; Schema: util; Owner: -
294
--
295

    
296
CREATE FUNCTION _lowercase(value text) RETURNS text
297
    LANGUAGE sql IMMUTABLE
298
    AS $_$
299
SELECT lower($1)
300
$_$;
301

    
302

    
303
--
304
-- Name: _map(hstore, anyelement); Type: FUNCTION; Schema: util; Owner: -
305
--
306

    
307
CREATE FUNCTION _map(map hstore, value anyelement) RETURNS anyelement
308
    LANGUAGE plpgsql IMMUTABLE STRICT
309
    AS $$
310
DECLARE
311
    result value%TYPE := util._map(map, value::text)::unknown;
312
BEGIN
313
    RETURN result;
314
END;
315
$$;
316

    
317

    
318
--
319
-- Name: _map(hstore, text); Type: FUNCTION; Schema: util; Owner: -
320
--
321

    
322
CREATE FUNCTION _map(map hstore, value text) RETURNS text
323
    LANGUAGE plpgsql IMMUTABLE STRICT
324
    AS $$
325
DECLARE
326
    match text := map -> value;
327
BEGIN
328
    IF match IS NULL AND NOT map ? value THEN -- no match rather than NULL match
329
        match := map -> '*'; -- use default entry
330
        IF match IS NULL AND NOT map ? '*' THEN match := '!'; -- no default
331
        END IF;
332
    END IF;
333
    
334
    -- Interpret result
335
    IF match = '!' THEN RAISE 'Value not in map: %', value USING ERRCODE = 'data_exception';
336
    ELSIF match = '*' THEN RETURN value;
337
    ELSE RETURN match;
338
    END IF;
339
END;
340
$$;
341

    
342

    
343
--
344
-- Name: _max(anyelement, anyelement, anyelement, anyelement, anyelement, anyelement, anyelement, anyelement, anyelement, anyelement); Type: FUNCTION; Schema: util; Owner: -
345
--
346

    
347
CREATE FUNCTION _max("0" anyelement DEFAULT NULL::unknown, "1" anyelement DEFAULT NULL::unknown, "2" anyelement DEFAULT NULL::unknown, "3" anyelement DEFAULT NULL::unknown, "4" anyelement DEFAULT NULL::unknown, "5" anyelement DEFAULT NULL::unknown, "6" anyelement DEFAULT NULL::unknown, "7" anyelement DEFAULT NULL::unknown, "8" anyelement DEFAULT NULL::unknown, "9" anyelement DEFAULT NULL::unknown) RETURNS anyelement
348
    LANGUAGE sql IMMUTABLE
349
    AS $_$
350
SELECT GREATEST($1, $2, $3, $4, $5, $6, $7, $8, $9, $10)
351
$_$;
352

    
353

    
354
--
355
-- Name: _merge(anyelement, anyelement, anyelement, anyelement, anyelement, anyelement, anyelement, anyelement, anyelement, anyelement); Type: FUNCTION; Schema: util; Owner: -
356
--
357

    
358
CREATE FUNCTION _merge("0" anyelement DEFAULT NULL::unknown, "1" anyelement DEFAULT NULL::unknown, "2" anyelement DEFAULT NULL::unknown, "3" anyelement DEFAULT NULL::unknown, "4" anyelement DEFAULT NULL::unknown, "5" anyelement DEFAULT NULL::unknown, "6" anyelement DEFAULT NULL::unknown, "7" anyelement DEFAULT NULL::unknown, "8" anyelement DEFAULT NULL::unknown, "9" anyelement DEFAULT NULL::unknown) RETURNS anyelement
359
    LANGUAGE sql IMMUTABLE
360
    AS $_$
361
SELECT util.join_strs(value, '; ')
362
FROM
363
(
364
    SELECT *
365
    FROM
366
    (
367
        SELECT
368
        DISTINCT ON (value)
369
        *
370
        FROM
371
        (VALUES
372
              (1, $1)
373
            , (2, $2)
374
            , (3, $3)
375
            , (4, $4)
376
            , (5, $5)
377
            , (6, $6)
378
            , (7, $7)
379
            , (8, $8)
380
            , (9, $9)
381
            , (10, $10)
382
        )
383
        AS v (sort_order, value)
384
        WHERE value IS NOT NULL
385
    )
386
    AS v
387
    ORDER BY sort_order
388
)
389
AS v
390
$_$;
391

    
392

    
393
--
394
-- Name: _merge_prefix(text, text); Type: FUNCTION; Schema: util; Owner: -
395
--
396

    
397
CREATE FUNCTION _merge_prefix(prefix text DEFAULT NULL::text, value text DEFAULT NULL::text) RETURNS text
398
    LANGUAGE sql IMMUTABLE
399
    AS $_$
400
SELECT _join_words((CASE WHEN $2 ~ ('^'||$1||E'\\y') THEN NULL ELSE $1 END), $2)
401
$_$;
402

    
403

    
404
--
405
-- Name: _merge_words(anyelement, anyelement, anyelement, anyelement, anyelement, anyelement, anyelement, anyelement, anyelement, anyelement); Type: FUNCTION; Schema: util; Owner: -
406
--
407

    
408
CREATE FUNCTION _merge_words("0" anyelement DEFAULT NULL::unknown, "1" anyelement DEFAULT NULL::unknown, "2" anyelement DEFAULT NULL::unknown, "3" anyelement DEFAULT NULL::unknown, "4" anyelement DEFAULT NULL::unknown, "5" anyelement DEFAULT NULL::unknown, "6" anyelement DEFAULT NULL::unknown, "7" anyelement DEFAULT NULL::unknown, "8" anyelement DEFAULT NULL::unknown, "9" anyelement DEFAULT NULL::unknown) RETURNS anyelement
409
    LANGUAGE sql IMMUTABLE
410
    AS $_$
411
SELECT util.join_strs(value, ' ')
412
FROM
413
(
414
    SELECT *
415
    FROM
416
    (
417
        SELECT
418
        DISTINCT ON (value)
419
        *
420
        FROM
421
        (VALUES
422
              (1, $1)
423
            , (2, $2)
424
            , (3, $3)
425
            , (4, $4)
426
            , (5, $5)
427
            , (6, $6)
428
            , (7, $7)
429
            , (8, $8)
430
            , (9, $9)
431
            , (10, $10)
432
        )
433
        AS v (sort_order, value)
434
        WHERE value IS NOT NULL
435
    )
436
    AS v
437
    ORDER BY sort_order
438
)
439
AS v
440
$_$;
441

    
442

    
443
--
444
-- Name: _min(anyelement, anyelement, anyelement, anyelement, anyelement, anyelement, anyelement, anyelement, anyelement, anyelement); Type: FUNCTION; Schema: util; Owner: -
445
--
446

    
447
CREATE FUNCTION _min("0" anyelement DEFAULT NULL::unknown, "1" anyelement DEFAULT NULL::unknown, "2" anyelement DEFAULT NULL::unknown, "3" anyelement DEFAULT NULL::unknown, "4" anyelement DEFAULT NULL::unknown, "5" anyelement DEFAULT NULL::unknown, "6" anyelement DEFAULT NULL::unknown, "7" anyelement DEFAULT NULL::unknown, "8" anyelement DEFAULT NULL::unknown, "9" anyelement DEFAULT NULL::unknown) RETURNS anyelement
448
    LANGUAGE sql IMMUTABLE
449
    AS $_$
450
SELECT LEAST($1, $2, $3, $4, $5, $6, $7, $8, $9, $10)
451
$_$;
452

    
453

    
454
--
455
-- Name: _not(boolean); Type: FUNCTION; Schema: util; Owner: -
456
--
457

    
458
CREATE FUNCTION _not(value boolean) RETURNS boolean
459
    LANGUAGE sql IMMUTABLE
460
    AS $_$
461
SELECT NOT $1
462
$_$;
463

    
464

    
465
--
466
-- Name: _now(); Type: FUNCTION; Schema: util; Owner: -
467
--
468

    
469
CREATE FUNCTION _now() RETURNS timestamp with time zone
470
    LANGUAGE sql STABLE
471
    AS $$
472
SELECT now()
473
$$;
474

    
475

    
476
--
477
-- Name: _nullIf(anyelement, text, text); Type: FUNCTION; Schema: util; Owner: -
478
--
479

    
480
CREATE FUNCTION "_nullIf"(value anyelement, "null" text, type text) RETURNS anyelement
481
    LANGUAGE sql IMMUTABLE
482
    AS $_$
483
SELECT util."_nullIf"($1, $2, $3::util.datatype)
484
$_$;
485

    
486

    
487
--
488
-- Name: _nullIf(anyelement, text, datatype); Type: FUNCTION; Schema: util; Owner: -
489
--
490

    
491
CREATE FUNCTION "_nullIf"(value anyelement, "null" text, type datatype DEFAULT 'str'::datatype) RETURNS anyelement
492
    LANGUAGE plpgsql IMMUTABLE
493
    AS $$
494
DECLARE
495
    type util.datatype NOT NULL := type; -- add NOT NULL
496
BEGIN
497
    IF type = 'str' THEN RETURN nullif(value::text, "null");
498
    -- Invalid value is ignored, but invalid null value generates error
499
    ELSIF type = 'float' THEN
500
        DECLARE
501
            -- Outside the try block so that invalid null value generates error
502
            "null" double precision := "null"::double precision;
503
        BEGIN
504
            RETURN nullif(value::double precision, "null");
505
        EXCEPTION
506
            WHEN data_exception THEN RETURN value; -- ignore invalid value
507
        END;
508
    END IF;
509
END;
510
$$;
511

    
512

    
513
--
514
-- Name: _or(boolean, boolean, boolean, boolean, boolean); Type: FUNCTION; Schema: util; Owner: -
515
--
516

    
517
CREATE FUNCTION _or("0" boolean DEFAULT NULL::boolean, "1" boolean DEFAULT NULL::boolean, "2" boolean DEFAULT NULL::boolean, "3" boolean DEFAULT NULL::boolean, "4" boolean DEFAULT NULL::boolean) RETURNS boolean
518
    LANGUAGE sql IMMUTABLE
519
    AS $_$
520
SELECT bool_or(value)
521
FROM
522
(VALUES
523
      ($1)
524
    , ($2)
525
    , ($3)
526
    , ($4)
527
    , ($5)
528
)
529
AS v (value)
530
$_$;
531

    
532

    
533
--
534
-- Name: FUNCTION _or("0" boolean, "1" boolean, "2" boolean, "3" boolean, "4" boolean); Type: COMMENT; Schema: util; Owner: -
535
--
536

    
537
COMMENT ON FUNCTION _or("0" boolean, "1" boolean, "2" boolean, "3" boolean, "4" boolean) IS '
538
_or() ignores NULL values, while OR combines them with the other values to potentially convert false to NULL. OR should be used with required fields, and _or() with optional fields.
539
';
540

    
541

    
542
--
543
-- Name: _range(double precision, double precision); Type: FUNCTION; Schema: util; Owner: -
544
--
545

    
546
CREATE FUNCTION _range("from" double precision DEFAULT NULL::double precision, "to" double precision DEFAULT NULL::double precision) RETURNS double precision
547
    LANGUAGE sql IMMUTABLE
548
    AS $_$
549
SELECT $2 - $1
550
$_$;
551

    
552

    
553
--
554
-- Name: _split(text, text); Type: FUNCTION; Schema: util; Owner: -
555
--
556

    
557
CREATE FUNCTION _split(value text DEFAULT NULL::text, separator text DEFAULT '; '::text) RETURNS SETOF text
558
    LANGUAGE sql IMMUTABLE
559
    AS $_$
560
SELECT regexp_split_to_table($1, $2)
561
$_$;
562

    
563

    
564
--
565
-- Name: added_cols(regclass, regclass); Type: FUNCTION; Schema: util; Owner: -
566
--
567

    
568
CREATE FUNCTION added_cols(table_ regclass, names regclass) RETURNS SETOF text
569
    LANGUAGE sql STABLE
570
    AS $_$
571
SELECT util.derived_cols($1, $2)
572
UNION
573
SELECT util.eval2set($$
574
SELECT col
575
FROM util.col_names($$||quote_nullable($1)||$$::regclass) f (col)
576
JOIN $$||$2||$$ ON "to" = col
577
WHERE "from" LIKE ':%'
578
$$, NULL::text)
579
$_$;
580

    
581

    
582
--
583
-- Name: FUNCTION added_cols(table_ regclass, names regclass); Type: COMMENT; Schema: util; Owner: -
584
--
585

    
586
COMMENT ON FUNCTION added_cols(table_ regclass, names regclass) IS '
587
gets table_''s added columns (all the columns not in the original data)
588
';
589

    
590

    
591
--
592
-- Name: all_same_final(anyarray); Type: FUNCTION; Schema: util; Owner: -
593
--
594

    
595
CREATE FUNCTION all_same_final(state anyarray) RETURNS boolean
596
    LANGUAGE sql IMMUTABLE
597
    AS $_$
598
SELECT $1 IS NULL/*no rows*/ OR util.not_empty($1)/*not no_match_sentinel*/
599
$_$;
600

    
601

    
602
--
603
-- Name: all_same_transform(anyarray, anyelement); Type: FUNCTION; Schema: util; Owner: -
604
--
605

    
606
CREATE FUNCTION all_same_transform(state anyarray, value anyelement) RETURNS anyarray
607
    LANGUAGE plpgsql IMMUTABLE
608
    AS $$
609
DECLARE
610
	value_cmp         state%TYPE = ARRAY[value];
611
	state             state%TYPE = COALESCE(state, value_cmp);
612
	no_match_sentinel state%TYPE = value_cmp[1:0]/*=ARRAY[]::state%TYPE*/;
613
BEGIN
614
	RETURN (CASE WHEN value_cmp IS NOT DISTINCT FROM state THEN state ELSE no_match_sentinel END);
615
END;
616
$$;
617

    
618

    
619
--
620
-- Name: analyze_(regclass); Type: FUNCTION; Schema: util; Owner: -
621
--
622

    
623
CREATE FUNCTION analyze_(table_ regclass) RETURNS void
624
    LANGUAGE sql
625
    AS $_$
626
SELECT util.eval($$ANALYZE VERBOSE $$||$1)
627
$_$;
628

    
629

    
630
--
631
-- Name: append_comment(regclass, text); Type: FUNCTION; Schema: util; Owner: -
632
--
633

    
634
CREATE FUNCTION append_comment(table_ regclass, comment text) RETURNS void
635
    LANGUAGE sql
636
    AS $_$
637
SELECT util.set_comment($1, concat(util.comment($1), $2))
638
$_$;
639

    
640

    
641
--
642
-- Name: FUNCTION append_comment(table_ regclass, comment text); Type: COMMENT; Schema: util; Owner: -
643
--
644

    
645
COMMENT ON FUNCTION append_comment(table_ regclass, comment text) IS '
646
comment: must start and end with a newline
647
';
648

    
649

    
650
--
651
-- Name: array_fill(anyelement, integer); Type: FUNCTION; Schema: util; Owner: -
652
--
653

    
654
CREATE FUNCTION array_fill(value anyelement, length integer) RETURNS anyarray
655
    LANGUAGE sql IMMUTABLE
656
    AS $_$
657
SELECT pg_catalog.array_fill($1, ARRAY[$2])
658
$_$;
659

    
660

    
661
--
662
-- Name: array_length(anyarray); Type: FUNCTION; Schema: util; Owner: -
663
--
664

    
665
CREATE FUNCTION array_length("array" anyarray) RETURNS integer
666
    LANGUAGE sql IMMUTABLE
667
    AS $_$
668
SELECT util.array_length($1, 1)
669
$_$;
670

    
671

    
672
--
673
-- Name: array_length(anyarray, integer); Type: FUNCTION; Schema: util; Owner: -
674
--
675

    
676
CREATE FUNCTION array_length("array" anyarray, dimension integer) RETURNS integer
677
    LANGUAGE sql IMMUTABLE
678
    AS $_$
679
SELECT CASE WHEN $1 IS NULL THEN NULL ELSE COALESCE(pg_catalog.array_length($1, $2), 0) END
680
$_$;
681

    
682

    
683
--
684
-- Name: FUNCTION array_length("array" anyarray, dimension integer); Type: COMMENT; Schema: util; Owner: -
685
--
686

    
687
COMMENT ON FUNCTION array_length("array" anyarray, dimension integer) IS '
688
returns 0 instead of NULL for empty arrays
689
';
690

    
691

    
692
--
693
-- Name: array_reverse(anyarray); Type: FUNCTION; Schema: util; Owner: -
694
--
695

    
696
CREATE FUNCTION array_reverse("array" anyarray) RETURNS anyarray
697
    LANGUAGE sql IMMUTABLE
698
    AS $_$
699
SELECT array(SELECT * FROM util.in_reverse($1))
700
$_$;
701

    
702

    
703
--
704
-- Name: auto_rm_freq(regclass[], text); Type: FUNCTION; Schema: util; Owner: -
705
--
706

    
707
CREATE FUNCTION auto_rm_freq(tables regclass[], freq_col text DEFAULT 'copies'::text) RETURNS void
708
    LANGUAGE sql
709
    AS $_$
710
SELECT CASE WHEN util.freq_always_1($1, $2)
711
THEN util.rm_freq($1, $2)
712
ELSE util.try_create($$ALTER TABLE $$||util.parent($1[1])||$$ ADD COLUMN $$||quote_ident($2)||$$ bigint NOT NULL$$)
713
END
714
$_$;
715

    
716

    
717
--
718
-- Name: cast(text, anyelement); Type: FUNCTION; Schema: util; Owner: -
719
--
720

    
721
CREATE FUNCTION "cast"(value text, ret_type_null anyelement) RETURNS anyelement
722
    LANGUAGE plpgsql IMMUTABLE
723
    AS $$
724
/* must use LANGUAGE plpgsql because LANGUAGE sql does not assignment-cast the
725
return value, causing a type mismatch */
726
BEGIN
727
	-- will then be assignment-cast to return type via INOUT
728
	RETURN value::cstring;
729
END;
730
$$;
731

    
732

    
733
--
734
-- Name: FUNCTION "cast"(value text, ret_type_null anyelement); Type: COMMENT; Schema: util; Owner: -
735
--
736

    
737
COMMENT ON FUNCTION "cast"(value text, ret_type_null anyelement) IS '
738
allows casting to an arbitrary type without eval()
739

    
740
usage:
741
SELECT util.cast(''value'', NULL::integer);
742

    
743
note that there does *not* need to be a cast from text to the output type,
744
because an INOUT cast is used instead
745
(http://www.postgresql.org/docs/9.3/static/sql-createcast.html#AEN69507)
746

    
747
ret_type_null: NULL::ret_type
748
';
749

    
750

    
751
--
752
-- Name: cluster_index(regclass); Type: FUNCTION; Schema: util; Owner: -
753
--
754

    
755
CREATE FUNCTION cluster_index(table_ regclass) RETURNS regclass
756
    LANGUAGE sql STABLE
757
    AS $_$
758
SELECT indexrelid FROM pg_index WHERE indrelid = $1 AND indisclustered
759
$_$;
760

    
761

    
762
--
763
-- Name: cluster_once(regclass, regclass); Type: FUNCTION; Schema: util; Owner: -
764
--
765

    
766
CREATE FUNCTION cluster_once(table_ regclass, index regclass) RETURNS void
767
    LANGUAGE plpgsql STRICT
768
    AS $_$
769
BEGIN
770
    -- not yet clustered (ARRAY[] compares NULLs literally)
771
    IF ARRAY[util.cluster_index(table_)] != ARRAY[index] THEN
772
        EXECUTE $$CLUSTER $$||table_||$$ USING $$||index;
773
    END IF;
774
END;
775
$_$;
776

    
777

    
778
--
779
-- Name: FUNCTION cluster_once(table_ regclass, index regclass); Type: COMMENT; Schema: util; Owner: -
780
--
781

    
782
COMMENT ON FUNCTION cluster_once(table_ regclass, index regclass) IS '
783
idempotent
784
';
785

    
786

    
787
--
788
-- Name: coalesce(anyarray); Type: FUNCTION; Schema: util; Owner: -
789
--
790

    
791
CREATE FUNCTION "coalesce"(VARIADIC values_ anyarray) RETURNS anyelement
792
    LANGUAGE sql IMMUTABLE
793
    AS $_$
794
SELECT value
795
FROM unnest($1) value
796
WHERE value IS NOT NULL
797
LIMIT 1
798
$_$;
799

    
800

    
801
--
802
-- Name: FUNCTION "coalesce"(VARIADIC values_ anyarray); Type: COMMENT; Schema: util; Owner: -
803
--
804

    
805
COMMENT ON FUNCTION "coalesce"(VARIADIC values_ anyarray) IS '
806
uses:
807
* coalescing array elements or rows together
808
* forcing evaluation of all values of a COALESCE()
809
';
810

    
811

    
812
--
813
-- Name: col__min(col_ref); Type: FUNCTION; Schema: util; Owner: -
814
--
815

    
816
CREATE FUNCTION col__min(col col_ref) RETURNS integer
817
    LANGUAGE sql STABLE
818
    AS $_$
819
SELECT util.eval2val($$
820
SELECT $$||quote_ident($1.name)||$$
821
FROM $$||$1.table_||$$
822
ORDER BY $$||quote_ident($1.name)||$$ ASC
823
LIMIT 1
824
$$, NULL::integer)
825
$_$;
826

    
827

    
828
--
829
-- Name: col_comment(col_ref); Type: FUNCTION; Schema: util; Owner: -
830
--
831

    
832
CREATE FUNCTION col_comment(col col_ref) RETURNS text
833
    LANGUAGE plpgsql STABLE STRICT
834
    AS $$
835
DECLARE
836
	comment text;
837
BEGIN
838
	SELECT description
839
	FROM pg_attribute
840
	LEFT JOIN pg_description ON objoid = attrelid
841
		AND classoid = 'pg_class'::regclass AND objsubid = attnum
842
	WHERE attrelid = col.table_ AND attname = col.name
843
	INTO STRICT comment
844
	;
845
	RETURN comment;
846
EXCEPTION
847
	WHEN no_data_found THEN PERFORM util.raise_undefined_column(col);
848
END;
849
$$;
850

    
851

    
852
--
853
-- Name: col_default_sql(col_ref); Type: FUNCTION; Schema: util; Owner: -
854
--
855

    
856
CREATE FUNCTION col_default_sql(col col_ref) RETURNS text
857
    LANGUAGE plpgsql STABLE STRICT
858
    AS $$
859
DECLARE
860
	default_sql text;
861
BEGIN
862
	SELECT adsrc
863
	FROM pg_attribute
864
	LEFT JOIN pg_attrdef ON adrelid = attrelid AND adnum = attnum
865
	WHERE attrelid = col.table_ AND attname = col.name
866
	INTO STRICT default_sql
867
	;
868
	RETURN default_sql;
869
EXCEPTION
870
	WHEN no_data_found THEN PERFORM util.raise_undefined_column(col);
871
END;
872
$$;
873

    
874

    
875
--
876
-- Name: col_default_value(col_ref, anyelement); Type: FUNCTION; Schema: util; Owner: -
877
--
878

    
879
CREATE FUNCTION col_default_value(col col_ref, ret_type_null anyelement DEFAULT NULL::text) RETURNS anyelement
880
    LANGUAGE sql STABLE
881
    AS $_$
882
SELECT util.eval_expr_passthru(util.col_default_sql($1), $2)
883
$_$;
884

    
885

    
886
--
887
-- Name: FUNCTION col_default_value(col col_ref, ret_type_null anyelement); Type: COMMENT; Schema: util; Owner: -
888
--
889

    
890
COMMENT ON FUNCTION col_default_value(col col_ref, ret_type_null anyelement) IS '
891
ret_type_null: NULL::ret_type
892
';
893

    
894

    
895
--
896
-- Name: col_exists(col_ref); Type: FUNCTION; Schema: util; Owner: -
897
--
898

    
899
CREATE FUNCTION col_exists(col col_ref) RETURNS boolean
900
    LANGUAGE plpgsql STRICT
901
    AS $$
902
BEGIN
903
    PERFORM util.col_type(col);
904
    RETURN true;
905
EXCEPTION
906
    WHEN undefined_column THEN RETURN false;
907
END;
908
$$;
909

    
910

    
911
--
912
-- Name: col_global_names(regtype); Type: FUNCTION; Schema: util; Owner: -
913
--
914

    
915
CREATE FUNCTION col_global_names(type regtype, OUT name text, OUT global_name text) RETURNS SETOF record
916
    LANGUAGE plpgsql STABLE STRICT
917
    AS $$
918
DECLARE
919
    prefix text := util.name(type)||'.';
920
BEGIN
921
    RETURN QUERY
922
        SELECT name_, (CASE WHEN util.contains(search_for:='.', in_str:=name_) THEN '' ELSE prefix END)||name_
923
        FROM util.col_names(type) f (name_);
924
END;
925
$$;
926

    
927

    
928
--
929
-- Name: col_names(regclass); Type: FUNCTION; Schema: util; Owner: -
930
--
931

    
932
CREATE FUNCTION col_names(table_ regclass) RETURNS SETOF text
933
    LANGUAGE sql STABLE
934
    AS $_$
935
SELECT attname::text
936
FROM pg_attribute
937
WHERE attrelid = $1 AND attnum >= 1 AND NOT attisdropped
938
ORDER BY attnum
939
$_$;
940

    
941

    
942
--
943
-- Name: col_names(regtype); Type: FUNCTION; Schema: util; Owner: -
944
--
945

    
946
CREATE FUNCTION col_names(type regtype) RETURNS SETOF text
947
    LANGUAGE plpgsql STABLE STRICT
948
    AS $_$
949
BEGIN
950
    RETURN QUERY EXECUTE $$SELECT skeys(hstore(NULL::$$||type||$$))$$;
951
END;
952
$_$;
953

    
954

    
955
--
956
-- Name: col_type(col_ref); Type: FUNCTION; Schema: util; Owner: -
957
--
958

    
959
CREATE FUNCTION col_type(col col_ref) RETURNS regtype
960
    LANGUAGE plpgsql STABLE STRICT
961
    AS $$
962
DECLARE
963
    type regtype;
964
BEGIN
965
    SELECT atttypid FROM pg_attribute
966
    WHERE attrelid = col.table_ AND attname = col.name
967
    INTO STRICT type
968
    ;
969
    RETURN type;
970
EXCEPTION
971
    WHEN no_data_found THEN
972
        RAISE undefined_column USING MESSAGE =
973
            concat('undefined column: ', col.name);
974
END;
975
$$;
976

    
977

    
978
--
979
-- Name: comment(oid); Type: FUNCTION; Schema: util; Owner: -
980
--
981

    
982
CREATE FUNCTION comment(element oid) RETURNS text
983
    LANGUAGE sql STABLE
984
    AS $_$
985
SELECT description FROM pg_description WHERE objoid = $1
986
$_$;
987

    
988

    
989
--
990
-- Name: concat_esc(text, text); Type: FUNCTION; Schema: util; Owner: -
991
--
992

    
993
CREATE FUNCTION concat_esc("left" text, "right" text) RETURNS text
994
    LANGUAGE sql IMMUTABLE
995
    AS $_$
996
SELECT util.esc_name__append($2, $1)
997
$_$;
998

    
999

    
1000
--
1001
-- Name: contains(text, text); Type: FUNCTION; Schema: util; Owner: -
1002
--
1003

    
1004
CREATE FUNCTION contains(search_for text, in_str text) RETURNS boolean
1005
    LANGUAGE sql IMMUTABLE
1006
    AS $_$
1007
SELECT position($1 in $2) > 0 /*1-based offset*/
1008
$_$;
1009

    
1010

    
1011
--
1012
-- Name: copy_struct(regclass, text); Type: FUNCTION; Schema: util; Owner: -
1013
--
1014

    
1015
CREATE FUNCTION copy_struct(from_ regclass, to_ text) RETURNS void
1016
    LANGUAGE sql
1017
    AS $_$
1018
SELECT util.eval($$CREATE TABLE $$||$2||$$ (LIKE $$||$1||$$ INCLUDING ALL)$$)
1019
$_$;
1020

    
1021

    
1022
--
1023
-- Name: copy_types_and_data(regclass, text); Type: FUNCTION; Schema: util; Owner: -
1024
--
1025

    
1026
CREATE FUNCTION copy_types_and_data(from_ regclass, to_ text) RETURNS void
1027
    LANGUAGE sql
1028
    AS $_$
1029
SELECT util.materialize_view($2, $1)
1030
$_$;
1031

    
1032

    
1033
--
1034
-- Name: create_if_not_exists(text); Type: FUNCTION; Schema: util; Owner: -
1035
--
1036

    
1037
CREATE FUNCTION create_if_not_exists(sql text) RETURNS void
1038
    LANGUAGE plpgsql STRICT
1039
    AS $$
1040
BEGIN
1041
	PERFORM util.eval(sql);
1042
EXCEPTION
1043
WHEN   duplicate_table
1044
	OR duplicate_object -- eg. constraint
1045
	OR duplicate_column
1046
	OR duplicate_function
1047
THEN NULL;
1048
WHEN invalid_table_definition THEN
1049
	IF SQLERRM LIKE 'multiple primary keys for table % are not allowed' THEN NULL;
1050
	ELSE RAISE;
1051
	END IF;
1052
END;
1053
$$;
1054

    
1055

    
1056
--
1057
-- Name: FUNCTION create_if_not_exists(sql text); Type: COMMENT; Schema: util; Owner: -
1058
--
1059

    
1060
COMMENT ON FUNCTION create_if_not_exists(sql text) IS '
1061
idempotent
1062
';
1063

    
1064

    
1065
--
1066
-- Name: curr_func(text, anyelement); Type: FUNCTION; Schema: util; Owner: -
1067
--
1068

    
1069
CREATE FUNCTION curr_func(func text, schema_anchor anyelement) RETURNS text
1070
    LANGUAGE sql STABLE
1071
    AS $$
1072
SELECT util.schema_esc(schema_anchor)||'.'||quote_ident(func)
1073
$$;
1074

    
1075

    
1076
--
1077
-- Name: debug_print_func_call(text); Type: FUNCTION; Schema: util; Owner: -
1078
--
1079

    
1080
CREATE FUNCTION debug_print_func_call(func_call text) RETURNS void
1081
    LANGUAGE sql IMMUTABLE
1082
    AS $_$
1083
SELECT util.raise('NOTICE', $$SELECT $$||$1)
1084
$_$;
1085

    
1086

    
1087
--
1088
-- Name: debug_print_return_value(anyelement, boolean); Type: FUNCTION; Schema: util; Owner: -
1089
--
1090

    
1091
CREATE FUNCTION debug_print_return_value(value anyelement, encode boolean DEFAULT false) RETURNS anyelement
1092
    LANGUAGE sql IMMUTABLE
1093
    AS $_$
1094
SELECT util.debug_print_value('returns: ', $1, $2);
1095
SELECT $1;
1096
$_$;
1097

    
1098

    
1099
--
1100
-- Name: debug_print_sql(text); Type: FUNCTION; Schema: util; Owner: -
1101
--
1102

    
1103
CREATE FUNCTION debug_print_sql(sql text) RETURNS void
1104
    LANGUAGE sql IMMUTABLE
1105
    AS $_$
1106
/* newline before so the query starts at the beginning of the line.
1107
newline after to visually separate queries from one another. */
1108
SELECT util.raise('NOTICE', $$
1109
$$||util.runnable_sql($1)||$$
1110
$$||COALESCE(util.explain2notice_msg_if_can($1), ''))
1111
$_$;
1112

    
1113

    
1114
--
1115
-- Name: debug_print_value(text, anyelement, boolean); Type: FUNCTION; Schema: util; Owner: -
1116
--
1117

    
1118
CREATE FUNCTION debug_print_value(label text, value anyelement, encode boolean DEFAULT false) RETURNS void
1119
    LANGUAGE sql IMMUTABLE
1120
    AS $_$
1121
SELECT util.raise('NOTICE', concat($1,
1122
(CASE WHEN $3 THEN util.quote_typed($2) ELSE $2::text END))||$$
1123
$$)
1124
$_$;
1125

    
1126

    
1127
--
1128
-- Name: debug_print_var(text, anyelement, boolean); Type: FUNCTION; Schema: util; Owner: -
1129
--
1130

    
1131
CREATE FUNCTION debug_print_var(var text, value anyelement, encode boolean DEFAULT false) RETURNS void
1132
    LANGUAGE sql IMMUTABLE
1133
    AS $_$
1134
/* can't use EXECUTE in the caller because "No substitution of PL/pgSQL
1135
variables is done on the computed command string"
1136
(http://www.postgresql.org/docs/9.3/static/plpgsql-statements.html#PLPGSQL-STATEMENTS-EXECUTING-DYN) */
1137
SELECT util.debug_print_value($1||' = ', $2, $3)
1138
$_$;
1139

    
1140

    
1141
--
1142
-- Name: derived_cols(regclass, regclass); Type: FUNCTION; Schema: util; Owner: -
1143
--
1144

    
1145
CREATE FUNCTION derived_cols(table_ regclass, names regclass) RETURNS SETOF text
1146
    LANGUAGE sql STABLE
1147
    AS $_$
1148
SELECT util.eval2set($$
1149
SELECT col
1150
FROM util.col_names($$||quote_nullable($1)||$$::regclass) f (col)
1151
LEFT JOIN $$||$2||$$ ON "to" = col
1152
WHERE "from" IS NULL
1153
$$, NULL::text)
1154
$_$;
1155

    
1156

    
1157
--
1158
-- Name: FUNCTION derived_cols(table_ regclass, names regclass); Type: COMMENT; Schema: util; Owner: -
1159
--
1160

    
1161
COMMENT ON FUNCTION derived_cols(table_ regclass, names regclass) IS '
1162
gets table_''s derived columns (all the columns not in the names table)
1163
';
1164

    
1165

    
1166
--
1167
-- Name: diff(regclass, regclass, anyelement); Type: FUNCTION; Schema: util; Owner: -
1168
--
1169

    
1170
CREATE FUNCTION diff(left_table regclass, right_table regclass, col_type_null anyelement, OUT left_ anyelement, OUT right_ anyelement) RETURNS SETOF record
1171
    LANGUAGE sql
1172
    AS $_$
1173
-- create a diff when the # of copies of a row differs between the tables
1174
SELECT util.to_freq($1);
1175
SELECT util.to_freq($2);
1176
SELECT util.auto_rm_freq(ARRAY[$1, $2]);
1177

    
1178
SELECT * FROM util.diff($1, $2, $3, has_freq := true)
1179
$_$;
1180

    
1181

    
1182
--
1183
-- Name: FUNCTION diff(left_table regclass, right_table regclass, col_type_null anyelement, OUT left_ anyelement, OUT right_ anyelement); Type: COMMENT; Schema: util; Owner: -
1184
--
1185

    
1186
COMMENT ON FUNCTION diff(left_table regclass, right_table regclass, col_type_null anyelement, OUT left_ anyelement, OUT right_ anyelement) IS '
1187
usage:
1188
SELECT * FROM util.diff(''"left_table"''::regclass, ''"right_table"''::regclass, NULL::shared_base_type)
1189

    
1190
col_type_null (*required*): NULL::shared_base_type
1191
';
1192

    
1193

    
1194
--
1195
-- Name: diff(text, text, anyelement, boolean); Type: FUNCTION; Schema: util; Owner: -
1196
--
1197

    
1198
CREATE FUNCTION diff(left__ text, right__ text, col_type_null anyelement, single_row boolean DEFAULT false, OUT left_ anyelement, OUT right_ anyelement) RETURNS SETOF record
1199
    LANGUAGE plpgsql
1200
    SET search_path TO pg_temp
1201
    AS $_$
1202
/* must use LANGUAGE plpgsql because LANGUAGE sql does not support runtime
1203
changes of search_path (schema elements are bound at inline time rather than
1204
runtime) */
1205
/* function option search_path is needed to limit the effects of
1206
`SET LOCAL search_path` to the current function */
1207
BEGIN
1208
	PERFORM util.use_schema($3); -- includes util.%== as default/fallback
1209
	
1210
	PERFORM util.mk_keys_func(pg_typeof($3));
1211
	RETURN QUERY
1212
		SELECT * FROM util.eval2col_pair(util.mk_diff_query($1, $2,
1213
$$/* need to explicitly cast each side to the return type because this does not
1214
happen automatically even when an implicit cast is available */
1215
  left_::$$||util.typeof($3)||$$
1216
, right_::$$||util.typeof($3)
1217
/* when using the util.%==(anyelement, anyelement) operator, you must cast to
1218
the *same* base type, *even though* this is optional when using a custom %== */
1219
, util._if($4, $$true/*= CROSS JOIN*/$$,
1220
$$ left_::$$||util.typeof($3)||$$
1221
%== right_::$$||util.typeof($3)||$$
1222
	-- refer to EXPLAIN output for expansion of %==$$
1223
)
1224
,     $$         left_::$$||util.typeof($3)||$$
1225
IS DISTINCT FROM right_::$$||util.typeof($3)
1226
), $3)
1227
	;
1228
END;
1229
$_$;
1230

    
1231

    
1232
--
1233
-- Name: FUNCTION diff(left__ text, right__ text, col_type_null anyelement, single_row boolean, OUT left_ anyelement, OUT right_ anyelement); Type: COMMENT; Schema: util; Owner: -
1234
--
1235

    
1236
COMMENT ON FUNCTION diff(left__ text, right__ text, col_type_null anyelement, single_row boolean, OUT left_ anyelement, OUT right_ anyelement) IS '
1237
col_type_null (*required*): NULL::col_type
1238
single_row: whether the tables consist of a single row, which should be
1239
	displayed side-by-side
1240

    
1241
to match up rows using a subset of the columns, create a custom keys() function
1242
which returns this subset as a record:
1243
-- note that OUT parameters for the returned fields are *not* needed
1244
CREATE OR REPLACE FUNCTION [schema].keys(value [schema].[base_type])
1245
  RETURNS record AS
1246
$BODY$
1247
SELECT ($1.key_field_0, $1.key_field_1)
1248
$BODY$
1249
  LANGUAGE sql IMMUTABLE
1250
  COST 100;
1251

    
1252

    
1253
to run EXPLAIN on the FULL JOIN query:
1254
# run this function
1255
# look for a NOTICE containing the expanded query that it ran
1256
# run EXPLAIN on this expanded query
1257
';
1258

    
1259

    
1260
--
1261
-- Name: diff(regclass, regclass, anyelement, boolean); Type: FUNCTION; Schema: util; Owner: -
1262
--
1263

    
1264
CREATE FUNCTION diff(left_table regclass, right_table regclass, col_type_null anyelement, has_freq boolean, OUT left_ anyelement, OUT right_ anyelement) RETURNS SETOF record
1265
    LANGUAGE sql
1266
    AS $_$
1267
SELECT * FROM util.diff($1::text, $2::text, $3,
1268
	single_row := util.has_single_row($1) AND util.has_single_row($2))
1269
$_$;
1270

    
1271

    
1272
--
1273
-- Name: FUNCTION diff(left_table regclass, right_table regclass, col_type_null anyelement, has_freq boolean, OUT left_ anyelement, OUT right_ anyelement); Type: COMMENT; Schema: util; Owner: -
1274
--
1275

    
1276
COMMENT ON FUNCTION diff(left_table regclass, right_table regclass, col_type_null anyelement, has_freq boolean, OUT left_ anyelement, OUT right_ anyelement) IS '
1277
helper function used by diff(regclass, regclass)
1278

    
1279
usage:
1280
SELECT * FROM util.diff(''"left_freq_table"''::regclass, ''"right_freq_table"''::regclass, NULL::shared_base_type, has_freq := true)
1281

    
1282
col_type_null (*required*): NULL::shared_base_type
1283
';
1284

    
1285

    
1286
--
1287
-- Name: do_optionally_ignore(text, boolean); Type: FUNCTION; Schema: util; Owner: -
1288
--
1289

    
1290
CREATE FUNCTION do_optionally_ignore(sql text, ignore boolean) RETURNS void
1291
    LANGUAGE sql
1292
    AS $_$
1293
SELECT CASE WHEN $2 THEN util.try_create($1) ELSE util.create_if_not_exists($1) END
1294
$_$;
1295

    
1296

    
1297
--
1298
-- Name: FUNCTION do_optionally_ignore(sql text, ignore boolean); Type: COMMENT; Schema: util; Owner: -
1299
--
1300

    
1301
COMMENT ON FUNCTION do_optionally_ignore(sql text, ignore boolean) IS '
1302
idempotent
1303
';
1304

    
1305

    
1306
--
1307
-- Name: drop_column(col_ref, boolean); Type: FUNCTION; Schema: util; Owner: -
1308
--
1309

    
1310
CREATE FUNCTION drop_column(col col_ref, force boolean DEFAULT false) RETURNS void
1311
    LANGUAGE sql
1312
    AS $_$
1313
SELECT util.eval($$ALTER TABLE $$||$1.table_||$$ DROP COLUMN IF EXISTS $$||
1314
quote_ident($1.name)||util._if($2, $$ CASCADE$$, ''::text))
1315
$_$;
1316

    
1317

    
1318
--
1319
-- Name: FUNCTION drop_column(col col_ref, force boolean); Type: COMMENT; Schema: util; Owner: -
1320
--
1321

    
1322
COMMENT ON FUNCTION drop_column(col col_ref, force boolean) IS '
1323
idempotent
1324
';
1325

    
1326

    
1327
--
1328
-- Name: drop_column(regclass[], text, boolean); Type: FUNCTION; Schema: util; Owner: -
1329
--
1330

    
1331
CREATE FUNCTION drop_column(tables regclass[], col text, force boolean DEFAULT false) RETURNS void
1332
    LANGUAGE sql
1333
    AS $_$
1334
SELECT util.drop_column((table_, $2), $3) FROM unnest($1) table_;
1335
SELECT NULL::void; -- don't fold away functions called in previous query
1336
$_$;
1337

    
1338

    
1339
--
1340
-- Name: FUNCTION drop_column(tables regclass[], col text, force boolean); Type: COMMENT; Schema: util; Owner: -
1341
--
1342

    
1343
COMMENT ON FUNCTION drop_column(tables regclass[], col text, force boolean) IS '
1344
idempotent
1345
';
1346

    
1347

    
1348
--
1349
-- Name: drop_relation(anyelement, boolean); Type: FUNCTION; Schema: util; Owner: -
1350
--
1351

    
1352
CREATE FUNCTION drop_relation(relation anyelement, force boolean DEFAULT false) RETURNS void
1353
    LANGUAGE sql
1354
    AS $_$
1355
/* use util.qual_name() instead of ::text so that the schema qualifier is always
1356
included in the debug SQL */
1357
SELECT util.drop_relation(util.relation_type($1), util.qual_name($1), $2)
1358
$_$;
1359

    
1360

    
1361
--
1362
-- Name: drop_relation(text, text, boolean); Type: FUNCTION; Schema: util; Owner: -
1363
--
1364

    
1365
CREATE FUNCTION drop_relation(type text, relation_esc text, force boolean DEFAULT false) RETURNS void
1366
    LANGUAGE sql
1367
    AS $_$
1368
SELECT util.eval($$DROP $$||$1||$$ IF EXISTS $$||$2
1369
||util._if($3, $$ CASCADE$$, ''::text))
1370
$_$;
1371

    
1372

    
1373
--
1374
-- Name: FUNCTION drop_relation(type text, relation_esc text, force boolean); Type: COMMENT; Schema: util; Owner: -
1375
--
1376

    
1377
COMMENT ON FUNCTION drop_relation(type text, relation_esc text, force boolean) IS '
1378
idempotent
1379
';
1380

    
1381

    
1382
--
1383
-- Name: drop_relations_like(text, anyelement, boolean); Type: FUNCTION; Schema: util; Owner: -
1384
--
1385

    
1386
CREATE FUNCTION drop_relations_like(name_regexp text, schema_anchor anyelement, force boolean DEFAULT false) RETURNS void
1387
    LANGUAGE sql
1388
    AS $_$
1389
SELECT util.drop_relations_like($1, util.schema_regexp($2), $3)
1390
$_$;
1391

    
1392

    
1393
--
1394
-- Name: drop_relations_like(text, text, boolean); Type: FUNCTION; Schema: util; Owner: -
1395
--
1396

    
1397
CREATE FUNCTION drop_relations_like(name_regexp text, schema_regexp text DEFAULT ''::text, force boolean DEFAULT false) RETURNS void
1398
    LANGUAGE sql
1399
    AS $_$
1400
SELECT util.debug_print_func_call(util.quote_func_call(
1401
'util.drop_relations_like', util.quote_typed($1), util.quote_typed($2),
1402
util.quote_typed($3)))
1403
;
1404
SELECT util.drop_relation(relation, $3)
1405
FROM util.show_relations_like($1, $2) relation
1406
;
1407
SELECT NULL::void; -- don't fold away functions called in previous query
1408
$_$;
1409

    
1410

    
1411
--
1412
-- Name: drop_table(text, boolean); Type: FUNCTION; Schema: util; Owner: -
1413
--
1414

    
1415
CREATE FUNCTION drop_table(table_ text, force boolean DEFAULT false) RETURNS void
1416
    LANGUAGE sql
1417
    AS $_$
1418
SELECT util.drop_relation('TABLE', $1, $2)
1419
$_$;
1420

    
1421

    
1422
--
1423
-- Name: FUNCTION drop_table(table_ text, force boolean); Type: COMMENT; Schema: util; Owner: -
1424
--
1425

    
1426
COMMENT ON FUNCTION drop_table(table_ text, force boolean) IS '
1427
idempotent
1428
';
1429

    
1430

    
1431
--
1432
-- Name: drop_view(text, boolean); Type: FUNCTION; Schema: util; Owner: -
1433
--
1434

    
1435
CREATE FUNCTION drop_view(view_ text, force boolean DEFAULT false) RETURNS void
1436
    LANGUAGE sql
1437
    AS $_$
1438
SELECT util.drop_relation('VIEW', $1, $2)
1439
$_$;
1440

    
1441

    
1442
--
1443
-- Name: FUNCTION drop_view(view_ text, force boolean); Type: COMMENT; Schema: util; Owner: -
1444
--
1445

    
1446
COMMENT ON FUNCTION drop_view(view_ text, force boolean) IS '
1447
idempotent
1448
';
1449

    
1450

    
1451
--
1452
-- Name: empty_array(anyelement); Type: FUNCTION; Schema: util; Owner: -
1453
--
1454

    
1455
CREATE FUNCTION empty_array(elem_type_null anyelement DEFAULT NULL::text) RETURNS anyarray
1456
    LANGUAGE sql IMMUTABLE
1457
    AS $_$
1458
SELECT util.array_fill($1, 0)
1459
$_$;
1460

    
1461

    
1462
--
1463
-- Name: FUNCTION empty_array(elem_type_null anyelement); Type: COMMENT; Schema: util; Owner: -
1464
--
1465

    
1466
COMMENT ON FUNCTION empty_array(elem_type_null anyelement) IS '
1467
constructs proper empty 1-dimensional array whose dimensions are not NULL ( ''{}''::text[] does not do this)
1468
';
1469

    
1470

    
1471
--
1472
-- Name: ensure_prefix(text, text); Type: FUNCTION; Schema: util; Owner: -
1473
--
1474

    
1475
CREATE FUNCTION ensure_prefix(prefix text, str text) RETURNS text
1476
    LANGUAGE sql IMMUTABLE
1477
    AS $_$
1478
SELECT (CASE WHEN util.has_prefix($1, $2) THEN $2 ELSE $1||$2 END)
1479
$_$;
1480

    
1481

    
1482
--
1483
-- Name: esc_name__append(text, text); Type: FUNCTION; Schema: util; Owner: -
1484
--
1485

    
1486
CREATE FUNCTION esc_name__append(suffix text, esc_name text) RETURNS text
1487
    LANGUAGE sql IMMUTABLE
1488
    AS $_$
1489
SELECT regexp_replace($2, '("?)$', $1||'\1')
1490
$_$;
1491

    
1492

    
1493
--
1494
-- Name: eval(text[]); Type: FUNCTION; Schema: util; Owner: -
1495
--
1496

    
1497
CREATE FUNCTION eval(queries text[]) RETURNS void
1498
    LANGUAGE sql
1499
    AS $_$
1500
SELECT util.eval(query) FROM unnest($1) query;
1501
SELECT NULL::void; -- don't fold away functions called in previous query
1502
$_$;
1503

    
1504

    
1505
--
1506
-- Name: eval(text, boolean); Type: FUNCTION; Schema: util; Owner: -
1507
--
1508

    
1509
CREATE FUNCTION eval(sql text, verbose_ boolean DEFAULT true) RETURNS void
1510
    LANGUAGE plpgsql
1511
    AS $$
1512
BEGIN
1513
	IF verbose_ THEN PERFORM util.debug_print_sql(sql); END IF;
1514
	EXECUTE sql;
1515
END;
1516
$$;
1517

    
1518

    
1519
--
1520
-- Name: eval2col_pair(text, anyelement); Type: FUNCTION; Schema: util; Owner: -
1521
--
1522

    
1523
CREATE FUNCTION eval2col_pair(sql text, col_type_null anyelement, OUT left_ anyelement, OUT right_ anyelement) RETURNS SETOF record
1524
    LANGUAGE plpgsql
1525
    AS $$
1526
BEGIN
1527
	PERFORM util.debug_print_sql(sql);
1528
	RETURN QUERY EXECUTE sql;
1529
END;
1530
$$;
1531

    
1532

    
1533
--
1534
-- Name: FUNCTION eval2col_pair(sql text, col_type_null anyelement, OUT left_ anyelement, OUT right_ anyelement); Type: COMMENT; Schema: util; Owner: -
1535
--
1536

    
1537
COMMENT ON FUNCTION eval2col_pair(sql text, col_type_null anyelement, OUT left_ anyelement, OUT right_ anyelement) IS '
1538
col_type_null (*required*): NULL::col_type
1539
';
1540

    
1541

    
1542
--
1543
-- Name: eval2records(text); Type: FUNCTION; Schema: util; Owner: -
1544
--
1545

    
1546
CREATE FUNCTION eval2records(sql text) RETURNS SETOF record
1547
    LANGUAGE plpgsql
1548
    AS $$
1549
BEGIN
1550
	PERFORM util.debug_print_sql(sql);
1551
	RETURN QUERY EXECUTE sql;
1552
END;
1553
$$;
1554

    
1555

    
1556
--
1557
-- Name: eval2set(text, anyelement, boolean); Type: FUNCTION; Schema: util; Owner: -
1558
--
1559

    
1560
CREATE FUNCTION eval2set(sql text, ret_type_null anyelement DEFAULT NULL::text, verbose_ boolean DEFAULT true) RETURNS SETOF anyelement
1561
    LANGUAGE plpgsql
1562
    AS $$
1563
BEGIN
1564
	IF verbose_ THEN PERFORM util.debug_print_sql(sql); END IF;
1565
	RETURN QUERY EXECUTE sql;
1566
END;
1567
$$;
1568

    
1569

    
1570
--
1571
-- Name: eval2val(text, anyelement); Type: FUNCTION; Schema: util; Owner: -
1572
--
1573

    
1574
CREATE FUNCTION eval2val(sql text, ret_type_null anyelement DEFAULT NULL::text) RETURNS anyelement
1575
    LANGUAGE plpgsql STABLE
1576
    AS $$
1577
DECLARE
1578
	ret_val ret_type_null%TYPE;
1579
BEGIN
1580
	PERFORM util.debug_print_sql(sql);
1581
	EXECUTE sql INTO STRICT ret_val;
1582
	RETURN ret_val;
1583
END;
1584
$$;
1585

    
1586

    
1587
--
1588
-- Name: FUNCTION eval2val(sql text, ret_type_null anyelement); Type: COMMENT; Schema: util; Owner: -
1589
--
1590

    
1591
COMMENT ON FUNCTION eval2val(sql text, ret_type_null anyelement) IS '
1592
ret_type_null: NULL::ret_type
1593
';
1594

    
1595

    
1596
--
1597
-- Name: eval_expr(text, anyelement); Type: FUNCTION; Schema: util; Owner: -
1598
--
1599

    
1600
CREATE FUNCTION eval_expr(sql text, ret_type_null anyelement DEFAULT NULL::text) RETURNS anyelement
1601
    LANGUAGE sql
1602
    AS $_$
1603
SELECT util.eval2val($$SELECT $$||$1, $2)
1604
$_$;
1605

    
1606

    
1607
--
1608
-- Name: FUNCTION eval_expr(sql text, ret_type_null anyelement); Type: COMMENT; Schema: util; Owner: -
1609
--
1610

    
1611
COMMENT ON FUNCTION eval_expr(sql text, ret_type_null anyelement) IS '
1612
ret_type_null: NULL::ret_type
1613
';
1614

    
1615

    
1616
--
1617
-- Name: eval_expr_passthru(text, anyelement); Type: FUNCTION; Schema: util; Owner: -
1618
--
1619

    
1620
CREATE FUNCTION eval_expr_passthru(sql text, ret_type_null anyelement DEFAULT NULL::text) RETURNS anyelement
1621
    LANGUAGE sql
1622
    AS $_$
1623
SELECT CASE WHEN $1 IS NULL THEN NULL ELSE util.eval_expr($1, $2) END
1624
$_$;
1625

    
1626

    
1627
--
1628
-- Name: FUNCTION eval_expr_passthru(sql text, ret_type_null anyelement); Type: COMMENT; Schema: util; Owner: -
1629
--
1630

    
1631
COMMENT ON FUNCTION eval_expr_passthru(sql text, ret_type_null anyelement) IS '
1632
sql: can be NULL, which will be passed through
1633
ret_type_null: NULL::ret_type
1634
';
1635

    
1636

    
1637
--
1638
-- Name: existing_cols(regclass, text[]); Type: FUNCTION; Schema: util; Owner: -
1639
--
1640

    
1641
CREATE FUNCTION existing_cols(table_ regclass, VARIADIC col_names text[]) RETURNS SETOF text
1642
    LANGUAGE sql STABLE
1643
    AS $_$
1644
SELECT col_name
1645
FROM unnest($2) s (col_name)
1646
WHERE util.col_exists(($1, col_name))
1647
$_$;
1648

    
1649

    
1650
--
1651
-- Name: explain(text); Type: FUNCTION; Schema: util; Owner: -
1652
--
1653

    
1654
CREATE FUNCTION explain(sql text) RETURNS SETOF text
1655
    LANGUAGE sql
1656
    AS $_$
1657
SELECT util.eval2set($$EXPLAIN $$||$1, verbose_ := false)
1658
$_$;
1659

    
1660

    
1661
--
1662
-- Name: explain2notice(text); Type: FUNCTION; Schema: util; Owner: -
1663
--
1664

    
1665
CREATE FUNCTION explain2notice(sql text) RETURNS void
1666
    LANGUAGE sql
1667
    AS $_$
1668
SELECT util.raise('NOTICE', util.explain2notice_msg($1))
1669
$_$;
1670

    
1671

    
1672
--
1673
-- Name: explain2notice_msg(text); Type: FUNCTION; Schema: util; Owner: -
1674
--
1675

    
1676
CREATE FUNCTION explain2notice_msg(sql text) RETURNS text
1677
    LANGUAGE sql
1678
    AS $_$
1679
-- newline before and after to visually separate it from other debug info
1680
SELECT COALESCE($$
1681
EXPLAIN:
1682
$$||util.fold_explain_msg(util.explain2str($1))||$$
1683
$$, '')
1684
$_$;
1685

    
1686

    
1687
--
1688
-- Name: explain2notice_msg_if_can(text); Type: FUNCTION; Schema: util; Owner: -
1689
--
1690

    
1691
CREATE FUNCTION explain2notice_msg_if_can(sql text) RETURNS text
1692
    LANGUAGE plpgsql
1693
    AS $$
1694
BEGIN
1695
	RETURN util.explain2notice_msg(sql);
1696
EXCEPTION
1697
WHEN syntax_error THEN RETURN NULL; -- non-explainable query
1698
	/* don't use util.is_explainable() because the list provided by Postgres
1699
	(http://www.postgresql.org/docs/9.3/static/sql-explain.html#AEN77691)
1700
	excludes some query types that are in fact EXPLAIN-able */
1701
END;
1702
$$;
1703

    
1704

    
1705
--
1706
-- Name: explain2str(text); Type: FUNCTION; Schema: util; Owner: -
1707
--
1708

    
1709
CREATE FUNCTION explain2str(sql text) RETURNS text
1710
    LANGUAGE sql
1711
    AS $_$
1712
SELECT util.join_strs(explain, $$
1713
$$) FROM util.explain($1)
1714
$_$;
1715

    
1716

    
1717
SET default_tablespace = '';
1718

    
1719
SET default_with_oids = false;
1720

    
1721
--
1722
-- Name: explain; Type: TABLE; Schema: util; Owner: -; Tablespace: 
1723
--
1724

    
1725
CREATE TABLE explain (
1726
    line text NOT NULL
1727
);
1728

    
1729

    
1730
--
1731
-- Name: explain2table(text, regclass); Type: FUNCTION; Schema: util; Owner: -
1732
--
1733

    
1734
CREATE FUNCTION explain2table(sql text, table_ regclass DEFAULT 'explain'::regclass) RETURNS void
1735
    LANGUAGE sql
1736
    AS $_$
1737
SELECT util.eval($$INSERT INTO $$||$2||$$ SELECT util.explain(
1738
$$||quote_nullable($1)||$$
1739
)$$)
1740
$_$;
1741

    
1742

    
1743
--
1744
-- Name: FUNCTION explain2table(sql text, table_ regclass); Type: COMMENT; Schema: util; Owner: -
1745
--
1746

    
1747
COMMENT ON FUNCTION explain2table(sql text, table_ regclass) IS '
1748
usage:
1749
PERFORM util.explain2table($$
1750
query
1751
$$);
1752
';
1753

    
1754

    
1755
--
1756
-- Name: first_word(text); Type: FUNCTION; Schema: util; Owner: -
1757
--
1758

    
1759
CREATE FUNCTION first_word(str text) RETURNS text
1760
    LANGUAGE sql IMMUTABLE
1761
    AS $_$
1762
SELECT match[1] FROM regexp_matches(util.ltrim_nl($1), '^(\S*)') match
1763
$_$;
1764

    
1765

    
1766
--
1767
-- Name: fix_array(anyarray); Type: FUNCTION; Schema: util; Owner: -
1768
--
1769

    
1770
CREATE FUNCTION fix_array("array" anyarray) RETURNS anyarray
1771
    LANGUAGE sql IMMUTABLE
1772
    AS $_$
1773
SELECT CASE WHEN $1 IS NULL THEN NULL ELSE (
1774
	CASE WHEN pg_catalog.array_ndims($1) IS NULL THEN util.empty_array($1[1]) ELSE $1 END
1775
) END
1776
$_$;
1777

    
1778

    
1779
--
1780
-- Name: FUNCTION fix_array("array" anyarray); Type: COMMENT; Schema: util; Owner: -
1781
--
1782

    
1783
COMMENT ON FUNCTION fix_array("array" anyarray) IS '
1784
ensures that an array will always have proper non-NULL dimensions
1785
';
1786

    
1787

    
1788
--
1789
-- Name: fold_explain_msg(text); Type: FUNCTION; Schema: util; Owner: -
1790
--
1791

    
1792
CREATE FUNCTION fold_explain_msg(explain text) RETURNS text
1793
    LANGUAGE sql IMMUTABLE
1794
    AS $_$
1795
SELECT (CASE WHEN util.first_word($1) = 'Result' THEN NULL ELSE $1 END)
1796
$_$;
1797

    
1798

    
1799
--
1800
-- Name: force_recreate(text, text[]); Type: FUNCTION; Schema: util; Owner: -
1801
--
1802

    
1803
CREATE FUNCTION force_recreate(cmd text, users text[] DEFAULT NULL::text[]) RETURNS void
1804
    LANGUAGE plpgsql
1805
    AS $_$
1806
DECLARE
1807
	PG_EXCEPTION_DETAIL text;
1808
	recreate_users_cmds text[] = util.save_drop_views(users);
1809
BEGIN
1810
	PERFORM util.eval(cmd);
1811
	PERFORM util.create_if_not_exists(recreate_cmd)
1812
	FROM unnest(recreate_users_cmds) recreate_cmd;
1813
		/* create_if_not_exists() rather than eval(), because cmd might manually
1814
		re-create a deleted dependent view, causing it to already exist */
1815
EXCEPTION
1816
WHEN dependent_objects_still_exist THEN
1817
	IF users IS NOT NULL THEN RAISE; END IF; -- save_drop_views() didn't fix it
1818
	GET STACKED DIAGNOSTICS PG_EXCEPTION_DETAIL = PG_EXCEPTION_DETAIL;
1819
	users = util.array_reverse(array(SELECT * FROM util.regexp_matches_group(
1820
		PG_EXCEPTION_DETAIL, '(?m)^view (.*) depends on [[:lower:]]+ .*$')));
1821
	PERFORM util.debug_print_var('PG_EXCEPTION_DETAIL', PG_EXCEPTION_DETAIL);
1822
	PERFORM util.debug_print_var('users', users);
1823
	IF util.is_empty(users) THEN RAISE; END IF;
1824
	PERFORM util.force_recreate(cmd, users);
1825
END;
1826
$_$;
1827

    
1828

    
1829
--
1830
-- Name: FUNCTION force_recreate(cmd text, users text[]); Type: COMMENT; Schema: util; Owner: -
1831
--
1832

    
1833
COMMENT ON FUNCTION force_recreate(cmd text, users text[]) IS '
1834
idempotent
1835

    
1836
users: not necessary to provide this because it will be autopopulated
1837
';
1838

    
1839

    
1840
--
1841
-- Name: force_update_view(text, text); Type: FUNCTION; Schema: util; Owner: -
1842
--
1843

    
1844
CREATE FUNCTION force_update_view(view_ text, query text) RETURNS void
1845
    LANGUAGE plpgsql STRICT
1846
    AS $_$
1847
DECLARE
1848
	mk_view text = $$CREATE OR REPLACE VIEW $$||view_||$$ AS
1849
$$||query;
1850
BEGIN
1851
	EXECUTE mk_view;
1852
EXCEPTION
1853
WHEN invalid_table_definition THEN
1854
	IF SQLERRM = 'cannot drop columns from view'
1855
	OR SQLERRM LIKE 'cannot change name of view column "%" to "%"'
1856
	THEN
1857
		EXECUTE $$DROP VIEW $$||view_||$$ CASCADE$$;
1858
		EXECUTE mk_view;
1859
	ELSE RAISE USING ERRCODE = SQLSTATE, MESSAGE = SQLERRM;
1860
	END IF;
1861
END;
1862
$_$;
1863

    
1864

    
1865
--
1866
-- Name: FUNCTION force_update_view(view_ text, query text); Type: COMMENT; Schema: util; Owner: -
1867
--
1868

    
1869
COMMENT ON FUNCTION force_update_view(view_ text, query text) IS '
1870
idempotent
1871
';
1872

    
1873

    
1874
--
1875
-- Name: freq_always_1(regclass, text); Type: FUNCTION; Schema: util; Owner: -
1876
--
1877

    
1878
CREATE FUNCTION freq_always_1(table_ regclass, freq_col text DEFAULT 'copies'::text) RETURNS boolean
1879
    LANGUAGE sql STABLE
1880
    AS $_$
1881
SELECT util.eval2val(
1882
$$SELECT NOT EXISTS( -- there is no row that is != 1
1883
	SELECT NULL
1884
	FROM $$||$1||$$
1885
	WHERE $$||quote_ident(freq_col)||$$ IS DISTINCT FROM 1
1886
	LIMIT 1
1887
)
1888
$$, NULL::boolean)
1889
$_$;
1890

    
1891

    
1892
--
1893
-- Name: freq_always_1(regclass[], text); Type: FUNCTION; Schema: util; Owner: -
1894
--
1895

    
1896
CREATE FUNCTION freq_always_1(tables regclass[], freq_col text DEFAULT 'copies'::text) RETURNS boolean
1897
    LANGUAGE sql STABLE
1898
    AS $_$
1899
SELECT bool_and(util.freq_always_1(table_, $2)) FROM unnest($1) table_
1900
$_$;
1901

    
1902

    
1903
--
1904
-- Name: grants_users(); Type: FUNCTION; Schema: util; Owner: -
1905
--
1906

    
1907
CREATE FUNCTION grants_users() RETURNS SETOF text
1908
    LANGUAGE sql IMMUTABLE
1909
    AS $$
1910
VALUES ('bien_read'), ('public_')
1911
$$;
1912

    
1913

    
1914
--
1915
-- Name: has_prefix(text, text); Type: FUNCTION; Schema: util; Owner: -
1916
--
1917

    
1918
CREATE FUNCTION has_prefix(prefix text, str text) RETURNS boolean
1919
    LANGUAGE sql IMMUTABLE
1920
    AS $_$
1921
SELECT substring($2 for length($1)) = $1
1922
$_$;
1923

    
1924

    
1925
--
1926
-- Name: has_single_row(regclass); Type: FUNCTION; Schema: util; Owner: -
1927
--
1928

    
1929
CREATE FUNCTION has_single_row(table_ regclass) RETURNS boolean
1930
    LANGUAGE sql STABLE
1931
    AS $_$
1932
SELECT util.eval2val($$SELECT COUNT(*) = 1 FROM $$||$1, NULL::boolean)
1933
$_$;
1934

    
1935

    
1936
--
1937
-- Name: hstore(text[], text); Type: FUNCTION; Schema: util; Owner: -
1938
--
1939

    
1940
CREATE FUNCTION hstore(keys text[], value text) RETURNS hstore
1941
    LANGUAGE sql IMMUTABLE
1942
    AS $_$
1943
SELECT hstore(util.fix_array($1), util.array_fill($2, util.array_length($1)))
1944
$_$;
1945

    
1946

    
1947
--
1948
-- Name: FUNCTION hstore(keys text[], value text); Type: COMMENT; Schema: util; Owner: -
1949
--
1950

    
1951
COMMENT ON FUNCTION hstore(keys text[], value text) IS '
1952
avoids repeating the same value for each key
1953
';
1954

    
1955

    
1956
--
1957
-- Name: ifnull(anyelement, anyelement); Type: FUNCTION; Schema: util; Owner: -
1958
--
1959

    
1960
CREATE FUNCTION ifnull(value anyelement, null_ anyelement) RETURNS anyelement
1961
    LANGUAGE sql IMMUTABLE
1962
    AS $_$
1963
SELECT COALESCE($1, $2)
1964
$_$;
1965

    
1966

    
1967
--
1968
-- Name: FUNCTION ifnull(value anyelement, null_ anyelement); Type: COMMENT; Schema: util; Owner: -
1969
--
1970

    
1971
COMMENT ON FUNCTION ifnull(value anyelement, null_ anyelement) IS '
1972
equivalent to MySQL''s IFNULL() (Postgres auto-lowercases the name)
1973
';
1974

    
1975

    
1976
--
1977
-- Name: in_reverse(anyarray); Type: FUNCTION; Schema: util; Owner: -
1978
--
1979

    
1980
CREATE FUNCTION in_reverse("array" anyarray) RETURNS SETOF anyelement
1981
    LANGUAGE sql IMMUTABLE
1982
    AS $_$
1983
SELECT * FROM unnest($1) ORDER BY row_number() OVER () DESC
1984
$_$;
1985

    
1986

    
1987
--
1988
-- Name: inherit(regclass, regclass); Type: FUNCTION; Schema: util; Owner: -
1989
--
1990

    
1991
CREATE FUNCTION inherit(derived regclass, base regclass) RETURNS void
1992
    LANGUAGE sql
1993
    AS $_$
1994
SELECT util.eval($$ALTER TABLE $$||$1||$$ INHERIT $$||$2)
1995
$_$;
1996

    
1997

    
1998
--
1999
-- Name: is_castable(text, anyelement); Type: FUNCTION; Schema: util; Owner: -
2000
--
2001

    
2002
CREATE FUNCTION is_castable(value text, ret_type_null anyelement) RETURNS boolean
2003
    LANGUAGE plpgsql IMMUTABLE
2004
    AS $$
2005
BEGIN
2006
	PERFORM util.cast(value, ret_type_null);
2007
	-- must happen *after* cast check, because NULL is not valid for some types
2008
	IF value IS NULL THEN RETURN NULL; END IF; -- pass NULL through
2009
	RETURN true;
2010
EXCEPTION
2011
WHEN data_exception THEN RETURN false;
2012
END;
2013
$$;
2014

    
2015

    
2016
--
2017
-- Name: FUNCTION is_castable(value text, ret_type_null anyelement); Type: COMMENT; Schema: util; Owner: -
2018
--
2019

    
2020
COMMENT ON FUNCTION is_castable(value text, ret_type_null anyelement) IS '
2021
passes NULL through. however, if NULL is not valid for the type, false will be
2022
returned instead.
2023

    
2024
ret_type_null: NULL::ret_type
2025
';
2026

    
2027

    
2028
--
2029
-- Name: is_constant(col_ref); Type: FUNCTION; Schema: util; Owner: -
2030
--
2031

    
2032
CREATE FUNCTION is_constant(col col_ref) RETURNS boolean
2033
    LANGUAGE sql STABLE
2034
    AS $_$
2035
SELECT COALESCE(util.col_comment($1) LIKE '
2036
constant
2037
%', false)
2038
$_$;
2039

    
2040

    
2041
--
2042
-- Name: is_empty(anyarray); Type: FUNCTION; Schema: util; Owner: -
2043
--
2044

    
2045
CREATE FUNCTION is_empty("array" anyarray) RETURNS boolean
2046
    LANGUAGE sql IMMUTABLE
2047
    AS $_$
2048
SELECT util.array_length($1) = 0
2049
$_$;
2050

    
2051

    
2052
--
2053
-- Name: is_explain(text); Type: FUNCTION; Schema: util; Owner: -
2054
--
2055

    
2056
CREATE FUNCTION is_explain(sql text) RETURNS boolean
2057
    LANGUAGE sql IMMUTABLE
2058
    AS $_$
2059
SELECT upper(util.first_word($1)) = 'EXPLAIN'
2060
$_$;
2061

    
2062

    
2063
--
2064
-- Name: is_explainable(text); Type: FUNCTION; Schema: util; Owner: -
2065
--
2066

    
2067
CREATE FUNCTION is_explainable(sql text) RETURNS boolean
2068
    LANGUAGE sql IMMUTABLE
2069
    AS $_$
2070
SELECT upper(util.first_word($1)) = ANY(
2071
'{SELECT,INSERT,UPDATE,DELETE,VALUES,EXECUTE,DECLARE}'::text[]
2072
/*from http://www.postgresql.org/docs/9.3/static/sql-explain.html#AEN77691*/
2073
)
2074
$_$;
2075

    
2076

    
2077
--
2078
-- Name: is_more_complete_than(anyelement, anyelement); Type: FUNCTION; Schema: util; Owner: -
2079
--
2080

    
2081
CREATE FUNCTION is_more_complete_than("left" anyelement, "right" anyelement) RETURNS boolean
2082
    LANGUAGE sql IMMUTABLE
2083
    AS $_$
2084
SELECT $1 IS NOT DISTINCT FROM $2 OR ($1 IS NOT NULL AND $2 IS NULL)
2085
$_$;
2086

    
2087

    
2088
--
2089
-- Name: is_populated_more_often_than(anyelement, anyelement); Type: FUNCTION; Schema: util; Owner: -
2090
--
2091

    
2092
CREATE FUNCTION is_populated_more_often_than("left" anyelement, "right" anyelement) RETURNS boolean
2093
    LANGUAGE sql IMMUTABLE
2094
    AS $_$
2095
SELECT $1 IS NOT NULL >= $2 IS NOT NULL -- true > false
2096
$_$;
2097

    
2098

    
2099
--
2100
-- Name: is_set_stmt(text); Type: FUNCTION; Schema: util; Owner: -
2101
--
2102

    
2103
CREATE FUNCTION is_set_stmt(sql text) RETURNS boolean
2104
    LANGUAGE sql IMMUTABLE
2105
    AS $_$
2106
SELECT upper(util.first_word($1)) = 'SET'
2107
$_$;
2108

    
2109

    
2110
--
2111
-- Name: is_table(regclass); Type: FUNCTION; Schema: util; Owner: -
2112
--
2113

    
2114
CREATE FUNCTION is_table(relation regclass) RETURNS boolean
2115
    LANGUAGE sql STABLE
2116
    AS $_$
2117
SELECT relkind = 'r' FROM pg_class WHERE oid = $1
2118
$_$;
2119

    
2120

    
2121
--
2122
-- Name: is_view(regclass); Type: FUNCTION; Schema: util; Owner: -
2123
--
2124

    
2125
CREATE FUNCTION is_view(relation regclass) RETURNS boolean
2126
    LANGUAGE sql STABLE
2127
    AS $_$
2128
SELECT relkind = 'v' FROM pg_class WHERE oid = $1
2129
$_$;
2130

    
2131

    
2132
--
2133
-- Name: join_strs_transform(text, text, text); Type: FUNCTION; Schema: util; Owner: -
2134
--
2135

    
2136
CREATE FUNCTION join_strs_transform(state text, value text, delim text) RETURNS text
2137
    LANGUAGE sql IMMUTABLE STRICT
2138
    AS $_$
2139
SELECT $1 || $3 || $2
2140
$_$;
2141

    
2142

    
2143
--
2144
-- Name: FUNCTION join_strs_transform(state text, value text, delim text); Type: COMMENT; Schema: util; Owner: -
2145
--
2146

    
2147
COMMENT ON FUNCTION join_strs_transform(state text, value text, delim text) IS '
2148
must be declared STRICT to use the special handling of STRICT aggregating functions
2149
';
2150

    
2151

    
2152
--
2153
-- Name: keys(anyelement); Type: FUNCTION; Schema: util; Owner: -
2154
--
2155

    
2156
CREATE FUNCTION keys(value anyelement) RETURNS anyelement
2157
    LANGUAGE sql IMMUTABLE
2158
    AS $_$
2159
SELECT $1 -- compare on the entire value
2160
$_$;
2161

    
2162

    
2163
--
2164
-- Name: limit2row_num(integer, integer, integer); Type: FUNCTION; Schema: util; Owner: -
2165
--
2166

    
2167
CREATE FUNCTION limit2row_num(limit_ integer, offset_ integer DEFAULT NULL::integer, min_row_num integer DEFAULT 1) RETURNS integer
2168
    LANGUAGE sql IMMUTABLE
2169
    AS $_$
2170
SELECT COALESCE(util.offset2row_num($2, $3) + $1 - 1, 2147483647)
2171
$_$;
2172

    
2173

    
2174
--
2175
-- Name: loop_ignore_errors(text, text, anyelement); Type: FUNCTION; Schema: util; Owner: -
2176
--
2177

    
2178
CREATE FUNCTION loop_ignore_errors(iter_sql text, loop_body_sql text, loop_type_null anyelement DEFAULT NULL::text) RETURNS void
2179
    LANGUAGE plpgsql
2180
    AS $$
2181
DECLARE
2182
	errors_ct integer = 0;
2183
	loop_var loop_type_null%TYPE;
2184
BEGIN
2185
	FOR loop_var IN SELECT * FROM util.eval2set(iter_sql, loop_type_null)
2186
	LOOP
2187
		BEGIN
2188
			EXECUTE loop_body_sql USING loop_var;
2189
		EXCEPTION
2190
		WHEN OTHERS THEN
2191
			errors_ct = errors_ct+1;
2192
			PERFORM util.raise_error_warning(SQLERRM);
2193
		END;
2194
	END LOOP;
2195
	IF errors_ct > 0 THEN
2196
		-- can't raise exception because this would roll back the transaction
2197
		PERFORM util.raise_error_warning('there were '||errors_ct
2198
			||' errors: see the WARNINGs for details');
2199
	END IF;
2200
END;
2201
$$;
2202

    
2203

    
2204
--
2205
-- Name: ltrim_nl(text); Type: FUNCTION; Schema: util; Owner: -
2206
--
2207

    
2208
CREATE FUNCTION ltrim_nl(str text) RETURNS text
2209
    LANGUAGE sql IMMUTABLE
2210
    AS $_$
2211
SELECT ltrim($1, $$
2212
$$)
2213
$_$;
2214

    
2215

    
2216
--
2217
-- Name: map_filter_insert(); Type: FUNCTION; Schema: util; Owner: -
2218
--
2219

    
2220
CREATE FUNCTION map_filter_insert() RETURNS trigger
2221
    LANGUAGE plpgsql
2222
    AS $$
2223
BEGIN
2224
	IF new."from" LIKE ':%' THEN RETURN NULL; END IF; -- exclude metadata values
2225
	RETURN new;
2226
END;
2227
$$;
2228

    
2229

    
2230
--
2231
-- Name: map_get(regclass, text); Type: FUNCTION; Schema: util; Owner: -
2232
--
2233

    
2234
CREATE FUNCTION map_get(map regclass, key text) RETURNS text
2235
    LANGUAGE plpgsql STABLE STRICT
2236
    AS $_$
2237
DECLARE
2238
    value text;
2239
BEGIN
2240
    EXECUTE $$SELECT "to" FROM $$||map||$$ WHERE "from" = $1$$
2241
        INTO value USING key;
2242
    RETURN value;
2243
END;
2244
$_$;
2245

    
2246

    
2247
--
2248
-- Name: map_nulls(text[], anyelement); Type: FUNCTION; Schema: util; Owner: -
2249
--
2250

    
2251
CREATE FUNCTION map_nulls(nulls text[], value anyelement) RETURNS anyelement
2252
    LANGUAGE sql IMMUTABLE
2253
    AS $_$
2254
SELECT util._map(util.nulls_map($1), $2)
2255
$_$;
2256

    
2257

    
2258
--
2259
-- Name: FUNCTION map_nulls(nulls text[], value anyelement); Type: COMMENT; Schema: util; Owner: -
2260
--
2261

    
2262
COMMENT ON FUNCTION map_nulls(nulls text[], value anyelement) IS '
2263
due to dynamic inlining[1], this is just as fast as util._map() which it wraps[2].
2264

    
2265
[1] inlining of function calls, which is different from constant folding
2266
[2] _map()''s profiling query
2267
SELECT util._map(''"1"=>NULL, "2"=>NULL, "3"=>NULL, *=>*'', v) FROM unnest(array_fill(1, array[100000])) f (v)
2268
and map_nulls()''s profiling query
2269
SELECT util.map_nulls(array[1, 2, 3]::text[], v) FROM unnest(array_fill(1, array[100000])) f (v)
2270
both take ~920 ms.
2271
also, /inputs/REMIB/Specimen/postprocess.sql > country takes the same amount of time (56000 ms) to build with map_nulls() as with a literal hstore.
2272
';
2273

    
2274

    
2275
--
2276
-- Name: map_values(regclass); Type: FUNCTION; Schema: util; Owner: -
2277
--
2278

    
2279
CREATE FUNCTION map_values(map regclass) RETURNS SETOF text
2280
    LANGUAGE plpgsql STABLE STRICT
2281
    AS $_$
2282
BEGIN
2283
    RETURN QUERY EXECUTE $$SELECT "to" FROM $$||map;
2284
END;
2285
$_$;
2286

    
2287

    
2288
--
2289
-- Name: materialize_query(text, text); Type: FUNCTION; Schema: util; Owner: -
2290
--
2291

    
2292
CREATE FUNCTION materialize_query(table_esc text, sql text) RETURNS void
2293
    LANGUAGE sql
2294
    AS $_$
2295
SELECT util.create_if_not_exists($$CREATE TABLE $$||$1||$$ AS
2296
$$||util.ltrim_nl($2));
2297
-- make sure the created table has the correct estimated row count
2298
SELECT util.analyze_($1);
2299

    
2300
SELECT util.append_comment($1, '
2301
contents generated from:
2302
'||util.ltrim_nl(util.runnable_sql($2))||';
2303
');
2304
$_$;
2305

    
2306

    
2307
--
2308
-- Name: FUNCTION materialize_query(table_esc text, sql text); Type: COMMENT; Schema: util; Owner: -
2309
--
2310

    
2311
COMMENT ON FUNCTION materialize_query(table_esc text, sql text) IS '
2312
idempotent
2313
';
2314

    
2315

    
2316
--
2317
-- Name: materialize_view(text, regclass); Type: FUNCTION; Schema: util; Owner: -
2318
--
2319

    
2320
CREATE FUNCTION materialize_view(table_esc text, view_ regclass) RETURNS void
2321
    LANGUAGE sql
2322
    AS $_$
2323
SELECT util.materialize_query($1, $$SELECT * FROM $$||$2)
2324
$_$;
2325

    
2326

    
2327
--
2328
-- Name: FUNCTION materialize_view(table_esc text, view_ regclass); Type: COMMENT; Schema: util; Owner: -
2329
--
2330

    
2331
COMMENT ON FUNCTION materialize_view(table_esc text, view_ regclass) IS '
2332
idempotent
2333
';
2334

    
2335

    
2336
--
2337
-- Name: mk_const_col(col_ref, anyelement); Type: FUNCTION; Schema: util; Owner: -
2338
--
2339

    
2340
CREATE FUNCTION mk_const_col(col col_ref, value anyelement) RETURNS void
2341
    LANGUAGE sql
2342
    AS $_$
2343
SELECT util.create_if_not_exists($$
2344
ALTER TABLE $$||$1.table_||$$ ADD COLUMN $$
2345
||quote_ident($1.name)||$$ $$||pg_typeof($2)||util.type_qual($2)||$$ DEFAULT $$
2346
||quote_literal($2)||$$;
2347
COMMENT ON COLUMN $$||$1.table_||$$.$$||quote_ident($1.name)||$$ IS '
2348
constant
2349
';
2350
$$)
2351
$_$;
2352

    
2353

    
2354
--
2355
-- Name: FUNCTION mk_const_col(col col_ref, value anyelement); Type: COMMENT; Schema: util; Owner: -
2356
--
2357

    
2358
COMMENT ON FUNCTION mk_const_col(col col_ref, value anyelement) IS '
2359
idempotent
2360
';
2361

    
2362

    
2363
--
2364
-- Name: mk_derived_col(col_ref, text, boolean); Type: FUNCTION; Schema: util; Owner: -
2365
--
2366

    
2367
CREATE FUNCTION mk_derived_col(col col_ref, expr text, overwrite boolean DEFAULT false) RETURNS void
2368
    LANGUAGE plpgsql STRICT
2369
    AS $_$
2370
DECLARE
2371
    type regtype = util.typeof(expr, col.table_::text::regtype);
2372
    col_name_sql text = quote_ident(col.name);
2373
BEGIN
2374
    PERFORM util.create_if_not_exists((CASE WHEN overwrite THEN '' ELSE $$
2375
ALTER TABLE $$||col.table_||$$ ADD   COLUMN $$||col_name_sql||$$      $$||type||$$;$$ END)||$$
2376
ALTER TABLE $$||col.table_||$$ ALTER COLUMN $$||col_name_sql||$$ TYPE $$||type||$$ USING
2377
$$||expr||$$;
2378
$$);
2379
END;
2380
$_$;
2381

    
2382

    
2383
--
2384
-- Name: FUNCTION mk_derived_col(col col_ref, expr text, overwrite boolean); Type: COMMENT; Schema: util; Owner: -
2385
--
2386

    
2387
COMMENT ON FUNCTION mk_derived_col(col col_ref, expr text, overwrite boolean) IS '
2388
idempotent
2389
';
2390

    
2391

    
2392
--
2393
-- Name: mk_diff_query(text, text, text, text, text); Type: FUNCTION; Schema: util; Owner: -
2394
--
2395

    
2396
CREATE FUNCTION mk_diff_query(left_ text, right_ text, cols text DEFAULT 'left_, right_'::text, join_cond text DEFAULT 'left_ %== right_ -- refer to EXPLAIN output for expansion of %=='::text, filter text DEFAULT 'left_ IS DISTINCT FROM right_'::text) RETURNS text
2397
    LANGUAGE sql IMMUTABLE
2398
    AS $_$
2399
SELECT
2400
$$SELECT
2401
$$||$3||$$
2402
FROM      $$||$1||$$ left_
2403
FULL JOIN $$||$2||$$ right_
2404
ON $$||$4||$$
2405
WHERE $$||$5||$$
2406
ORDER BY left_, right_
2407
$$
2408
$_$;
2409

    
2410

    
2411
--
2412
-- Name: mk_keys_func(regtype); Type: FUNCTION; Schema: util; Owner: -
2413
--
2414

    
2415
CREATE FUNCTION mk_keys_func(type regtype) RETURNS void
2416
    LANGUAGE sql
2417
    AS $_$
2418
-- keys()
2419
SELECT util.mk_keys_func($1, ARRAY(
2420
SELECT col FROM util.typed_cols($1) col
2421
WHERE (col).type != ANY('{bigint}'::regtype[]) -- not a count col
2422
));
2423

    
2424
-- values_()
2425
SELECT util.mk_keys_func($1, COALESCE(
2426
	NULLIF(ARRAY(
2427
	SELECT col FROM util.typed_cols($1) col
2428
	WHERE (col).type = ANY('{bigint}'::regtype[]) -- is a count col
2429
	), ARRAY[]::util.col_cast[])
2430
, ARRAY(SELECT util.typed_cols($1))) -- no count cols, so use all cols
2431
, 'values_');
2432
$_$;
2433

    
2434

    
2435
--
2436
-- Name: mk_keys_func(regtype, col_cast[], text); Type: FUNCTION; Schema: util; Owner: -
2437
--
2438

    
2439
CREATE FUNCTION mk_keys_func(type regtype, cols col_cast[], name text DEFAULT 'keys'::text) RETURNS void
2440
    LANGUAGE sql
2441
    AS $_$
2442
SELECT util.create_if_not_exists($$
2443
CREATE TYPE $$||util.prefixed_name($3||'_', $1)||$$ AS
2444
($$||util.mk_typed_cols_list($2)||$$);
2445
COMMENT ON TYPE $$||util.prefixed_name($3||'_', $1)||$$ IS '
2446
autogenerated
2447
';
2448
$$);
2449

    
2450
SELECT util.mk_keys_func($1, util.prefixed_name($3||'_', $1)::regtype, $3);
2451
$_$;
2452

    
2453

    
2454
--
2455
-- Name: mk_keys_func(regtype, regtype, text); Type: FUNCTION; Schema: util; Owner: -
2456
--
2457

    
2458
CREATE FUNCTION mk_keys_func(type regtype, return_type regtype, name text DEFAULT 'keys'::text) RETURNS void
2459
    LANGUAGE sql
2460
    AS $_$
2461
SELECT util.create_if_not_exists($$
2462
CREATE FUNCTION $$||util.qual_name(util.schema($1), $3)||$$(value $$
2463
||util.qual_name($1)||$$)
2464
  RETURNS $$||util.qual_name($2)||$$ AS
2465
$BODY1$
2466
SELECT ROW($$||
2467
(SELECT COALESCE(string_agg($$$1.$$||quote_ident((col).col_name), ', '), '')
2468
FROM util.typed_cols($2) col) ||$$)::$$||util.qual_name($2)||$$
2469
$BODY1$
2470
  LANGUAGE sql IMMUTABLE
2471
  COST 100;
2472
$$);
2473
$_$;
2474

    
2475

    
2476
--
2477
-- Name: mk_map_table(text); Type: FUNCTION; Schema: util; Owner: -
2478
--
2479

    
2480
CREATE FUNCTION mk_map_table(table_ text) RETURNS void
2481
    LANGUAGE sql
2482
    AS $_$
2483
SELECT util.create_if_not_exists($$
2484
CREATE TABLE $$||$1||$$
2485
(
2486
    LIKE util.map INCLUDING ALL
2487
);
2488

    
2489
CREATE TRIGGER map_filter_insert
2490
  BEFORE INSERT
2491
  ON $$||$1||$$
2492
  FOR EACH ROW
2493
  EXECUTE PROCEDURE util.map_filter_insert();
2494
$$)
2495
$_$;
2496

    
2497

    
2498
--
2499
-- Name: mk_not_null(text); Type: FUNCTION; Schema: util; Owner: -
2500
--
2501

    
2502
CREATE FUNCTION mk_not_null(text) RETURNS text
2503
    LANGUAGE sql IMMUTABLE
2504
    AS $_$
2505
SELECT COALESCE($1, '<NULL>')
2506
$_$;
2507

    
2508

    
2509
--
2510
-- Name: mk_out_params(col_cast[]); Type: FUNCTION; Schema: util; Owner: -
2511
--
2512

    
2513
CREATE FUNCTION mk_out_params(cols col_cast[]) RETURNS text
2514
    LANGUAGE sql IMMUTABLE
2515
    AS $_$
2516
SELECT COALESCE(string_agg($$, OUT $$||(unnest).col_name||$$ $$||
2517
util.qual_name((unnest).type), ''), '')
2518
FROM unnest($1)
2519
$_$;
2520

    
2521

    
2522
--
2523
-- Name: mk_search_path(text[]); Type: FUNCTION; Schema: util; Owner: -
2524
--
2525

    
2526
CREATE FUNCTION mk_search_path(VARIADIC schemas text[]) RETURNS text
2527
    LANGUAGE sql IMMUTABLE
2528
    AS $_$
2529
SELECT string_agg(quote_ident(unnest), ', ') FROM unnest($1||'util'::text)
2530
$_$;
2531

    
2532

    
2533
--
2534
-- Name: FUNCTION mk_search_path(VARIADIC schemas text[]); Type: COMMENT; Schema: util; Owner: -
2535
--
2536

    
2537
COMMENT ON FUNCTION mk_search_path(VARIADIC schemas text[]) IS '
2538
auto-appends util to the search_path to enable use of util operators
2539
';
2540

    
2541

    
2542
--
2543
-- Name: mk_set_search_path(boolean); Type: FUNCTION; Schema: util; Owner: -
2544
--
2545

    
2546
CREATE FUNCTION mk_set_search_path(for_printing boolean DEFAULT false) RETURNS text
2547
    LANGUAGE sql IMMUTABLE
2548
    AS $_$
2549
SELECT util.mk_set_search_path(current_setting('search_path'), $1)
2550
$_$;
2551

    
2552

    
2553
--
2554
-- Name: mk_set_search_path(text, boolean); Type: FUNCTION; Schema: util; Owner: -
2555
--
2556

    
2557
CREATE FUNCTION mk_set_search_path(search_path text, for_printing boolean DEFAULT false) RETURNS text
2558
    LANGUAGE sql IMMUTABLE
2559
    AS $_$
2560
/* debug_print_return_value() needed because this function is used with EXECUTE
2561
rather than util.eval() (in order to affect the calling function), so the
2562
search_path would not otherwise be printed */
2563
SELECT $$SET$$||util._if($2, $$ /*LOCAL*/$$::text, $$ LOCAL$$)
2564
||$$ search_path TO $$||$1
2565
$_$;
2566

    
2567

    
2568
--
2569
-- Name: mk_source_col(regclass); Type: FUNCTION; Schema: util; Owner: -
2570
--
2571

    
2572
CREATE FUNCTION mk_source_col(table_ regclass) RETURNS void
2573
    LANGUAGE sql
2574
    AS $_$
2575
SELECT util.mk_const_col(($1, 'source'), util.schema($1))
2576
$_$;
2577

    
2578

    
2579
--
2580
-- Name: FUNCTION mk_source_col(table_ regclass); Type: COMMENT; Schema: util; Owner: -
2581
--
2582

    
2583
COMMENT ON FUNCTION mk_source_col(table_ regclass) IS '
2584
idempotent
2585
';
2586

    
2587

    
2588
--
2589
-- Name: mk_subset_by_row_num_func(regclass); Type: FUNCTION; Schema: util; Owner: -
2590
--
2591

    
2592
CREATE FUNCTION mk_subset_by_row_num_func(view_ regclass) RETURNS void
2593
    LANGUAGE plpgsql STRICT
2594
    AS $_$
2595
DECLARE
2596
	view_qual_name text = util.qual_name(view_);
2597
BEGIN
2598
	EXECUTE $$
2599
CREATE OR REPLACE FUNCTION $$||view_||$$(limit_ integer DEFAULT NULL::integer, offset_ integer DEFAULT NULL)
2600
  RETURNS SETOF $$||view_||$$ AS
2601
$BODY1$
2602
SELECT * FROM $$||view_qual_name||$$
2603
ORDER BY sort_col
2604
LIMIT $1 OFFSET $2
2605
$BODY1$
2606
  LANGUAGE sql STABLE
2607
  COST 100
2608
  ROWS 1000
2609
$$;
2610
	
2611
	PERFORM util.mk_subset_by_row_num_no_sort_func(view_);
2612
END;
2613
$_$;
2614

    
2615

    
2616
--
2617
-- Name: mk_subset_by_row_num_func(regclass, text); Type: FUNCTION; Schema: util; Owner: -
2618
--
2619

    
2620
CREATE FUNCTION mk_subset_by_row_num_func(view_ regclass, row_num_col text) RETURNS void
2621
    LANGUAGE plpgsql STRICT
2622
    AS $_$
2623
DECLARE
2624
	view_qual_name text = util.qual_name(view_);
2625
	row_num__min__fn text = util.esc_name__append('__row_num__min', view_qual_name);
2626
BEGIN
2627
	EXECUTE $$
2628
CREATE OR REPLACE FUNCTION $$||row_num__min__fn||$$()
2629
  RETURNS integer AS
2630
$BODY1$
2631
SELECT $$||quote_ident(row_num_col)||$$
2632
FROM $$||view_qual_name||$$
2633
ORDER BY $$||quote_ident(row_num_col)||$$ ASC
2634
LIMIT 1
2635
$BODY1$
2636
  LANGUAGE sql STABLE
2637
  COST 100;
2638
$$;
2639
	
2640
	EXECUTE $$
2641
CREATE OR REPLACE FUNCTION $$||view_||$$(limit_ integer DEFAULT NULL::integer, offset_ integer DEFAULT NULL)
2642
  RETURNS SETOF $$||view_||$$ AS
2643
$BODY1$
2644
SELECT * FROM $$||view_qual_name||$$
2645
WHERE $$||quote_ident(row_num_col)||$$ BETWEEN
2646
	util.offset2row_num(    $2, $$||row_num__min__fn||$$())
2647
AND util.limit2row_num ($1, $2, $$||row_num__min__fn||$$())
2648
ORDER BY $$||quote_ident(row_num_col)||$$
2649
$BODY1$
2650
  LANGUAGE sql STABLE
2651
  COST 100
2652
  ROWS 1000
2653
$$;
2654
	
2655
	PERFORM util.mk_subset_by_row_num_no_sort_func(view_);
2656
END;
2657
$_$;
2658

    
2659

    
2660
--
2661
-- Name: mk_subset_by_row_num_no_sort_func(regclass); Type: FUNCTION; Schema: util; Owner: -
2662
--
2663

    
2664
CREATE FUNCTION mk_subset_by_row_num_no_sort_func(view_ regclass) RETURNS void
2665
    LANGUAGE plpgsql STRICT
2666
    AS $_$
2667
DECLARE
2668
	view_qual_name text = util.qual_name(view_);
2669
BEGIN
2670
	EXECUTE $$
2671
CREATE OR REPLACE FUNCTION $$||view_||$$(no_sort boolean, limit_ integer DEFAULT NULL::integer, offset_ integer DEFAULT NULL)
2672
  RETURNS SETOF $$||view_||$$
2673
  SET enable_sort TO 'off'
2674
  AS
2675
$BODY1$
2676
SELECT * FROM $$||view_qual_name||$$($2, $3)
2677
$BODY1$
2678
  LANGUAGE sql STABLE
2679
  COST 100
2680
  ROWS 1000
2681
;
2682
COMMENT ON FUNCTION $$||view_||$$(no_sort boolean, limit_ integer, offset_ integer) IS '
2683
Use this for limit values greater than ~100,000 to avoid unwanted slow sorts.
2684
If you want to run EXPLAIN and get expanded output, use the regular subset
2685
function instead. (When a config param is set on a function, EXPLAIN produces
2686
just a function scan.)
2687
';
2688
$$;
2689
END;
2690
$_$;
2691

    
2692

    
2693
--
2694
-- Name: FUNCTION mk_subset_by_row_num_no_sort_func(view_ regclass); Type: COMMENT; Schema: util; Owner: -
2695
--
2696

    
2697
COMMENT ON FUNCTION mk_subset_by_row_num_no_sort_func(view_ regclass) IS '
2698
creates subset function which turns off enable_sort
2699
';
2700

    
2701

    
2702
--
2703
-- Name: mk_typed_cols_list(col_cast[]); Type: FUNCTION; Schema: util; Owner: -
2704
--
2705

    
2706
CREATE FUNCTION mk_typed_cols_list(cols col_cast[]) RETURNS text
2707
    LANGUAGE sql IMMUTABLE
2708
    AS $_$
2709
SELECT COALESCE(string_agg(quote_ident((unnest).col_name)||$$ $$||
2710
util.qual_name((unnest).type), ', '), '')
2711
FROM unnest($1)
2712
$_$;
2713

    
2714

    
2715
--
2716
-- Name: name(regclass); Type: FUNCTION; Schema: util; Owner: -
2717
--
2718

    
2719
CREATE FUNCTION name(table_ regclass) RETURNS text
2720
    LANGUAGE sql STABLE
2721
    AS $_$
2722
SELECT relname::text FROM pg_class WHERE oid = $1
2723
$_$;
2724

    
2725

    
2726
--
2727
-- Name: name(regtype); Type: FUNCTION; Schema: util; Owner: -
2728
--
2729

    
2730
CREATE FUNCTION name(type regtype) RETURNS text
2731
    LANGUAGE sql STABLE
2732
    AS $_$
2733
SELECT typname::text FROM pg_type WHERE oid = $1
2734
$_$;
2735

    
2736

    
2737
--
2738
-- Name: name_was_truncated(text, integer); Type: FUNCTION; Schema: util; Owner: -
2739
--
2740

    
2741
CREATE FUNCTION name_was_truncated(name_ text, max_prefix_len integer DEFAULT 0) RETURNS boolean
2742
    LANGUAGE sql IMMUTABLE
2743
    AS $_$
2744
SELECT octet_length($1) >= util.namedatalen() - $2
2745
$_$;
2746

    
2747

    
2748
--
2749
-- Name: namedatalen(); Type: FUNCTION; Schema: util; Owner: -
2750
--
2751

    
2752
CREATE FUNCTION namedatalen() RETURNS integer
2753
    LANGUAGE sql IMMUTABLE
2754
    AS $$
2755
SELECT octet_length(repeat('_', 1024/*>63*/)::name::text)
2756
$$;
2757

    
2758

    
2759
--
2760
-- Name: not_empty(anyarray); Type: FUNCTION; Schema: util; Owner: -
2761
--
2762

    
2763
CREATE FUNCTION not_empty(value anyarray) RETURNS boolean
2764
    LANGUAGE sql IMMUTABLE
2765
    AS $_$
2766
SELECT $1 IS NOT NULL AND util.array_length($1) > 0
2767
$_$;
2768

    
2769

    
2770
--
2771
-- Name: not_null(anyelement); Type: FUNCTION; Schema: util; Owner: -
2772
--
2773

    
2774
CREATE FUNCTION not_null(value anyelement) RETURNS boolean
2775
    LANGUAGE sql IMMUTABLE
2776
    AS $_$
2777
SELECT $1 IS NOT NULL
2778
$_$;
2779

    
2780

    
2781
--
2782
-- Name: nulls_map(text[]); Type: FUNCTION; Schema: util; Owner: -
2783
--
2784

    
2785
CREATE FUNCTION nulls_map(nulls text[]) RETURNS hstore
2786
    LANGUAGE sql IMMUTABLE
2787
    AS $_$
2788
SELECT util.hstore($1, NULL) || '*=>*'
2789
$_$;
2790

    
2791

    
2792
--
2793
-- Name: FUNCTION nulls_map(nulls text[]); Type: COMMENT; Schema: util; Owner: -
2794
--
2795

    
2796
COMMENT ON FUNCTION nulls_map(nulls text[]) IS '
2797
for use with _map()
2798
';
2799

    
2800

    
2801
--
2802
-- Name: offset2row_num(integer, integer); Type: FUNCTION; Schema: util; Owner: -
2803
--
2804

    
2805
CREATE FUNCTION offset2row_num(offset_ integer, min_row_num integer DEFAULT 1) RETURNS integer
2806
    LANGUAGE sql IMMUTABLE
2807
    AS $_$
2808
SELECT $2 + COALESCE($1, 0)
2809
$_$;
2810

    
2811

    
2812
--
2813
-- Name: parent(regclass); Type: FUNCTION; Schema: util; Owner: -
2814
--
2815

    
2816
CREATE FUNCTION parent(table_ regclass) RETURNS regclass
2817
    LANGUAGE sql STABLE
2818
    AS $_$
2819
SELECT inhparent FROM pg_inherits WHERE inhrelid = $1
2820
$_$;
2821

    
2822

    
2823
--
2824
-- Name: populate_table(regclass, text); Type: FUNCTION; Schema: util; Owner: -
2825
--
2826

    
2827
CREATE FUNCTION populate_table(table_ regclass, sql text) RETURNS void
2828
    LANGUAGE sql
2829
    AS $_$
2830
SELECT util.eval($$INSERT INTO $$||$1||$$
2831
$$||util.ltrim_nl($2));
2832
-- make sure the created table has the correct estimated row count
2833
SELECT util.analyze_($1);
2834
$_$;
2835

    
2836

    
2837
--
2838
-- Name: prefixed_name(text, anyelement); Type: FUNCTION; Schema: util; Owner: -
2839
--
2840

    
2841
CREATE FUNCTION prefixed_name(prefix text, type anyelement) RETURNS text
2842
    LANGUAGE sql IMMUTABLE
2843
    AS $_$
2844
SELECT util.qual_name(util.schema($2), $1||util.name($2))
2845
$_$;
2846

    
2847

    
2848
--
2849
-- Name: prepend_comment(regclass, text); Type: FUNCTION; Schema: util; Owner: -
2850
--
2851

    
2852
CREATE FUNCTION prepend_comment(table_ regclass, comment text) RETURNS void
2853
    LANGUAGE sql
2854
    AS $_$
2855
SELECT util.set_comment($1, concat($2, util.comment($1)))
2856
$_$;
2857

    
2858

    
2859
--
2860
-- Name: FUNCTION prepend_comment(table_ regclass, comment text); Type: COMMENT; Schema: util; Owner: -
2861
--
2862

    
2863
COMMENT ON FUNCTION prepend_comment(table_ regclass, comment text) IS '
2864
comment: must start and end with a newline
2865
';
2866

    
2867

    
2868
--
2869
-- Name: qual_name(text[]); Type: FUNCTION; Schema: util; Owner: -
2870
--
2871

    
2872
CREATE FUNCTION qual_name(VARIADIC elems text[]) RETURNS text
2873
    LANGUAGE sql IMMUTABLE
2874
    AS $_$
2875
SELECT string_agg(quote_ident(unnest), '.') FROM unnest($1)
2876
$_$;
2877

    
2878

    
2879
--
2880
-- Name: qual_name(regclass); Type: FUNCTION; Schema: util; Owner: -
2881
--
2882

    
2883
CREATE FUNCTION qual_name(table_ regclass) RETURNS text
2884
    LANGUAGE sql STABLE
2885
    SET search_path TO pg_temp
2886
    AS $_$
2887
SELECT $1::text
2888
$_$;
2889

    
2890

    
2891
--
2892
-- Name: qual_name(regtype); Type: FUNCTION; Schema: util; Owner: -
2893
--
2894

    
2895
CREATE FUNCTION qual_name(type regtype) RETURNS text
2896
    LANGUAGE sql STABLE
2897
    SET search_path TO pg_temp
2898
    AS $_$
2899
SELECT $1::text
2900
$_$;
2901

    
2902

    
2903
--
2904
-- Name: FUNCTION qual_name(type regtype); Type: COMMENT; Schema: util; Owner: -
2905
--
2906

    
2907
COMMENT ON FUNCTION qual_name(type regtype) IS '
2908
a type''s schema-qualified name
2909
';
2910

    
2911

    
2912
--
2913
-- Name: qual_name(unknown); Type: FUNCTION; Schema: util; Owner: -
2914
--
2915

    
2916
CREATE FUNCTION qual_name(type unknown) RETURNS text
2917
    LANGUAGE sql STABLE
2918
    AS $_$
2919
SELECT util.qual_name($1::text::regtype)
2920
$_$;
2921

    
2922

    
2923
--
2924
-- Name: quote_func_call(regprocedure, text[]); Type: FUNCTION; Schema: util; Owner: -
2925
--
2926

    
2927
CREATE FUNCTION quote_func_call(func regprocedure, VARIADIC args_esc text[]) RETURNS text
2928
    LANGUAGE sql IMMUTABLE
2929
    AS $_$
2930
SELECT util.quote_func_call($1::regproc::text, VARIADIC $2)
2931
$_$;
2932

    
2933

    
2934
--
2935
-- Name: quote_func_call(text, text[]); Type: FUNCTION; Schema: util; Owner: -
2936
--
2937

    
2938
CREATE FUNCTION quote_func_call(func_esc text, VARIADIC args_esc text[]) RETURNS text
2939
    LANGUAGE sql IMMUTABLE
2940
    AS $_$
2941
SELECT $1||'('||concat_ws(', ', VARIADIC $2)||')'
2942
$_$;
2943

    
2944

    
2945
--
2946
-- Name: quote_typed(anyelement); Type: FUNCTION; Schema: util; Owner: -
2947
--
2948

    
2949
CREATE FUNCTION quote_typed(value anyelement) RETURNS text
2950
    LANGUAGE sql IMMUTABLE
2951
    AS $_$
2952
SELECT quote_nullable($1)||$$::$$||util.qual_name(pg_typeof($1))
2953
$_$;
2954

    
2955

    
2956
--
2957
-- Name: raise(text, text); Type: FUNCTION; Schema: util; Owner: -
2958
--
2959

    
2960
CREATE FUNCTION raise(type text, msg text) RETURNS void
2961
    LANGUAGE sql IMMUTABLE
2962
    AS $_X$
2963
SELECT util.eval($$
2964
CREATE OR REPLACE FUNCTION pg_temp.__raise()
2965
  RETURNS void AS
2966
-- $__BODY1$ in case msg contains $BODY1$ (in SQL)
2967
$__BODY1$
2968
BEGIN
2969
	RAISE $$||$1||$$ USING MESSAGE = $$||quote_nullable($2)||$$;
2970
END;
2971
$__BODY1$
2972
  LANGUAGE plpgsql IMMUTABLE
2973
  COST 100;
2974
$$, verbose_ := false);
2975

    
2976
SELECT util.eval($$SELECT pg_temp.__raise()$$, verbose_ := false);
2977
$_X$;
2978

    
2979

    
2980
--
2981
-- Name: FUNCTION raise(type text, msg text); Type: COMMENT; Schema: util; Owner: -
2982
--
2983

    
2984
COMMENT ON FUNCTION raise(type text, msg text) IS '
2985
type: a log level from
2986
http://www.postgresql.org/docs/9.3/static/plpgsql-errors-and-messages.html
2987
or a condition name from
2988
http://www.postgresql.org/docs/9.3/static/errcodes-appendix.html
2989
';
2990

    
2991

    
2992
--
2993
-- Name: raise_error_warning(text); Type: FUNCTION; Schema: util; Owner: -
2994
--
2995

    
2996
CREATE FUNCTION raise_error_warning(msg text) RETURNS void
2997
    LANGUAGE sql IMMUTABLE
2998
    AS $_$
2999
SELECT util.raise('WARNING', 'ERROR:  '||$1)
3000
$_$;
3001

    
3002

    
3003
--
3004
-- Name: raise_undefined_column(col_ref); Type: FUNCTION; Schema: util; Owner: -
3005
--
3006

    
3007
CREATE FUNCTION raise_undefined_column(col col_ref) RETURNS text
3008
    LANGUAGE plpgsql IMMUTABLE STRICT
3009
    AS $$
3010
BEGIN
3011
	RAISE undefined_column USING MESSAGE = concat('undefined column: ', col.name);
3012
END;
3013
$$;
3014

    
3015

    
3016
--
3017
-- Name: regexp_matches_group(text, text, integer); Type: FUNCTION; Schema: util; Owner: -
3018
--
3019

    
3020
CREATE FUNCTION regexp_matches_group(str text, re text, group_ integer DEFAULT 1) RETURNS SETOF text
3021
    LANGUAGE sql IMMUTABLE
3022
    AS $_$
3023
SELECT regexp_matches[$3] FROM regexp_matches($1, $2, 'g')
3024
$_$;
3025

    
3026

    
3027
--
3028
-- Name: regexp_quote(text); Type: FUNCTION; Schema: util; Owner: -
3029
--
3030

    
3031
CREATE FUNCTION regexp_quote(str text) RETURNS text
3032
    LANGUAGE sql IMMUTABLE
3033
    AS $_$
3034
SELECT regexp_replace($1, '\W', /*\char*/'\\\&', 'g')
3035
$_$;
3036

    
3037

    
3038
--
3039
-- Name: regprocedure(text); Type: FUNCTION; Schema: util; Owner: -
3040
--
3041

    
3042
CREATE FUNCTION regprocedure(func text) RETURNS regprocedure
3043
    LANGUAGE sql IMMUTABLE
3044
    AS $_$
3045
SELECT (CASE WHEN right($1, 1) = ')'
3046
THEN $1::regprocedure ELSE $1::regproc::regprocedure END)
3047
$_$;
3048

    
3049

    
3050
--
3051
-- Name: relation_type(regclass); Type: FUNCTION; Schema: util; Owner: -
3052
--
3053

    
3054
CREATE FUNCTION relation_type(relation regclass) RETURNS text
3055
    LANGUAGE sql STABLE
3056
    AS $_$
3057
SELECT util.relation_type(util.relation_type_char($1))
3058
$_$;
3059

    
3060

    
3061
--
3062
-- Name: relation_type("char"); Type: FUNCTION; Schema: util; Owner: -
3063
--
3064

    
3065
CREATE FUNCTION relation_type(relation_type_char "char") RETURNS text
3066
    LANGUAGE sql IMMUTABLE
3067
    AS $_$
3068
SELECT 'c=>TYPE, r=>TABLE, v=>VIEW'::hstore -> $1
3069
$_$;
3070

    
3071

    
3072
--
3073
-- Name: relation_type(regtype); Type: FUNCTION; Schema: util; Owner: -
3074
--
3075

    
3076
CREATE FUNCTION relation_type(type regtype) RETURNS text
3077
    LANGUAGE sql IMMUTABLE
3078
    AS $$
3079
SELECT 'TYPE'::text
3080
$$;
3081

    
3082

    
3083
--
3084
-- Name: relation_type_char(regclass); Type: FUNCTION; Schema: util; Owner: -
3085
--
3086

    
3087
CREATE FUNCTION relation_type_char(relation regclass) RETURNS "char"
3088
    LANGUAGE sql STABLE
3089
    AS $_$
3090
SELECT relkind FROM pg_class WHERE oid = $1
3091
$_$;
3092

    
3093

    
3094
--
3095
-- Name: remake_diff_table(text, regclass, regclass, text); Type: FUNCTION; Schema: util; Owner: -
3096
--
3097

    
3098
CREATE FUNCTION remake_diff_table(diff_table text, left_table regclass, right_table regclass, type_table text) RETURNS void
3099
    LANGUAGE sql
3100
    AS $_$
3101
/* can't have in_table/out_table inherit from *each other*, because inheritance
3102
also causes the rows of the parent table to be included in the child table.
3103
instead, they need to inherit from a common, empty table. */
3104
SELECT util.create_if_not_exists($$SELECT $$||util.quote_func_call(
3105
'util.copy_struct', util.quote_typed($2), util.quote_typed($4)));
3106
SELECT util.rm_freq(ARRAY[$4]); -- left/right_table don't have freq yet
3107
SELECT util.inherit($2, $4);
3108
SELECT util.inherit($3, $4);
3109

    
3110
SELECT util.rematerialize_query($1, $$
3111
SELECT * FROM util.diff(
3112
  $$||util.quote_typed($2)||$$
3113
, $$||util.quote_typed($3)||$$
3114
, NULL::$$||$4||$$)
3115
$$);
3116

    
3117
/* the table unfortunately cannot be *materialized* in human-readable form,
3118
because this would create column name collisions between the two sides */
3119
SELECT util.prepend_comment($1, '
3120
to view this table in human-readable form (with each side''s tuple column
3121
expanded to its component fields):
3122
SELECT (left_).*, ('||util.schema($4::regclass)||'.values_(right_)).* FROM '||$1||';
3123

    
3124
to display NULL values that are extra or missing:
3125
SELECT * FROM '||$1||';
3126
');
3127
$_$;
3128

    
3129

    
3130
--
3131
-- Name: FUNCTION remake_diff_table(diff_table text, left_table regclass, right_table regclass, type_table text); Type: COMMENT; Schema: util; Owner: -
3132
--
3133

    
3134
COMMENT ON FUNCTION remake_diff_table(diff_table text, left_table regclass, right_table regclass, type_table text) IS '
3135
type_table (*required*): table to create as the shared base type
3136
';
3137

    
3138

    
3139
--
3140
-- Name: rematerialize_query(text, text); Type: FUNCTION; Schema: util; Owner: -
3141
--
3142

    
3143
CREATE FUNCTION rematerialize_query(table_esc text, sql text) RETURNS void
3144
    LANGUAGE sql
3145
    AS $_$
3146
SELECT util.drop_table($1);
3147
SELECT util.materialize_query($1, $2);
3148
$_$;
3149

    
3150

    
3151
--
3152
-- Name: FUNCTION rematerialize_query(table_esc text, sql text); Type: COMMENT; Schema: util; Owner: -
3153
--
3154

    
3155
COMMENT ON FUNCTION rematerialize_query(table_esc text, sql text) IS '
3156
idempotent, but repeats action each time
3157
';
3158

    
3159

    
3160
--
3161
-- Name: rematerialize_view(text, regclass); Type: FUNCTION; Schema: util; Owner: -
3162
--
3163

    
3164
CREATE FUNCTION rematerialize_view(table_esc text, view_ regclass) RETURNS void
3165
    LANGUAGE sql
3166
    AS $_$
3167
SELECT util.drop_table($1);
3168
SELECT util.materialize_view($1, $2);
3169
$_$;
3170

    
3171

    
3172
--
3173
-- Name: FUNCTION rematerialize_view(table_esc text, view_ regclass); Type: COMMENT; Schema: util; Owner: -
3174
--
3175

    
3176
COMMENT ON FUNCTION rematerialize_view(table_esc text, view_ regclass) IS '
3177
idempotent, but repeats action each time
3178
';
3179

    
3180

    
3181
--
3182
-- Name: rename_cols(regclass, anyelement); Type: FUNCTION; Schema: util; Owner: -
3183
--
3184

    
3185
CREATE FUNCTION rename_cols(table_ regclass, renames anyelement) RETURNS void
3186
    LANGUAGE sql
3187
    AS $_$
3188
SELECT util.try_create($$ALTER TABLE $$||$1||$$ RENAME $$
3189
||quote_ident(name)||$$ TO $$||quote_ident($2 -> name))
3190
FROM util.col_names($1::text::regtype) f (name);
3191
SELECT NULL::void; -- don't fold away functions called in previous query
3192
$_$;
3193

    
3194

    
3195
--
3196
-- Name: FUNCTION rename_cols(table_ regclass, renames anyelement); Type: COMMENT; Schema: util; Owner: -
3197
--
3198

    
3199
COMMENT ON FUNCTION rename_cols(table_ regclass, renames anyelement) IS '
3200
idempotent
3201
';
3202

    
3203

    
3204
--
3205
-- Name: rename_relation(regclass, text); Type: FUNCTION; Schema: util; Owner: -
3206
--
3207

    
3208
CREATE FUNCTION rename_relation(from_ regclass, to_ text) RETURNS void
3209
    LANGUAGE sql
3210
    AS $_$
3211
/* use util.qual_name() instead of ::text so that the schema qualifier is always
3212
included in the debug SQL */
3213
SELECT util.rename_relation(util.qual_name($1), $2)
3214
$_$;
3215

    
3216

    
3217
--
3218
-- Name: rename_relation(text, text); Type: FUNCTION; Schema: util; Owner: -
3219
--
3220

    
3221
CREATE FUNCTION rename_relation(from_esc text, to_name text) RETURNS void
3222
    LANGUAGE sql
3223
    AS $_$
3224
/* 'ALTER TABLE can be used with views too'
3225
(http://www.postgresql.org/docs/9.3/static/sql-alterview.html) */
3226
SELECT util.eval($$ALTER TABLE IF EXISTS $$||$1||$$ RENAME TO $$
3227
||quote_ident($2))
3228
$_$;
3229

    
3230

    
3231
--
3232
-- Name: FUNCTION rename_relation(from_esc text, to_name text); Type: COMMENT; Schema: util; Owner: -
3233
--
3234

    
3235
COMMENT ON FUNCTION rename_relation(from_esc text, to_name text) IS '
3236
idempotent
3237
';
3238

    
3239

    
3240
--
3241
-- Name: replace_suffix(text, text, text, integer); Type: FUNCTION; Schema: util; Owner: -
3242
--
3243

    
3244
CREATE FUNCTION replace_suffix(str text, old_suffix text, new_suffix text, max_prefix_len integer DEFAULT 0) RETURNS text
3245
    LANGUAGE sql IMMUTABLE
3246
    AS $_$
3247
SELECT regexp_replace($1, util.truncated_prefixed_name_regexp($2, $4), '\1'||$3)
3248
$_$;
3249

    
3250

    
3251
--
3252
-- Name: FUNCTION replace_suffix(str text, old_suffix text, new_suffix text, max_prefix_len integer); Type: COMMENT; Schema: util; Owner: -
3253
--
3254

    
3255
COMMENT ON FUNCTION replace_suffix(str text, old_suffix text, new_suffix text, max_prefix_len integer) IS '
3256
max_prefix_len: when str may have been truncated (eg. as a table name) due to the prepending of a prefix, support prefixes up to this length 
3257
';
3258

    
3259

    
3260
--
3261
-- Name: reset_col_names(regclass, regclass); Type: FUNCTION; Schema: util; Owner: -
3262
--
3263

    
3264
CREATE FUNCTION reset_col_names(table_ regclass, names regclass) RETURNS void
3265
    LANGUAGE sql
3266
    AS $_$
3267
SELECT util.eval($$DELETE FROM $$||$2||$$ WHERE "from" LIKE ':%'$$);
3268
SELECT util.mk_derived_col(($2, 'to'), $$"from"$$, overwrite := true);
3269
SELECT util.set_col_names($1, $2);
3270
$_$;
3271

    
3272

    
3273
--
3274
-- Name: FUNCTION reset_col_names(table_ regclass, names regclass); Type: COMMENT; Schema: util; Owner: -
3275
--
3276

    
3277
COMMENT ON FUNCTION reset_col_names(table_ regclass, names regclass) IS '
3278
idempotent.
3279
alters the names table, so it will need to be repopulated after running this function.
3280
';
3281

    
3282

    
3283
--
3284
-- Name: reset_map_table(text); Type: FUNCTION; Schema: util; Owner: -
3285
--
3286

    
3287
CREATE FUNCTION reset_map_table(table_ text) RETURNS void
3288
    LANGUAGE sql
3289
    AS $_$
3290
SELECT util.drop_table($1);
3291
SELECT util.mk_map_table($1);
3292
$_$;
3293

    
3294

    
3295
--
3296
-- Name: rm_freq(regclass[], text); Type: FUNCTION; Schema: util; Owner: -
3297
--
3298

    
3299
CREATE FUNCTION rm_freq(tables regclass[], freq_col text DEFAULT 'copies'::text) RETURNS void
3300
    LANGUAGE sql
3301
    AS $_$
3302
SELECT util.drop_column($1, $2, force := true)
3303
$_$;
3304

    
3305

    
3306
--
3307
-- Name: rtrim_n(text, integer); Type: FUNCTION; Schema: util; Owner: -
3308
--
3309

    
3310
CREATE FUNCTION rtrim_n(str text, count integer) RETURNS text
3311
    LANGUAGE sql IMMUTABLE
3312
    AS $_$
3313
SELECT (CASE WHEN $2 <= 0 THEN $1 ELSE left($1, -$2) END)
3314
$_$;
3315

    
3316

    
3317
--
3318
-- Name: runnable_sql(text); Type: FUNCTION; Schema: util; Owner: -
3319
--
3320

    
3321
CREATE FUNCTION runnable_sql(sql text) RETURNS text
3322
    LANGUAGE sql IMMUTABLE
3323
    AS $_$
3324
SELECT (CASE WHEN util.is_set_stmt($1) THEN ''
3325
ELSE util.mk_set_search_path(for_printing := true)||$$;
3326
$$ END)||$1
3327
$_$;
3328

    
3329

    
3330
--
3331
-- Name: save_drop_view(text); Type: FUNCTION; Schema: util; Owner: -
3332
--
3333

    
3334
CREATE FUNCTION save_drop_view(view_ text) RETURNS text
3335
    LANGUAGE plpgsql STRICT
3336
    AS $$
3337
DECLARE
3338
	result text = NULL;
3339
BEGIN
3340
	BEGIN
3341
		result = util.show_create_view(view_, replace := false);
3342
			/* replace: no `OR REPLACE` because that causes nonuniform errors
3343
			(eg. invalid_table_definition), instead of the standard
3344
			duplicate_table exception caught by util.create_if_not_exists() */
3345
		PERFORM util.drop_view(view_);
3346
	EXCEPTION
3347
		WHEN undefined_table THEN NULL;
3348
	END;
3349
	RETURN result;
3350
END;
3351
$$;
3352

    
3353

    
3354
--
3355
-- Name: save_drop_views(text[]); Type: FUNCTION; Schema: util; Owner: -
3356
--
3357

    
3358
CREATE FUNCTION save_drop_views(views text[]) RETURNS text[]
3359
    LANGUAGE sql
3360
    AS $_$
3361
SELECT array(SELECT util.save_drop_view(view_) FROM unnest($1) view_)
3362
$_$;
3363

    
3364

    
3365
--
3366
-- Name: schema(oid); Type: FUNCTION; Schema: util; Owner: -
3367
--
3368

    
3369
CREATE FUNCTION schema(pg_namespace_oid oid) RETURNS text
3370
    LANGUAGE sql STABLE
3371
    AS $_$
3372
SELECT nspname::text FROM pg_namespace WHERE pg_namespace.oid = $1
3373
$_$;
3374

    
3375

    
3376
--
3377
-- Name: schema(regclass); Type: FUNCTION; Schema: util; Owner: -
3378
--
3379

    
3380
CREATE FUNCTION schema(table_ regclass) RETURNS text
3381
    LANGUAGE sql STABLE
3382
    AS $_$
3383
SELECT util.schema(relnamespace) FROM pg_class WHERE oid = $1
3384
$_$;
3385

    
3386

    
3387
--
3388
-- Name: schema(regtype); Type: FUNCTION; Schema: util; Owner: -
3389
--
3390

    
3391
CREATE FUNCTION schema(type regtype) RETURNS text
3392
    LANGUAGE sql STABLE
3393
    AS $_$
3394
SELECT util.schema(typnamespace) FROM pg_type WHERE oid = $1
3395
$_$;
3396

    
3397

    
3398
--
3399
-- Name: schema(anyelement); Type: FUNCTION; Schema: util; Owner: -
3400
--
3401

    
3402
CREATE FUNCTION schema(type_null anyelement) RETURNS text
3403
    LANGUAGE sql STABLE
3404
    AS $_$
3405
SELECT util.schema(pg_typeof($1))
3406
$_$;
3407

    
3408

    
3409
--
3410
-- Name: schema_bundle_get_schemas(text); Type: FUNCTION; Schema: util; Owner: -
3411
--
3412

    
3413
CREATE FUNCTION schema_bundle_get_schemas(schema_bundle text) RETURNS SETOF text
3414
    LANGUAGE sql STABLE
3415
    AS $_$
3416
SELECT nspname::text FROM pg_namespace WHERE nspname ~ ('^'||$1||'(?=\y|_)')
3417
$_$;
3418

    
3419

    
3420
--
3421
-- Name: FUNCTION schema_bundle_get_schemas(schema_bundle text); Type: COMMENT; Schema: util; Owner: -
3422
--
3423

    
3424
COMMENT ON FUNCTION schema_bundle_get_schemas(schema_bundle text) IS '
3425
a schema bundle is a group of schemas with a common prefix
3426
';
3427

    
3428

    
3429
--
3430
-- Name: schema_bundle_rename(text, text); Type: FUNCTION; Schema: util; Owner: -
3431
--
3432

    
3433
CREATE FUNCTION schema_bundle_rename(old text, new text) RETURNS void
3434
    LANGUAGE sql
3435
    AS $_$
3436
SELECT util.schema_rename(old_schema,
3437
	overlay(old_schema placing new from 1 for length(old))) -- replace prefix
3438
FROM util.schema_bundle_get_schemas($1) f (old_schema);
3439
SELECT NULL::void; -- don't fold away functions called in previous query
3440
$_$;
3441

    
3442

    
3443
--
3444
-- Name: schema_bundle_replace(text, text); Type: FUNCTION; Schema: util; Owner: -
3445
--
3446

    
3447
CREATE FUNCTION schema_bundle_replace(replace text, with_ text) RETURNS void
3448
    LANGUAGE plpgsql
3449
    AS $$
3450
BEGIN
3451
	-- don't schema_bundle_rm() the schema_bundle to keep!
3452
	IF replace = with_ THEN RETURN; END IF;
3453
	
3454
	PERFORM util.schema_bundle_rm(replace);
3455
	PERFORM util.schema_bundle_rename(with_, replace);
3456
END;
3457
$$;
3458

    
3459

    
3460
--
3461
-- Name: schema_bundle_rm(text); Type: FUNCTION; Schema: util; Owner: -
3462
--
3463

    
3464
CREATE FUNCTION schema_bundle_rm(schema_bundle text) RETURNS void
3465
    LANGUAGE sql
3466
    AS $_$
3467
SELECT util.schema_rm(schema)
3468
FROM util.schema_bundle_get_schemas($1) f (schema);
3469
SELECT NULL::void; -- don't fold away functions called in previous query
3470
$_$;
3471

    
3472

    
3473
--
3474
-- Name: schema_esc(anyelement); Type: FUNCTION; Schema: util; Owner: -
3475
--
3476

    
3477
CREATE FUNCTION schema_esc(type_null anyelement) RETURNS text
3478
    LANGUAGE sql STABLE
3479
    AS $_$
3480
SELECT quote_ident(util.schema($1))
3481
$_$;
3482

    
3483

    
3484
--
3485
-- Name: schema_matches(text, text); Type: FUNCTION; Schema: util; Owner: -
3486
--
3487

    
3488
CREATE FUNCTION schema_matches(schema text, schema_regexp text) RETURNS boolean
3489
    LANGUAGE sql IMMUTABLE
3490
    AS $_$
3491
SELECT $1 ~ $2 AND /*in userspace*/$1 !~ '^(?:information_schema|pg_.*)$'
3492
$_$;
3493

    
3494

    
3495
--
3496
-- Name: schema_oid(text); Type: FUNCTION; Schema: util; Owner: -
3497
--
3498

    
3499
CREATE FUNCTION schema_oid(schema text) RETURNS oid
3500
    LANGUAGE sql STABLE
3501
    AS $_$
3502
SELECT oid FROM pg_namespace WHERE nspname = $1
3503
$_$;
3504

    
3505

    
3506
--
3507
-- Name: schema_regexp(regclass); Type: FUNCTION; Schema: util; Owner: -
3508
--
3509

    
3510
CREATE FUNCTION schema_regexp(relation regclass) RETURNS text
3511
    LANGUAGE sql IMMUTABLE
3512
    AS $_$
3513
SELECT util.schema_regexp(schema_anchor := $1)
3514
$_$;
3515

    
3516

    
3517
--
3518
-- Name: schema_regexp(anyelement); Type: FUNCTION; Schema: util; Owner: -
3519
--
3520

    
3521
CREATE FUNCTION schema_regexp(schema_anchor anyelement) RETURNS text
3522
    LANGUAGE sql IMMUTABLE
3523
    AS $_$
3524
SELECT util.str_equality_regexp(util.schema($1))
3525
$_$;
3526

    
3527

    
3528
--
3529
-- Name: schema_rename(text, text); Type: FUNCTION; Schema: util; Owner: -
3530
--
3531

    
3532
CREATE FUNCTION schema_rename(old text, new text) RETURNS void
3533
    LANGUAGE sql
3534
    AS $_$
3535
SELECT util.eval($$ALTER SCHEMA $$||quote_ident($1)||$$ RENAME TO $$||quote_ident($2));
3536
$_$;
3537

    
3538

    
3539
--
3540
-- Name: schema_replace(text, text); Type: FUNCTION; Schema: util; Owner: -
3541
--
3542

    
3543
CREATE FUNCTION schema_replace(replace text, with_ text) RETURNS void
3544
    LANGUAGE plpgsql
3545
    AS $$
3546
BEGIN
3547
	-- don't schema_rm() the schema to keep!
3548
	IF replace = with_ THEN RETURN; END IF;
3549
	
3550
	PERFORM util.schema_rm(replace);
3551
	PERFORM util.schema_rename(with_, replace);
3552
END;
3553
$$;
3554

    
3555

    
3556
--
3557
-- Name: schema_rm(text); Type: FUNCTION; Schema: util; Owner: -
3558
--
3559

    
3560
CREATE FUNCTION schema_rm(schema text) RETURNS void
3561
    LANGUAGE sql
3562
    AS $_$
3563
SELECT util.eval($$DROP SCHEMA IF EXISTS $$||quote_ident($1)||$$ CASCADE$$);
3564
$_$;
3565

    
3566

    
3567
--
3568
-- Name: search_path_append(text); Type: FUNCTION; Schema: util; Owner: -
3569
--
3570

    
3571
CREATE FUNCTION search_path_append(schemas text) RETURNS void
3572
    LANGUAGE sql
3573
    AS $_$
3574
SELECT util.eval(
3575
$$SET search_path TO $$||current_setting('search_path')||$$, $$||$1);
3576
$_$;
3577

    
3578

    
3579
--
3580
-- Name: set_col_names(regclass, regclass); Type: FUNCTION; Schema: util; Owner: -
3581
--
3582

    
3583
CREATE FUNCTION set_col_names(table_ regclass, names regclass) RETURNS void
3584
    LANGUAGE plpgsql STRICT
3585
    AS $_$
3586
DECLARE
3587
    old text[] = ARRAY(SELECT util.col_names(table_));
3588
    new text[] = ARRAY(SELECT util.map_values(names));
3589
BEGIN
3590
    old = old[1:array_length(new, 1)]; -- truncate to same length
3591
    PERFORM util.eval($$ALTER TABLE $$||$1||$$ RENAME $$||quote_ident(key)
3592
||$$ TO $$||quote_ident(value))
3593
    FROM each(hstore(old, new))
3594
    WHERE value != key -- not same name
3595
    ;
3596
END;
3597
$_$;
3598

    
3599

    
3600
--
3601
-- Name: FUNCTION set_col_names(table_ regclass, names regclass); Type: COMMENT; Schema: util; Owner: -
3602
--
3603

    
3604
COMMENT ON FUNCTION set_col_names(table_ regclass, names regclass) IS '
3605
idempotent
3606
';
3607

    
3608

    
3609
--
3610
-- Name: set_col_names_with_metadata(regclass, regclass); Type: FUNCTION; Schema: util; Owner: -
3611
--
3612

    
3613
CREATE FUNCTION set_col_names_with_metadata(table_ regclass, names regclass) RETURNS void
3614
    LANGUAGE plpgsql STRICT
3615
    AS $_$
3616
DECLARE
3617
	row_ util.map;
3618
BEGIN
3619
	-- rename any metadata cols rather than re-adding them with new names
3620
	BEGIN
3621
		PERFORM util.set_col_names(table_, names);
3622
	EXCEPTION
3623
		WHEN array_subscript_error THEN -- selective suppress
3624
			IF SQLERRM LIKE 'arrays must have same bounds' THEN NULL;
3625
				-- metadata cols not yet added
3626
			ELSE RAISE;
3627
			END IF;
3628
	END;
3629
	
3630
	FOR row_ IN EXECUTE $$SELECT * FROM $$||names||$$ WHERE "from" LIKE ':%'$$
3631
	LOOP
3632
		PERFORM util.mk_const_col((table_, row_."to"),
3633
			substring(row_."from" from 2));
3634
	END LOOP;
3635
	
3636
	PERFORM util.set_col_names(table_, names);
3637
END;
3638
$_$;
3639

    
3640

    
3641
--
3642
-- Name: FUNCTION set_col_names_with_metadata(table_ regclass, names regclass); Type: COMMENT; Schema: util; Owner: -
3643
--
3644

    
3645
COMMENT ON FUNCTION set_col_names_with_metadata(table_ regclass, names regclass) IS '
3646
idempotent.
3647
the metadata mappings must be *last* in the names table.
3648
';
3649

    
3650

    
3651
--
3652
-- Name: set_col_types(regclass, col_cast[]); Type: FUNCTION; Schema: util; Owner: -
3653
--
3654

    
3655
CREATE FUNCTION set_col_types(table_ regclass, col_casts col_cast[]) RETURNS void
3656
    LANGUAGE sql
3657
    AS $_$
3658
SELECT util.eval(COALESCE(
3659
$$ALTER TABLE $$||$1||$$
3660
$$||(
3661
	SELECT
3662
	string_agg($$ALTER COLUMN $$||col_name_sql||$$ TYPE $$||target_type
3663
	||$$ USING $$||col_name_sql||$$::$$||target_type, $$
3664
, $$)
3665
	FROM
3666
	(
3667
		SELECT
3668
		  quote_ident(col_name) AS col_name_sql
3669
		, util.col_type(($1, col_name)) AS curr_type
3670
		, type AS target_type
3671
		FROM unnest($2)
3672
	) s
3673
	WHERE curr_type != target_type
3674
), ''))
3675
$_$;
3676

    
3677

    
3678
--
3679
-- Name: FUNCTION set_col_types(table_ regclass, col_casts col_cast[]); Type: COMMENT; Schema: util; Owner: -
3680
--
3681

    
3682
COMMENT ON FUNCTION set_col_types(table_ regclass, col_casts col_cast[]) IS '
3683
idempotent
3684
';
3685

    
3686

    
3687
--
3688
-- Name: set_comment(regclass, text); Type: FUNCTION; Schema: util; Owner: -
3689
--
3690

    
3691
CREATE FUNCTION set_comment(table_ regclass, comment text) RETURNS void
3692
    LANGUAGE sql
3693
    AS $_$
3694
SELECT util.eval($$COMMENT ON TABLE $$||$1||$$ IS $$||quote_nullable($2))
3695
$_$;
3696

    
3697

    
3698
--
3699
-- Name: set_search_path(text, boolean); Type: FUNCTION; Schema: util; Owner: -
3700
--
3701

    
3702
CREATE FUNCTION set_search_path(search_path text, for_session boolean DEFAULT false) RETURNS void
3703
    LANGUAGE sql
3704
    AS $_$
3705
SELECT util.eval(util.mk_set_search_path($1, $2))
3706
$_$;
3707

    
3708

    
3709
--
3710
-- Name: show_create_view(regclass, boolean); Type: FUNCTION; Schema: util; Owner: -
3711
--
3712

    
3713
CREATE FUNCTION show_create_view(view_ regclass, replace boolean DEFAULT true) RETURNS text
3714
    LANGUAGE sql STABLE
3715
    AS $_$
3716
SELECT $$CREATE$$||(CASE WHEN $2 THEN $$ OR REPLACE$$ ELSE '' END)||$$ VIEW $$
3717
||$1||$$ AS
3718
$$||pg_get_viewdef($1)||/*no ; because pg_get_viewdef() includes one*/$$
3719
$$||util.show_grants_for($1)
3720
$_$;
3721

    
3722

    
3723
--
3724
-- Name: show_grants_for(regclass); Type: FUNCTION; Schema: util; Owner: -
3725
--
3726

    
3727
CREATE FUNCTION show_grants_for(table_ regclass) RETURNS text
3728
    LANGUAGE sql STABLE
3729
    AS $_$
3730
SELECT string_agg(cmd, '')
3731
FROM
3732
(
3733
	SELECT (CASE WHEN has_table_privilege(user_, $1, 'SELECT') THEN
3734
$$GRANT SELECT ON TABLE $$||$1||$$ TO $$||quote_ident(user_)||$$;
3735
$$ ELSE '' END) AS cmd
3736
	FROM util.grants_users() f (user_)
3737
) s
3738
$_$;
3739

    
3740

    
3741
--
3742
-- Name: show_relations_like(text, text, character[]); Type: FUNCTION; Schema: util; Owner: -
3743
--
3744

    
3745
CREATE FUNCTION show_relations_like(name_regexp text, schema_regexp text DEFAULT ''::text, types character[] DEFAULT ARRAY['c'::text, 'r'::text, 'v'::text]) RETURNS SETOF regclass
3746
    LANGUAGE sql STABLE
3747
    AS $_$
3748
SELECT oid FROM pg_class
3749
WHERE relkind = ANY($3) AND relname ~ $1
3750
AND util.schema_matches(util.schema(relnamespace), $2)
3751
ORDER BY relname
3752
$_$;
3753

    
3754

    
3755
--
3756
-- Name: show_types_like(text, text); Type: FUNCTION; Schema: util; Owner: -
3757
--
3758

    
3759
CREATE FUNCTION show_types_like(name_regexp text DEFAULT ''::text, schema_regexp text DEFAULT ''::text) RETURNS SETOF regtype
3760
    LANGUAGE sql STABLE
3761
    AS $_$
3762
SELECT oid
3763
FROM pg_type
3764
WHERE typname ~ $1 AND util.schema_matches(util.schema(typnamespace), $2)
3765
ORDER BY typname
3766
$_$;
3767

    
3768

    
3769
--
3770
-- Name: show_views_like(text, text); Type: FUNCTION; Schema: util; Owner: -
3771
--
3772

    
3773
CREATE FUNCTION show_views_like(name_regexp text, schema_regexp text DEFAULT ''::text) RETURNS SETOF regclass
3774
    LANGUAGE sql STABLE
3775
    AS $_$
3776
SELECT * FROM util.show_relations_like($1, $2, ARRAY['v'])
3777
$_$;
3778

    
3779

    
3780
--
3781
-- Name: str_equality_regexp(text); Type: FUNCTION; Schema: util; Owner: -
3782
--
3783

    
3784
CREATE FUNCTION str_equality_regexp(literal text) RETURNS text
3785
    LANGUAGE sql IMMUTABLE
3786
    AS $_$
3787
SELECT '^'||util.regexp_quote($1)||'$'
3788
$_$;
3789

    
3790

    
3791
--
3792
-- Name: table2hstore(regclass); Type: FUNCTION; Schema: util; Owner: -
3793
--
3794

    
3795
CREATE FUNCTION table2hstore(table_ regclass) RETURNS hstore
3796
    LANGUAGE plpgsql STABLE STRICT
3797
    AS $_$
3798
DECLARE
3799
    hstore hstore;
3800
BEGIN
3801
    EXECUTE $$SELECT hstore(ARRAY(SELECT unnest(ARRAY["from", "to"]) FROM $$||
3802
        table_||$$))$$ INTO STRICT hstore;
3803
    RETURN hstore;
3804
END;
3805
$_$;
3806

    
3807

    
3808
--
3809
-- Name: table_flag__get(regclass, text); Type: FUNCTION; Schema: util; Owner: -
3810
--
3811

    
3812
CREATE FUNCTION table_flag__get(table_ regclass, flag text) RETURNS boolean
3813
    LANGUAGE sql STABLE
3814
    AS $_$
3815
SELECT COUNT(*) > 0 FROM pg_constraint
3816
WHERE conrelid = $1 AND contype = 'c' AND conname = $2
3817
$_$;
3818

    
3819

    
3820
--
3821
-- Name: FUNCTION table_flag__get(table_ regclass, flag text); Type: COMMENT; Schema: util; Owner: -
3822
--
3823

    
3824
COMMENT ON FUNCTION table_flag__get(table_ regclass, flag text) IS '
3825
gets whether a status flag is set by the presence of a table constraint
3826
';
3827

    
3828

    
3829
--
3830
-- Name: table_flag__set(regclass, text); Type: FUNCTION; Schema: util; Owner: -
3831
--
3832

    
3833
CREATE FUNCTION table_flag__set(table_ regclass, flag text) RETURNS void
3834
    LANGUAGE sql
3835
    AS $_$
3836
SELECT util.create_if_not_exists($$ALTER TABLE $$||$1||$$ ADD CONSTRAINT $$
3837
||quote_ident($2)||$$ CHECK (true)$$)
3838
$_$;
3839

    
3840

    
3841
--
3842
-- Name: FUNCTION table_flag__set(table_ regclass, flag text); Type: COMMENT; Schema: util; Owner: -
3843
--
3844

    
3845
COMMENT ON FUNCTION table_flag__set(table_ regclass, flag text) IS '
3846
stores a status flag by the presence of a table constraint.
3847
idempotent.
3848
';
3849

    
3850

    
3851
--
3852
-- Name: table_nulls_mapped__get(regclass); Type: FUNCTION; Schema: util; Owner: -
3853
--
3854

    
3855
CREATE FUNCTION table_nulls_mapped__get(table_ regclass) RETURNS boolean
3856
    LANGUAGE sql STABLE
3857
    AS $_$
3858
SELECT util.table_flag__get($1, 'nulls_mapped')
3859
$_$;
3860

    
3861

    
3862
--
3863
-- Name: FUNCTION table_nulls_mapped__get(table_ regclass); Type: COMMENT; Schema: util; Owner: -
3864
--
3865

    
3866
COMMENT ON FUNCTION table_nulls_mapped__get(table_ regclass) IS '
3867
gets whether a table''s NULL-equivalent strings have been replaced with NULL
3868
';
3869

    
3870

    
3871
--
3872
-- Name: table_nulls_mapped__set(regclass); Type: FUNCTION; Schema: util; Owner: -
3873
--
3874

    
3875
CREATE FUNCTION table_nulls_mapped__set(table_ regclass) RETURNS void
3876
    LANGUAGE sql
3877
    AS $_$
3878
SELECT util.table_flag__set($1, 'nulls_mapped')
3879
$_$;
3880

    
3881

    
3882
--
3883
-- Name: FUNCTION table_nulls_mapped__set(table_ regclass); Type: COMMENT; Schema: util; Owner: -
3884
--
3885

    
3886
COMMENT ON FUNCTION table_nulls_mapped__set(table_ regclass) IS '
3887
sets that a table''s NULL-equivalent strings have been replaced with NULL.
3888
idempotent.
3889
';
3890

    
3891

    
3892
--
3893
-- Name: to_freq(regclass); Type: FUNCTION; Schema: util; Owner: -
3894
--
3895

    
3896
CREATE FUNCTION to_freq(table_ regclass) RETURNS void
3897
    LANGUAGE sql
3898
    AS $_$
3899
-- save data before truncating main table
3900
SELECT util.copy_types_and_data($1, 'pg_temp.__copy');
3901

    
3902
-- repopulate main table w/ copies column
3903
SELECT util.truncate($1);
3904
SELECT util.eval($$ALTER TABLE $$||$1||$$ ADD COLUMN copies bigint NOT NULL$$);
3905
SELECT util.populate_table($1, $$
3906
SELECT (table_).*, copies
3907
FROM (
3908
	SELECT table_, COUNT(*) AS copies
3909
	FROM pg_temp.__copy table_
3910
	GROUP BY table_
3911
) s
3912
$$);
3913

    
3914
-- delete temp table so it doesn't stay around until end of connection
3915
SELECT util.drop_table('pg_temp.__copy');
3916
$_$;
3917

    
3918

    
3919
--
3920
-- Name: to_global_col_names(regclass); Type: FUNCTION; Schema: util; Owner: -
3921
--
3922

    
3923
CREATE FUNCTION to_global_col_names(table_ regclass) RETURNS void
3924
    LANGUAGE plpgsql STRICT
3925
    AS $_$
3926
DECLARE
3927
    row record;
3928
BEGIN
3929
    FOR row IN SELECT * FROM util.col_global_names(table_::text::regtype)
3930
    LOOP
3931
        IF row.global_name != row.name THEN
3932
            EXECUTE $$ALTER TABLE $$||table_||$$ RENAME $$
3933
                ||quote_ident(row.name)||$$ TO $$||quote_ident(row.global_name);
3934
        END IF;
3935
    END LOOP;
3936
END;
3937
$_$;
3938

    
3939

    
3940
--
3941
-- Name: FUNCTION to_global_col_names(table_ regclass); Type: COMMENT; Schema: util; Owner: -
3942
--
3943

    
3944
COMMENT ON FUNCTION to_global_col_names(table_ regclass) IS '
3945
idempotent
3946
';
3947

    
3948

    
3949
--
3950
-- Name: trim(regclass, regclass, boolean); Type: FUNCTION; Schema: util; Owner: -
3951
--
3952

    
3953
CREATE FUNCTION "trim"(table_ regclass, names regclass, force boolean DEFAULT true) RETURNS void
3954
    LANGUAGE sql
3955
    AS $_$
3956
SELECT util.drop_column(($1, col), $3) FROM util.added_cols($1, $2) f (col);
3957
SELECT NULL::void; -- don't fold away functions called in previous query
3958
$_$;
3959

    
3960

    
3961
--
3962
-- Name: FUNCTION "trim"(table_ regclass, names regclass, force boolean); Type: COMMENT; Schema: util; Owner: -
3963
--
3964

    
3965
COMMENT ON FUNCTION "trim"(table_ regclass, names regclass, force boolean) IS '
3966
trims table_ to include only columns in the original data
3967

    
3968
by default, cascadingly drops dependent columns so that they don''t prevent
3969
trim() from succeeding. note that this requires the dependent columns to then be
3970
manually re-created.
3971

    
3972
idempotent
3973
';
3974

    
3975

    
3976
--
3977
-- Name: truncate(regclass); Type: FUNCTION; Schema: util; Owner: -
3978
--
3979

    
3980
CREATE FUNCTION truncate(table_ regclass) RETURNS void
3981
    LANGUAGE plpgsql STRICT
3982
    AS $_$
3983
BEGIN
3984
    EXECUTE $$TRUNCATE $$||table_||$$ CASCADE$$;
3985
END;
3986
$_$;
3987

    
3988

    
3989
--
3990
-- Name: FUNCTION truncate(table_ regclass); Type: COMMENT; Schema: util; Owner: -
3991
--
3992

    
3993
COMMENT ON FUNCTION truncate(table_ regclass) IS '
3994
idempotent
3995
';
3996

    
3997

    
3998
--
3999
-- Name: truncated_prefixed_name_regexp(text, integer); Type: FUNCTION; Schema: util; Owner: -
4000
--
4001

    
4002
CREATE FUNCTION truncated_prefixed_name_regexp(name text, max_prefix_len integer) RETURNS text
4003
    LANGUAGE sql IMMUTABLE
4004
    AS $_$
4005
SELECT '^(.*)'||util._if(util.name_was_truncated($1, $2),
4006
util.regexp_quote(util.rtrim_n($1, $2))||'.*', util.regexp_quote($1)) ||'$'
4007
$_$;
4008

    
4009

    
4010
--
4011
-- Name: try_cast(text, anyelement); Type: FUNCTION; Schema: util; Owner: -
4012
--
4013

    
4014
CREATE FUNCTION try_cast(value text, ret_type_null anyelement) RETURNS anyelement
4015
    LANGUAGE plpgsql IMMUTABLE
4016
    AS $$
4017
BEGIN
4018
	/* need explicit cast because some types not implicitly-castable, and also
4019
	to make the cast happen inside the try block. (*implicit* casts to the
4020
	return type happen at the end of the function, outside any block.) */
4021
	RETURN util.cast(value, ret_type_null);
4022
EXCEPTION
4023
WHEN data_exception THEN
4024
	PERFORM util.raise('WARNING', SQLERRM);
4025
	RETURN NULL;
4026
END;
4027
$$;
4028

    
4029

    
4030
--
4031
-- Name: FUNCTION try_cast(value text, ret_type_null anyelement); Type: COMMENT; Schema: util; Owner: -
4032
--
4033

    
4034
COMMENT ON FUNCTION try_cast(value text, ret_type_null anyelement) IS '
4035
ret_type_null: NULL::ret_type
4036
';
4037

    
4038

    
4039
--
4040
-- Name: try_create(text); Type: FUNCTION; Schema: util; Owner: -
4041
--
4042

    
4043
CREATE FUNCTION try_create(sql text) RETURNS void
4044
    LANGUAGE plpgsql STRICT
4045
    AS $$
4046
BEGIN
4047
	PERFORM util.eval(sql);
4048
EXCEPTION
4049
WHEN   not_null_violation
4050
		/* trying to add NOT NULL column to parent table, which cascades to
4051
		child table whose values for the new column will be NULL */
4052
	OR wrong_object_type -- trying to alter a view's columns
4053
	OR undefined_column
4054
	OR duplicate_column
4055
THEN NULL;
4056
WHEN datatype_mismatch THEN
4057
	IF SQLERRM LIKE 'child table is missing column %' THEN NULL;
4058
	ELSE RAISE; -- rethrow
4059
	END IF;
4060
END;
4061
$$;
4062

    
4063

    
4064
--
4065
-- Name: FUNCTION try_create(sql text); Type: COMMENT; Schema: util; Owner: -
4066
--
4067

    
4068
COMMENT ON FUNCTION try_create(sql text) IS '
4069
idempotent
4070
';
4071

    
4072

    
4073
--
4074
-- Name: try_mk_derived_col(col_ref, text); Type: FUNCTION; Schema: util; Owner: -
4075
--
4076

    
4077
CREATE FUNCTION try_mk_derived_col(col col_ref, expr text) RETURNS void
4078
    LANGUAGE sql
4079
    AS $_$
4080
SELECT util.try_create($$SELECT util.mk_derived_col($$||quote_literal($1)||$$, $$||quote_literal($2)||$$)$$)
4081
$_$;
4082

    
4083

    
4084
--
4085
-- Name: FUNCTION try_mk_derived_col(col col_ref, expr text); Type: COMMENT; Schema: util; Owner: -
4086
--
4087

    
4088
COMMENT ON FUNCTION try_mk_derived_col(col col_ref, expr text) IS '
4089
idempotent
4090
';
4091

    
4092

    
4093
--
4094
-- Name: type_qual(anyelement); Type: FUNCTION; Schema: util; Owner: -
4095
--
4096

    
4097
CREATE FUNCTION type_qual(value anyelement) RETURNS text
4098
    LANGUAGE sql IMMUTABLE
4099
    AS $_$
4100
SELECT CASE WHEN $1 IS NULL THEN '' ELSE $$ NOT NULL$$ END
4101
$_$;
4102

    
4103

    
4104
--
4105
-- Name: FUNCTION type_qual(value anyelement); Type: COMMENT; Schema: util; Owner: -
4106
--
4107

    
4108
COMMENT ON FUNCTION type_qual(value anyelement) IS '
4109
a type''s NOT NULL qualifier
4110
';
4111

    
4112

    
4113
--
4114
-- Name: typed_cols(regtype); Type: FUNCTION; Schema: util; Owner: -
4115
--
4116

    
4117
CREATE FUNCTION typed_cols(type regtype) RETURNS SETOF col_cast
4118
    LANGUAGE sql STABLE
4119
    AS $_$
4120
SELECT (attname::text, atttypid)::util.col_cast
4121
FROM pg_attribute
4122
WHERE attrelid = $1::text::regclass AND attnum >= 1 AND NOT attisdropped
4123
ORDER BY attnum
4124
$_$;
4125

    
4126

    
4127
--
4128
-- Name: typeof(anyelement); Type: FUNCTION; Schema: util; Owner: -
4129
--
4130

    
4131
CREATE FUNCTION typeof(value anyelement) RETURNS text
4132
    LANGUAGE sql IMMUTABLE
4133
    AS $_$
4134
SELECT util.qual_name(pg_typeof($1))
4135
$_$;
4136

    
4137

    
4138
--
4139
-- Name: typeof(text, regtype); Type: FUNCTION; Schema: util; Owner: -
4140
--
4141

    
4142
CREATE FUNCTION typeof(expr text, table_ regtype DEFAULT NULL::regtype) RETURNS regtype
4143
    LANGUAGE plpgsql STABLE
4144
    AS $_$
4145
DECLARE
4146
    type regtype;
4147
BEGIN
4148
    EXECUTE $$SELECT pg_typeof($$||expr||$$)$$||
4149
COALESCE($$ FROM (SELECT (NULL::$$||table_||$$).*) _s$$, '') INTO STRICT type;
4150
    RETURN type;
4151
END;
4152
$_$;
4153

    
4154

    
4155
--
4156
-- Name: use_schema(anyelement); Type: FUNCTION; Schema: util; Owner: -
4157
--
4158

    
4159
CREATE FUNCTION use_schema(schema_anchor anyelement) RETURNS void
4160
    LANGUAGE sql
4161
    AS $_$
4162
SELECT util.set_search_path(util.mk_search_path(util.schema($1)))
4163
$_$;
4164

    
4165

    
4166
--
4167
-- Name: FUNCTION use_schema(schema_anchor anyelement); Type: COMMENT; Schema: util; Owner: -
4168
--
4169

    
4170
COMMENT ON FUNCTION use_schema(schema_anchor anyelement) IS '
4171
auto-appends util to the search_path to enable use of util operators
4172
';
4173

    
4174

    
4175
--
4176
-- Name: all_same(anyelement); Type: AGGREGATE; Schema: util; Owner: -
4177
--
4178

    
4179
CREATE AGGREGATE all_same(anyelement) (
4180
    SFUNC = all_same_transform,
4181
    STYPE = anyarray,
4182
    FINALFUNC = all_same_final
4183
);
4184

    
4185

    
4186
--
4187
-- Name: AGGREGATE all_same(anyelement); Type: COMMENT; Schema: util; Owner: -
4188
--
4189

    
4190
COMMENT ON AGGREGATE all_same(anyelement) IS '
4191
includes NULLs in comparison
4192
';
4193

    
4194

    
4195
--
4196
-- Name: join_strs(text, text); Type: AGGREGATE; Schema: util; Owner: -
4197
--
4198

    
4199
CREATE AGGREGATE join_strs(text, text) (
4200
    SFUNC = join_strs_transform,
4201
    STYPE = text
4202
);
4203

    
4204

    
4205
--
4206
-- Name: %==; Type: OPERATOR; Schema: util; Owner: -
4207
--
4208

    
4209
CREATE OPERATOR %== (
4210
    PROCEDURE = "%==",
4211
    LEFTARG = anyelement,
4212
    RIGHTARG = anyelement
4213
);
4214

    
4215

    
4216
--
4217
-- Name: OPERATOR %== (anyelement, anyelement); Type: COMMENT; Schema: util; Owner: -
4218
--
4219

    
4220
COMMENT ON OPERATOR %== (anyelement, anyelement) IS '
4221
returns whether the map-keys of the compared values are the same
4222
(mnemonic: % is the Perl symbol for a hash map)
4223

    
4224
should be overridden for types that store both keys and values
4225

    
4226
used in a FULL JOIN to select which columns to join on
4227
';
4228

    
4229

    
4230
--
4231
-- Name: ->; Type: OPERATOR; Schema: util; Owner: -
4232
--
4233

    
4234
CREATE OPERATOR -> (
4235
    PROCEDURE = map_get,
4236
    LEFTARG = regclass,
4237
    RIGHTARG = text
4238
);
4239

    
4240

    
4241
--
4242
-- Name: =>; Type: OPERATOR; Schema: util; Owner: -
4243
--
4244

    
4245
CREATE OPERATOR => (
4246
    PROCEDURE = hstore,
4247
    LEFTARG = text[],
4248
    RIGHTARG = text
4249
);
4250

    
4251

    
4252
--
4253
-- Name: OPERATOR => (text[], text); Type: COMMENT; Schema: util; Owner: -
4254
--
4255

    
4256
COMMENT ON OPERATOR => (text[], text) IS '
4257
usage: array[''key1'', ...]::text[] => ''value''
4258
';
4259

    
4260

    
4261
--
4262
-- Name: ?*>=; Type: OPERATOR; Schema: util; Owner: -
4263
--
4264

    
4265
CREATE OPERATOR ?*>= (
4266
    PROCEDURE = is_populated_more_often_than,
4267
    LEFTARG = anyelement,
4268
    RIGHTARG = anyelement
4269
);
4270

    
4271

    
4272
--
4273
-- Name: ?>=; Type: OPERATOR; Schema: util; Owner: -
4274
--
4275

    
4276
CREATE OPERATOR ?>= (
4277
    PROCEDURE = is_more_complete_than,
4278
    LEFTARG = anyelement,
4279
    RIGHTARG = anyelement
4280
);
4281

    
4282

    
4283
--
4284
-- Name: ||%; Type: OPERATOR; Schema: util; Owner: -
4285
--
4286

    
4287
CREATE OPERATOR ||% (
4288
    PROCEDURE = concat_esc,
4289
    LEFTARG = text,
4290
    RIGHTARG = text
4291
);
4292

    
4293

    
4294
--
4295
-- Name: OPERATOR ||% (text, text); Type: COMMENT; Schema: util; Owner: -
4296
--
4297

    
4298
COMMENT ON OPERATOR ||% (text, text) IS '
4299
% indicates an identifier, as in Perl hashes and one of the x86 assembler syntaxes for registers
4300
';
4301

    
4302

    
4303
--
4304
-- Name: map; Type: TABLE; Schema: util; Owner: -; Tablespace: 
4305
--
4306

    
4307
CREATE TABLE map (
4308
    "from" text NOT NULL,
4309
    "to" text,
4310
    filter text,
4311
    notes text
4312
);
4313

    
4314

    
4315
--
4316
-- Data for Name: explain; Type: TABLE DATA; Schema: util; Owner: -
4317
--
4318

    
4319

    
4320

    
4321
--
4322
-- Data for Name: map; Type: TABLE DATA; Schema: util; Owner: -
4323
--
4324

    
4325

    
4326

    
4327
--
4328
-- Name: map__unique__from; Type: CONSTRAINT; Schema: util; Owner: -; Tablespace: 
4329
--
4330

    
4331
ALTER TABLE ONLY map
4332
    ADD CONSTRAINT map__unique__from UNIQUE ("from");
4333

    
4334

    
4335
--
4336
-- Name: map__unique__to; Type: CONSTRAINT; Schema: util; Owner: -; Tablespace: 
4337
--
4338

    
4339
ALTER TABLE ONLY map
4340
    ADD CONSTRAINT map__unique__to UNIQUE ("to");
4341

    
4342

    
4343
--
4344
-- Name: map_filter_insert; Type: TRIGGER; Schema: util; Owner: -
4345
--
4346

    
4347
CREATE TRIGGER map_filter_insert BEFORE INSERT ON map FOR EACH ROW EXECUTE PROCEDURE map_filter_insert();
4348

    
4349

    
4350
--
4351
-- PostgreSQL database dump complete
4352
--
4353

    
(21-21/31)