Skip to content

Commit df9c5fb

Browse files
committed
Fix DROP DATABASE for databases with many ACLs
Commit c66a7d7 modified DROP DATABASE so that if interrupted, the database is known to be in an invalid state and can only be dropped. This is done by setting a flag using an in-place update, so that it's not lost in case of rollback. For databases with many ACLs, this may however fail like this: ERROR: wrong tuple length This happens because with many ACLs, the pg_database.datacl attribute gets TOASTed. The dropdb() code reads the tuple from the syscache, which means it's detoasted. But the in-place update expects the tuple length to match the on-disk tuple. Fixed by reading the tuple from the catalog directly, not from syscache. Report and fix by Ayush Tiwari. Backpatch to 12. The DROP DATABASE fix was backpatched to 11, but 11 is EOL at this point. Reported-by: Ayush Tiwari Author: Ayush Tiwari Reviewed-by: Tomas Vondra Backpatch-through: 12 Discussion: https://postgr.es/m/CAJTYsWWNkCt+-UnMhg=BiCD3Mh8c2JdHLofPxsW3m2dkDFw8RA@mail.gmail.com
1 parent e8f3641 commit df9c5fb

File tree

1 file changed

+16
-1
lines changed

1 file changed

+16
-1
lines changed

src/backend/commands/dbcommands.c

Lines changed: 16 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1523,6 +1523,8 @@ dropdb(const char *dbname, bool missing_ok, bool force)
15231523
bool db_istemplate;
15241524
Relation pgdbrel;
15251525
HeapTuple tup;
1526+
ScanKeyData scankey;
1527+
SysScanDesc scan;
15261528
Form_pg_database datform;
15271529
int notherbackends;
15281530
int npreparedxacts;
@@ -1660,7 +1662,18 @@ dropdb(const char *dbname, bool missing_ok, bool force)
16601662
*/
16611663
pgstat_drop_database(db_id);
16621664

1663-
tup = SearchSysCacheCopy1(DATABASEOID, ObjectIdGetDatum(db_id));
1665+
/*
1666+
* Update the database's pg_database tuple
1667+
*/
1668+
ScanKeyInit(&scankey,
1669+
Anum_pg_database_datname,
1670+
BTEqualStrategyNumber, F_NAMEEQ,
1671+
CStringGetDatum(dbname));
1672+
1673+
scan = systable_beginscan(pgdbrel, DatabaseNameIndexId, true,
1674+
NULL, 1, &scankey);
1675+
1676+
tup = systable_getnext(scan);
16641677
if (!HeapTupleIsValid(tup))
16651678
elog(ERROR, "cache lookup failed for database %u", db_id);
16661679
datform = (Form_pg_database) GETSTRUCT(tup);
@@ -1686,6 +1699,8 @@ dropdb(const char *dbname, bool missing_ok, bool force)
16861699
*/
16871700
CatalogTupleDelete(pgdbrel, &tup->t_self);
16881701

1702+
systable_endscan(scan);
1703+
16891704
/*
16901705
* Drop db-specific replication slots.
16911706
*/

0 commit comments

Comments
 (0)