Prevent altering partitioned table's rowtype, if it's used elsewhere.
authorTom Lane <[email protected]>
Thu, 6 Jan 2022 21:46:46 +0000 (16:46 -0500)
committerTom Lane <[email protected]>
Thu, 6 Jan 2022 21:46:46 +0000 (16:46 -0500)
We disallow altering a column datatype within a regular table,
if the table's rowtype is used as a column type elsewhere,
because we lack code to go around and rewrite the other tables.
This restriction should apply to partitioned tables as well, but it
was not checked because ATRewriteTables and ATPrepAlterColumnType
were not on the same page about who should do it for which relkinds.

Per bug #17351 from Alexander Lakhin.  Back-patch to all supported
branches.

Discussion: https://postgr.es/m/17351-6db1870f3f4f612a@postgresql.org

src/backend/commands/tablecmds.c
src/test/regress/expected/alter_table.out
src/test/regress/sql/alter_table.sql

index ef88a10f0174abc7ce0cc3bd64b35f2e3a2c8a44..b0a7e3221f17fd0e532efa88dfd1b96d7e1a30b6 100644 (file)
@@ -8847,11 +8847,12 @@ ATPrepAlterColumnType(List **wqueue,
                        RelationGetRelationName(rel))));
 
    if (tab->relkind == RELKIND_COMPOSITE_TYPE ||
-       tab->relkind == RELKIND_FOREIGN_TABLE)
+       tab->relkind == RELKIND_FOREIGN_TABLE ||
+       tab->relkind == RELKIND_PARTITIONED_TABLE)
    {
        /*
-        * For composite types, do this check now.  Tables will check it later
-        * when the table is being rewritten.
+        * For relations without storage, do this check now.  Regular tables
+        * will check it later when the table is being rewritten.
         */
        find_composite_type_dependencies(rel->rd_rel->reltype, rel, NULL);
    }
index c22ae567549c2955a98a83797e51413e73407520..d566716089472c6cb393c07e4cc0eda10e047f62 100644 (file)
@@ -2036,11 +2036,24 @@ select * from another;
 (3 rows)
 
 drop table another;
--- table's row type
-create table tab1 (a int, b text);
-create table tab2 (x int, y tab1);
-alter table tab1 alter column b type varchar; -- fails
-ERROR:  cannot alter table "tab1" because column "tab2.y" uses its row type
+-- We disallow changing table's row type if it's used for storage
+create table at_tab1 (a int, b text);
+create table at_tab2 (x int, y at_tab1);
+alter table at_tab1 alter column b type varchar; -- fails
+ERROR:  cannot alter table "at_tab1" because column "at_tab2.y" uses its row type
+drop table at_tab2;
+-- Use of row type in an expression is defended differently
+create table at_tab2 (x int, y text, check((x,y)::at_tab1 = (1,'42')::at_tab1));
+alter table at_tab1 alter column b type varchar; -- allowed, but ...
+insert into at_tab2 values(1,'42'); -- ... this will fail
+ERROR:  ROW() column has type text instead of type character varying
+drop table at_tab1, at_tab2;
+-- Check it for a partitioned table, too
+create table at_tab1 (a int, b text) partition by list(a);
+create table at_tab2 (x int, y at_tab1);
+alter table at_tab1 alter column b type varchar; -- fails
+ERROR:  cannot alter table "at_tab1" because column "at_tab2.y" uses its row type
+drop table at_tab1, at_tab2;
 -- disallow recursive containment of row types
 create temp table recur1 (f1 int);
 alter table recur1 add column f2 recur1; -- fails
index 67c7fa131efccdd2009ecea31fa5ca0ef64daff7..4968b1f7a354c9794495392580d878d6b59b6ebf 100644 (file)
@@ -1370,10 +1370,21 @@ select * from another;
 
 drop table another;
 
--- table's row type
-create table tab1 (a int, b text);
-create table tab2 (x int, y tab1);
-alter table tab1 alter column b type varchar; -- fails
+-- We disallow changing table's row type if it's used for storage
+create table at_tab1 (a int, b text);
+create table at_tab2 (x int, y at_tab1);
+alter table at_tab1 alter column b type varchar; -- fails
+drop table at_tab2;
+-- Use of row type in an expression is defended differently
+create table at_tab2 (x int, y text, check((x,y)::at_tab1 = (1,'42')::at_tab1));
+alter table at_tab1 alter column b type varchar; -- allowed, but ...
+insert into at_tab2 values(1,'42'); -- ... this will fail
+drop table at_tab1, at_tab2;
+-- Check it for a partitioned table, too
+create table at_tab1 (a int, b text) partition by list(a);
+create table at_tab2 (x int, y at_tab1);
+alter table at_tab1 alter column b type varchar; -- fails
+drop table at_tab1, at_tab2;
 
 -- disallow recursive containment of row types
 create temp table recur1 (f1 int);