Remove support for unlogged on partitioned tables
authorMichael Paquier <michael@paquier.xyz>
Thu, 3 Oct 2024 01:55:02 +0000 (10:55 +0900)
committerMichael Paquier <michael@paquier.xyz>
Thu, 3 Oct 2024 01:55:02 +0000 (10:55 +0900)
The following commands were allowed on partitioned tables, with
different effects:
1) ALTER TABLE SET [UN]LOGGED did not issue an error, and did not update
pg_class.relpersistence.
2) CREATE UNLOGGED TABLE was working with pg_class.relpersistence marked
as initially defined, but partitions did not inherit the UNLOGGED
property, which was confusing.

This commit causes the commands mentioned above to fail for partitioned
tables, instead.

pg_dump is tweaked so as partitioned tables marked as UNLOGGED ignore
the option when dumped from older server versions.  pgbench needs a
tweak for --unlogged and --partitions=N to ignore the UNLOGGED option on
the partitioned tables created, its partitions still being unlogged.

Author: Michael Paquier
Reviewed-by: Nathan Bossart
Discussion: https://postgr.es/m/ZiiyGFTBNkqcMQi_@paquier.xyz

doc/src/sgml/ref/alter_table.sgml
doc/src/sgml/ref/create_table.sgml
src/backend/commands/tablecmds.c
src/bin/pg_dump/pg_dump.c
src/bin/pgbench/pgbench.c
src/test/regress/expected/create_table.out
src/test/regress/sql/create_table.sql

index c5a14674804f3d4e24b6da98f42cc52c2bcb1147..36770c012a64860b455087cf2fc3122c2d95eb9c 100644 (file)
@@ -797,6 +797,10 @@ WITH ( MODULUS <replaceable class="parameter">numeric_literal</replaceable>, REM
       (for identity or serial columns).  However, it is also possible to
       change the persistence of such sequences separately.
      </para>
+
+     <para>
+      This form is not supported for partitioned tables.
+     </para>
     </listitem>
    </varlistentry>
 
index c1855b8d827b531b8337e71f0cac2e4ada554e88..83859bac76f77a2278fa110acdbedb0c8430efbe 100644 (file)
@@ -220,6 +220,10 @@ WITH ( MODULUS <replaceable class="parameter">numeric_literal</replaceable>, REM
       If this is specified, any sequences created together with the unlogged
       table (for identity or serial columns) are also created as unlogged.
      </para>
+
+     <para>
+      This form is not supported for partitioned tables.
+     </para>
     </listitem>
    </varlistentry>
 
index 96373323b8df886b68924ff99758d762802bb831..16c86ffcc0f9b7948edb5c38973783dadd49ac2f 100644 (file)
@@ -734,6 +734,12 @@ DefineRelation(CreateStmt *stmt, char relkind, Oid ownerId,
        else
                partitioned = false;
 
+       if (relkind == RELKIND_PARTITIONED_TABLE &&
+               stmt->relation->relpersistence == RELPERSISTENCE_UNLOGGED)
+               ereport(ERROR,
+                               (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
+                                errmsg("partitioned tables cannot be unlogged")));
+
        /*
         * Look up the namespace in which we are supposed to create the relation,
         * check we have permission to create there, lock it against concurrent
@@ -4993,8 +4999,7 @@ ATPrepCmd(List **wqueue, Relation rel, AlterTableCmd *cmd,
                        break;
                case AT_SetLogged:              /* SET LOGGED */
                case AT_SetUnLogged:    /* SET UNLOGGED */
-                       ATSimplePermissions(cmd->subtype, rel,
-                                                               ATT_TABLE | ATT_PARTITIONED_TABLE | ATT_SEQUENCE);
+                       ATSimplePermissions(cmd->subtype, rel, ATT_TABLE | ATT_SEQUENCE);
                        if (tab->chgPersistence)
                                ereport(ERROR,
                                                (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
index 4dfb7d1abe4dbeeaf40d83f65d537e29bd542db7..1b47c388ced053dd9319f5b28b5d70aa88a1f5bf 100644 (file)
@@ -15915,8 +15915,13 @@ dumpTableSchema(Archive *fout, const TableInfo *tbinfo)
                        binary_upgrade_set_pg_class_oids(fout, q,
                                                                                         tbinfo->dobj.catId.oid);
 
+               /*
+                * PostgreSQL 18 has disabled UNLOGGED for partitioned tables, so
+                * ignore it when dumping if it was set in this case.
+                */
                appendPQExpBuffer(q, "CREATE %s%s %s",
-                                                 tbinfo->relpersistence == RELPERSISTENCE_UNLOGGED ?
+                                                 (tbinfo->relpersistence == RELPERSISTENCE_UNLOGGED &&
+                                                  tbinfo->relkind != RELKIND_PARTITIONED_TABLE) ?
                                                  "UNLOGGED " : "",
                                                  reltypename,
                                                  qualrelname);
index 61618f2e188fc8f5585bca96c3408f96b68a2056..e658d060add93effb58298f00e5796b415cff6c1 100644 (file)
@@ -4865,7 +4865,7 @@ initCreateTables(PGconn *con)
 
                /* Construct new create table statement. */
                printfPQExpBuffer(&query, "create%s table %s(%s)",
-                                                 unlogged_tables ? " unlogged" : "",
+                                                 (unlogged_tables && partition_method == PART_NONE) ? " unlogged" : "",
                                                  ddl->table,
                                                  (scale >= SCALE_32BIT_THRESHOLD) ? ddl->bigcols : ddl->smcols);
 
index 284a7fb85c8f96e5404389eb26b3b16efe62ec1a..c45e02d42f928313076c2e5ecf2bc21c90247b02 100644 (file)
@@ -50,6 +50,16 @@ ERROR:  cannot create temporary relation in non-temporary schema
 LINE 1: CREATE TEMP TABLE public.temp_to_perm (a int primary key);
                           ^
 DROP TABLE unlogged1, public.unlogged2;
+CREATE UNLOGGED TABLE unlogged1 (a int) PARTITION BY RANGE (a); -- fail
+ERROR:  partitioned tables cannot be unlogged
+CREATE TABLE unlogged1 (a int) PARTITION BY RANGE (a); -- ok
+ALTER TABLE unlogged1 SET LOGGED; -- fails
+ERROR:  ALTER action SET LOGGED cannot be performed on relation "unlogged1"
+DETAIL:  This operation is not supported for partitioned tables.
+ALTER TABLE unlogged1 SET UNLOGGED; -- fails
+ERROR:  ALTER action SET UNLOGGED cannot be performed on relation "unlogged1"
+DETAIL:  This operation is not supported for partitioned tables.
+DROP TABLE unlogged1;
 CREATE TABLE as_select1 AS SELECT * FROM pg_class WHERE relkind = 'r';
 CREATE TABLE as_select1 AS SELECT * FROM pg_class WHERE relkind = 'r';
 ERROR:  relation "as_select1" already exists
index 1fd4cbfa7ef0d71699bfdaf449d8c65bfcbfbee2..37a227148e9c2270b83bf8e53a04aab1f5901720 100644 (file)
@@ -30,6 +30,12 @@ CREATE TEMP TABLE pg_temp.doubly_temp (a int primary key);           -- also OK
 CREATE TEMP TABLE public.temp_to_perm (a int primary key);             -- not OK
 DROP TABLE unlogged1, public.unlogged2;
 
+CREATE UNLOGGED TABLE unlogged1 (a int) PARTITION BY RANGE (a); -- fail
+CREATE TABLE unlogged1 (a int) PARTITION BY RANGE (a); -- ok
+ALTER TABLE unlogged1 SET LOGGED; -- fails
+ALTER TABLE unlogged1 SET UNLOGGED; -- fails
+DROP TABLE unlogged1;
+
 CREATE TABLE as_select1 AS SELECT * FROM pg_class WHERE relkind = 'r';
 CREATE TABLE as_select1 AS SELECT * FROM pg_class WHERE relkind = 'r';
 CREATE TABLE IF NOT EXISTS as_select1 AS SELECT * FROM pg_class WHERE relkind = 'r';