2 * Copyright (C) 2008 Equinox Software, Inc.
3 * Mike Rylander <miker@esilibrary.com.com>
5 * This program is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU General Public License
7 * as published by the Free Software Foundation; either version 2
8 * of the License, or (at your option) any later version.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
19 -------------------------------------------------------------------
20 /* new materialized view for reporting schema -- ok if it fails */
21 -------------------------------------------------------------------
23 CREATE TABLE reporter.materialized_simple_record AS SELECT * FROM reporter.super_simple_record WHERE 1=0;
25 INSERT INTO reporter.materialized_simple_record
26 (id,fingerprint,quality,tcn_source,tcn_value,title,author,publisher,pubdate,isbn,issn)
27 SELECT DISTINCT ON (id) * FROM reporter.super_simple_record;
29 ALTER TABLE reporter.materialized_simple_record ADD PRIMARY KEY (id);
31 CREATE OR REPLACE VIEW reporter.super_simple_record AS SELECT * FROM reporter.materialized_simple_record;
33 CREATE OR REPLACE VIEW reporter.old_super_simple_record AS
40 FIRST(author.value) AS author,
41 publisher.value AS publisher,
42 SUBSTRING(pubdate.value FROM $$\d+$$) AS pubdate,
43 ARRAY_ACCUM( SUBSTRING(isbn.value FROM $$^\S+$$) ) AS isbn,
44 ARRAY_ACCUM( SUBSTRING(issn.value FROM $$^\S+$$) ) AS issn
45 FROM biblio.record_entry r
46 LEFT JOIN metabib.full_rec title ON (r.id = title.record AND title.tag = '245' AND title.subfield = 'a')
47 LEFT JOIN metabib.full_rec author ON (r.id = author.record AND author.tag IN ('100','110','111') AND author.subfield = 'a')
48 LEFT JOIN metabib.full_rec publisher ON (r.id = publisher.record AND publisher.tag = '260' AND publisher.subfield = 'b')
49 LEFT JOIN metabib.full_rec pubdate ON (r.id = pubdate.record AND pubdate.tag = '260' AND pubdate.subfield = 'c')
50 LEFT JOIN metabib.full_rec isbn ON (r.id = isbn.record AND isbn.tag IN ('024', '020') AND isbn.subfield IN ('a','z'))
51 LEFT JOIN metabib.full_rec issn ON (r.id = issn.record AND issn.tag = '022' AND issn.subfield = 'a')
52 GROUP BY 1,2,3,4,5,6,8,9;
54 CREATE OR REPLACE FUNCTION reporter.simple_rec_sync () RETURNS TRIGGER AS $$
59 IF TG_OP IN ('DELETE') THEN
65 SELECT * INTO new_data FROM reporter.materialized_simple_record WHERE id = r_id FOR UPDATE;
66 DELETE FROM reporter.materialized_simple_record WHERE id = r_id;
68 IF TG_OP IN ('DELETE') THEN
71 INSERT INTO reporter.materialized_simple_record SELECT DISTINCT ON (id) * FROM reporter.old_super_simple_record WHERE id = NEW.record;
78 CREATE TRIGGER zzz_update_materialized_simple_record_tgr
79 AFTER INSERT OR UPDATE OR DELETE ON metabib.full_rec
80 FOR EACH ROW EXECUTE PROCEDURE reporter.simple_rec_sync();
85 DROP VIEW reporter.overdue_reports;
86 DROP VIEW reporter.pending_reports;
87 DROP VIEW reporter.currently_running;
91 -------------------------------------------------------------------
92 /* convenience views for report management */
93 -------------------------------------------------------------------
95 CREATE OR REPLACE VIEW reporter.overdue_reports AS
96 SELECT s.id, c.barcode AS runner_barcode, r.name, s.run_time, s.run_time - now() AS scheduled_wait_time
97 FROM reporter.schedule s
98 JOIN reporter.report r ON r.id = s.report
99 JOIN actor.usr u ON s.runner = u.id
100 JOIN actor.card c ON c.id = u.card
101 WHERE s.start_time IS NULL AND s.run_time < now();
103 CREATE OR REPLACE VIEW reporter.pending_reports AS
104 SELECT s.id, c.barcode AS runner_barcode, r.name, s.run_time, s.run_time - now() AS scheduled_wait_time
105 FROM reporter.schedule s
106 JOIN reporter.report r ON r.id = s.report
107 JOIN actor.usr u ON s.runner = u.id
108 JOIN actor.card c ON c.id = u.card
109 WHERE s.start_time IS NULL;
111 CREATE OR REPLACE VIEW reporter.currently_running AS
112 SELECT s.id, c.barcode AS runner_barcode, r.name, s.run_time, s.run_time - now() AS scheduled_wait_time
113 FROM reporter.schedule s
114 JOIN reporter.report r ON r.id = s.report
115 JOIN actor.usr u ON s.runner = u.id
116 JOIN actor.card c ON c.id = u.card
117 WHERE s.start_time IS NOT NULL AND s.complete_time IS NULL;
120 -------------------------------------------------------------------
121 /* view for restricting circ counts by circ_mod */
122 -------------------------------------------------------------------
124 CREATE OR REPLACE VIEW action.open_circ_count_by_circ_mod AS
128 FROM action.circulation circ
129 JOIN asset.copy cp ON (circ.target_copy = cp.id)
130 WHERE circ.checkin_time IS NULL
131 AND ( circ.stop_fines IN ('LOST','LONGOVERDUE','CLAIMSRETURNED') OR circ.stop_fines IS NULL )
135 -------------------------------------------------------------------
136 /* reporting functions for new (and fixed) transforms */
137 -------------------------------------------------------------------
139 CREATE OR REPLACE FUNCTION public.first_word ( TEXT ) RETURNS TEXT AS $$
140 SELECT SUBSTRING( $1 FROM $_$^\S+$_$);
143 CREATE OR REPLACE FUNCTION public.first5 ( TEXT ) RETURNS TEXT AS $$
144 SELECT SUBSTRING( $1, 1, 5);
147 CREATE OR REPLACE FUNCTION public.call_number_dewey( TEXT ) RETURNS TEXT AS $$
150 $txt =~ s/[\[\]\{\}\(\)`'"#<>\*\?\-\+\$\\]+//o; #' To help vim in SQL mode
152 if ($txt =~ /(\d{3}(?:\.\d+)?)/o) {
155 return (split /\s+/, $txt)[0];
157 $$ LANGUAGE 'plperlu' STRICT IMMUTABLE;
161 DROP SCHEMA search CASCADE;
164 -------------------------------------------------------------------
165 /* staged search -- also applied by 300.schema.staged_search.sql */
166 -------------------------------------------------------------------
168 CREATE SCHEMA search;
170 CREATE TABLE search.relevance_adjustment (
171 id SERIAL PRIMARY KEY,
172 active BOOL NOT NULL DEFAULT TRUE,
173 field INT NOT NULL REFERENCES config.metabib_field (id) DEFERRABLE INITIALLY DEFERRED,
174 bump_type TEXT NOT NULL CHECK (bump_type IN ('word_order','first_word','full_match')),
175 multiplier NUMERIC NOT NULL DEFAULT 1.0
177 CREATE UNIQUE INDEX bump_once_per_field_idx ON search.relevance_adjustment ( field, bump_type );
179 INSERT INTO search.relevance_adjustment (field, bump_type, multiplier) VALUES(1, 'first_word', 1.5);
180 INSERT INTO search.relevance_adjustment (field, bump_type, multiplier) VALUES(1, 'full_match', 20);
181 INSERT INTO search.relevance_adjustment (field, bump_type, multiplier) VALUES(2, 'first_word', 1.5);
182 INSERT INTO search.relevance_adjustment (field, bump_type, multiplier) VALUES(2, 'word_order', 10);
183 INSERT INTO search.relevance_adjustment (field, bump_type, multiplier) VALUES(2, 'full_match', 20);
184 INSERT INTO search.relevance_adjustment (field, bump_type, multiplier) VALUES(3, 'first_word', 1.5);
185 INSERT INTO search.relevance_adjustment (field, bump_type, multiplier) VALUES(3, 'word_order', 10);
186 INSERT INTO search.relevance_adjustment (field, bump_type, multiplier) VALUES(3, 'full_match', 20);
187 INSERT INTO search.relevance_adjustment (field, bump_type, multiplier) VALUES(4, 'first_word', 1.5);
188 INSERT INTO search.relevance_adjustment (field, bump_type, multiplier) VALUES(4, 'word_order', 10);
189 INSERT INTO search.relevance_adjustment (field, bump_type, multiplier) VALUES(4, 'full_match', 20);
190 INSERT INTO search.relevance_adjustment (field, bump_type, multiplier) VALUES(5, 'first_word', 1.5);
191 INSERT INTO search.relevance_adjustment (field, bump_type, multiplier) VALUES(5, 'word_order', 10);
192 INSERT INTO search.relevance_adjustment (field, bump_type, multiplier) VALUES(5, 'full_match', 20);
193 INSERT INTO search.relevance_adjustment (field, bump_type, multiplier) VALUES(6, 'first_word', 1.5);
194 INSERT INTO search.relevance_adjustment (field, bump_type, multiplier) VALUES(7, 'first_word', 1.5);
195 INSERT INTO search.relevance_adjustment (field, bump_type, multiplier) VALUES(8, 'first_word', 1.5);
196 INSERT INTO search.relevance_adjustment (field, bump_type, multiplier) VALUES(9, 'first_word', 1.5);
197 INSERT INTO search.relevance_adjustment (field, bump_type, multiplier) VALUES(14, 'word_order', 10);
199 CREATE OR REPLACE FUNCTION search.pick_table (TEXT) RETURNS TEXT AS $$
201 WHEN $1 = 'author' THEN 'metabib.author_field_entry'
202 WHEN $1 = 'title' THEN 'metabib.title_field_entry'
203 WHEN $1 = 'subject' THEN 'metabib.subject_field_entry'
204 WHEN $1 = 'keyword' THEN 'metabib.keyword_field_entry'
205 WHEN $1 = 'series' THEN 'metabib.series_field_entry'
209 CREATE TYPE search.search_result AS ( id BIGINT, rel NUMERIC, record INT, total INT, checked INT, visible INT, deleted INT, excluded INT );
210 CREATE TYPE search.search_args AS ( id INT, field_class TEXT, field_name TEXT, table_alias TEXT, term TEXT, term_type TEXT );
212 CREATE OR REPLACE FUNCTION search.staged_fts (
216 param_searches TEXT, -- JSON hash, to be turned into a resultset via search.parse_search_args
217 param_statuses INT[],
218 param_audience TEXT[],
219 param_language TEXT[],
220 param_lit_form TEXT[],
223 param_vformats TEXT[],
224 param_pref_lang TEXT,
225 param_pref_lang_multiplier REAL,
227 param_sort_desc BOOL,
234 ) RETURNS SETOF search.search_result AS $func$
237 current_res search.search_result%ROWTYPE;
238 query_part search.search_args%ROWTYPE;
239 phrase_query_part search.search_args%ROWTYPE;
244 rank_adjust search.relevance_adjustment%ROWTYPE;
249 ranks TEXT[] := '{}';
250 query_table_alias TEXT;
251 from_alias_array TEXT[] := '{}';
252 used_ranks TEXT[] := '{}';
255 search_org_list INT[];
256 select_clause TEXT := 'SELECT';
257 from_clause TEXT := ' FROM metabib.metarecord_source_map m JOIN metabib.rec_descriptor mrd ON (m.source = mrd.record) ';
258 where_clause TEXT := ' WHERE 1=1 ';
259 mrd_used BOOL := FALSE;
260 sort_desc BOOL := FALSE;
263 core_cursor REFCURSOR;
265 vis_limit_query TEXT;
266 inner_where_clause TEXT;
268 total_count INT := 0;
269 check_count INT := 0;
270 deleted_count INT := 0;
271 visible_count INT := 0;
272 excluded_count INT := 0;
276 core_rel_limit := COALESCE( param_rel_limit, 25000 );
277 core_chk_limit := COALESCE( param_chk_limit, 1000 );
278 core_skip_chk := COALESCE( param_skip_chk, 1 );
281 select_clause := select_clause || ' m.metarecord as id, array_accum(distinct m.source) as records,';
283 select_clause := select_clause || ' m.source as id, array_accum(distinct m.source) as records,';
286 -- first we need to construct the base query
287 FOR query_part IN SELECT * FROM search.parse_search_args(param_searches) WHERE term_type = 'fts_query' LOOP
289 inner_where_clause := 'index_vector @@ ' || query_part.term;
291 IF query_part.field_name IS NOT NULL THEN
293 SELECT id INTO mb_field
294 FROM config.metabib_field
295 WHERE field_class = query_part.field_class
296 AND name = query_part.field_name;
299 inner_where_clause := inner_where_clause ||
300 ' AND ' || 'field = ' || mb_field;
305 -- moving on to the rank ...
306 SELECT * INTO query_part
307 FROM search.parse_search_args(param_searches)
308 WHERE term_type = 'fts_rank'
309 AND table_alias = query_part.table_alias;
311 current_rank := query_part.term || ' * ' || query_part.table_alias || '_weight.weight';
313 IF query_part.field_name IS NOT NULL THEN
315 SELECT array_accum(distinct id) INTO mb_field_list
316 FROM config.metabib_field
317 WHERE field_class = query_part.field_class
318 AND name = query_part.field_name;
322 SELECT array_accum(distinct id) INTO mb_field_list
323 FROM config.metabib_field
324 WHERE field_class = query_part.field_class;
328 FOR rank_adjust IN SELECT * FROM search.relevance_adjustment WHERE active AND field IN ( SELECT * FROM search.explode_array( mb_field_list ) ) LOOP
330 IF NOT rank_adjust.bump_type = ANY (used_ranks) THEN
332 IF rank_adjust.bump_type = 'first_word' THEN
333 SELECT term INTO tmp_text
334 FROM search.parse_search_args(param_searches)
335 WHERE table_alias = query_part.table_alias AND term_type = 'word'
339 tmp_text := query_part.table_alias || '.value ILIKE ' || quote_literal( tmp_text || '%' );
341 ELSIF rank_adjust.bump_type = 'word_order' THEN
342 SELECT array_to_string( array_accum( term ), '%' ) INTO tmp_text
343 FROM search.parse_search_args(param_searches)
344 WHERE table_alias = query_part.table_alias AND term_type = 'word';
346 tmp_text := query_part.table_alias || '.value ILIKE ' || quote_literal( '%' || tmp_text || '%' );
348 ELSIF rank_adjust.bump_type = 'full_match' THEN
349 SELECT array_to_string( array_accum( term ), E'\\s+' ) INTO tmp_text
350 FROM search.parse_search_args(param_searches)
351 WHERE table_alias = query_part.table_alias AND term_type = 'word';
353 tmp_text := query_part.table_alias || '.value ~ ' || quote_literal( '^' || tmp_text || E'\\W*$' );
357 current_rank := current_rank || ' * ( CASE WHEN ' || tmp_text ||
358 ' THEN ' || rank_adjust.multiplier || '::REAL ELSE 1.0 END )';
360 used_ranks := array_append( used_ranks, rank_adjust.bump_type );
366 ranks := array_append( ranks, current_rank );
369 FOR phrase_query_part IN
371 FROM search.parse_search_args(param_searches)
372 WHERE term_type = 'phrase'
373 AND table_alias = query_part.table_alias LOOP
375 tmp_text := replace( phrase_query_part.term, '*', E'\\*' );
376 tmp_text := replace( tmp_text, '?', E'\\?' );
377 tmp_text := replace( tmp_text, '+', E'\\+' );
378 tmp_text := replace( tmp_text, '|', E'\\|' );
379 tmp_text := replace( tmp_text, '(', E'\\(' );
380 tmp_text := replace( tmp_text, ')', E'\\)' );
381 tmp_text := replace( tmp_text, '[', E'\\[' );
382 tmp_text := replace( tmp_text, ']', E'\\]' );
384 inner_where_clause := inner_where_clause || ' AND ' || 'value ~* ' || quote_literal( E'(^|\\W+)' || regexp_replace(tmp_text, E'\\s+',E'\\\\s+','g') || E'(\\W+|\$)' );
388 query_table := search.pick_table(query_part.field_class);
390 from_clause := from_clause ||
391 ' JOIN ( SELECT * FROM ' || query_table || ' WHERE ' || inner_where_clause ||
392 CASE WHEN core_rel_limit > 0 THEN ' LIMIT ' || core_rel_limit::TEXT ELSE '' END || ' ) AS ' || query_part.table_alias ||
393 ' ON ( m.source = ' || query_part.table_alias || '.source )' ||
394 ' JOIN config.metabib_field AS ' || query_part.table_alias || '_weight' ||
395 ' ON ( ' || query_part.table_alias || '.field = ' || query_part.table_alias || '_weight.id AND ' || query_part.table_alias || '_weight.search_field)';
397 from_alias_array := array_append(from_alias_array, query_part.table_alias);
401 IF param_pref_lang IS NOT NULL AND param_pref_lang_multiplier IS NOT NULL THEN
402 current_rank := ' CASE WHEN mrd.item_lang = ' || quote_literal( param_pref_lang ) ||
403 ' THEN ' || param_pref_lang_multiplier || '::REAL ELSE 1.0 END ';
405 --ranks := array_append( ranks, current_rank );
408 current_rank := ' AVG( ( (' || array_to_string( ranks, ') + (' ) || ') ) * ' || current_rank || ' ) ';
409 select_clause := select_clause || current_rank || ' AS rel,';
411 sort_desc = param_sort_desc;
413 IF param_sort = 'pubdate' THEN
415 tmp_text := '999999';
416 IF param_sort_desc THEN tmp_text := '0'; END IF;
420 SELECT SUBSTRING(frp.value FROM E'\\d{4}')
421 FROM metabib.full_rec frp
422 WHERE frp.record = m.source
424 AND frp.subfield = 'c'
426 )), $$ || quote_literal(tmp_text) || $$ )::INT )
429 ELSIF param_sort = 'title' THEN
431 tmp_text := 'zzzzzz';
432 IF param_sort_desc THEN tmp_text := ' '; END IF;
436 SELECT LTRIM(SUBSTR( frt.value, COALESCE(SUBSTRING(frt.ind2 FROM E'\\d+'),'0')::INT + 1 ))
437 FROM metabib.full_rec frt
438 WHERE frt.record = m.source
440 AND frt.subfield = 'a'
442 )),$$ || quote_literal(tmp_text) || $$))
445 ELSIF param_sort = 'author' THEN
447 tmp_text := 'zzzzzz';
448 IF param_sort_desc THEN tmp_text := ' '; END IF;
452 SELECT LTRIM(fra.value)
453 FROM metabib.full_rec fra
454 WHERE fra.record = m.source
455 AND fra.tag LIKE '1%'
456 AND fra.subfield = 'a'
457 ORDER BY fra.tag::text::int
459 )),$$ || quote_literal(tmp_text) || $$))
462 ELSIF param_sort = 'create_date' THEN
463 current_rank := $$( FIRST (( SELECT create_date FROM biblio.record_entry rbr WHERE rbr.id = m.source)) )$$;
464 ELSIF param_sort = 'edit_date' THEN
465 current_rank := $$( FIRST (( SELECT edit_date FROM biblio.record_entry rbr WHERE rbr.id = m.source)) )$$;
467 sort_desc := NOT COALESCE(param_sort_desc, FALSE);
470 select_clause := select_clause || current_rank || ' AS rank';
472 -- now add the other qualifiers
473 IF param_audience IS NOT NULL AND array_upper(param_audience, 1) > 0 THEN
474 where_clause = where_clause || $$ AND mrd.audience IN ('$$ || array_to_string(param_audience, $$','$$) || $$') $$;
477 IF param_language IS NOT NULL AND array_upper(param_language, 1) > 0 THEN
478 where_clause = where_clause || $$ AND mrd.item_lang IN ('$$ || array_to_string(param_language, $$','$$) || $$') $$;
481 IF param_lit_form IS NOT NULL AND array_upper(param_lit_form, 1) > 0 THEN
482 where_clause = where_clause || $$ AND mrd.lit_form IN ('$$ || array_to_string(param_lit_form, $$','$$) || $$') $$;
485 IF param_types IS NOT NULL AND array_upper(param_types, 1) > 0 THEN
486 where_clause = where_clause || $$ AND mrd.item_type IN ('$$ || array_to_string(param_types, $$','$$) || $$') $$;
489 IF param_forms IS NOT NULL AND array_upper(param_forms, 1) > 0 THEN
490 where_clause = where_clause || $$ AND mrd.item_form IN ('$$ || array_to_string(param_forms, $$','$$) || $$') $$;
493 IF param_vformats IS NOT NULL AND array_upper(param_vformats, 1) > 0 THEN
494 where_clause = where_clause || $$ AND mrd.vr_format IN ('$$ || array_to_string(param_vformats, $$','$$) || $$') $$;
497 core_rel_query := select_clause || from_clause || where_clause ||
498 ' GROUP BY 1 ORDER BY 4' || CASE WHEN sort_desc THEN ' DESC' ELSE ' ASC' END || ';';
499 --RAISE NOTICE 'Base Query: %', core_rel_query;
501 IF param_depth IS NOT NULL THEN
502 SELECT array_accum(distinct id) INTO search_org_list FROM actor.org_unit_descendants( param_search_ou, param_depth );
504 SELECT array_accum(distinct id) INTO search_org_list FROM actor.org_unit_descendants( param_search_ou );
507 OPEN core_cursor FOR EXECUTE core_rel_query;
511 FETCH core_cursor INTO core_result;
514 IF total_count % 1000 = 0 THEN
515 -- RAISE NOTICE ' % total, % checked so far ... ', total_count, check_count;
518 IF core_chk_limit > 0 AND total_count - core_skip_chk + 1 >= core_chk_limit THEN
519 total_count := total_count + 1;
523 total_count := total_count + 1;
525 CONTINUE WHEN param_skip_chk IS NOT NULL and total_count < param_skip_chk;
527 check_count := check_count + 1;
529 PERFORM 1 FROM biblio.record_entry b WHERE NOT b.deleted AND b.id IN ( SELECT * FROM search.explode_array( core_result.records ) );
531 -- RAISE NOTICE ' % were all deleted ... ', core_result.records;
532 deleted_count := deleted_count + 1;
537 FROM biblio.record_entry b
538 JOIN config.bib_source s ON (b.source = s.id)
540 AND b.id IN ( SELECT * FROM search.explode_array( core_result.records ) );
543 -- RAISE NOTICE ' % were all transcendant ... ', core_result.records;
544 visible_count := visible_count + 1;
546 current_res.id = core_result.id;
547 current_res.rel = core_result.rel;
551 SELECT COUNT(DISTINCT s.source) INTO tmp_int FROM metabib.metarecord_source_map s WHERE s.metarecord = core_result.id;
555 current_res.record = core_result.records[1];
557 current_res.record = NULL;
560 RETURN NEXT current_res;
565 IF param_statuses IS NOT NULL AND array_upper(param_statuses, 1) > 0 THEN
568 FROM asset.call_number cn
569 JOIN asset.copy cp ON (cp.call_number = cn.id)
572 AND cp.status IN ( SELECT * FROM search.explode_array( param_statuses ) )
573 AND cn.record IN ( SELECT * FROM search.explode_array( core_result.records ) )
574 AND cp.circ_lib IN ( SELECT * FROM search.explode_array( search_org_list ) )
578 -- RAISE NOTICE ' % were all status-excluded ... ', core_result.records;
579 excluded_count := excluded_count + 1;
585 IF staff IS NULL OR NOT staff THEN
588 FROM asset.call_number cn
589 JOIN asset.copy cp ON (cp.call_number = cn.id)
590 JOIN actor.org_unit a ON (cp.circ_lib = a.id)
591 JOIN asset.copy_location cl ON (cp.location = cl.id)
592 JOIN config.copy_status cs ON (cp.status = cs.id)
599 AND cp.circ_lib IN ( SELECT * FROM search.explode_array( search_org_list ) )
600 AND cn.record IN ( SELECT * FROM search.explode_array( core_result.records ) )
604 -- RAISE NOTICE ' % were all visibility-excluded ... ', core_result.records;
605 excluded_count := excluded_count + 1;
612 FROM asset.call_number cn
613 JOIN asset.copy cp ON (cp.call_number = cn.id)
614 JOIN actor.org_unit a ON (cp.circ_lib = a.id)
615 JOIN asset.copy_location cl ON (cp.location = cl.id)
616 JOIN config.copy_status cs ON (cp.status = cs.id)
619 AND cp.circ_lib IN ( SELECT * FROM search.explode_array( search_org_list ) )
620 AND cn.record IN ( SELECT * FROM search.explode_array( core_result.records ) )
626 FROM asset.call_number cn
627 WHERE cn.record IN ( SELECT * FROM search.explode_array( core_result.records ) )
631 -- RAISE NOTICE ' % were all visibility-excluded ... ', core_result.records;
632 excluded_count := excluded_count + 1;
640 visible_count := visible_count + 1;
642 current_res.id = core_result.id;
643 current_res.rel = core_result.rel;
647 SELECT COUNT(DISTINCT s.source) INTO tmp_int FROM metabib.metarecord_source_map s WHERE s.metarecord = core_result.id;
651 current_res.record = core_result.records[1];
653 current_res.record = NULL;
656 RETURN NEXT current_res;
658 IF visible_count % 1000 = 0 THEN
659 -- RAISE NOTICE ' % visible so far ... ', visible_count;
664 current_res.id = NULL;
665 current_res.rel = NULL;
666 current_res.record = NULL;
667 current_res.total = total_count;
668 current_res.checked = check_count;
669 current_res.deleted = deleted_count;
670 current_res.visible = visible_count;
671 current_res.excluded = excluded_count;
675 RETURN NEXT current_res;
678 $func$ LANGUAGE PLPGSQL;
680 CREATE OR REPLACE FUNCTION search.explode_array(anyarray) RETURNS SETOF anyelement AS $BODY$
681 SELECT ($1)[s] FROM generate_series(1, array_upper($1, 1)) AS s;
683 LANGUAGE 'sql' IMMUTABLE;
685 CREATE OR REPLACE FUNCTION search.parse_search_args (TEXT) RETURNS SETOF search.search_args AS $perlcode$
689 my $args = decode_json( $json );
693 for my $k ( keys %$args ) {
694 (my $alias = $k) =~ s/\|/_/gso;
695 my ($class, $field) = split /\|/, $k;
696 my $part = $args->{$k};
697 for my $p ( keys %$part ) {
698 my $data = $part->{$p};
699 $data = [$data] if (!ref($data));
700 for my $datum ( @$data ) {
702 { field_class => $class,
703 field_name => $field,
705 table_alias => $alias,
717 $perlcode$ LANGUAGE PLPERLU;