Released: 19 June 2026
This release of EDB Postgres® AI for CloudNativePG™ Cluster includes the following:
Highlights
The cluster reference is now immutable on the Database, Pooler,
Publication, Subscription, and ScheduledBackup resources. Pointing one
of these objects at a different cluster has no well-defined semantics and
previously left the controllers in an inconsistent state; the update is now
rejected at the API server via a CEL validation rule.
(#10743)
Features
| Description | Addresses |
|---|---|
Primary | #10627 |
| #6155 |
TLS client certificates for declarative rolesa | #10896 |
PgBouncer image management via image catalogsthe | #10568 |
Enhancements
| Description | Addresses |
|---|---|
Enabled | #10366 |
Added TLS support for the | #10466 |
Added a label selector to the | #8996 |
| The operator now emits a `Warning` `PrimaryStatusCheckFailed` event on the `Cluster` when the primary pod is `Ready` from the kubelet perspective but the operator's `/pg/status` check fails and failover is deferred, giving users visibility into the deferral via `kubectl describe cluster`. | #10509 |
Added the | #10420 |
| The operator now reloads a CNPG-i plugin automatically when its pods are rolled: it watches the `EndpointSlices` backing plugin `Services` and re-enqueues every cluster using the plugin once the new pods become `Ready`, so an upgraded plugin is picked up without waiting for the next resync. | #10836 |
Instance serial numbers are now assigned by reusing the lowest free slot among existing instance names, instead of always incrementing a global counter.Pod and PVC names stay stable across instance recreation (for example, an instance recreated after a node drain comes back with the same name), and serials freed by deleted instances are reclaimed. A new | #10548 |
Defaulting and validation now run during reconciliation as a fallback when admission webhooks are unavailable, or configured to ignore failures, so the operator no longer reconciles invalid or incomplete specs.Missing defaults are applied directly, and validation failures are surfaced in the resource status instead of failing silently later. | #10874 |
Security Fixes
| Description | Addresses |
|---|---|
| #10774 |
Authenticated operator-to-instance-manager callsthe instance manager's remote webserver previously accepted unauthenticated requests, so any process on the pod network could trigger backups, instance manager upgrades, or WAL archival. The operator now generates an in-memory ECDSA P-256 client certificate at startup and reconciles its SHA-256 fingerprint into the cluster status; the instance manager rejects requests to sensitive endpoints that do not present a matching certificate. | #10579 |
Operator-side SCRAM-SHA-256 password encodingthe operator now SCRAM-SHA-256 encodes cleartext role passwords before issuing | #10724 |
Changes
| Description | Addresses |
|---|---|
| Added support for Kubernetes 1.36. | #10900 |
| Updated the default PostgreSQL version to 18.4. | #10719 |
| Updated the Kubernetes versions used to test the operator on public cloud providers. | #10720, #10563 |
Bug Fixes
| Description | Addresses |
|---|---|
| Fixed declarative `Database`, `Publication`, and `Subscription` objects reporting a stale primary-side status forever after their cluster was demoted to a replica; the controller now re-checks the replica condition and watches the `Cluster` so a demotion is detected promptly. | #10871 |
| Fixed non-sequential pod names (for example `-1`, `-3`) caused by the instance serial counter being advanced before the corresponding `Job` and PVCs were created; the bump is now persisted only after those resources exist. | #10491 |
Fixed deletion of a | #10853 |
| Fixed a conflicting duplicate `Database` or `Subscription` with a `delete` reclaim policy dropping the PostgreSQL object owned by the surviving CR; the drop is now gated on a recorded reconciliation. | #10870 |
Fixed the | #10834 |
| Fixed backups getting stuck in the `started` phase when the instance manager running them was restarted (for example by the in-place upgrade following an operator upgrade) before the backup reached `running`; the reconciliation is now rescheduled so the lost session is detected. | #10859 |
Fixed | #10876, #10933 |
Fixed resource leaks when concurrent | #10747 |
| Fixed role reconciliation clearing the password on a PostgreSQL role when the referenced Secret could not be fetched; the role is now left untouched until the Secret becomes available, and per-action errors are aggregated for better visibility. | #10053 |
Fixed a bootstrap failure where a metrics-exporter setup error (commonly a duplicate-key race with the controller) rolled back | #10749 |
| Fixed a `ScheduledBackup` controller loop that occurred when a `Backup` was created but its status patch never landed; the controller now adopts an existing `Backup` for the next iteration instead of looping on `AlreadyExists`. | #10612 |
| Fixed a nil-pointer panic when reconciling a `Pooler` whose `Cluster` has been deleted. | #10667 |
| Fixed bootstrap log handling so that all named log pipes (`postgres`, `postgres.csv`, and `postgres.json`) get consumers during `WithActiveInstance`, preventing regular files from being created in place of the named pipes. | #10043 |
Fixed generation of invalid IPv6 URLs by wrapping the address in square brackets.Contributed by @Infinoid. | #10682 |
| Fixed an external cluster plugin still being treated as active when its configuration set `enabled: false`. | #10932 |
Fixed a race during bootstrap recovery from an object store where the restore job could read a stale | #10818 |