Hi,

On Sat, Jan 10, 2026 at 12:14 PM Tatsuro Yamada <[email protected]>
wrote:

> The next patch will include the following:
> - Rebased version
> - Expanded regression tests (kept to a minimum)
>

I revised the patch based on my previous email.

It includes:
  - Rebased version
  - Cosmetic changes
  - Add incorrect option check
  - Use validateSQLNamePattern() instead of processSQLNamePattern()
  - Fix bug \dcs with 'x' (expanded display) option
  - Add table name into ORDER BY to get consistent results
  - Add regression test cases
    - some edge cases suggested by Jim (Thanks!)
    - test cases for using validateSQLNamePattern()
    - incorrect option check, and so on

Comments and suggestions are welcome.

Regards,
Tatsuro Yamada
From e56619698712ee3395d27b47ffc5f421e27c91a2 Mon Sep 17 00:00:00 2001
From: Tatsuro Yamada <[email protected]>
Date: Mon, 12 Jan 2026 14:26:20 +0900
Subject: [PATCH] Add list constraints meta-command \dcs on psql

\dcs shows all kind of constraints by using pg_constraint.
You can filter constraints by appending c/f/n/p/t/u/e to \dcs.
For example, \dcsc will show only check constraints.

This patch also includes:
  - document
  - regression test
  - tab completion

Changes since the last patch:
  - Rebased version
  - Cosmetic changes
  - Add incorrect option check
  - Use validateSQLNamePattern() instead of processSQLNamePattern()
  - Fix bug \dcs with x (expanded display) option
  - Use c.relname instead cst.conrelid::pg_catalog.regclass AS "Table"
    on sql
  - Add table name into ORDER BY clause to get consistent results
  - Add regression test cases
    - some edge cases suggested by Jim (Thanks!)
    - test cases for using validateSQLNamePattern()
    - incorrect option check, and so on
---
 doc/src/sgml/ref/psql-ref.sgml     |  20 +-
 src/bin/psql/command.c             |  20 ++
 src/bin/psql/describe.c            | 132 +++++++++
 src/bin/psql/describe.h            |   4 +
 src/bin/psql/help.c                |   2 +
 src/bin/psql/tab-complete.in.c     |   4 +-
 src/test/regress/expected/psql.out | 451 +++++++++++++++++++++++++++++
 src/test/regress/sql/psql.sql      | 110 +++++++
 8 files changed, 741 insertions(+), 2 deletions(-)

diff --git a/doc/src/sgml/ref/psql-ref.sgml b/doc/src/sgml/ref/psql-ref.sgml
index f56c70263e0..3a6c56d7adc 100644
--- a/doc/src/sgml/ref/psql-ref.sgml
+++ b/doc/src/sgml/ref/psql-ref.sgml
@@ -1548,6 +1548,25 @@ SELECT $1 \parse stmt1
         </listitem>
       </varlistentry>
 
+      <varlistentry id="app-psql-meta-command-dcs">
+        <term><literal>\dcs[cfnptue][Sx+] [ <link linkend="app-psql-patterns"><replaceable class="parameter">pattern</replaceable></link> ]</literal></term>
+        <listitem>
+        <para>
+        Lists constraints.
+        If <replaceable class="parameter">pattern</replaceable>
+        is specified, only entries whose name matches the pattern are listed.
+        The modifiers <literal>c</literal> (check), <literal>f</literal> (foreign key),
+        <literal>n</literal> (not-null), <literal>p</literal> (primary key),
+        <literal>t</literal> (trigger), <literal>u</literal> (unique),
+        <literal>e</literal> (exclusion) can be appended to the command,
+        filtering the kind of constraints to list.
+        By default, only user-created constraints are shown; supply the
+        <literal>S</literal> modifier to include system objects.
+        If <literal>+</literal> is appended to the command name, each object
+        is listed with its associated description.
+        </para>
+        </listitem>
+      </varlistentry>
 
       <varlistentry id="app-psql-meta-command-dc-uc">
         <term><literal>\dC[x+] [ <link linkend="app-psql-patterns"><replaceable class="parameter">pattern</replaceable></link> ]</literal></term>
@@ -1566,7 +1585,6 @@ SELECT $1 \parse stmt1
         </listitem>
       </varlistentry>
 
-
       <varlistentry id="app-psql-meta-command-dd-lc">
         <term><literal>\dd[Sx] [ <link linkend="app-psql-patterns"><replaceable class="parameter">pattern</replaceable></link> ]</literal></term>
         <listitem>
diff --git a/src/bin/psql/command.c b/src/bin/psql/command.c
index 213d48500de..7fb6bde8746 100644
--- a/src/bin/psql/command.c
+++ b/src/bin/psql/command.c
@@ -1103,6 +1103,26 @@ exec_command_d(PsqlScanState scan_state, bool active_branch, const char *cmd)
 					success = describeConfigurationParameters(pattern,
 															  show_verbose,
 															  show_system);
+				else if (strncmp(cmd, "dcs", 3) == 0) /* Constraint */
+					switch (cmd[3])
+					{
+						case '\0':
+						case '+':
+						case 'S':
+						case 'c':
+						case 'f':
+						case 'n':
+						case 'p':
+						case 't':
+						case 'u':
+						case 'e':
+						case 'x':
+							success = listConstraints(&cmd[3], pattern, show_verbose, show_system);
+							break;
+						default:
+							status = PSQL_CMD_UNKNOWN;
+							break;
+					}
 				else
 					success = listConversions(pattern,
 											  show_verbose,
diff --git a/src/bin/psql/describe.c b/src/bin/psql/describe.c
index 3584c4e1428..0fff3207708 100644
--- a/src/bin/psql/describe.c
+++ b/src/bin/psql/describe.c
@@ -4980,6 +4980,138 @@ listExtendedStats(const char *pattern)
 	return true;
 }
 
+/*
+ * \dcs
+ * Describes constraints
+ *
+ * As with \d, you can specify the kinds of constraints you want:
+ *
+ * c for check
+ * f for foreign key
+ * n for not null
+ * p for primary key
+ * t for trigger
+ * u for unique
+ * e for exclusion
+ *
+ * and you can mix and match these in any order.
+ */
+bool
+listConstraints(const char *contypes, const char *pattern, bool verbose, bool showSystem)
+{
+	const char *dcs_options = "cfnptueSx+";
+	bool		showCheck = strchr(contypes, CONSTRAINT_CHECK) != NULL;
+	bool		showForeign = strchr(contypes, CONSTRAINT_FOREIGN) != NULL;
+	bool		showNotnull = strchr(contypes, CONSTRAINT_NOTNULL) != NULL;
+	bool		showPrimary = strchr(contypes, CONSTRAINT_PRIMARY) != NULL;
+	bool		showTrigger = strchr(contypes, CONSTRAINT_TRIGGER) != NULL;
+	bool		showUnique = strchr(contypes, CONSTRAINT_UNIQUE) != NULL;
+				/* 'x' is already used for expanded display, so use 'e' instead */
+	bool		showExclusion = strchr(contypes, 'e') != NULL;
+	bool		showAllkinds = false;
+	PQExpBufferData buf;
+	PGresult   *res;
+	printQueryOpt myopt = pset.popt;
+
+	if (strlen(contypes) != strspn(contypes, dcs_options))
+	{
+		pg_log_error("\\dcs only takes [%s] as options", dcs_options);
+		return true;
+	}
+
+	if (pset.sversion < 180000)
+	{
+		char		sverbuf[32];
+
+		pg_log_error("The server (version %s) does not support this meta-command on psql.",
+					 formatPGVersionNumber(pset.sversion, false,
+										   sverbuf, sizeof(sverbuf)));
+		return true;
+	}
+
+	/* If contypes were not selected, show them all */
+	if (!(showCheck || showForeign || showNotnull || showPrimary || showTrigger || showUnique || showExclusion))
+		showAllkinds = true;
+
+	initPQExpBuffer(&buf);
+	printfPQExpBuffer(&buf,
+					  "SELECT n.nspname AS \"%s\", \n"
+					  "       cst.conname AS \"%s\" ",
+					  gettext_noop("Schema"),
+					  gettext_noop("Name")
+					 );
+
+	if (verbose)
+		appendPQExpBuffer(&buf,
+						  ",\n       pg_catalog.pg_get_constraintdef(cst.oid) AS \"%s\", \n"
+						  "       c.relname AS \"%s\" ",
+						  gettext_noop("Definition"),
+						  gettext_noop("Table")
+						 );
+
+	appendPQExpBufferStr(&buf,
+						 "\nFROM pg_catalog.pg_constraint cst \n"
+						 "     JOIN pg_catalog.pg_namespace n ON n.oid = cst.connamespace \n"
+						 "     JOIN pg_catalog.pg_class c on c.oid = cst.conrelid \n"
+						);
+
+	if (!showSystem && !pattern)
+		appendPQExpBufferStr(&buf,
+							 "WHERE n.nspname <> 'pg_catalog' \n"
+							 "  AND n.nspname <> 'information_schema' \n");
+
+	if (!validateSQLNamePattern(&buf, pattern,
+								!showSystem && !pattern, false,
+								"n.nspname", "cst.conname", NULL,
+								"pg_catalog.pg_table_is_visible(cst.conrelid)",
+								NULL, 3))
+	{
+		termPQExpBuffer(&buf);
+		return false;
+	}
+
+	if (!showAllkinds)
+	{
+		appendPQExpBufferStr(&buf, "  AND cst.contype in (");
+
+		if (showCheck)
+			appendPQExpBufferStr(&buf, CppAsString2(CONSTRAINT_CHECK) ",");
+		if (showForeign)
+			appendPQExpBufferStr(&buf, CppAsString2(CONSTRAINT_FOREIGN) ",");
+		if (showNotnull)
+			appendPQExpBufferStr(&buf, CppAsString2(CONSTRAINT_NOTNULL) ",");
+		if (showPrimary)
+			appendPQExpBufferStr(&buf, CppAsString2(CONSTRAINT_PRIMARY) ",");
+		if (showTrigger)
+			appendPQExpBufferStr(&buf, CppAsString2(CONSTRAINT_TRIGGER) ",");
+		if (showUnique)
+			appendPQExpBufferStr(&buf, CppAsString2(CONSTRAINT_UNIQUE) ",");
+		if (showExclusion)
+			appendPQExpBufferStr(&buf, CppAsString2(CONSTRAINT_EXCLUSION) ",");
+
+		appendPQExpBufferStr(&buf, " ''");	/* dummy */
+		appendPQExpBufferStr(&buf, ")\n");
+	}
+
+	if (verbose)
+		appendPQExpBufferStr(&buf, "ORDER BY 1, 2, 4;");
+	else
+		appendPQExpBufferStr(&buf, "ORDER BY 1, 2;");
+
+	res = PSQLexec(buf.data);
+	termPQExpBuffer(&buf);
+	if (!res)
+		return false;
+
+	myopt.title = _("List of constraints");
+	myopt.translate_header = true;
+
+	printQuery(res, &myopt, pset.queryFout, false, pset.logfile);
+
+	PQclear(res);
+	return true;
+}
+
 /*
  * \dC
  *
diff --git a/src/bin/psql/describe.h b/src/bin/psql/describe.h
index b60a2ad0e14..7b92b1c1ad0 100644
--- a/src/bin/psql/describe.h
+++ b/src/bin/psql/describe.h
@@ -83,6 +83,10 @@ extern bool listConversions(const char *pattern, bool verbose, bool showSystem);
 extern bool describeConfigurationParameters(const char *pattern, bool verbose,
 											bool showSystem);
 
+/* \dcs */
+extern bool listConstraints(const char *contypes, const char *pattern, bool verbose,
+							bool showSystem);
+
 /* \dC */
 extern bool listCasts(const char *pattern, bool verbose);
 
diff --git a/src/bin/psql/help.c b/src/bin/psql/help.c
index dfd9dd73078..77dc1399149 100644
--- a/src/bin/psql/help.c
+++ b/src/bin/psql/help.c
@@ -230,6 +230,8 @@ slashUsage(unsigned short int pager)
 	HELP0("  \\db[x+]  [PATTERN]     list tablespaces\n");
 	HELP0("  \\dc[Sx+] [PATTERN]     list conversions\n");
 	HELP0("  \\dconfig[x+] [PATTERN] list configuration parameters\n");
+	HELP0("  \\dcs[cfnptue] [Sx+] [PATTERN] list [only check/foreign key/not-null/primary key\n"
+		  "                             /constraint trigger/unique key/exclusion] constraints\n");
 	HELP0("  \\dC[x+]  [PATTERN]     list casts\n");
 	HELP0("  \\dd[Sx]  [PATTERN]     show object descriptions not displayed elsewhere\n");
 	HELP0("  \\dD[Sx+] [PATTERN]     list domains\n");
diff --git a/src/bin/psql/tab-complete.in.c b/src/bin/psql/tab-complete.in.c
index 8b91bc00062..11e349ce0cf 100644
--- a/src/bin/psql/tab-complete.in.c
+++ b/src/bin/psql/tab-complete.in.c
@@ -1925,7 +1925,7 @@ psql_completion(const char *text, int start, int end)
 		"\\connect", "\\conninfo", "\\C", "\\cd", "\\close_prepared", "\\copy",
 		"\\copyright", "\\crosstabview",
 		"\\d", "\\da", "\\dA", "\\dAc", "\\dAf", "\\dAo", "\\dAp",
-		"\\db", "\\dc", "\\dconfig", "\\dC", "\\dd", "\\ddp", "\\dD",
+		"\\db", "\\dc", "\\dconfig", "\\dcs", "\\dC", "\\dd", "\\ddp", "\\dD",
 		"\\des", "\\det", "\\deu", "\\dew", "\\dE", "\\df",
 		"\\dF", "\\dFd", "\\dFp", "\\dFt", "\\dg", "\\di", "\\dl", "\\dL",
 		"\\dm", "\\dn", "\\do", "\\dO", "\\dp", "\\dP", "\\dPi", "\\dPt",
@@ -5474,6 +5474,8 @@ match_previous_words(int pattern_id,
 		COMPLETE_WITH_QUERY(Query_for_list_of_tablespaces);
 	else if (TailMatchesCS("\\dconfig*"))
 		COMPLETE_WITH_QUERY_VERBATIM(Query_for_list_of_show_vars);
+	else if (TailMatchesCS("\\dcs*"))
+		COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_constraints_with_schema);
 	else if (TailMatchesCS("\\dD*"))
 		COMPLETE_WITH_SCHEMA_QUERY(Query_for_list_of_domains);
 	else if (TailMatchesCS("\\des*"))
diff --git a/src/test/regress/expected/psql.out b/src/test/regress/expected/psql.out
index c8f3932edf0..334053f4066 100644
--- a/src/test/regress/expected/psql.out
+++ b/src/test/regress/expected/psql.out
@@ -5388,6 +5388,433 @@ List of configuration parameters
 (1 row)
 
 reset work_mem;
+-- check \dcs
+CREATE TABLE con_c (
+    primary_col SERIAL PRIMARY KEY
+);
+CREATE TABLE con_p (
+    primary_col SERIAL PRIMARY KEY,
+    notnull_col TEXT NOT NULL,
+    check_col INT CHECK (check_col >= 0),
+    foreign_col INT REFERENCES con_c(primary_col),
+    unique_col TEXT UNIQUE,
+    exclusion_col INT,
+    CONSTRAINT con_p_exclusion EXCLUDE USING btree (exclusion_col WITH =)
+);
+CREATE OR REPLACE FUNCTION trigger_hoge() RETURNS TRIGGER AS $$
+  BEGIN
+    RETURN NULL;
+  END;
+  $$ LANGUAGE PLPGSQL;
+CREATE CONSTRAINT TRIGGER con_p_trigger AFTER INSERT ON con_p
+  FOR EACH ROW EXECUTE PROCEDURE trigger_hoge();
+---- \dcs shows constraints
+\dcs con_*
+         List of constraints
+ Schema |            Name            
+--------+----------------------------
+ public | con_c_pkey
+ public | con_c_primary_col_not_null
+ public | con_p_check_col_check
+ public | con_p_exclusion
+ public | con_p_foreign_col_fkey
+ public | con_p_notnull_col_not_null
+ public | con_p_pkey
+ public | con_p_primary_col_not_null
+ public | con_p_trigger
+ public | con_p_unique_col_key
+(10 rows)
+
+\dcscfnptue con_*
+         List of constraints
+ Schema |            Name            
+--------+----------------------------
+ public | con_c_pkey
+ public | con_c_primary_col_not_null
+ public | con_p_check_col_check
+ public | con_p_exclusion
+ public | con_p_foreign_col_fkey
+ public | con_p_notnull_col_not_null
+ public | con_p_pkey
+ public | con_p_primary_col_not_null
+ public | con_p_trigger
+ public | con_p_unique_col_key
+(10 rows)
+
+\dcs+ con_*
+                                          List of constraints
+ Schema |            Name            |                       Definition                        | Table 
+--------+----------------------------+---------------------------------------------------------+-------
+ public | con_c_pkey                 | PRIMARY KEY (primary_col)                               | con_c
+ public | con_c_primary_col_not_null | NOT NULL primary_col                                    | con_c
+ public | con_p_check_col_check      | CHECK ((check_col >= 0))                                | con_p
+ public | con_p_exclusion            | EXCLUDE USING btree (exclusion_col WITH =)              | con_p
+ public | con_p_foreign_col_fkey     | FOREIGN KEY (foreign_col) REFERENCES con_c(primary_col) | con_p
+ public | con_p_notnull_col_not_null | NOT NULL notnull_col                                    | con_p
+ public | con_p_pkey                 | PRIMARY KEY (primary_col)                               | con_p
+ public | con_p_primary_col_not_null | NOT NULL primary_col                                    | con_p
+ public | con_p_trigger              | TRIGGER                                                 | con_p
+ public | con_p_unique_col_key       | UNIQUE (unique_col)                                     | con_p
+(10 rows)
+
+\dcscfnptue+ con_*
+                                          List of constraints
+ Schema |            Name            |                       Definition                        | Table 
+--------+----------------------------+---------------------------------------------------------+-------
+ public | con_c_pkey                 | PRIMARY KEY (primary_col)                               | con_c
+ public | con_c_primary_col_not_null | NOT NULL primary_col                                    | con_c
+ public | con_p_check_col_check      | CHECK ((check_col >= 0))                                | con_p
+ public | con_p_exclusion            | EXCLUDE USING btree (exclusion_col WITH =)              | con_p
+ public | con_p_foreign_col_fkey     | FOREIGN KEY (foreign_col) REFERENCES con_c(primary_col) | con_p
+ public | con_p_notnull_col_not_null | NOT NULL notnull_col                                    | con_p
+ public | con_p_pkey                 | PRIMARY KEY (primary_col)                               | con_p
+ public | con_p_primary_col_not_null | NOT NULL primary_col                                    | con_p
+ public | con_p_trigger              | TRIGGER                                                 | con_p
+ public | con_p_unique_col_key       | UNIQUE (unique_col)                                     | con_p
+(10 rows)
+
+\dcsc con_*
+      List of constraints
+ Schema |         Name          
+--------+-----------------------
+ public | con_p_check_col_check
+(1 row)
+
+\dcsc+ con_*
+                        List of constraints
+ Schema |         Name          |        Definition        | Table 
+--------+-----------------------+--------------------------+-------
+ public | con_p_check_col_check | CHECK ((check_col >= 0)) | con_p
+(1 row)
+
+\dcsf con_*
+       List of constraints
+ Schema |          Name          
+--------+------------------------
+ public | con_p_foreign_col_fkey
+(1 row)
+
+\dcsf+ con_*
+                                        List of constraints
+ Schema |          Name          |                       Definition                        | Table 
+--------+------------------------+---------------------------------------------------------+-------
+ public | con_p_foreign_col_fkey | FOREIGN KEY (foreign_col) REFERENCES con_c(primary_col) | con_p
+(1 row)
+
+\dcsn con_*
+         List of constraints
+ Schema |            Name            
+--------+----------------------------
+ public | con_c_primary_col_not_null
+ public | con_p_notnull_col_not_null
+ public | con_p_primary_col_not_null
+(3 rows)
+
+\dcsn+ con_*
+                        List of constraints
+ Schema |            Name            |      Definition      | Table 
+--------+----------------------------+----------------------+-------
+ public | con_c_primary_col_not_null | NOT NULL primary_col | con_c
+ public | con_p_notnull_col_not_null | NOT NULL notnull_col | con_p
+ public | con_p_primary_col_not_null | NOT NULL primary_col | con_p
+(3 rows)
+
+\dcsp con_*
+ List of constraints
+ Schema |    Name    
+--------+------------
+ public | con_c_pkey
+ public | con_p_pkey
+(2 rows)
+
+\dcsp+ con_*
+                   List of constraints
+ Schema |    Name    |        Definition         | Table 
+--------+------------+---------------------------+-------
+ public | con_c_pkey | PRIMARY KEY (primary_col) | con_c
+ public | con_p_pkey | PRIMARY KEY (primary_col) | con_p
+(2 rows)
+
+\dcst con_*
+  List of constraints
+ Schema |     Name      
+--------+---------------
+ public | con_p_trigger
+(1 row)
+
+\dcst+ con_*
+             List of constraints
+ Schema |     Name      | Definition | Table 
+--------+---------------+------------+-------
+ public | con_p_trigger | TRIGGER    | con_p
+(1 row)
+
+\dcsu con_*
+      List of constraints
+ Schema |         Name         
+--------+----------------------
+ public | con_p_unique_col_key
+(1 row)
+
+\dcsu+ con_*
+                     List of constraints
+ Schema |         Name         |     Definition      | Table 
+--------+----------------------+---------------------+-------
+ public | con_p_unique_col_key | UNIQUE (unique_col) | con_p
+(1 row)
+
+\dcse con_*
+   List of constraints
+ Schema |      Name       
+--------+-----------------
+ public | con_p_exclusion
+(1 row)
+
+\dcse+ con_*
+                              List of constraints
+ Schema |      Name       |                 Definition                 | Table 
+--------+-----------------+--------------------------------------------+-------
+ public | con_p_exclusion | EXCLUDE USING btree (exclusion_col WITH =) | con_p
+(1 row)
+
+\dcsx con_*
+List of constraints
+-[ RECORD 1 ]----------------------
+Schema | public
+Name   | con_c_pkey
+-[ RECORD 2 ]----------------------
+Schema | public
+Name   | con_c_primary_col_not_null
+-[ RECORD 3 ]----------------------
+Schema | public
+Name   | con_p_check_col_check
+-[ RECORD 4 ]----------------------
+Schema | public
+Name   | con_p_exclusion
+-[ RECORD 5 ]----------------------
+Schema | public
+Name   | con_p_foreign_col_fkey
+-[ RECORD 6 ]----------------------
+Schema | public
+Name   | con_p_notnull_col_not_null
+-[ RECORD 7 ]----------------------
+Schema | public
+Name   | con_p_pkey
+-[ RECORD 8 ]----------------------
+Schema | public
+Name   | con_p_primary_col_not_null
+-[ RECORD 9 ]----------------------
+Schema | public
+Name   | con_p_trigger
+-[ RECORD 10 ]---------------------
+Schema | public
+Name   | con_p_unique_col_key
+
+\dcsx+ con_*
+List of constraints
+-[ RECORD 1 ]-------------------------------------------------------
+Schema     | public
+Name       | con_c_pkey
+Definition | PRIMARY KEY (primary_col)
+Table      | con_c
+-[ RECORD 2 ]-------------------------------------------------------
+Schema     | public
+Name       | con_c_primary_col_not_null
+Definition | NOT NULL primary_col
+Table      | con_c
+-[ RECORD 3 ]-------------------------------------------------------
+Schema     | public
+Name       | con_p_check_col_check
+Definition | CHECK ((check_col >= 0))
+Table      | con_p
+-[ RECORD 4 ]-------------------------------------------------------
+Schema     | public
+Name       | con_p_exclusion
+Definition | EXCLUDE USING btree (exclusion_col WITH =)
+Table      | con_p
+-[ RECORD 5 ]-------------------------------------------------------
+Schema     | public
+Name       | con_p_foreign_col_fkey
+Definition | FOREIGN KEY (foreign_col) REFERENCES con_c(primary_col)
+Table      | con_p
+-[ RECORD 6 ]-------------------------------------------------------
+Schema     | public
+Name       | con_p_notnull_col_not_null
+Definition | NOT NULL notnull_col
+Table      | con_p
+-[ RECORD 7 ]-------------------------------------------------------
+Schema     | public
+Name       | con_p_pkey
+Definition | PRIMARY KEY (primary_col)
+Table      | con_p
+-[ RECORD 8 ]-------------------------------------------------------
+Schema     | public
+Name       | con_p_primary_col_not_null
+Definition | NOT NULL primary_col
+Table      | con_p
+-[ RECORD 9 ]-------------------------------------------------------
+Schema     | public
+Name       | con_p_trigger
+Definition | TRIGGER
+Table      | con_p
+-[ RECORD 10 ]------------------------------------------------------
+Schema     | public
+Name       | con_p_unique_col_key
+Definition | UNIQUE (unique_col)
+Table      | con_p
+
+\dcsS pg_constraint*
+                    List of constraints
+   Schema   |                     Name                      
+------------+-----------------------------------------------
+ pg_catalog | pg_constraint_conrelid_contypid_conname_index
+ pg_catalog | pg_constraint_oid_index
+ pg_catalog | pg_constraint_oid_not_null
+(3 rows)
+
+\dcsS+ pg_constraint*
+                                                List of constraints
+   Schema   |                     Name                      |              Definition              |     Table     
+------------+-----------------------------------------------+--------------------------------------+---------------
+ pg_catalog | pg_constraint_conrelid_contypid_conname_index | UNIQUE (conrelid, contypid, conname) | pg_constraint
+ pg_catalog | pg_constraint_oid_index                       | PRIMARY KEY (oid)                    | pg_constraint
+ pg_catalog | pg_constraint_oid_not_null                    | NOT NULL oid                         | pg_constraint
+(3 rows)
+
+\dcscfnpueS pg_constraint*
+                    List of constraints
+   Schema   |                     Name                      
+------------+-----------------------------------------------
+ pg_catalog | pg_constraint_conrelid_contypid_conname_index
+ pg_catalog | pg_constraint_oid_index
+ pg_catalog | pg_constraint_oid_not_null
+(3 rows)
+
+\dcscfnpueS+ pg_constraint*
+                                                List of constraints
+   Schema   |                     Name                      |              Definition              |     Table     
+------------+-----------------------------------------------+--------------------------------------+---------------
+ pg_catalog | pg_constraint_conrelid_contypid_conname_index | UNIQUE (conrelid, contypid, conname) | pg_constraint
+ pg_catalog | pg_constraint_oid_index                       | PRIMARY KEY (oid)                    | pg_constraint
+ pg_catalog | pg_constraint_oid_not_null                    | NOT NULL oid                         | pg_constraint
+(3 rows)
+
+\dcscfnpueS+x pg_constraint*
+List of constraints
+-[ RECORD 1 ]---------------------------------------------
+Schema     | pg_catalog
+Name       | pg_constraint_conrelid_contypid_conname_index
+Definition | UNIQUE (conrelid, contypid, conname)
+Table      | pg_constraint
+-[ RECORD 2 ]---------------------------------------------
+Schema     | pg_catalog
+Name       | pg_constraint_oid_index
+Definition | PRIMARY KEY (oid)
+Table      | pg_constraint
+-[ RECORD 3 ]---------------------------------------------
+Schema     | pg_catalog
+Name       | pg_constraint_oid_not_null
+Definition | NOT NULL oid
+Table      | pg_constraint
+
+---- \dcs doesn't show constraints related to domain,
+---- since \dD can be used to check them
+CREATE DOMAIN notnulldomain integer NOT NULL;
+CREATE TABLE table_used_domain (
+    col notnulldomain
+);
+\dcs+ notnulldomain*;
+        List of constraints
+ Schema | Name | Definition | Table 
+--------+------+------------+-------
+(0 rows)
+
+---- Incorrect options will result in an error
+\dcscz
+\dcs only takes [cfnptueSx+] as options
+---- test with search_path
+CREATE SCHEMA con_schema;
+CREATE TABLE con_schema.con_schema_test (
+    primary_col SERIAL PRIMARY KEY
+);
+SET SEARCH_PATH TO public, con_schema;
+\dcs con_*
+                List of constraints
+   Schema   |                 Name                 
+------------+--------------------------------------
+ con_schema | con_schema_test_pkey
+ con_schema | con_schema_test_primary_col_not_null
+ public     | con_c_pkey
+ public     | con_c_primary_col_not_null
+ public     | con_p_check_col_check
+ public     | con_p_exclusion
+ public     | con_p_foreign_col_fkey
+ public     | con_p_notnull_col_not_null
+ public     | con_p_pkey
+ public     | con_p_primary_col_not_null
+ public     | con_p_trigger
+ public     | con_p_unique_col_key
+(12 rows)
+
+SET SEARCH_PATH TO public;
+\dcs con_*
+         List of constraints
+ Schema |            Name            
+--------+----------------------------
+ public | con_c_pkey
+ public | con_c_primary_col_not_null
+ public | con_p_check_col_check
+ public | con_p_exclusion
+ public | con_p_foreign_col_fkey
+ public | con_p_notnull_col_not_null
+ public | con_p_pkey
+ public | con_p_primary_col_not_null
+ public | con_p_trigger
+ public | con_p_unique_col_key
+(10 rows)
+
+RESET search_path;
+---- test with inherits
+CREATE TABLE zoo_parent (
+  cage int PRIMARY KEY
+);
+CREATE TABLE zoo_child (
+    animal text
+) INHERITS (zoo_parent);
+\dcs+ zoo_parent_cage_not_null
+                      List of constraints
+ Schema |           Name           |  Definition   |   Table    
+--------+--------------------------+---------------+------------
+ public | zoo_parent_cage_not_null | NOT NULL cage | zoo_child
+ public | zoo_parent_cage_not_null | NOT NULL cage | zoo_parent
+(2 rows)
+
+---- test with partitioned table
+CREATE TABLE zoo_part (
+    cage int,
+    animal text,
+    CONSTRAINT zoo_part_pk PRIMARY KEY (cage)
+) PARTITION BY RANGE (cage);
+CREATE TABLE zoo_part_1
+    PARTITION OF zoo_part
+    FOR VALUES FROM (0) TO (100);
+\dcs+ zoo_part*
+                        List of constraints
+ Schema |          Name          |     Definition     |   Table    
+--------+------------------------+--------------------+------------
+ public | zoo_part_1_pkey        | PRIMARY KEY (cage) | zoo_part_1
+ public | zoo_part_cage_not_null | NOT NULL cage      | zoo_part
+ public | zoo_part_cage_not_null | NOT NULL cage      | zoo_part_1
+ public | zoo_part_pk            | PRIMARY KEY (cage) | zoo_part
+(4 rows)
+
+-- clean up for \dcs test cases
+DROP SCHEMA con_schema CASCADE;
+NOTICE:  drop cascades to table con_schema.con_schema_test
+DROP TABLE con_p, con_c, table_used_domain, zoo_parent, zoo_part CASCADE;
+NOTICE:  drop cascades to table zoo_child
+DROP FUNCTION trigger_hoge;
+DROP DOMAIN notnulldomain;
 -- check \df, \do with argument specifications
 \df *sqrt
                              List of functions
@@ -6100,6 +6527,10 @@ improper qualified name (too many dotted names): host.regression.public.conversi
 cross-database references are not implemented: (.public.conversion
 \dc nonesuch.public.conversion
 cross-database references are not implemented: nonesuch.public.conversion
+\dcs host.regression.public.constraint
+improper qualified name (too many dotted names): host.regression.public.constraint
+\dcs nonesuch.public.constraint
+cross-database references are not implemented: nonesuch.public.constraint
 \dC host.regression.pg_catalog.int8
 improper qualified name (too many dotted names): host.regression.pg_catalog.int8
 \dC ).pg_catalog.int8
@@ -6333,6 +6764,12 @@ List of access methods
 --------+------+--------+-------------+----------
 (0 rows)
 
+\dcs "no.such.constraint"
+List of constraints
+ Schema | Name 
+--------+------
+(0 rows)
+
 \dC "no.such.cast"
                   List of casts
  Source type | Target type | Function | Implicit? 
@@ -6543,6 +6980,12 @@ improper qualified name (too many dotted names): "no.such.schema"."no.such.table
 --------+------+--------+-------------+----------
 (0 rows)
 
+\dcs "no.such.schema"."no.such.constraint"
+List of constraints
+ Schema | Name 
+--------+------
+(0 rows)
+
 \dC "no.such.schema"."no.such.cast"
                   List of casts
  Source type | Target type | Function | Implicit? 
@@ -6700,6 +7143,12 @@ improper qualified name (too many dotted names): "no.such.schema"."no.such.event
 --------+------+--------+-------------+----------
 (0 rows)
 
+\dcs regression."no.such.schema"."no.such.constraint"
+List of constraints
+ Schema | Name 
+--------+------
+(0 rows)
+
 \dC regression."no.such.schema"."no.such.cast"
                   List of casts
  Source type | Target type | Function | Implicit? 
@@ -6821,6 +7270,8 @@ cross-database references are not implemented: "no.such.database"."no.such.schem
 cross-database references are not implemented: "no.such.database"."no.such.schema"."no.such.aggregate.function"
 \dc "no.such.database"."no.such.schema"."no.such.conversion"
 cross-database references are not implemented: "no.such.database"."no.such.schema"."no.such.conversion"
+\dcs "no.such.database"."no.such.schema"."no.such.constraint"
+cross-database references are not implemented: "no.such.database"."no.such.schema"."no.such.constraint"
 \dC "no.such.database"."no.such.schema"."no.such.cast"
 cross-database references are not implemented: "no.such.database"."no.such.schema"."no.such.cast"
 \dd "no.such.database"."no.such.schema"."no.such.object.description"
diff --git a/src/test/regress/sql/psql.sql b/src/test/regress/sql/psql.sql
index dcdbd4fc020..714e59573c2 100644
--- a/src/test/regress/sql/psql.sql
+++ b/src/test/regress/sql/psql.sql
@@ -1347,6 +1347,110 @@ set work_mem = 10240;
 \dconfig+ work*
 reset work_mem;
 
+-- check \dcs
+CREATE TABLE con_c (
+    primary_col SERIAL PRIMARY KEY
+);
+
+CREATE TABLE con_p (
+    primary_col SERIAL PRIMARY KEY,
+    notnull_col TEXT NOT NULL,
+    check_col INT CHECK (check_col >= 0),
+    foreign_col INT REFERENCES con_c(primary_col),
+    unique_col TEXT UNIQUE,
+    exclusion_col INT,
+    CONSTRAINT con_p_exclusion EXCLUDE USING btree (exclusion_col WITH =)
+);
+
+CREATE OR REPLACE FUNCTION trigger_hoge() RETURNS TRIGGER AS $$
+  BEGIN
+    RETURN NULL;
+  END;
+  $$ LANGUAGE PLPGSQL;
+
+CREATE CONSTRAINT TRIGGER con_p_trigger AFTER INSERT ON con_p
+  FOR EACH ROW EXECUTE PROCEDURE trigger_hoge();
+
+---- \dcs shows constraints
+\dcs con_*
+\dcscfnptue con_*
+\dcs+ con_*
+\dcscfnptue+ con_*
+\dcsc con_*
+\dcsc+ con_*
+\dcsf con_*
+\dcsf+ con_*
+\dcsn con_*
+\dcsn+ con_*
+\dcsp con_*
+\dcsp+ con_*
+\dcst con_*
+\dcst+ con_*
+\dcsu con_*
+\dcsu+ con_*
+\dcse con_*
+\dcse+ con_*
+\dcsx con_*
+\dcsx+ con_*
+\dcsS pg_constraint*
+\dcsS+ pg_constraint*
+\dcscfnpueS pg_constraint*
+\dcscfnpueS+ pg_constraint*
+\dcscfnpueS+x pg_constraint*
+
+---- \dcs doesn't show constraints related to domain,
+---- since \dD can be used to check them
+CREATE DOMAIN notnulldomain integer NOT NULL;
+CREATE TABLE table_used_domain (
+    col notnulldomain
+);
+\dcs+ notnulldomain*;
+
+---- Incorrect options will result in an error
+\dcscz
+
+---- test with search_path
+CREATE SCHEMA con_schema;
+CREATE TABLE con_schema.con_schema_test (
+    primary_col SERIAL PRIMARY KEY
+);
+
+SET SEARCH_PATH TO public, con_schema;
+\dcs con_*
+SET SEARCH_PATH TO public;
+\dcs con_*
+RESET search_path;
+
+---- test with inherits
+CREATE TABLE zoo_parent (
+  cage int PRIMARY KEY
+);
+
+CREATE TABLE zoo_child (
+    animal text
+) INHERITS (zoo_parent);
+
+\dcs+ zoo_parent_cage_not_null
+
+---- test with partitioned table
+CREATE TABLE zoo_part (
+    cage int,
+    animal text,
+    CONSTRAINT zoo_part_pk PRIMARY KEY (cage)
+) PARTITION BY RANGE (cage);
+
+CREATE TABLE zoo_part_1
+    PARTITION OF zoo_part
+    FOR VALUES FROM (0) TO (100);
+
+\dcs+ zoo_part*
+
+-- clean up for \dcs test cases
+DROP SCHEMA con_schema CASCADE;
+DROP TABLE con_p, con_c, table_used_domain, zoo_parent, zoo_part CASCADE;
+DROP FUNCTION trigger_hoge;
+DROP DOMAIN notnulldomain;
+
 -- check \df, \do with argument specifications
 \df *sqrt
 \df *sqrt num*
@@ -1683,6 +1787,8 @@ DROP FUNCTION psql_error;
 \dc host.regression.public.conversion
 \dc (.public.conversion
 \dc nonesuch.public.conversion
+\dcs host.regression.public.constraint
+\dcs nonesuch.public.constraint
 \dC host.regression.pg_catalog.int8
 \dC ).pg_catalog.int8
 \dC nonesuch.pg_catalog.int8
@@ -1783,6 +1889,7 @@ DROP FUNCTION psql_error;
 \dAp "no.such.operator.support.function.of.operator.family"
 \db "no.such.tablespace"
 \dc "no.such.conversion"
+\dcs "no.such.constraint"
 \dC "no.such.cast"
 \dd "no.such.object.description"
 \dD "no.such.domain"
@@ -1824,6 +1931,7 @@ DROP FUNCTION psql_error;
 \dAp "no.such.schema"."no.such.operator.support.function.of.operator.family"
 \db "no.such.schema"."no.such.tablespace"
 \dc "no.such.schema"."no.such.conversion"
+\dcs "no.such.schema"."no.such.constraint"
 \dC "no.such.schema"."no.such.cast"
 \dd "no.such.schema"."no.such.object.description"
 \dD "no.such.schema"."no.such.domain"
@@ -1858,6 +1966,7 @@ DROP FUNCTION psql_error;
 \dt regression."no.such.schema"."no.such.table.relation"
 \da regression."no.such.schema"."no.such.aggregate.function"
 \dc regression."no.such.schema"."no.such.conversion"
+\dcs regression."no.such.schema"."no.such.constraint"
 \dC regression."no.such.schema"."no.such.cast"
 \dd regression."no.such.schema"."no.such.object.description"
 \dD regression."no.such.schema"."no.such.domain"
@@ -1882,6 +1991,7 @@ DROP FUNCTION psql_error;
 \dt "no.such.database"."no.such.schema"."no.such.table.relation"
 \da "no.such.database"."no.such.schema"."no.such.aggregate.function"
 \dc "no.such.database"."no.such.schema"."no.such.conversion"
+\dcs "no.such.database"."no.such.schema"."no.such.constraint"
 \dC "no.such.database"."no.such.schema"."no.such.cast"
 \dd "no.such.database"."no.such.schema"."no.such.object.description"
 \dD "no.such.database"."no.such.schema"."no.such.domain"
-- 
2.43.5

Reply via email to