mirror of
https://github.com/postgres/postgres.git
synced 2025-05-05 09:19:17 +03:00
Block ALTER TABLE .. DROP NOT NULL on columns in replica identity index
Replica identities that depend directly on an index rely on a set of properties, one of them being that all the columns defined in this index have to be marked as NOT NULL. There was a hole in the logic with ALTER TABLE DROP NOT NULL, where it was possible to remove the NOT NULL property of a column part of an index used as replica identity, so block it to avoid problems with logical decoding down the road. The same check was already done columns part of a primary key, so the fix is straight-forward. Author: Haiying Tang, Hou Zhijie Reviewed-by: Dilip Kumar, Michael Paquier Discussion: https://postgr.es/m/OS0PR01MB6113338C102BEE8B2FFC5BD9FB619@OS0PR01MB6113.jpnprd01.prod.outlook.com Backpatch-through: 10
This commit is contained in:
parent
d2198b4593
commit
e415916e24
@ -7020,7 +7020,8 @@ ATExecDropNotNull(Relation rel, const char *colName, LOCKMODE lockmode)
|
|||||||
colName, RelationGetRelationName(rel))));
|
colName, RelationGetRelationName(rel))));
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Check that the attribute is not in a primary key
|
* Check that the attribute is not in a primary key or in an index used as
|
||||||
|
* a replica identity.
|
||||||
*
|
*
|
||||||
* Note: we'll throw error even if the pkey index is not valid.
|
* Note: we'll throw error even if the pkey index is not valid.
|
||||||
*/
|
*/
|
||||||
@ -7040,20 +7041,32 @@ ATExecDropNotNull(Relation rel, const char *colName, LOCKMODE lockmode)
|
|||||||
elog(ERROR, "cache lookup failed for index %u", indexoid);
|
elog(ERROR, "cache lookup failed for index %u", indexoid);
|
||||||
indexStruct = (Form_pg_index) GETSTRUCT(indexTuple);
|
indexStruct = (Form_pg_index) GETSTRUCT(indexTuple);
|
||||||
|
|
||||||
/* If the index is not a primary key, skip the check */
|
/*
|
||||||
if (indexStruct->indisprimary)
|
* If the index is not a primary key or an index used as replica
|
||||||
|
* identity, skip the check.
|
||||||
|
*/
|
||||||
|
if (indexStruct->indisprimary || indexStruct->indisreplident)
|
||||||
{
|
{
|
||||||
/*
|
/*
|
||||||
* Loop over each attribute in the primary key and see if it
|
* Loop over each attribute in the primary key or the index used
|
||||||
* matches the to-be-altered attribute
|
* as replica identity and see if it matches the to-be-altered
|
||||||
|
* attribute.
|
||||||
*/
|
*/
|
||||||
for (i = 0; i < indexStruct->indnkeyatts; i++)
|
for (i = 0; i < indexStruct->indnkeyatts; i++)
|
||||||
{
|
{
|
||||||
if (indexStruct->indkey.values[i] == attnum)
|
if (indexStruct->indkey.values[i] == attnum)
|
||||||
ereport(ERROR,
|
{
|
||||||
(errcode(ERRCODE_INVALID_TABLE_DEFINITION),
|
if (indexStruct->indisprimary)
|
||||||
errmsg("column \"%s\" is in a primary key",
|
ereport(ERROR,
|
||||||
colName)));
|
(errcode(ERRCODE_INVALID_TABLE_DEFINITION),
|
||||||
|
errmsg("column \"%s\" is in a primary key",
|
||||||
|
colName)));
|
||||||
|
else
|
||||||
|
ereport(ERROR,
|
||||||
|
(errcode(ERRCODE_INVALID_TABLE_DEFINITION),
|
||||||
|
errmsg("column \"%s\" is in index used as replica identity",
|
||||||
|
colName)));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -223,6 +223,10 @@ ALTER TABLE test_replica_identity3 ALTER COLUMN id TYPE bigint;
|
|||||||
Indexes:
|
Indexes:
|
||||||
"test_replica_identity3_id_key" UNIQUE, btree (id) REPLICA IDENTITY
|
"test_replica_identity3_id_key" UNIQUE, btree (id) REPLICA IDENTITY
|
||||||
|
|
||||||
|
-- ALTER TABLE DROP NOT NULL is not allowed for columns part of an index
|
||||||
|
-- used as replica identity.
|
||||||
|
ALTER TABLE test_replica_identity3 ALTER COLUMN id DROP NOT NULL;
|
||||||
|
ERROR: column "id" is in index used as replica identity
|
||||||
DROP TABLE test_replica_identity;
|
DROP TABLE test_replica_identity;
|
||||||
DROP TABLE test_replica_identity2;
|
DROP TABLE test_replica_identity2;
|
||||||
DROP TABLE test_replica_identity3;
|
DROP TABLE test_replica_identity3;
|
||||||
|
@ -94,6 +94,10 @@ ALTER TABLE test_replica_identity3 REPLICA IDENTITY USING INDEX test_replica_ide
|
|||||||
ALTER TABLE test_replica_identity3 ALTER COLUMN id TYPE bigint;
|
ALTER TABLE test_replica_identity3 ALTER COLUMN id TYPE bigint;
|
||||||
\d test_replica_identity3
|
\d test_replica_identity3
|
||||||
|
|
||||||
|
-- ALTER TABLE DROP NOT NULL is not allowed for columns part of an index
|
||||||
|
-- used as replica identity.
|
||||||
|
ALTER TABLE test_replica_identity3 ALTER COLUMN id DROP NOT NULL;
|
||||||
|
|
||||||
DROP TABLE test_replica_identity;
|
DROP TABLE test_replica_identity;
|
||||||
DROP TABLE test_replica_identity2;
|
DROP TABLE test_replica_identity2;
|
||||||
DROP TABLE test_replica_identity3;
|
DROP TABLE test_replica_identity3;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user