2 * psql - the PostgreSQL interactive terminal
4 * Support for the various \d ("describe") commands. Note that the current
5 * expectation is that all functions in this file will succeed when working
6 * with servers of versions 7.4 and up. It's okay to omit irrelevant
7 * information for an old server, but not to fail outright.
9 * Copyright (c) 2000-2008, PostgreSQL Global Development Group
13 #include "postgres_fe.h"
19 #include "dumputils.h"
23 #include "variables.h"
26 static bool describeOneTableDetails(const char *schemaname
,
27 const char *relationname
,
30 static void add_tablespace_footer(printTableContent
*const cont
, char relkind
,
31 Oid tablespace
, const bool newline
);
32 static void add_role_attribute(PQExpBuffer buf
, const char *const str
);
33 static bool listTSParsersVerbose(const char *pattern
);
34 static bool describeOneTSParser(const char *oid
, const char *nspname
,
36 static bool listTSConfigsVerbose(const char *pattern
);
37 static bool describeOneTSConfig(const char *oid
, const char *nspname
,
39 const char *pnspname
, const char *prsname
);
43 * Handlers for various slash commands displaying some sort of list
44 * of things in the database.
46 * Note: try to format the queries to look nice in -E output.
52 * Takes an optional regexp to select particular aggregates
55 describeAggregates(const char *pattern
, bool verbose
)
59 printQueryOpt myopt
= pset
.popt
;
61 initPQExpBuffer(&buf
);
63 printfPQExpBuffer(&buf
,
64 "SELECT n.nspname as \"%s\",\n"
65 " p.proname AS \"%s\",\n"
66 " pg_catalog.format_type(p.prorettype, NULL) AS \"%s\",\n",
67 gettext_noop("Schema"),
69 gettext_noop("Result data type"));
71 if (pset
.sversion
>= 80200)
72 appendPQExpBuffer(&buf
,
73 " CASE WHEN p.pronargs = 0\n"
74 " THEN CAST('*' AS pg_catalog.text)\n"
76 " pg_catalog.array_to_string(ARRAY(\n"
78 " pg_catalog.format_type(p.proargtypes[s.i], NULL)\n"
80 " pg_catalog.generate_series(0, pg_catalog.array_upper(p.proargtypes, 1)) AS s(i)\n"
83 gettext_noop("Argument data types"));
85 appendPQExpBuffer(&buf
,
86 " pg_catalog.format_type(p.proargtypes[0], NULL) AS \"%s\",\n",
87 gettext_noop("Argument data types"));
89 appendPQExpBuffer(&buf
,
90 " pg_catalog.obj_description(p.oid, 'pg_proc') as \"%s\"\n"
91 "FROM pg_catalog.pg_proc p\n"
92 " LEFT JOIN pg_catalog.pg_namespace n ON n.oid = p.pronamespace\n"
94 gettext_noop("Description"));
96 processSQLNamePattern(pset
.db
, &buf
, pattern
, true, false,
97 "n.nspname", "p.proname", NULL
,
98 "pg_catalog.pg_function_is_visible(p.oid)");
100 appendPQExpBuffer(&buf
, "ORDER BY 1, 2, 4;");
102 res
= PSQLexec(buf
.data
, false);
103 termPQExpBuffer(&buf
);
107 myopt
.nullPrint
= NULL
;
108 myopt
.title
= _("List of aggregate functions");
109 myopt
.translate_header
= true;
111 printQuery(res
, &myopt
, pset
.queryFout
, pset
.logfile
);
118 * Takes an optional regexp to select particular tablespaces
121 describeTablespaces(const char *pattern
, bool verbose
)
125 printQueryOpt myopt
= pset
.popt
;
127 if (pset
.sversion
< 80000)
129 fprintf(stderr
, _("The server (version %d.%d) does not support tablespaces.\n"),
130 pset
.sversion
/ 10000, (pset
.sversion
/ 100) % 100);
134 initPQExpBuffer(&buf
);
136 printfPQExpBuffer(&buf
,
137 "SELECT spcname AS \"%s\",\n"
138 " pg_catalog.pg_get_userbyid(spcowner) AS \"%s\",\n"
139 " spclocation AS \"%s\"",
140 gettext_noop("Name"),
141 gettext_noop("Owner"),
142 gettext_noop("Location"));
145 appendPQExpBuffer(&buf
,
146 ",\n spcacl AS \"%s\"",
147 gettext_noop("Access privileges"));
148 if (verbose
&& pset
.sversion
>= 80200)
149 appendPQExpBuffer(&buf
,
150 ",\n pg_catalog.shobj_description(oid, 'pg_tablespace') AS \"%s\"",
151 gettext_noop("Description"));
153 appendPQExpBuffer(&buf
,
154 "\nFROM pg_catalog.pg_tablespace\n");
156 processSQLNamePattern(pset
.db
, &buf
, pattern
, false, false,
157 NULL
, "spcname", NULL
,
160 appendPQExpBuffer(&buf
, "ORDER BY 1;");
162 res
= PSQLexec(buf
.data
, false);
163 termPQExpBuffer(&buf
);
167 myopt
.nullPrint
= NULL
;
168 myopt
.title
= _("List of tablespaces");
169 myopt
.translate_header
= true;
171 printQuery(res
, &myopt
, pset
.queryFout
, pset
.logfile
);
179 * Takes an optional regexp to select particular functions
182 describeFunctions(const char *pattern
, bool verbose
)
186 printQueryOpt myopt
= pset
.popt
;
188 initPQExpBuffer(&buf
);
190 printfPQExpBuffer(&buf
,
191 "SELECT n.nspname as \"%s\",\n"
192 " p.proname as \"%s\",\n",
193 gettext_noop("Schema"),
194 gettext_noop("Name"));
196 if (pset
.sversion
>= 80400)
197 appendPQExpBuffer(&buf
,
198 " pg_catalog.pg_get_function_result(p.oid) as \"%s\",\n"
199 " pg_catalog.pg_get_function_arguments(p.oid) as \"%s\"",
200 gettext_noop("Result data type"),
201 gettext_noop("Argument data types"));
202 else if (pset
.sversion
>= 80100)
203 appendPQExpBuffer(&buf
,
204 " CASE WHEN p.proretset THEN 'SETOF ' ELSE '' END ||\n"
205 " pg_catalog.format_type(p.prorettype, NULL) as \"%s\",\n"
206 " CASE WHEN proallargtypes IS NOT NULL THEN\n"
207 " pg_catalog.array_to_string(ARRAY(\n"
210 " WHEN p.proargmodes[s.i] = 'i' THEN ''\n"
211 " WHEN p.proargmodes[s.i] = 'o' THEN 'OUT '\n"
212 " WHEN p.proargmodes[s.i] = 'b' THEN 'INOUT '\n"
213 " WHEN p.proargmodes[s.i] = 'v' THEN 'VARIADIC '\n"
216 " WHEN COALESCE(p.proargnames[s.i], '') = '' THEN ''\n"
217 " ELSE p.proargnames[s.i] || ' ' \n"
219 " pg_catalog.format_type(p.proallargtypes[s.i], NULL)\n"
221 " pg_catalog.generate_series(1, pg_catalog.array_upper(p.proallargtypes, 1)) AS s(i)\n"
224 " pg_catalog.array_to_string(ARRAY(\n"
227 " WHEN COALESCE(p.proargnames[s.i+1], '') = '' THEN ''\n"
228 " ELSE p.proargnames[s.i+1] || ' '\n"
230 " pg_catalog.format_type(p.proargtypes[s.i], NULL)\n"
232 " pg_catalog.generate_series(0, pg_catalog.array_upper(p.proargtypes, 1)) AS s(i)\n"
235 gettext_noop("Result data type"),
236 gettext_noop("Argument data types"));
238 appendPQExpBuffer(&buf
,
239 " CASE WHEN p.proretset THEN 'SETOF ' ELSE '' END ||\n"
240 " pg_catalog.format_type(p.prorettype, NULL) as \"%s\",\n"
241 " pg_catalog.oidvectortypes(p.proargtypes) as \"%s\"",
242 gettext_noop("Result data type"),
243 gettext_noop("Argument data types"));
246 appendPQExpBuffer(&buf
,
248 " WHEN p.provolatile = 'i' THEN 'immutable'\n"
249 " WHEN p.provolatile = 's' THEN 'stable'\n"
250 " WHEN p.provolatile = 'v' THEN 'volatile'\n"
252 ",\n pg_catalog.pg_get_userbyid(p.proowner) as \"%s\",\n"
253 " l.lanname as \"%s\",\n"
254 " p.prosrc as \"%s\",\n"
255 " pg_catalog.obj_description(p.oid, 'pg_proc') as \"%s\"",
256 gettext_noop("Volatility"),
257 gettext_noop("Owner"),
258 gettext_noop("Language"),
259 gettext_noop("Source code"),
260 gettext_noop("Description"));
262 appendPQExpBuffer(&buf
,
263 "\nFROM pg_catalog.pg_proc p"
264 "\n LEFT JOIN pg_catalog.pg_namespace n ON n.oid = p.pronamespace\n");
267 appendPQExpBuffer(&buf
,
268 " LEFT JOIN pg_catalog.pg_language l ON l.oid = p.prolang\n");
271 * we skip in/out funcs by excluding functions that take or return cstring
273 appendPQExpBuffer(&buf
,
274 "WHERE p.prorettype <> 'pg_catalog.cstring'::pg_catalog.regtype\n"
275 " AND p.proargtypes[0] IS DISTINCT FROM 'pg_catalog.cstring'::pg_catalog.regtype\n"
276 " AND NOT p.proisagg\n");
278 processSQLNamePattern(pset
.db
, &buf
, pattern
, true, false,
279 "n.nspname", "p.proname", NULL
,
280 "pg_catalog.pg_function_is_visible(p.oid)");
282 appendPQExpBuffer(&buf
, "ORDER BY 1, 2, 4;");
284 res
= PSQLexec(buf
.data
, false);
285 termPQExpBuffer(&buf
);
289 myopt
.nullPrint
= NULL
;
290 myopt
.title
= _("List of functions");
291 myopt
.translate_header
= true;
293 printQuery(res
, &myopt
, pset
.queryFout
, pset
.logfile
);
306 describeTypes(const char *pattern
, bool verbose
)
310 printQueryOpt myopt
= pset
.popt
;
312 initPQExpBuffer(&buf
);
314 printfPQExpBuffer(&buf
,
315 "SELECT n.nspname as \"%s\",\n"
316 " pg_catalog.format_type(t.oid, NULL) AS \"%s\",\n",
317 gettext_noop("Schema"),
318 gettext_noop("Name"));
320 appendPQExpBuffer(&buf
,
321 " t.typname AS \"%s\",\n"
322 " CASE WHEN t.typrelid != 0\n"
323 " THEN CAST('tuple' AS pg_catalog.text)\n"
324 " WHEN t.typlen < 0\n"
325 " THEN CAST('var' AS pg_catalog.text)\n"
326 " ELSE CAST(t.typlen AS pg_catalog.text)\n"
328 gettext_noop("Internal name"),
329 gettext_noop("Size"));
330 if (verbose
&& pset
.sversion
>= 80300)
331 appendPQExpBuffer(&buf
,
332 " pg_catalog.array_to_string(\n"
334 " SELECT e.enumlabel\n"
335 " FROM pg_catalog.pg_enum e\n"
336 " WHERE e.enumtypid = t.oid\n"
341 gettext_noop("Elements"));
343 appendPQExpBuffer(&buf
,
344 " pg_catalog.obj_description(t.oid, 'pg_type') as \"%s\"\n",
345 gettext_noop("Description"));
347 appendPQExpBuffer(&buf
, "FROM pg_catalog.pg_type t\n"
348 " LEFT JOIN pg_catalog.pg_namespace n ON n.oid = t.typnamespace\n");
351 * do not include complex types (typrelid!=0) unless they are standalone
354 appendPQExpBuffer(&buf
, "WHERE (t.typrelid = 0 ");
355 appendPQExpBuffer(&buf
, "OR (SELECT c.relkind = 'c' FROM pg_catalog.pg_class c "
356 "WHERE c.oid = t.typrelid))\n");
358 * do not include array types (before 8.3 we have to use the assumption
359 * that their names start with underscore)
361 if (pset
.sversion
>= 80300)
362 appendPQExpBuffer(&buf
, " AND NOT EXISTS(SELECT 1 FROM pg_catalog.pg_type el WHERE el.oid = t.typelem AND el.typarray = t.oid)\n");
364 appendPQExpBuffer(&buf
, " AND t.typname !~ '^_'\n");
366 /* Match name pattern against either internal or external name */
367 processSQLNamePattern(pset
.db
, &buf
, pattern
, true, false,
368 "n.nspname", "t.typname",
369 "pg_catalog.format_type(t.oid, NULL)",
370 "pg_catalog.pg_type_is_visible(t.oid)");
372 appendPQExpBuffer(&buf
, "ORDER BY 1, 2;");
374 res
= PSQLexec(buf
.data
, false);
375 termPQExpBuffer(&buf
);
379 myopt
.nullPrint
= NULL
;
380 myopt
.title
= _("List of data types");
381 myopt
.translate_header
= true;
383 printQuery(res
, &myopt
, pset
.queryFout
, pset
.logfile
);
393 describeOperators(const char *pattern
)
397 printQueryOpt myopt
= pset
.popt
;
399 initPQExpBuffer(&buf
);
401 printfPQExpBuffer(&buf
,
402 "SELECT n.nspname as \"%s\",\n"
403 " o.oprname AS \"%s\",\n"
404 " CASE WHEN o.oprkind='l' THEN NULL ELSE pg_catalog.format_type(o.oprleft, NULL) END AS \"%s\",\n"
405 " CASE WHEN o.oprkind='r' THEN NULL ELSE pg_catalog.format_type(o.oprright, NULL) END AS \"%s\",\n"
406 " pg_catalog.format_type(o.oprresult, NULL) AS \"%s\",\n"
407 " coalesce(pg_catalog.obj_description(o.oid, 'pg_operator'),\n"
408 " pg_catalog.obj_description(o.oprcode, 'pg_proc')) AS \"%s\"\n"
409 "FROM pg_catalog.pg_operator o\n"
410 " LEFT JOIN pg_catalog.pg_namespace n ON n.oid = o.oprnamespace\n",
411 gettext_noop("Schema"),
412 gettext_noop("Name"),
413 gettext_noop("Left arg type"),
414 gettext_noop("Right arg type"),
415 gettext_noop("Result type"),
416 gettext_noop("Description"));
418 processSQLNamePattern(pset
.db
, &buf
, pattern
, false, true,
419 "n.nspname", "o.oprname", NULL
,
420 "pg_catalog.pg_operator_is_visible(o.oid)");
422 appendPQExpBuffer(&buf
, "ORDER BY 1, 2, 3, 4;");
424 res
= PSQLexec(buf
.data
, false);
425 termPQExpBuffer(&buf
);
429 myopt
.nullPrint
= NULL
;
430 myopt
.title
= _("List of operators");
431 myopt
.translate_header
= true;
433 printQuery(res
, &myopt
, pset
.queryFout
, pset
.logfile
);
443 * for \l, \list, and -l switch
446 listAllDbs(bool verbose
)
450 printQueryOpt myopt
= pset
.popt
;
452 initPQExpBuffer(&buf
);
454 printfPQExpBuffer(&buf
,
455 "SELECT d.datname as \"%s\",\n"
456 " pg_catalog.pg_get_userbyid(d.datdba) as \"%s\",\n"
457 " pg_catalog.pg_encoding_to_char(d.encoding) as \"%s\",\n",
458 gettext_noop("Name"),
459 gettext_noop("Owner"),
460 gettext_noop("Encoding"));
461 if (pset
.sversion
>= 80400)
462 appendPQExpBuffer(&buf
,
463 " d.datcollate as \"%s\",\n"
464 " d.datctype as \"%s\",\n",
465 gettext_noop("Collation"),
466 gettext_noop("Ctype"));
467 appendPQExpBuffer(&buf
,
468 " d.datacl as \"%s\"",
469 gettext_noop("Access Privileges"));
470 if (verbose
&& pset
.sversion
>= 80200)
471 appendPQExpBuffer(&buf
,
472 ",\n CASE WHEN pg_catalog.has_database_privilege(d.datname, 'CONNECT')\n"
473 " THEN pg_catalog.pg_size_pretty(pg_catalog.pg_database_size(d.datname))\n"
474 " ELSE 'No Access'\n"
476 gettext_noop("Size"));
477 if (verbose
&& pset
.sversion
>= 80000)
478 appendPQExpBuffer(&buf
,
479 ",\n t.spcname as \"%s\"",
480 gettext_noop("Tablespace"));
481 if (verbose
&& pset
.sversion
>= 80200)
482 appendPQExpBuffer(&buf
,
483 ",\n pg_catalog.shobj_description(d.oid, 'pg_database') as \"%s\"",
484 gettext_noop("Description"));
485 appendPQExpBuffer(&buf
,
486 "\nFROM pg_catalog.pg_database d\n");
487 if (verbose
&& pset
.sversion
>= 80000)
488 appendPQExpBuffer(&buf
,
489 " JOIN pg_catalog.pg_tablespace t on d.dattablespace = t.oid\n");
490 appendPQExpBuffer(&buf
, "ORDER BY 1;");
491 res
= PSQLexec(buf
.data
, false);
492 termPQExpBuffer(&buf
);
496 myopt
.nullPrint
= NULL
;
497 myopt
.title
= _("List of databases");
498 myopt
.translate_header
= true;
500 printQuery(res
, &myopt
, pset
.queryFout
, pset
.logfile
);
508 * List Tables Grant/Revoke Permissions
509 * \z (now also \dp -- perhaps more mnemonic)
512 permissionsList(const char *pattern
)
516 printQueryOpt myopt
= pset
.popt
;
517 static const bool translate_columns
[] = {false, false, true, false};
519 initPQExpBuffer(&buf
);
522 * we ignore indexes and toast tables since they have no meaningful rights
524 printfPQExpBuffer(&buf
,
525 "SELECT n.nspname as \"%s\",\n"
526 " c.relname as \"%s\",\n"
527 " CASE c.relkind WHEN 'r' THEN '%s' WHEN 'v' THEN '%s' WHEN 'S' THEN '%s' END as \"%s\",\n",
528 gettext_noop("Schema"),
529 gettext_noop("Name"),
530 gettext_noop("table"), gettext_noop("view"), gettext_noop("sequence"),
531 gettext_noop("Type"));
533 if (pset
.sversion
>= 80100)
534 appendPQExpBuffer(&buf
, " pg_catalog.array_to_string(c.relacl, E'\\n') as \"%s\"\n",
535 gettext_noop("Access privileges"));
537 appendPQExpBuffer(&buf
, " pg_catalog.array_to_string(c.relacl, '\\n') as \"%s\"\n",
538 gettext_noop("Access privileges"));
540 appendPQExpBuffer(&buf
, "FROM pg_catalog.pg_class c\n"
541 " LEFT JOIN pg_catalog.pg_namespace n ON n.oid = c.relnamespace\n"
542 "WHERE c.relkind IN ('r', 'v', 'S')\n");
545 * Unless a schema pattern is specified, we suppress system and temp
546 * tables, since they normally aren't very interesting from a permissions
547 * point of view. You can see 'em by explicit request though, eg with \z
550 processSQLNamePattern(pset
.db
, &buf
, pattern
, true, false,
551 "n.nspname", "c.relname", NULL
,
552 "n.nspname !~ '^pg_' AND pg_catalog.pg_table_is_visible(c.oid)");
554 appendPQExpBuffer(&buf
, "ORDER BY 1, 2;");
556 res
= PSQLexec(buf
.data
, false);
559 termPQExpBuffer(&buf
);
563 myopt
.nullPrint
= NULL
;
564 printfPQExpBuffer(&buf
, _("Access privileges"));
565 myopt
.title
= buf
.data
;
566 myopt
.translate_header
= true;
567 myopt
.translate_columns
= translate_columns
;
569 printQuery(res
, &myopt
, pset
.queryFout
, pset
.logfile
);
571 termPQExpBuffer(&buf
);
579 * Get object comments
583 * Note: This only lists things that actually have a description. For complete
584 * lists of things, there are other \d? commands.
587 objectDescription(const char *pattern
)
591 printQueryOpt myopt
= pset
.popt
;
592 static const bool translate_columns
[] = {false, false, true, false};
594 initPQExpBuffer(&buf
);
596 appendPQExpBuffer(&buf
,
597 "SELECT DISTINCT tt.nspname AS \"%s\", tt.name AS \"%s\", tt.object AS \"%s\", d.description AS \"%s\"\n"
599 gettext_noop("Schema"),
600 gettext_noop("Name"),
601 gettext_noop("Object"),
602 gettext_noop("Description"));
604 /* Aggregate descriptions */
605 appendPQExpBuffer(&buf
,
606 " SELECT p.oid as oid, p.tableoid as tableoid,\n"
607 " n.nspname as nspname,\n"
608 " CAST(p.proname AS pg_catalog.text) as name,"
609 " CAST('%s' AS pg_catalog.text) as object\n"
610 " FROM pg_catalog.pg_proc p\n"
611 " LEFT JOIN pg_catalog.pg_namespace n ON n.oid = p.pronamespace\n"
612 " WHERE p.proisagg\n",
613 gettext_noop("aggregate"));
614 processSQLNamePattern(pset
.db
, &buf
, pattern
, true, false,
615 "n.nspname", "p.proname", NULL
,
616 "pg_catalog.pg_function_is_visible(p.oid)");
618 /* Function descriptions (except in/outs for datatypes) */
619 appendPQExpBuffer(&buf
,
621 " SELECT p.oid as oid, p.tableoid as tableoid,\n"
622 " n.nspname as nspname,\n"
623 " CAST(p.proname AS pg_catalog.text) as name,"
624 " CAST('%s' AS pg_catalog.text) as object\n"
625 " FROM pg_catalog.pg_proc p\n"
626 " LEFT JOIN pg_catalog.pg_namespace n ON n.oid = p.pronamespace\n"
628 " WHERE p.prorettype <> 'pg_catalog.cstring'::pg_catalog.regtype\n"
629 " AND (p.proargtypes[0] IS NULL\n"
630 " OR p.proargtypes[0] <> 'pg_catalog.cstring'::pg_catalog.regtype)\n"
631 " AND NOT p.proisagg\n",
632 gettext_noop("function"));
633 processSQLNamePattern(pset
.db
, &buf
, pattern
, true, false,
634 "n.nspname", "p.proname", NULL
,
635 "pg_catalog.pg_function_is_visible(p.oid)");
637 /* Operator descriptions (only if operator has its own comment) */
638 appendPQExpBuffer(&buf
,
640 " SELECT o.oid as oid, o.tableoid as tableoid,\n"
641 " n.nspname as nspname,\n"
642 " CAST(o.oprname AS pg_catalog.text) as name,"
643 " CAST('%s' AS pg_catalog.text) as object\n"
644 " FROM pg_catalog.pg_operator o\n"
645 " LEFT JOIN pg_catalog.pg_namespace n ON n.oid = o.oprnamespace\n",
646 gettext_noop("operator"));
647 processSQLNamePattern(pset
.db
, &buf
, pattern
, false, false,
648 "n.nspname", "o.oprname", NULL
,
649 "pg_catalog.pg_operator_is_visible(o.oid)");
651 /* Type description */
652 appendPQExpBuffer(&buf
,
654 " SELECT t.oid as oid, t.tableoid as tableoid,\n"
655 " n.nspname as nspname,\n"
656 " pg_catalog.format_type(t.oid, NULL) as name,"
657 " CAST('%s' AS pg_catalog.text) as object\n"
658 " FROM pg_catalog.pg_type t\n"
659 " LEFT JOIN pg_catalog.pg_namespace n ON n.oid = t.typnamespace\n",
660 gettext_noop("data type"));
661 processSQLNamePattern(pset
.db
, &buf
, pattern
, false, false,
662 "n.nspname", "pg_catalog.format_type(t.oid, NULL)",
664 "pg_catalog.pg_type_is_visible(t.oid)");
666 /* Relation (tables, views, indexes, sequences) descriptions */
667 appendPQExpBuffer(&buf
,
669 " SELECT c.oid as oid, c.tableoid as tableoid,\n"
670 " n.nspname as nspname,\n"
671 " CAST(c.relname AS pg_catalog.text) as name,\n"
673 " CASE c.relkind WHEN 'r' THEN '%s' WHEN 'v' THEN '%s' WHEN 'i' THEN '%s' WHEN 'S' THEN '%s' END"
674 " AS pg_catalog.text) as object\n"
675 " FROM pg_catalog.pg_class c\n"
676 " LEFT JOIN pg_catalog.pg_namespace n ON n.oid = c.relnamespace\n"
677 " WHERE c.relkind IN ('r', 'v', 'i', 'S')\n",
678 gettext_noop("table"),
679 gettext_noop("view"),
680 gettext_noop("index"),
681 gettext_noop("sequence"));
682 processSQLNamePattern(pset
.db
, &buf
, pattern
, true, false,
683 "n.nspname", "c.relname", NULL
,
684 "pg_catalog.pg_table_is_visible(c.oid)");
686 /* Rule description (ignore rules for views) */
687 appendPQExpBuffer(&buf
,
689 " SELECT r.oid as oid, r.tableoid as tableoid,\n"
690 " n.nspname as nspname,\n"
691 " CAST(r.rulename AS pg_catalog.text) as name,"
692 " CAST('%s' AS pg_catalog.text) as object\n"
693 " FROM pg_catalog.pg_rewrite r\n"
694 " JOIN pg_catalog.pg_class c ON c.oid = r.ev_class\n"
695 " LEFT JOIN pg_catalog.pg_namespace n ON n.oid = c.relnamespace\n"
696 " WHERE r.rulename != '_RETURN'\n",
697 gettext_noop("rule"));
698 /* XXX not sure what to do about visibility rule here? */
699 processSQLNamePattern(pset
.db
, &buf
, pattern
, true, false,
700 "n.nspname", "r.rulename", NULL
,
701 "pg_catalog.pg_table_is_visible(c.oid)");
703 /* Trigger description */
704 appendPQExpBuffer(&buf
,
706 " SELECT t.oid as oid, t.tableoid as tableoid,\n"
707 " n.nspname as nspname,\n"
708 " CAST(t.tgname AS pg_catalog.text) as name,"
709 " CAST('%s' AS pg_catalog.text) as object\n"
710 " FROM pg_catalog.pg_trigger t\n"
711 " JOIN pg_catalog.pg_class c ON c.oid = t.tgrelid\n"
712 " LEFT JOIN pg_catalog.pg_namespace n ON n.oid = c.relnamespace\n",
713 gettext_noop("trigger"));
714 /* XXX not sure what to do about visibility rule here? */
715 processSQLNamePattern(pset
.db
, &buf
, pattern
, false, false,
716 "n.nspname", "t.tgname", NULL
,
717 "pg_catalog.pg_table_is_visible(c.oid)");
719 appendPQExpBuffer(&buf
,
721 " JOIN pg_catalog.pg_description d ON (tt.oid = d.objoid AND tt.tableoid = d.classoid AND d.objsubid = 0)\n");
723 appendPQExpBuffer(&buf
, "ORDER BY 1, 2, 3;");
725 res
= PSQLexec(buf
.data
, false);
726 termPQExpBuffer(&buf
);
730 myopt
.nullPrint
= NULL
;
731 myopt
.title
= _("Object descriptions");
732 myopt
.translate_header
= true;
733 myopt
.translate_columns
= translate_columns
;
735 printQuery(res
, &myopt
, pset
.queryFout
, pset
.logfile
);
743 * describeTableDetails (for \d)
745 * This routine finds the tables to be displayed, and calls
746 * describeOneTableDetails for each one.
748 * verbose: if true, this is \d+
751 describeTableDetails(const char *pattern
, bool verbose
)
757 initPQExpBuffer(&buf
);
759 printfPQExpBuffer(&buf
,
763 "FROM pg_catalog.pg_class c\n"
764 " LEFT JOIN pg_catalog.pg_namespace n ON n.oid = c.relnamespace\n");
766 processSQLNamePattern(pset
.db
, &buf
, pattern
, false, false,
767 "n.nspname", "c.relname", NULL
,
768 "pg_catalog.pg_table_is_visible(c.oid)");
770 appendPQExpBuffer(&buf
, "ORDER BY 2, 3;");
772 res
= PSQLexec(buf
.data
, false);
773 termPQExpBuffer(&buf
);
777 if (PQntuples(res
) == 0)
780 fprintf(stderr
, _("Did not find any relation named \"%s\".\n"),
786 for (i
= 0; i
< PQntuples(res
); i
++)
792 oid
= PQgetvalue(res
, i
, 0);
793 nspname
= PQgetvalue(res
, i
, 1);
794 relname
= PQgetvalue(res
, i
, 2);
796 if (!describeOneTableDetails(nspname
, relname
, oid
, verbose
))
813 * describeOneTableDetails (for \d)
815 * Unfortunately, the information presented here is so complicated that it
816 * cannot be done in a single query. So we have to assemble the printed table
817 * by hand and pass it to the underlying printTable() function.
820 describeOneTableDetails(const char *schemaname
,
821 const char *relationname
,
826 PGresult
*res
= NULL
;
827 printTableOpt myopt
= pset
.popt
.topt
;
828 printTableContent cont
;
829 bool printTableInitialized
= false;
831 char *view_def
= NULL
;
833 char **seq_values
= NULL
;
834 char **modifiers
= NULL
;
836 PQExpBufferData title
;
837 PQExpBufferData tmpbuf
;
850 bool show_modifiers
= false;
855 /* This output looks confusing in expanded mode. */
856 myopt
.expanded
= false;
858 initPQExpBuffer(&buf
);
859 initPQExpBuffer(&title
);
860 initPQExpBuffer(&tmpbuf
);
862 /* Get general table info */
863 printfPQExpBuffer(&buf
,
864 "SELECT relchecks, relkind, relhasindex, relhasrules, %s, "
866 "FROM pg_catalog.pg_class WHERE oid = '%s'",
867 (pset
.sversion
>= 80400 ? "relhastriggers" : "reltriggers <> 0"),
868 (pset
.sversion
>= 80000 ? ", reltablespace" : ""),
870 res
= PSQLexec(buf
.data
, false);
874 /* Did we get anything? */
875 if (PQntuples(res
) == 0)
878 fprintf(stderr
, _("Did not find any relation with OID %s.\n"),
883 tableinfo
.checks
= atoi(PQgetvalue(res
, 0, 0));
884 tableinfo
.relkind
= *(PQgetvalue(res
, 0, 1));
885 tableinfo
.hasindex
= strcmp(PQgetvalue(res
, 0, 2), "t") == 0;
886 tableinfo
.hasrules
= strcmp(PQgetvalue(res
, 0, 3), "t") == 0;
887 tableinfo
.hastriggers
= strcmp(PQgetvalue(res
, 0, 4), "t") == 0;
888 tableinfo
.hasoids
= strcmp(PQgetvalue(res
, 0, 5), "t") == 0;
889 tableinfo
.tablespace
= (pset
.sversion
>= 80000) ?
890 atooid(PQgetvalue(res
, 0, 6)) : 0;
895 * If it's a sequence, fetch its values and store into an
896 * array that will be used later.
898 if (tableinfo
.relkind
== 'S')
902 #define SEQ_NUM_COLS 10
903 printfPQExpBuffer(&buf
,
904 "SELECT sequence_name, last_value,\n"
905 " start_value, increment_by,\n"
906 " max_value, min_value, cache_value,\n"
907 " log_cnt, is_cycled, is_called\n"
910 /* must be separate because fmtId isn't reentrant */
911 appendPQExpBuffer(&buf
, ".%s", fmtId(relationname
));
913 result
= PSQLexec(buf
.data
, false);
917 seq_values
= pg_malloc_zero((SEQ_NUM_COLS
+1) * sizeof(*seq_values
));
919 for (i
= 0; i
< SEQ_NUM_COLS
; i
++)
920 seq_values
[i
] = pg_strdup(PQgetvalue(result
, 0, i
));
925 /* Get column info (index requires additional checks) */
926 printfPQExpBuffer(&buf
, "SELECT a.attname,");
927 appendPQExpBuffer(&buf
, "\n pg_catalog.format_type(a.atttypid, a.atttypmod),"
928 "\n (SELECT substring(pg_catalog.pg_get_expr(d.adbin, d.adrelid) for 128)"
929 "\n FROM pg_catalog.pg_attrdef d"
930 "\n WHERE d.adrelid = a.attrelid AND d.adnum = a.attnum AND a.atthasdef),"
931 "\n a.attnotnull, a.attnum");
933 appendPQExpBuffer(&buf
, ", a.attstorage, pg_catalog.col_description(a.attrelid, a.attnum)");
934 appendPQExpBuffer(&buf
, "\nFROM pg_catalog.pg_attribute a");
935 if (tableinfo
.relkind
== 'i')
936 appendPQExpBuffer(&buf
, ", pg_catalog.pg_index i");
937 appendPQExpBuffer(&buf
, "\nWHERE a.attrelid = '%s' AND a.attnum > 0 AND NOT a.attisdropped", oid
);
938 if (tableinfo
.relkind
== 'i')
939 appendPQExpBuffer(&buf
, " AND a.attrelid = i.indexrelid");
940 appendPQExpBuffer(&buf
, "\nORDER BY a.attnum");
942 res
= PSQLexec(buf
.data
, false);
945 numrows
= PQntuples(res
);
948 switch (tableinfo
.relkind
)
951 printfPQExpBuffer(&title
, _("Table \"%s.%s\""),
952 schemaname
, relationname
);
955 printfPQExpBuffer(&title
, _("View \"%s.%s\""),
956 schemaname
, relationname
);
959 printfPQExpBuffer(&title
, _("Sequence \"%s.%s\""),
960 schemaname
, relationname
);
963 printfPQExpBuffer(&title
, _("Index \"%s.%s\""),
964 schemaname
, relationname
);
967 /* not used as of 8.2, but keep it for backwards compatibility */
968 printfPQExpBuffer(&title
, _("Special relation \"%s.%s\""),
969 schemaname
, relationname
);
972 printfPQExpBuffer(&title
, _("TOAST table \"%s.%s\""),
973 schemaname
, relationname
);
976 printfPQExpBuffer(&title
, _("Composite type \"%s.%s\""),
977 schemaname
, relationname
);
980 /* untranslated unknown relkind */
981 printfPQExpBuffer(&title
, "?%c? \"%s.%s\"",
982 tableinfo
.relkind
, schemaname
, relationname
);
986 /* Set the number of columns, and their names */
988 headers
[0] = gettext_noop("Column");
989 headers
[1] = gettext_noop("Type");
991 if (tableinfo
.relkind
== 'r' || tableinfo
.relkind
== 'v')
993 show_modifiers
= true;
994 headers
[cols
++] = gettext_noop("Modifiers");
995 modifiers
= pg_malloc_zero((numrows
+ 1) * sizeof(*modifiers
));
998 if (tableinfo
.relkind
== 'S')
999 headers
[cols
++] = gettext_noop("Value");
1003 headers
[cols
++] = gettext_noop("Storage");
1004 headers
[cols
++] = gettext_noop("Description");
1007 printTableInit(&cont
, &myopt
, title
.data
, cols
, numrows
);
1008 printTableInitialized
= true;
1010 for (i
= 0; i
< cols
; i
++)
1011 printTableAddHeader(&cont
, headers
[i
], true, 'l');
1013 /* Check if table is a view */
1014 if (tableinfo
.relkind
== 'v')
1018 printfPQExpBuffer(&buf
,
1019 "SELECT pg_catalog.pg_get_viewdef('%s'::pg_catalog.oid, true)",
1021 result
= PSQLexec(buf
.data
, false);
1025 if (PQntuples(result
) > 0)
1026 view_def
= pg_strdup(PQgetvalue(result
, 0, 0));
1031 /* Generate table cells to be printed */
1032 for (i
= 0; i
< numrows
; i
++)
1035 printTableAddCell(&cont
, PQgetvalue(res
, i
, 0), false);
1038 printTableAddCell(&cont
, PQgetvalue(res
, i
, 1), false);
1040 /* Modifiers: not null and default */
1043 resetPQExpBuffer(&tmpbuf
);
1044 if (strcmp(PQgetvalue(res
, i
, 3), "t") == 0)
1045 appendPQExpBufferStr(&tmpbuf
, _("not null"));
1047 /* handle "default" here */
1048 /* (note: above we cut off the 'default' string at 128) */
1049 if (strlen(PQgetvalue(res
, i
, 2)) != 0)
1052 appendPQExpBufferStr(&tmpbuf
, " ");
1053 /* translator: default values of column definitions */
1054 appendPQExpBuffer(&tmpbuf
, _("default %s"),
1055 PQgetvalue(res
, i
, 2));
1058 modifiers
[i
] = pg_strdup(tmpbuf
.data
);
1059 printTableAddCell(&cont
, modifiers
[i
], false);
1062 /* Value: for sequences only */
1063 if (tableinfo
.relkind
== 'S')
1064 printTableAddCell(&cont
, seq_values
[i
], false);
1066 /* Storage and Description */
1069 char *storage
= PQgetvalue(res
, i
, 5);
1071 /* these strings are literal in our syntax, so not translated. */
1072 printTableAddCell(&cont
, (storage
[0]=='p' ? "plain" :
1073 (storage
[0]=='m' ? "main" :
1074 (storage
[0]=='x' ? "extended" :
1075 (storage
[0]=='e' ? "external" :
1078 printTableAddCell(&cont
, PQgetvalue(res
, i
, 6), false);
1083 if (tableinfo
.relkind
== 'i')
1085 /* Footer information about an index */
1088 printfPQExpBuffer(&buf
,
1089 "SELECT i.indisunique, i.indisprimary, i.indisclustered, ");
1090 if (pset
.sversion
>= 80200)
1091 appendPQExpBuffer(&buf
, "i.indisvalid, ");
1093 appendPQExpBuffer(&buf
, "true as indisvalid, ");
1094 appendPQExpBuffer(&buf
, "a.amname, c2.relname,\n"
1095 " pg_catalog.pg_get_expr(i.indpred, i.indrelid, true)\n"
1096 "FROM pg_catalog.pg_index i, pg_catalog.pg_class c, pg_catalog.pg_class c2, pg_catalog.pg_am a\n"
1097 "WHERE i.indexrelid = c.oid AND c.oid = '%s' AND c.relam = a.oid\n"
1098 "AND i.indrelid = c2.oid",
1101 result
= PSQLexec(buf
.data
, false);
1104 else if (PQntuples(result
) != 1)
1111 char *indisunique
= PQgetvalue(result
, 0, 0);
1112 char *indisprimary
= PQgetvalue(result
, 0, 1);
1113 char *indisclustered
= PQgetvalue(result
, 0, 2);
1114 char *indisvalid
= PQgetvalue(result
, 0, 3);
1115 char *indamname
= PQgetvalue(result
, 0, 4);
1116 char *indtable
= PQgetvalue(result
, 0, 5);
1117 char *indpred
= PQgetvalue(result
, 0, 6);
1119 if (strcmp(indisprimary
, "t") == 0)
1120 printfPQExpBuffer(&tmpbuf
, _("primary key, "));
1121 else if (strcmp(indisunique
, "t") == 0)
1122 printfPQExpBuffer(&tmpbuf
, _("unique, "));
1124 resetPQExpBuffer(&tmpbuf
);
1125 appendPQExpBuffer(&tmpbuf
, "%s, ", indamname
);
1127 /* we assume here that index and table are in same schema */
1128 appendPQExpBuffer(&tmpbuf
, _("for table \"%s.%s\""),
1129 schemaname
, indtable
);
1131 if (strlen(indpred
))
1132 appendPQExpBuffer(&tmpbuf
, _(", predicate (%s)"), indpred
);
1134 if (strcmp(indisclustered
, "t") == 0)
1135 appendPQExpBuffer(&tmpbuf
, _(", clustered"));
1137 if (strcmp(indisvalid
, "t") != 0)
1138 appendPQExpBuffer(&tmpbuf
, _(", invalid"));
1140 printTableAddFooter(&cont
, tmpbuf
.data
);
1141 add_tablespace_footer(&cont
, tableinfo
.relkind
,
1142 tableinfo
.tablespace
, true);
1149 PGresult
*result
= NULL
;
1151 /* Footer information about a view */
1152 printTableAddFooter(&cont
, _("View definition:"));
1153 printTableAddFooter(&cont
, view_def
);
1156 if (tableinfo
.hasrules
)
1158 printfPQExpBuffer(&buf
,
1159 "SELECT r.rulename, trim(trailing ';' from pg_catalog.pg_get_ruledef(r.oid, true))\n"
1160 "FROM pg_catalog.pg_rewrite r\n"
1161 "WHERE r.ev_class = '%s' AND r.rulename != '_RETURN' ORDER BY 1",
1163 result
= PSQLexec(buf
.data
, false);
1167 if (PQntuples(result
) > 0)
1169 printTableAddFooter(&cont
, _("Rules:"));
1170 for (i
= 0; i
< PQntuples(result
); i
++)
1172 const char *ruledef
;
1174 /* Everything after "CREATE RULE" is echoed verbatim */
1175 ruledef
= PQgetvalue(result
, i
, 1);
1178 printfPQExpBuffer(&buf
, " %s", ruledef
);
1179 printTableAddFooter(&cont
, buf
.data
);
1185 else if (tableinfo
.relkind
== 'r')
1187 /* Footer information about a table */
1188 PGresult
*result
= NULL
;
1192 if (tableinfo
.hasindex
)
1194 printfPQExpBuffer(&buf
,
1195 "SELECT c2.relname, i.indisprimary, i.indisunique, i.indisclustered, ");
1196 if (pset
.sversion
>= 80200)
1197 appendPQExpBuffer(&buf
, "i.indisvalid, ");
1199 appendPQExpBuffer(&buf
, "true as indisvalid, ");
1200 appendPQExpBuffer(&buf
, "pg_catalog.pg_get_indexdef(i.indexrelid, 0, true)");
1201 if (pset
.sversion
>= 80000)
1202 appendPQExpBuffer(&buf
, ", c2.reltablespace");
1203 appendPQExpBuffer(&buf
,
1204 "\nFROM pg_catalog.pg_class c, pg_catalog.pg_class c2, pg_catalog.pg_index i\n"
1205 "WHERE c.oid = '%s' AND c.oid = i.indrelid AND i.indexrelid = c2.oid\n"
1206 "ORDER BY i.indisprimary DESC, i.indisunique DESC, c2.relname",
1208 result
= PSQLexec(buf
.data
, false);
1212 tuples
= PQntuples(result
);
1216 printTableAddFooter(&cont
, _("Indexes:"));
1217 for (i
= 0; i
< tuples
; i
++)
1219 const char *indexdef
;
1220 const char *usingpos
;
1222 /* untranslated index name */
1223 printfPQExpBuffer(&buf
, " \"%s\"",
1224 PQgetvalue(result
, i
, 0));
1226 /* Label as primary key or unique (but not both) */
1227 appendPQExpBuffer(&buf
,
1228 strcmp(PQgetvalue(result
, i
, 1), "t") == 0
1230 (strcmp(PQgetvalue(result
, i
, 2), "t") == 0
1233 /* Everything after "USING" is echoed verbatim */
1234 indexdef
= PQgetvalue(result
, i
, 5);
1235 usingpos
= strstr(indexdef
, " USING ");
1237 indexdef
= usingpos
+ 7;
1239 appendPQExpBuffer(&buf
, " %s", indexdef
);
1241 if (strcmp(PQgetvalue(result
, i
, 3), "t") == 0)
1242 appendPQExpBuffer(&buf
, " CLUSTER");
1244 if (strcmp(PQgetvalue(result
, i
, 4), "t") != 0)
1245 appendPQExpBuffer(&buf
, " INVALID");
1247 printTableAddFooter(&cont
, buf
.data
);
1249 /* Print tablespace of the index on the same line */
1250 if (pset
.sversion
>= 80000)
1251 add_tablespace_footer(&cont
, 'i',
1252 atooid(PQgetvalue(result
, i
, 6)),
1259 /* print table (and column) check constraints */
1260 if (tableinfo
.checks
)
1262 printfPQExpBuffer(&buf
,
1263 "SELECT r.conname, "
1264 "pg_catalog.pg_get_constraintdef(r.oid, true)\n"
1265 "FROM pg_catalog.pg_constraint r\n"
1266 "WHERE r.conrelid = '%s' AND r.contype = 'c'\nORDER BY 1",
1268 result
= PSQLexec(buf
.data
, false);
1272 tuples
= PQntuples(result
);
1276 printTableAddFooter(&cont
, _("Check constraints:"));
1277 for (i
= 0; i
< tuples
; i
++)
1279 /* untranslated contraint name and def */
1280 printfPQExpBuffer(&buf
, " \"%s\" %s",
1281 PQgetvalue(result
, i
, 0),
1282 PQgetvalue(result
, i
, 1));
1284 printTableAddFooter(&cont
, buf
.data
);
1290 /* print foreign-key constraints (there are none if no triggers) */
1291 if (tableinfo
.hastriggers
)
1293 printfPQExpBuffer(&buf
,
1295 " pg_catalog.pg_get_constraintdef(r.oid, true) as condef\n"
1296 "FROM pg_catalog.pg_constraint r\n"
1297 "WHERE r.conrelid = '%s' AND r.contype = 'f' ORDER BY 1",
1299 result
= PSQLexec(buf
.data
, false);
1303 tuples
= PQntuples(result
);
1307 printTableAddFooter(&cont
, _("Foreign-key constraints:"));
1308 for (i
= 0; i
< tuples
; i
++)
1310 /* untranslated constraint name and def */
1311 printfPQExpBuffer(&buf
, " \"%s\" %s",
1312 PQgetvalue(result
, i
, 0),
1313 PQgetvalue(result
, i
, 1));
1315 printTableAddFooter(&cont
, buf
.data
);
1321 /* print incoming foreign-key references (none if no triggers) */
1322 if (tableinfo
.hastriggers
)
1324 printfPQExpBuffer(&buf
,
1325 "SELECT conname, conrelid::pg_catalog.regclass,\n"
1326 " pg_catalog.pg_get_constraintdef(c.oid, true) as condef\n"
1327 "FROM pg_catalog.pg_constraint c\n"
1328 "WHERE c.confrelid = '%s' AND c.contype = 'f' ORDER BY 1",
1330 result
= PSQLexec(buf
.data
, false);
1334 tuples
= PQntuples(result
);
1338 printTableAddFooter(&cont
, _("Referenced by:"));
1339 for (i
= 0; i
< tuples
; i
++)
1341 /* translator: the first %s is a FK name, the following are
1342 * a table name and the FK definition */
1343 printfPQExpBuffer(&buf
, _(" \"%s\" IN %s %s"),
1344 PQgetvalue(result
, i
, 0),
1345 PQgetvalue(result
, i
, 1),
1346 PQgetvalue(result
, i
, 2));
1348 printTableAddFooter(&cont
, buf
.data
);
1355 if (tableinfo
.hasrules
)
1357 if (pset
.sversion
>= 80300)
1359 printfPQExpBuffer(&buf
,
1360 "SELECT r.rulename, trim(trailing ';' from pg_catalog.pg_get_ruledef(r.oid, true)), "
1362 "FROM pg_catalog.pg_rewrite r\n"
1363 "WHERE r.ev_class = '%s' ORDER BY 1",
1368 printfPQExpBuffer(&buf
,
1369 "SELECT r.rulename, trim(trailing ';' from pg_catalog.pg_get_ruledef(r.oid, true)), "
1370 "'O'::char AS ev_enabled\n"
1371 "FROM pg_catalog.pg_rewrite r\n"
1372 "WHERE r.ev_class = '%s' ORDER BY 1",
1375 result
= PSQLexec(buf
.data
, false);
1379 tuples
= PQntuples(result
);
1386 for (category
= 0; category
< 4; category
++)
1388 have_heading
= false;
1390 for (i
= 0; i
< tuples
; i
++)
1392 const char *ruledef
;
1393 bool list_rule
= false;
1398 if (*PQgetvalue(result
, i
, 2) == 'O')
1402 if (*PQgetvalue(result
, i
, 2) == 'D')
1406 if (*PQgetvalue(result
, i
, 2) == 'A')
1410 if (*PQgetvalue(result
, i
, 2) == 'R')
1422 printfPQExpBuffer(&buf
, _("Rules:"));
1425 printfPQExpBuffer(&buf
, _("Disabled rules:"));
1428 printfPQExpBuffer(&buf
, _("Rules firing always:"));
1431 printfPQExpBuffer(&buf
, _("Rules firing on replica only:"));
1434 printTableAddFooter(&cont
, buf
.data
);
1435 have_heading
= true;
1438 /* Everything after "CREATE RULE" is echoed verbatim */
1439 ruledef
= PQgetvalue(result
, i
, 1);
1441 printfPQExpBuffer(&buf
, " %s", ruledef
);
1442 printTableAddFooter(&cont
, buf
.data
);
1449 /* print triggers (but ignore foreign-key triggers) */
1450 if (tableinfo
.hastriggers
)
1452 printfPQExpBuffer(&buf
,
1454 "pg_catalog.pg_get_triggerdef(t.oid), "
1456 "FROM pg_catalog.pg_trigger t\n"
1457 "WHERE t.tgrelid = '%s' AND ",
1459 if (pset
.sversion
>= 80300)
1460 appendPQExpBuffer(&buf
, "t.tgconstraint = 0");
1462 appendPQExpBuffer(&buf
,
1463 "(NOT tgisconstraint "
1465 " (SELECT 1 FROM pg_catalog.pg_depend d "
1466 " JOIN pg_catalog.pg_constraint c ON (d.refclassid = c.tableoid AND d.refobjid = c.oid) "
1467 " WHERE d.classid = t.tableoid AND d.objid = t.oid AND d.deptype = 'i' AND c.contype = 'f'))");
1468 appendPQExpBuffer(&buf
, "\nORDER BY 1");
1470 result
= PSQLexec(buf
.data
, false);
1474 tuples
= PQntuples(result
);
1482 * split the output into 4 different categories. Enabled triggers,
1483 * disabled triggers and the two special ALWAYS and REPLICA
1486 for (category
= 0; category
< 4; category
++)
1488 have_heading
= false;
1489 for (i
= 0; i
< tuples
; i
++)
1493 const char *usingpos
;
1494 const char *tgenabled
;
1496 /* Check if this trigger falls into the current category */
1497 tgenabled
= PQgetvalue(result
, i
, 2);
1498 list_trigger
= false;
1502 if (*tgenabled
== 'O' || *tgenabled
== 't')
1503 list_trigger
= true;
1506 if (*tgenabled
== 'D' || *tgenabled
== 'f')
1507 list_trigger
= true;
1510 if (*tgenabled
== 'A')
1511 list_trigger
= true;
1514 if (*tgenabled
== 'R')
1515 list_trigger
= true;
1518 if (list_trigger
== false)
1521 /* Print the category heading once */
1522 if (have_heading
== false)
1527 printfPQExpBuffer(&buf
, _("Triggers:"));
1530 printfPQExpBuffer(&buf
, _("Disabled triggers:"));
1533 printfPQExpBuffer(&buf
, _("Triggers firing always:"));
1536 printfPQExpBuffer(&buf
, _("Triggers firing on replica only:"));
1540 printTableAddFooter(&cont
, buf
.data
);
1541 have_heading
= true;
1544 /* Everything after "TRIGGER" is echoed verbatim */
1545 tgdef
= PQgetvalue(result
, i
, 1);
1546 usingpos
= strstr(tgdef
, " TRIGGER ");
1548 tgdef
= usingpos
+ 9;
1550 printfPQExpBuffer(&buf
, " %s", tgdef
);
1551 printTableAddFooter(&cont
, buf
.data
);
1558 /* print inherited tables */
1559 printfPQExpBuffer(&buf
, "SELECT c.oid::pg_catalog.regclass FROM pg_catalog.pg_class c, pg_catalog.pg_inherits i WHERE c.oid=i.inhparent AND i.inhrelid = '%s' ORDER BY inhseqno", oid
);
1561 result
= PSQLexec(buf
.data
, false);
1565 tuples
= PQntuples(result
);
1567 for (i
= 0; i
< tuples
; i
++)
1569 const char *s
= _("Inherits");
1572 printfPQExpBuffer(&buf
, "%s: %s", s
, PQgetvalue(result
, i
, 0));
1574 printfPQExpBuffer(&buf
, "%*s %s", (int) strlen(s
), "", PQgetvalue(result
, i
, 0));
1576 appendPQExpBuffer(&buf
, ",");
1578 printTableAddFooter(&cont
, buf
.data
);
1584 const char *s
= _("Has OIDs");
1586 printfPQExpBuffer(&buf
, "%s: %s", s
,
1587 (tableinfo
.hasoids
? _("yes") : _("no")));
1588 printTableAddFooter(&cont
, buf
.data
);
1591 add_tablespace_footer(&cont
, tableinfo
.relkind
, tableinfo
.tablespace
,
1595 printTable(&cont
, pset
.queryFout
, pset
.logfile
);
1596 printTableCleanup(&cont
);
1603 if (printTableInitialized
)
1604 printTableCleanup(&cont
);
1605 termPQExpBuffer(&buf
);
1606 termPQExpBuffer(&title
);
1607 termPQExpBuffer(&tmpbuf
);
1611 for (ptr
= seq_values
; *ptr
; ptr
++)
1618 for (ptr
= modifiers
; *ptr
; ptr
++)
1633 * Add a tablespace description to a footer. If 'newline' is true, it is added
1634 * in a new line; otherwise it's appended to the current value of the last
1638 add_tablespace_footer(printTableContent
*const cont
, char relkind
,
1639 Oid tablespace
, const bool newline
)
1641 /* relkinds for which we support tablespaces */
1642 if (relkind
== 'r' || relkind
== 'i')
1645 * We ignore the database default tablespace so that users not using
1646 * tablespaces don't need to know about them. This case also covers
1647 * pre-8.0 servers, for which tablespace will always be 0.
1649 if (tablespace
!= 0)
1651 PGresult
*result
= NULL
;
1652 PQExpBufferData buf
;
1654 initPQExpBuffer(&buf
);
1655 printfPQExpBuffer(&buf
,
1656 "SELECT spcname FROM pg_catalog.pg_tablespace\n"
1657 "WHERE oid = '%u'", tablespace
);
1658 result
= PSQLexec(buf
.data
, false);
1661 /* Should always be the case, but.... */
1662 if (PQntuples(result
) > 0)
1666 /* Add the tablespace as a new footer */
1667 printfPQExpBuffer(&buf
, _("Tablespace: \"%s\""),
1668 PQgetvalue(result
, 0, 0));
1669 printTableAddFooter(cont
, buf
.data
);
1673 /* Append the tablespace to the latest footer */
1674 printfPQExpBuffer(&buf
, "%s", cont
->footer
->data
);
1675 /* translator: before this string there's an index
1676 * description like '"foo_pkey" PRIMARY KEY, btree (a)' */
1677 appendPQExpBuffer(&buf
, _(", tablespace \"%s\""),
1678 PQgetvalue(result
, 0, 0));
1679 printTableSetFooter(cont
, buf
.data
);
1683 termPQExpBuffer(&buf
);
1691 * Describes roles. Any schema portion of the pattern is ignored.
1694 describeRoles(const char *pattern
, bool verbose
)
1696 PQExpBufferData buf
;
1698 printTableContent cont
;
1699 printTableOpt myopt
= pset
.popt
.topt
;
1704 const char align
= 'l';
1707 initPQExpBuffer(&buf
);
1709 if (pset
.sversion
>= 80100)
1711 printfPQExpBuffer(&buf
,
1712 "SELECT r.rolname, r.rolsuper, r.rolinherit,\n"
1713 " r.rolcreaterole, r.rolcreatedb, r.rolcanlogin,\n"
1714 " r.rolconnlimit,\n"
1715 " ARRAY(SELECT b.rolname\n"
1716 " FROM pg_catalog.pg_auth_members m\n"
1717 " JOIN pg_catalog.pg_roles b ON (m.roleid = b.oid)\n"
1718 " WHERE m.member = r.oid) as memberof");
1720 if (verbose
&& pset
.sversion
>= 80200)
1722 appendPQExpBufferStr(&buf
, "\n, pg_catalog.shobj_description(r.oid, 'pg_authid') AS description");
1726 appendPQExpBufferStr(&buf
, "\nFROM pg_catalog.pg_roles r\n");
1728 processSQLNamePattern(pset
.db
, &buf
, pattern
, false, false,
1729 NULL
, "r.rolname", NULL
, NULL
);
1733 printfPQExpBuffer(&buf
,
1734 "SELECT u.usename AS rolname,\n"
1735 " u.usesuper AS rolsuper,\n"
1736 " true AS rolinherit, false AS rolcreaterole,\n"
1737 " u.usecreatedb AS rolcreatedb, true AS rolcanlogin,\n"
1738 " -1 AS rolconnlimit,\n"
1739 " ARRAY(SELECT g.groname FROM pg_catalog.pg_group g WHERE u.usesysid = ANY(g.grolist)) as memberof"
1740 "\nFROM pg_catalog.pg_user u\n");
1742 processSQLNamePattern(pset
.db
, &buf
, pattern
, false, false,
1743 NULL
, "u.usename", NULL
, NULL
);
1746 appendPQExpBuffer(&buf
, "ORDER BY 1;");
1748 res
= PSQLexec(buf
.data
, false);
1752 nrows
= PQntuples(res
);
1753 attr
= pg_malloc_zero((nrows
+ 1) * sizeof(*attr
));
1755 printTableInit(&cont
, &myopt
, _("List of roles"), ncols
, nrows
);
1757 printTableAddHeader(&cont
, gettext_noop("Role name"), true, align
);
1758 printTableAddHeader(&cont
, gettext_noop("Attributes"), true, align
);
1759 printTableAddHeader(&cont
, gettext_noop("Member of"), true, align
);
1761 if (verbose
&& pset
.sversion
>= 80200)
1762 printTableAddHeader(&cont
, gettext_noop("Description"), true, align
);
1764 for (i
= 0; i
< nrows
; i
++)
1766 printTableAddCell(&cont
, PQgetvalue(res
, i
, 0), false);
1768 resetPQExpBuffer(&buf
);
1769 if (strcmp(PQgetvalue(res
, i
, 1), "t") == 0)
1770 add_role_attribute(&buf
, _("Superuser"));
1772 if (strcmp(PQgetvalue(res
, i
, 2), "t") != 0)
1773 add_role_attribute(&buf
, _("No inheritance"));
1775 if (strcmp(PQgetvalue(res
, i
, 3), "t") == 0)
1776 add_role_attribute(&buf
, _("Create role"));
1778 if (strcmp(PQgetvalue(res
, i
, 4), "t") == 0)
1779 add_role_attribute(&buf
, _("Create DB"));
1781 if (strcmp(PQgetvalue(res
, i
, 5), "t") != 0)
1782 add_role_attribute(&buf
, _("Cannot login"));
1784 conns
= atoi(PQgetvalue(res
, i
, 6));
1788 appendPQExpBufferStr(&buf
, "\n");
1791 appendPQExpBuffer(&buf
, _("No connections"));
1792 else if (conns
== 1)
1793 appendPQExpBuffer(&buf
, _("1 connection"));
1795 appendPQExpBuffer(&buf
, _("%d connections"), conns
);
1798 attr
[i
] = pg_strdup(buf
.data
);
1800 printTableAddCell(&cont
, attr
[i
], false);
1802 printTableAddCell(&cont
, PQgetvalue(res
, i
, 7), false);
1804 if (verbose
&& pset
.sversion
>= 80200)
1805 printTableAddCell(&cont
, PQgetvalue(res
, i
, 8), false);
1807 termPQExpBuffer(&buf
);
1809 printTable(&cont
, pset
.queryFout
, pset
.logfile
);
1810 printTableCleanup(&cont
);
1812 for (i
= 0; i
< nrows
; i
++)
1821 add_role_attribute(PQExpBuffer buf
, const char *const str
)
1824 appendPQExpBufferStr(buf
, "\n");
1826 appendPQExpBufferStr(buf
, str
);
1833 * handler for \d, \dt, etc.
1835 * tabtypes is an array of characters, specifying what info is desired:
1840 * S - system tables (pg_catalog)
1841 * (any order of the above is fine)
1844 listTables(const char *tabtypes
, const char *pattern
, bool verbose
)
1846 bool showTables
= strchr(tabtypes
, 't') != NULL
;
1847 bool showIndexes
= strchr(tabtypes
, 'i') != NULL
;
1848 bool showViews
= strchr(tabtypes
, 'v') != NULL
;
1849 bool showSeq
= strchr(tabtypes
, 's') != NULL
;
1850 bool showSystem
= strchr(tabtypes
, 'S') != NULL
;
1852 PQExpBufferData buf
;
1854 printQueryOpt myopt
= pset
.popt
;
1855 static const bool translate_columns
[] = {false, false, true, false, false, false};
1857 if (!(showTables
|| showIndexes
|| showViews
|| showSeq
))
1858 showTables
= showViews
= showSeq
= true;
1860 initPQExpBuffer(&buf
);
1863 * Note: as of Pg 8.2, we no longer use relkind 's', but we keep it here
1864 * for backwards compatibility.
1866 printfPQExpBuffer(&buf
,
1867 "SELECT n.nspname as \"%s\",\n"
1868 " c.relname as \"%s\",\n"
1869 " CASE c.relkind WHEN 'r' THEN '%s' WHEN 'v' THEN '%s' WHEN 'i' THEN '%s' WHEN 'S' THEN '%s' WHEN 's' THEN '%s' END as \"%s\",\n"
1870 " pg_catalog.pg_get_userbyid(c.relowner) as \"%s\"",
1871 gettext_noop("Schema"),
1872 gettext_noop("Name"),
1873 gettext_noop("table"),
1874 gettext_noop("view"),
1875 gettext_noop("index"),
1876 gettext_noop("sequence"),
1877 gettext_noop("special"),
1878 gettext_noop("Type"),
1879 gettext_noop("Owner"));
1882 appendPQExpBuffer(&buf
,
1883 ",\n c2.relname as \"%s\"",
1884 gettext_noop("Table"));
1886 if (verbose
&& pset
.sversion
>= 80100)
1887 appendPQExpBuffer(&buf
,
1888 ",\n pg_catalog.pg_size_pretty(pg_catalog.pg_relation_size(c.oid)) as \"%s\"",
1889 gettext_noop("Size"));
1891 appendPQExpBuffer(&buf
,
1892 ",\n pg_catalog.obj_description(c.oid, 'pg_class') as \"%s\"",
1893 gettext_noop("Description"));
1895 appendPQExpBuffer(&buf
,
1896 "\nFROM pg_catalog.pg_class c"
1897 "\n LEFT JOIN pg_catalog.pg_namespace n ON n.oid = c.relnamespace");
1899 appendPQExpBuffer(&buf
,
1900 "\n LEFT JOIN pg_catalog.pg_index i ON i.indexrelid = c.oid"
1901 "\n LEFT JOIN pg_catalog.pg_class c2 ON i.indrelid = c2.oid");
1903 appendPQExpBuffer(&buf
, "\nWHERE c.relkind IN (");
1905 appendPQExpBuffer(&buf
, "'r',");
1907 appendPQExpBuffer(&buf
, "'v',");
1909 appendPQExpBuffer(&buf
, "'i',");
1911 appendPQExpBuffer(&buf
, "'S',");
1912 if (showSystem
&& showTables
)
1913 appendPQExpBuffer(&buf
, "'s',");
1914 appendPQExpBuffer(&buf
, "''"); /* dummy */
1915 appendPQExpBuffer(&buf
, ")\n");
1918 * If showSystem is specified, show only system objects (those in
1919 * pg_catalog). Otherwise, suppress system objects, including those in
1920 * pg_catalog and pg_toast. (We don't want to hide temp tables though.)
1923 appendPQExpBuffer(&buf
,
1924 " AND n.nspname = 'pg_catalog'\n");
1926 appendPQExpBuffer(&buf
,
1927 " AND n.nspname <> 'pg_catalog'\n"
1928 " AND n.nspname !~ '^pg_toast'\n");
1930 processSQLNamePattern(pset
.db
, &buf
, pattern
, true, false,
1931 "n.nspname", "c.relname", NULL
,
1932 "pg_catalog.pg_table_is_visible(c.oid)");
1934 appendPQExpBuffer(&buf
, "ORDER BY 1,2;");
1936 res
= PSQLexec(buf
.data
, false);
1937 termPQExpBuffer(&buf
);
1941 if (PQntuples(res
) == 0 && !pset
.quiet
)
1944 fprintf(pset
.queryFout
, _("No matching relations found.\n"));
1946 fprintf(pset
.queryFout
, _("No relations found.\n"));
1950 myopt
.nullPrint
= NULL
;
1951 myopt
.title
= _("List of relations");
1952 myopt
.translate_header
= true;
1953 myopt
.translate_columns
= translate_columns
;
1955 printQuery(res
, &myopt
, pset
.queryFout
, pset
.logfile
);
1966 * Describes domains.
1969 listDomains(const char *pattern
)
1971 PQExpBufferData buf
;
1973 printQueryOpt myopt
= pset
.popt
;
1975 initPQExpBuffer(&buf
);
1977 printfPQExpBuffer(&buf
,
1978 "SELECT n.nspname as \"%s\",\n"
1979 " t.typname as \"%s\",\n"
1980 " pg_catalog.format_type(t.typbasetype, t.typtypmod) as \"%s\",\n"
1981 " CASE WHEN t.typnotnull AND t.typdefault IS NOT NULL THEN 'not null default '||t.typdefault\n"
1982 " WHEN t.typnotnull AND t.typdefault IS NULL THEN 'not null'\n"
1983 " WHEN NOT t.typnotnull AND t.typdefault IS NOT NULL THEN 'default '||t.typdefault\n"
1986 " pg_catalog.pg_get_constraintdef(r.oid, true) as \"%s\"\n"
1987 "FROM pg_catalog.pg_type t\n"
1988 " LEFT JOIN pg_catalog.pg_namespace n ON n.oid = t.typnamespace\n"
1989 " LEFT JOIN pg_catalog.pg_constraint r ON t.oid = r.contypid\n"
1990 "WHERE t.typtype = 'd'\n",
1991 gettext_noop("Schema"),
1992 gettext_noop("Name"),
1993 gettext_noop("Type"),
1994 gettext_noop("Modifier"),
1995 gettext_noop("Check"));
1997 processSQLNamePattern(pset
.db
, &buf
, pattern
, true, false,
1998 "n.nspname", "t.typname", NULL
,
1999 "pg_catalog.pg_type_is_visible(t.oid)");
2001 appendPQExpBuffer(&buf
, "ORDER BY 1, 2;");
2003 res
= PSQLexec(buf
.data
, false);
2004 termPQExpBuffer(&buf
);
2008 myopt
.nullPrint
= NULL
;
2009 myopt
.title
= _("List of domains");
2010 myopt
.translate_header
= true;
2012 printQuery(res
, &myopt
, pset
.queryFout
, pset
.logfile
);
2021 * Describes conversions.
2024 listConversions(const char *pattern
)
2026 PQExpBufferData buf
;
2028 printQueryOpt myopt
= pset
.popt
;
2029 static const bool translate_columns
[] = {false, false, false, false, true};
2031 initPQExpBuffer(&buf
);
2033 printfPQExpBuffer(&buf
,
2034 "SELECT n.nspname AS \"%s\",\n"
2035 " c.conname AS \"%s\",\n"
2036 " pg_catalog.pg_encoding_to_char(c.conforencoding) AS \"%s\",\n"
2037 " pg_catalog.pg_encoding_to_char(c.contoencoding) AS \"%s\",\n"
2038 " CASE WHEN c.condefault THEN '%s'\n"
2039 " ELSE '%s' END AS \"%s\"\n"
2040 "FROM pg_catalog.pg_conversion c, pg_catalog.pg_namespace n\n"
2041 "WHERE n.oid = c.connamespace\n",
2042 gettext_noop("Schema"),
2043 gettext_noop("Name"),
2044 gettext_noop("Source"),
2045 gettext_noop("Destination"),
2046 gettext_noop("yes"), gettext_noop("no"),
2047 gettext_noop("Default?"));
2049 processSQLNamePattern(pset
.db
, &buf
, pattern
, true, false,
2050 "n.nspname", "c.conname", NULL
,
2051 "pg_catalog.pg_conversion_is_visible(c.oid)");
2053 appendPQExpBuffer(&buf
, "ORDER BY 1, 2;");
2055 res
= PSQLexec(buf
.data
, false);
2056 termPQExpBuffer(&buf
);
2060 myopt
.nullPrint
= NULL
;
2061 myopt
.title
= _("List of conversions");
2062 myopt
.translate_header
= true;
2063 myopt
.translate_columns
= translate_columns
;
2065 printQuery(res
, &myopt
, pset
.queryFout
, pset
.logfile
);
2077 listCasts(const char *pattern
)
2079 PQExpBufferData buf
;
2081 printQueryOpt myopt
= pset
.popt
;
2082 static const bool translate_columns
[] = {false, false, false, true};
2084 initPQExpBuffer(&buf
);
2086 * We need a left join to pg_proc for binary casts; the others are just
2087 * paranoia. Also note that we don't attempt to localize '(binary
2088 * coercible)', because there's too much risk of gettext translating a
2089 * function name that happens to match some string in the PO database.
2091 printfPQExpBuffer(&buf
,
2092 "SELECT pg_catalog.format_type(castsource, NULL) AS \"%s\",\n"
2093 " pg_catalog.format_type(casttarget, NULL) AS \"%s\",\n"
2094 " CASE WHEN castfunc = 0 THEN '(binary coercible)'\n"
2097 " CASE WHEN c.castcontext = 'e' THEN '%s'\n"
2098 " WHEN c.castcontext = 'a' THEN '%s'\n"
2101 "FROM pg_catalog.pg_cast c LEFT JOIN pg_catalog.pg_proc p\n"
2102 " ON c.castfunc = p.oid\n"
2103 " LEFT JOIN pg_catalog.pg_type ts\n"
2104 " ON c.castsource = ts.oid\n"
2105 " LEFT JOIN pg_catalog.pg_namespace ns\n"
2106 " ON ns.oid = ts.typnamespace\n"
2107 " LEFT JOIN pg_catalog.pg_type tt\n"
2108 " ON c.casttarget = tt.oid\n"
2109 " LEFT JOIN pg_catalog.pg_namespace nt\n"
2110 " ON nt.oid = tt.typnamespace\n"
2112 gettext_noop("Source type"),
2113 gettext_noop("Target type"),
2114 gettext_noop("Function"),
2115 gettext_noop("no"), gettext_noop("in assignment"), gettext_noop("yes"),
2116 gettext_noop("Implicit?"));
2119 * Match name pattern against either internal or external name of either
2120 * castsource or casttarget
2122 processSQLNamePattern(pset
.db
, &buf
, pattern
, true, false,
2123 "ns.nspname", "ts.typname",
2124 "pg_catalog.format_type(ts.oid, NULL)",
2125 "pg_catalog.pg_type_is_visible(ts.oid)");
2127 appendPQExpBuffer(&buf
, ") OR (true");
2129 processSQLNamePattern(pset
.db
, &buf
, pattern
, true, false,
2130 "nt.nspname", "tt.typname",
2131 "pg_catalog.format_type(tt.oid, NULL)",
2132 "pg_catalog.pg_type_is_visible(tt.oid)");
2134 appendPQExpBuffer(&buf
, ")\nORDER BY 1, 2;");
2136 res
= PSQLexec(buf
.data
, false);
2137 termPQExpBuffer(&buf
);
2141 myopt
.nullPrint
= NULL
;
2142 myopt
.title
= _("List of casts");
2143 myopt
.translate_header
= true;
2144 myopt
.translate_columns
= translate_columns
;
2146 printQuery(res
, &myopt
, pset
.queryFout
, pset
.logfile
);
2155 * Describes schemas (namespaces)
2158 listSchemas(const char *pattern
, bool verbose
)
2160 PQExpBufferData buf
;
2162 printQueryOpt myopt
= pset
.popt
;
2164 initPQExpBuffer(&buf
);
2165 printfPQExpBuffer(&buf
,
2166 "SELECT n.nspname AS \"%s\",\n"
2167 " pg_catalog.pg_get_userbyid(n.nspowner) AS \"%s\"",
2168 gettext_noop("Name"),
2169 gettext_noop("Owner"));
2172 appendPQExpBuffer(&buf
,
2173 ",\n n.nspacl as \"%s\","
2174 " pg_catalog.obj_description(n.oid, 'pg_namespace') as \"%s\"",
2175 gettext_noop("Access privileges"),
2176 gettext_noop("Description"));
2178 appendPQExpBuffer(&buf
,
2179 "\nFROM pg_catalog.pg_namespace n\n"
2180 "WHERE (n.nspname !~ '^pg_temp_' OR\n"
2181 " n.nspname = (pg_catalog.current_schemas(true))[1])\n"); /* temp schema is first */
2183 processSQLNamePattern(pset
.db
, &buf
, pattern
, true, false,
2184 NULL
, "n.nspname", NULL
,
2187 appendPQExpBuffer(&buf
, "ORDER BY 1;");
2189 res
= PSQLexec(buf
.data
, false);
2190 termPQExpBuffer(&buf
);
2194 myopt
.nullPrint
= NULL
;
2195 myopt
.title
= _("List of schemas");
2196 myopt
.translate_header
= true;
2198 printQuery(res
, &myopt
, pset
.queryFout
, pset
.logfile
);
2207 * list text search parsers
2210 listTSParsers(const char *pattern
, bool verbose
)
2212 PQExpBufferData buf
;
2214 printQueryOpt myopt
= pset
.popt
;
2216 if (pset
.sversion
< 80300)
2218 fprintf(stderr
, _("The server (version %d.%d) does not support full text search.\n"),
2219 pset
.sversion
/ 10000, (pset
.sversion
/ 100) % 100);
2224 return listTSParsersVerbose(pattern
);
2226 initPQExpBuffer(&buf
);
2228 printfPQExpBuffer(&buf
,
2230 " n.nspname as \"%s\",\n"
2231 " p.prsname as \"%s\",\n"
2232 " pg_catalog.obj_description(p.oid, 'pg_ts_parser') as \"%s\"\n"
2233 "FROM pg_catalog.pg_ts_parser p \n"
2234 "LEFT JOIN pg_catalog.pg_namespace n ON n.oid = p.prsnamespace\n",
2235 gettext_noop("Schema"),
2236 gettext_noop("Name"),
2237 gettext_noop("Description")
2240 processSQLNamePattern(pset
.db
, &buf
, pattern
, false, false,
2241 "n.nspname", "p.prsname", NULL
,
2242 "pg_catalog.pg_ts_parser_is_visible(p.oid)");
2244 appendPQExpBuffer(&buf
, "ORDER BY 1, 2;");
2246 res
= PSQLexec(buf
.data
, false);
2247 termPQExpBuffer(&buf
);
2251 myopt
.nullPrint
= NULL
;
2252 myopt
.title
= _("List of text search parsers");
2253 myopt
.translate_header
= true;
2255 printQuery(res
, &myopt
, pset
.queryFout
, pset
.logfile
);
2262 * full description of parsers
2265 listTSParsersVerbose(const char *pattern
)
2267 PQExpBufferData buf
;
2271 initPQExpBuffer(&buf
);
2273 printfPQExpBuffer(&buf
,
2277 "FROM pg_catalog.pg_ts_parser p\n"
2278 "LEFT JOIN pg_catalog.pg_namespace n ON n.oid = p.prsnamespace\n"
2281 processSQLNamePattern(pset
.db
, &buf
, pattern
, false, false,
2282 "n.nspname", "p.prsname", NULL
,
2283 "pg_catalog.pg_ts_parser_is_visible(p.oid)");
2285 appendPQExpBuffer(&buf
, "ORDER BY 1, 2;");
2287 res
= PSQLexec(buf
.data
, false);
2288 termPQExpBuffer(&buf
);
2292 if (PQntuples(res
) == 0)
2295 fprintf(stderr
, _("Did not find any text search parser named \"%s\".\n"),
2301 for (i
= 0; i
< PQntuples(res
); i
++)
2304 const char *nspname
= NULL
;
2305 const char *prsname
;
2307 oid
= PQgetvalue(res
, i
, 0);
2308 if (!PQgetisnull(res
, i
, 1))
2309 nspname
= PQgetvalue(res
, i
, 1);
2310 prsname
= PQgetvalue(res
, i
, 2);
2312 if (!describeOneTSParser(oid
, nspname
, prsname
))
2330 describeOneTSParser(const char *oid
, const char *nspname
, const char *prsname
)
2332 PQExpBufferData buf
;
2335 printQueryOpt myopt
= pset
.popt
;
2336 static const bool translate_columns
[] = {true, false, false};
2338 initPQExpBuffer(&buf
);
2340 printfPQExpBuffer(&buf
,
2341 "SELECT '%s' AS \"%s\", \n"
2342 " p.prsstart::pg_catalog.regproc AS \"%s\", \n"
2343 " pg_catalog.obj_description(p.prsstart, 'pg_proc') as \"%s\" \n"
2344 " FROM pg_catalog.pg_ts_parser p \n"
2345 " WHERE p.oid = '%s' \n"
2348 " p.prstoken::pg_catalog.regproc, \n"
2349 " pg_catalog.obj_description(p.prstoken, 'pg_proc') \n"
2350 " FROM pg_catalog.pg_ts_parser p \n"
2351 " WHERE p.oid = '%s' \n"
2354 " p.prsend::pg_catalog.regproc, \n"
2355 " pg_catalog.obj_description(p.prsend, 'pg_proc') \n"
2356 " FROM pg_catalog.pg_ts_parser p \n"
2357 " WHERE p.oid = '%s' \n"
2360 " p.prsheadline::pg_catalog.regproc, \n"
2361 " pg_catalog.obj_description(p.prsheadline, 'pg_proc') \n"
2362 " FROM pg_catalog.pg_ts_parser p \n"
2363 " WHERE p.oid = '%s' \n"
2366 " p.prslextype::pg_catalog.regproc, \n"
2367 " pg_catalog.obj_description(p.prslextype, 'pg_proc') \n"
2368 " FROM pg_catalog.pg_ts_parser p \n"
2369 " WHERE p.oid = '%s' \n",
2370 gettext_noop("Start parse"),
2371 gettext_noop("Method"),
2372 gettext_noop("Function"),
2373 gettext_noop("Description"),
2375 gettext_noop("Get next token"),
2377 gettext_noop("End parse"),
2379 gettext_noop("Get headline"),
2381 gettext_noop("Get token types"),
2384 res
= PSQLexec(buf
.data
, false);
2385 termPQExpBuffer(&buf
);
2389 myopt
.nullPrint
= NULL
;
2391 sprintf(title
, _("Text search parser \"%s.%s\""), nspname
, prsname
);
2393 sprintf(title
, _("Text search parser \"%s\""), prsname
);
2394 myopt
.title
= title
;
2395 myopt
.footers
= NULL
;
2396 myopt
.default_footer
= false;
2397 myopt
.translate_header
= true;
2398 myopt
.translate_columns
= translate_columns
;
2400 printQuery(res
, &myopt
, pset
.queryFout
, pset
.logfile
);
2404 initPQExpBuffer(&buf
);
2406 printfPQExpBuffer(&buf
,
2407 "SELECT t.alias as \"%s\", \n"
2408 " t.description as \"%s\" \n"
2409 "FROM pg_catalog.ts_token_type( '%s'::pg_catalog.oid ) as t \n"
2411 gettext_noop("Token name"),
2412 gettext_noop("Description"),
2415 res
= PSQLexec(buf
.data
, false);
2416 termPQExpBuffer(&buf
);
2420 myopt
.nullPrint
= NULL
;
2422 sprintf(title
, _("Token types for parser \"%s.%s\""), nspname
, prsname
);
2424 sprintf(title
, _("Token types for parser \"%s\""), prsname
);
2425 myopt
.title
= title
;
2426 myopt
.footers
= NULL
;
2427 myopt
.default_footer
= true;
2428 myopt
.translate_header
= true;
2429 myopt
.translate_columns
= NULL
;
2431 printQuery(res
, &myopt
, pset
.queryFout
, pset
.logfile
);
2440 * list text search dictionaries
2443 listTSDictionaries(const char *pattern
, bool verbose
)
2445 PQExpBufferData buf
;
2447 printQueryOpt myopt
= pset
.popt
;
2449 if (pset
.sversion
< 80300)
2451 fprintf(stderr
, _("The server (version %d.%d) does not support full text search.\n"),
2452 pset
.sversion
/ 10000, (pset
.sversion
/ 100) % 100);
2456 initPQExpBuffer(&buf
);
2458 printfPQExpBuffer(&buf
,
2460 " n.nspname as \"%s\",\n"
2461 " d.dictname as \"%s\",\n",
2462 gettext_noop("Schema"),
2463 gettext_noop("Name"));
2467 appendPQExpBuffer(&buf
,
2468 " ( SELECT COALESCE(nt.nspname, '(null)')::pg_catalog.text || '.' || t.tmplname FROM \n"
2469 " pg_catalog.pg_ts_template t \n"
2470 " LEFT JOIN pg_catalog.pg_namespace nt ON nt.oid = t.tmplnamespace \n"
2471 " WHERE d.dicttemplate = t.oid ) AS \"%s\", \n"
2472 " d.dictinitoption as \"%s\", \n",
2473 gettext_noop("Template"),
2474 gettext_noop("Init options"));
2477 appendPQExpBuffer(&buf
,
2478 " pg_catalog.obj_description(d.oid, 'pg_ts_dict') as \"%s\"\n",
2479 gettext_noop("Description"));
2481 appendPQExpBuffer(&buf
, "FROM pg_catalog.pg_ts_dict d\n"
2482 "LEFT JOIN pg_catalog.pg_namespace n ON n.oid = d.dictnamespace\n");
2484 processSQLNamePattern(pset
.db
, &buf
, pattern
, false, false,
2485 "n.nspname", "d.dictname", NULL
,
2486 "pg_catalog.pg_ts_dict_is_visible(d.oid)");
2488 appendPQExpBuffer(&buf
, "ORDER BY 1, 2;");
2490 res
= PSQLexec(buf
.data
, false);
2491 termPQExpBuffer(&buf
);
2495 myopt
.nullPrint
= NULL
;
2496 myopt
.title
= _("List of text search dictionaries");
2497 myopt
.translate_header
= true;
2499 printQuery(res
, &myopt
, pset
.queryFout
, pset
.logfile
);
2508 * list text search templates
2511 listTSTemplates(const char *pattern
, bool verbose
)
2513 PQExpBufferData buf
;
2515 printQueryOpt myopt
= pset
.popt
;
2517 if (pset
.sversion
< 80300)
2519 fprintf(stderr
, _("The server (version %d.%d) does not support full text search.\n"),
2520 pset
.sversion
/ 10000, (pset
.sversion
/ 100) % 100);
2524 initPQExpBuffer(&buf
);
2527 printfPQExpBuffer(&buf
,
2529 " n.nspname AS \"%s\",\n"
2530 " t.tmplname AS \"%s\",\n"
2531 " t.tmplinit::pg_catalog.regproc AS \"%s\",\n"
2532 " t.tmpllexize::pg_catalog.regproc AS \"%s\",\n"
2533 " pg_catalog.obj_description(t.oid, 'pg_ts_template') AS \"%s\"\n",
2534 gettext_noop("Schema"),
2535 gettext_noop("Name"),
2536 gettext_noop("Init"),
2537 gettext_noop("Lexize"),
2538 gettext_noop("Description"));
2540 printfPQExpBuffer(&buf
,
2542 " n.nspname AS \"%s\",\n"
2543 " t.tmplname AS \"%s\",\n"
2544 " pg_catalog.obj_description(t.oid, 'pg_ts_template') AS \"%s\"\n",
2545 gettext_noop("Schema"),
2546 gettext_noop("Name"),
2547 gettext_noop("Description"));
2549 appendPQExpBuffer(&buf
, "FROM pg_catalog.pg_ts_template t\n"
2550 "LEFT JOIN pg_catalog.pg_namespace n ON n.oid = t.tmplnamespace\n");
2552 processSQLNamePattern(pset
.db
, &buf
, pattern
, false, false,
2553 "n.nspname", "t.tmplname", NULL
,
2554 "pg_catalog.pg_ts_template_is_visible(t.oid)");
2556 appendPQExpBuffer(&buf
, "ORDER BY 1, 2;");
2558 res
= PSQLexec(buf
.data
, false);
2559 termPQExpBuffer(&buf
);
2563 myopt
.nullPrint
= NULL
;
2564 myopt
.title
= _("List of text search templates");
2565 myopt
.translate_header
= true;
2567 printQuery(res
, &myopt
, pset
.queryFout
, pset
.logfile
);
2576 * list text search configurations
2579 listTSConfigs(const char *pattern
, bool verbose
)
2581 PQExpBufferData buf
;
2583 printQueryOpt myopt
= pset
.popt
;
2585 if (pset
.sversion
< 80300)
2587 fprintf(stderr
, _("The server (version %d.%d) does not support full text search.\n"),
2588 pset
.sversion
/ 10000, (pset
.sversion
/ 100) % 100);
2593 return listTSConfigsVerbose(pattern
);
2595 initPQExpBuffer(&buf
);
2597 printfPQExpBuffer(&buf
,
2599 " n.nspname as \"%s\",\n"
2600 " c.cfgname as \"%s\",\n"
2601 " pg_catalog.obj_description(c.oid, 'pg_ts_config') as \"%s\"\n"
2602 "FROM pg_catalog.pg_ts_config c\n"
2603 "LEFT JOIN pg_catalog.pg_namespace n ON n.oid = c.cfgnamespace \n",
2604 gettext_noop("Schema"),
2605 gettext_noop("Name"),
2606 gettext_noop("Description")
2609 processSQLNamePattern(pset
.db
, &buf
, pattern
, false, false,
2610 "n.nspname", "c.cfgname", NULL
,
2611 "pg_catalog.pg_ts_config_is_visible(c.oid)");
2613 appendPQExpBuffer(&buf
, "ORDER BY 1, 2;");
2615 res
= PSQLexec(buf
.data
, false);
2616 termPQExpBuffer(&buf
);
2620 myopt
.nullPrint
= NULL
;
2621 myopt
.title
= _("List of text search configurations");
2622 myopt
.translate_header
= true;
2624 printQuery(res
, &myopt
, pset
.queryFout
, pset
.logfile
);
2631 listTSConfigsVerbose(const char *pattern
)
2633 PQExpBufferData buf
;
2637 initPQExpBuffer(&buf
);
2639 printfPQExpBuffer(&buf
,
2640 "SELECT c.oid, c.cfgname,\n"
2643 " np.nspname as pnspname \n"
2644 "FROM pg_catalog.pg_ts_config c \n"
2645 " LEFT JOIN pg_catalog.pg_namespace n ON n.oid = c.cfgnamespace, \n"
2646 " pg_catalog.pg_ts_parser p \n"
2647 " LEFT JOIN pg_catalog.pg_namespace np ON np.oid = p.prsnamespace \n"
2648 "WHERE p.oid = c.cfgparser\n"
2651 processSQLNamePattern(pset
.db
, &buf
, pattern
, true, false,
2652 "n.nspname", "c.cfgname", NULL
,
2653 "pg_catalog.pg_ts_config_is_visible(c.oid)");
2655 appendPQExpBuffer(&buf
, "ORDER BY 3, 2;");
2657 res
= PSQLexec(buf
.data
, false);
2658 termPQExpBuffer(&buf
);
2662 if (PQntuples(res
) == 0)
2665 fprintf(stderr
, _("Did not find any text search configuration named \"%s\".\n"),
2671 for (i
= 0; i
< PQntuples(res
); i
++)
2674 const char *cfgname
;
2675 const char *nspname
= NULL
;
2676 const char *prsname
;
2677 const char *pnspname
= NULL
;
2679 oid
= PQgetvalue(res
, i
, 0);
2680 cfgname
= PQgetvalue(res
, i
, 1);
2681 if (!PQgetisnull(res
, i
, 2))
2682 nspname
= PQgetvalue(res
, i
, 2);
2683 prsname
= PQgetvalue(res
, i
, 3);
2684 if (!PQgetisnull(res
, i
, 4))
2685 pnspname
= PQgetvalue(res
, i
, 4);
2687 if (!describeOneTSConfig(oid
, nspname
, cfgname
, pnspname
, prsname
))
2705 describeOneTSConfig(const char *oid
, const char *nspname
, const char *cfgname
,
2706 const char *pnspname
, const char *prsname
)
2708 PQExpBufferData buf
,
2711 printQueryOpt myopt
= pset
.popt
;
2713 initPQExpBuffer(&buf
);
2715 printfPQExpBuffer(&buf
,
2717 " ( SELECT t.alias FROM \n"
2718 " pg_catalog.ts_token_type(c.cfgparser) AS t \n"
2719 " WHERE t.tokid = m.maptokentype ) AS \"%s\", \n"
2720 " pg_catalog.btrim( \n"
2721 " ARRAY( SELECT mm.mapdict::pg_catalog.regdictionary \n"
2722 " FROM pg_catalog.pg_ts_config_map AS mm \n"
2723 " WHERE mm.mapcfg = m.mapcfg AND mm.maptokentype = m.maptokentype \n"
2724 " ORDER BY mapcfg, maptokentype, mapseqno \n"
2725 " ) :: pg_catalog.text , \n"
2726 " '{}') AS \"%s\" \n"
2727 "FROM pg_catalog.pg_ts_config AS c, pg_catalog.pg_ts_config_map AS m \n"
2728 "WHERE c.oid = '%s' AND m.mapcfg = c.oid \n"
2729 "GROUP BY m.mapcfg, m.maptokentype, c.cfgparser \n"
2731 gettext_noop("Token"),
2732 gettext_noop("Dictionaries"),
2735 res
= PSQLexec(buf
.data
, false);
2736 termPQExpBuffer(&buf
);
2740 initPQExpBuffer(&title
);
2743 appendPQExpBuffer(&title
, _("Text search configuration \"%s.%s\""),
2746 appendPQExpBuffer(&title
, _("Text search configuration \"%s\""),
2750 appendPQExpBuffer(&title
, _("\nParser: \"%s.%s\""),
2753 appendPQExpBuffer(&title
, _("\nParser: \"%s\""),
2756 myopt
.nullPrint
= NULL
;
2757 myopt
.title
= title
.data
;
2758 myopt
.footers
= NULL
;
2759 myopt
.default_footer
= false;
2760 myopt
.translate_header
= true;
2762 printQuery(res
, &myopt
, pset
.queryFout
, pset
.logfile
);
2764 termPQExpBuffer(&title
);