E.28. Release 13.22
Release date: 2025-08-14
This release contains a variety of fixes from 13.21. For information about new features in major release 13, see Section E.50.
The PostgreSQL community will stop releasing updates for the 13.X release series in November 2025. Users are encouraged to update to a newer release branch soon.
E.28.1. Migration to Version 13.22
A dump/restore is not required for those running 13.X.
However, if you are upgrading from a version earlier than 13.21, see Section E.29.
E.28.2. Changes
- Tighten security checks in planner estimation functions (Dean Rasheed) § - The fix for CVE-2017-7484, plus followup fixes, intended to prevent leaky functions from being applied to statistics data for columns that the calling user does not have permission to read. Two gaps in that protection have been found. One gap applies to partitioning and inheritance hierarchies where RLS policies on the tables should restrict access to statistics data, but did not. - The other gap applies to cases where the query accesses a table via a view, and the view owner has permissions to read the underlying table but the calling user does not have permissions on the view. The view owner's permissions satisfied the security checks, and the leaky function would get applied to the underlying table's statistics before we check the calling user's permissions on the view. This has been fixed by making security checks on views occur at the start of planning. That might cause permissions failures to occur earlier than before. - The PostgreSQL Project thanks Dean Rasheed for reporting this problem. (CVE-2025-8713) 
- Prevent pg_dump scripts from being used to attack the user running the restore (Nathan Bossart) § § - Since dump/restore operations typically involve running SQL commands as superuser, the target database installation must trust the source server. However, it does not follow that the operating system user who executes psql to perform the restore should have to trust the source server. The risk here is that an attacker who has gained superuser-level control over the source server might be able to cause it to emit text that would be interpreted as psql meta-commands. That would provide shell-level access to the restoring user's own account, independently of access to the target database. - To provide a positive guarantee that this can't happen, extend psql with a - \restrictcommand that prevents execution of further meta-commands, and teach pg_dump to issue that before any data coming from the source server.- The PostgreSQL Project thanks Martin Rakhmanov, Matthieu Denais, and RyotaK for reporting this problem. (CVE-2025-8714) 
- Convert newlines to spaces in names included in comments in pg_dump output (Noah Misch) § - Object names containing newlines offered the ability to inject arbitrary SQL commands into the output script. (Without the preceding fix, injection of psql meta-commands would also be possible this way.) CVE-2012-0868 fixed this class of problem at the time, but later work reintroduced several cases. - The PostgreSQL Project thanks Noah Misch for reporting this problem. (CVE-2025-8715) 
- Avoid regression in the size of XML input that we will accept (Michael Paquier, Erik Wienhold) § § - Our workaround for a bug in early 2.13.x releases of libxml2 made use of a code path that rejects text chunks exceeding 10MB, whereas the previous coding did not. Those early releases are presumably extinct in the wild by now, so revert to the previous coding. 
- Allow tables with statement-level triggers to become partitions or inheritance children (Etsuro Fujita) § - We do not allow partitions or inheritance child tables to have row-level triggers with transition tables, because an operation on the whole inheritance tree would need to maintain a separate transition table for each such child table. But that problem does not apply for statement-level triggers, because only the parent's statement-level triggers will be fired. The code that checks whether an existing table can become a partition or inheritance child nonetheless rejected both kinds of trigger. 
- Disallow collecting transition tuples from child foreign tables (Etsuro Fujita) § - We do not support triggers with transition tables on foreign tables. However, the case of a partition or inheritance child that is a foreign table was overlooked. If the parent has such a trigger, incorrect transition tuples were collected from the foreign child. Instead throw an error, reporting that the case is not supported. 
- Shorten the race condition window for creating indexes with conflicting names (Tom Lane) § - When choosing an auto-generated name for an index, avoid conflicting with not-yet-committed - pg_classrows as well as fully-valid ones. This avoids possibly choosing the same name as some concurrent- CREATE INDEXdid, when that command is still in process of filling its index, or is done but is part of a not-yet-committed transaction. There's still a window for trouble, but it's only as long as the time needed to validate a new index's parameters and insert its- pg_classrow.
- Prevent usage of incorrect - VACUUMoptions in some cases where multiple tables are vacuumed in a single command (Nathan Bossart, Michael Paquier) §- The - TRUNCATEand- INDEX_CLEANUPoptions of one table could be applied to others.
- Fix processing of character classes within - SIMILAR TOregular expressions (Laurenz Albe) § §- The code that translates - SIMILAR TOpattern matching expressions to POSIX-style regular expressions did not consider that square brackets can be nested. For example, in a pattern like- [[:alpha:]%_], the code treated the- %and- _characters as metacharacters when they should be literals.
- When deparsing queries, always add parentheses around the expression in - FETCH FIRSTclauses (Heikki Linnakangas) §- expressionROWS WITH TIES- This avoids some cases where the deparsed result wasn't syntactically valid. 
- Limit the checkpointer process's fsync request queue size (Alexander Korotkov, Xuneng Zhou) § § - With very large - shared_bufferssettings, it was possible for the checkpointer to attempt to allocate more than 1GB for fsync requests, leading to failure and an infinite loop. Clamp the queue size to prevent this scenario.
- Avoid infinite wait in logical decoding when reading a partially-written WAL record (Vignesh C) § - If the server crashes after writing the first part of a WAL record that would span multiple pages, subsequent logical decoding of the WAL stream would wait for data to arrive on the next WAL page. That might never happen if the server is now idle. 
- Fix inconsistent quoting of role names in ACL strings (Tom Lane) § - The previous quoting rule was locale-sensitive, which could lead to portability problems when transferring - aclitemvalues across installations. (pg_dump does not do that, but other tools might.) To ensure consistency, always quote non-ASCII characters in- aclitemoutput; but to preserve backward compatibility, never require that they be quoted during- acliteminput.
- Reject equal signs ( - =) in the names of relation options and foreign-data options (Tom Lane) §- There's no evident use-case for option names like this, and allowing them creates ambiguity in the stored representation. 
- Avoid a rare scenario where a btree index scan could mark the wrong index entries as dead (Peter Geoghegan) § 
- Avoid re-distributing cache invalidation messages from other transactions during logical replication (vignesh C) § § - Our previous round of minor releases included a bug fix to ensure that replication receiver processes would respond to cross-process cache invalidation messages, preventing them from using stale catalog data while performing replication updates. However, the fix unintentionally made them also redistribute those messages again, leading to an exponential increase in the number of invalidation messages, which would often end in a memory allocation failure. Fix by not redistributing received messages. 
- Avoid premature removal of old WAL during checkpoints (Vitaly Davydov) § - If a replication slot's restart point is advanced while a checkpoint is in progress, no-longer-needed WAL segments could get removed too soon, leading to recovery failure if the database crashes immediately afterwards. Fix by keeping them for one additional checkpoint cycle. 
- Never move a replication slot's confirmed-flush position backwards (Shveta Malik) § - In some cases a replication client could acknowledge an LSN that's past what it has stored persistently, and then perhaps send an older LSN after a restart. We consider this not-a-bug so long as the client did not have anything it needed to do for the WAL between the two points. However, we should not re-send that WAL for fear of data duplication, so make sure we always believe the latest confirmed LSN for a given slot. 
- Allow waiting for a transaction on a standby server to be interrupted (Kevin K Biju) § - Creation of a replication slot on a standby server may require waiting for some active transaction(s) to finish on the primary and then be replayed on the standby. Since that could be an indefinite wait, it's desirable to allow the operation to be cancelled, but there was no check for query cancel in the loop. 
- Fix some places that might try to fetch toasted fields of system catalogs without any snapshot (Nathan Bossart) § - This could result in an assertion failure or “cannot fetch toast data without an active snapshot” error. 
- Avoid assertion failure during cross-table constraint updates (Tom Lane, Jian He) § § 
- Remove faulty assertion that a command tag must have been determined by the end of - PortalRunMulti()(Álvaro Herrera) §- This failed in edge cases such as an empty prepared statement. 
- Restore the ability to run PL/pgSQL expressions in parallel (Dipesh Dhameliya) § - PL/pgSQL's notion of an “expression” is very broad, encompassing any SQL - SELECTquery that returns a single column and no more than one row. So there are cases, for example evaluation of an aggregate function, where the query involves significant work and it'd be useful to run it with parallel workers. This used to be possible, but a previous bug fix unintentionally disabled it.
- Fix edge-case resource leaks in PL/Python error reporting (Tom Lane) § § - An out-of-memory failure while reporting an error from Python could result in failure to drop reference counts on Python objects, leading to session-lifespan memory leakage. 
- Fix libpq's - PQport()function to never return NULL unless the passed connection is NULL (Daniele Varrazzo) §- This is the documented behavior, but recent libpq versions would return NULL in some cases where the user had not provided a port specification. Revert to our historical behavior of returning an empty string in such cases. (v18 and later will return the compiled-in default port number, typically - "5432", instead.)
- Avoid failure when GSSAPI authentication requires packets larger than 16kB (Jacob Champion, Tom Lane) § - Larger authentication packets are needed for Active Directory users who belong to many AD groups. This limitation manifested in connection failures with unintelligible error messages, typically “GSSAPI context establishment error: The routine must be called again to complete its function: Unknown error”. 
- Fix timing-dependent failures in SSL and GSSAPI data transmission (Tom Lane) § - When using SSL or GSSAPI encryption in non-blocking mode, libpq sometimes failed with “SSL error: bad length” or “GSSAPI caller failed to retransmit all data needing to be retried”. 
- Avoid null-pointer dereference during connection lookup in ecpg applications (Aleksander Alekseev) § - The case could occur only if the application has some connections that are named and some that are not. 
- Ensure that pg_dump dumps comments on domain constraints in a valid order (Jian He) § - In some cases the comment command could appear before creation of the constraint. 
- Ensure stable sort ordering in pg_dump for all types of database objects (Noah Misch, Andreas Karlsson) § § § - pg_dump sorts objects by their logical names before performing dependency-driven reordering. This sort did not account for the full unique key identifying certain object types such as rules and constraints, and thus it could produce dissimilar sort orders for logically-identical databases. That made it difficult to compare databases by diff'ing pg_dump output, so improve the logic to ensure stable sort ordering in all cases. 
- In pg_upgrade, check for inconsistent inherited not-null constraints (Ali Akbar) § § § § § § - PostgreSQL versions before 18 allow an inherited column not-null constraint to be dropped. However, this results in a schema that cannot be restored, leading to failure in pg_upgrade. Detect such cases during pg_upgrade's preflight checks to allow users to fix them before initiating the upgrade. 
- Avoid assertion failure if - track_commit_timestampis enabled during initdb (Hayato Kuroda, Andy Fan) §
- Avoid possible leak of the open connection during - contrib/dblinkconnection establishment (Tom Lane) §- In the rare scenario where we hit out-of-memory while inserting the new connection object into dblink's hashtable, the open connection would be leaked until end of session, leaving an idle session sitting on the remote server. 
- Make - contrib/pg_prewarmcope with very large- shared_bufferssettings (Daria Shanina) §- Autoprewarm failed with a memory allocation error if - shared_bufferswas larger than about 50 million buffers (400GB).
- In - contrib/pg_stat_statements, avoid leaving gaps in the set of parameter numbers used in a normalized query (Sami Imseih) §
- Fix memory leakage in - contrib/postgres_fdw's DirectModify methods (Tom Lane) §- The - PGresultholding the results of the remote modify command would be leaked for the rest of the session if the query fails between invocations of the DirectModify methods, which could happen when there's- RETURNINGdata to process.
- Ensure that directories listed in configure's - --with-includesand- --with-librariesoptions are searched before system-supplied directories (Tom Lane) §- A common reason for using these options is to allow a user-built version of some library to override the system-supplied version. However, that failed to work in some environments because of careless ordering of switches in the commands issued by the makefiles. 
- Fix configure's checks for - __cpuid()and- __cpuidex()(Lukas Fittl, Michael Paquier) §- configure failed to detect these Windows-specific functions, so that they would not be used, leading to slower-than-necessary CRC computations since the availability of hardware instructions could not be verified. The practical impact of this error was limited, because production builds for Windows typically do not use the Autoconf toolchain. 
- Fix build failure with - --with-pamoption on Solaris-based platforms (Tom Lane) §- Solaris is inconsistent with other Unix platforms about the API for PAM authentication. This manifested as an “inconsistent pointer” compiler warning, which we never did anything about. But as of GCC 14 it's an error not warning by default, so fix it. 
- Make our code portable to GNU Hurd (Michael Banck, Christoph Berg, Samuel Thibault) § - Fix assumptions about - IOV_MAXand- O_RDONLYthat don't hold on Hurd.
- Make our usage of - memset_s()conform strictly to the C11 standard (Tom Lane) §- This avoids compile failures on some platforms. 
- Prevent uninitialized-value compiler warnings in JSONB comparison code (Tom Lane) § 
- Avoid deprecation warnings when building with libxml2 2.14 and later (Michael Paquier) § 
- Avoid problems when compiling - pg_locale.hunder C++ (John Naylor) §- PostgreSQL header files generally need to be wrapped in - extern "C" { ... }in order to be included in extensions written in C++. This failed for- pg_locale.hbecause of its use of libicu headers, but we can work around that by suppressing C++-only declarations in those headers. C++ extensions that want to use libicu's C++ APIs can do so by including the libicu headers ahead of- pg_locale.h.