Using pg_tde with Cosmian KMS and PostgreSQL 17 (Percona)¶
This guide demonstrates how to configure PostgreSQL 17 with Percona’s pg_tde extension to use Cosmian KMS for transparent data encryption (TDE).
- Using pg_tde with Cosmian KMS and PostgreSQL 17 (Percona)- Prerequisites
- Configuration Steps- 1. Configure PostgreSQL
- 2. Enable TDE Extension
- 3. Configure the KMS Key Provider
- 4. Set the Default Encryption Key
- 5. Ensure event triggers are set (usually created on extension install)
- 6. Enable Wal Encrypt and TDE Enforce Encryption (not mandatory but strongly recommended)
- 7. Create encrypted tables
- 8. Verify if a table is encrypted
- 9. Insert and query data transparently
- 10. Check current encryption settings
 
- Troubleshooting \& Notes
 
Prerequisites¶
Before starting, ensure you have:
- PostgreSQL 17 (Percona Server for PostgreSQL 17.5.2 or later)
- pg_tdeextension installed
- Access to a running Cosmian KMS server
- Appropriate SSL certificates for KMIP communication
Configuration Steps¶
1. Configure PostgreSQL¶
Edit your postgresql.conf file to activate tde extension:
Important: Changes to shared_preload_libraries require a PostgreSQL restart to take effect.
sudo systemctl restart [email protected]
2. Enable TDE Extension¶
Create the pg_tde extension in your target database(s):
3. Configure the KMS Key Provider¶
Connect to your PostgreSQL database and add the Cosmian KMS as a key provider using the KMIP protocol:
SELECT pg_tde_add_global_key_provider_kmip(
  'kms_provider',                -- Provider name (can be customized)
  'kms-host.example.com',        -- Your KMS server hostname
  5696,                          -- KMIP port (default is 5696)
  '/path/to/client_cert.pem',    -- Client certificate file path
  '/path/to/client_key.pem',     -- Client private key file path
  '/path/to/ca_cert.pem'         -- Certificate Authority file path
);
Note: Replace the placeholder values with your actual KMS server details and certificate paths.
4. Set the Default Encryption Key¶
Configure the default encryption key using the KMS provider:
SELECT pg_tde_create_key_using_global_key_provider('key_01', 'kms_provider');
SELECT pg_tde_set_server_key_using_global_key_provider('key_01', 'kms_provider');
SELECT pg_tde_set_default_key_using_global_key_provider('key_01', 'kms_provider');
The first parameter (key_01) is the key identifier, and the second parameter (kms_provider) must match the provider name from step 3.
5. Ensure event triggers are set (usually created on extension install)¶
CREATE EVENT TRIGGER pg_tde_ddl_start ON ddl_command_start
  EXECUTE FUNCTION pg_tde_ddl_command_start_capture();
CREATE EVENT TRIGGER pg_tde_ddl_end ON ddl_command_end
  EXECUTE FUNCTION pg_tde_ddl_command_end_capture();
If the trigger already exists, this error can be safely ignored.
6. Enable Wal Encrypt and TDE Enforce Encryption (not mandatory but strongly recommended)¶
Edit your postgresql.conf again and add:
Important: Changes to pg_tde.wal_encrypt or pg_tde.enforce_encryption require a PostgreSQL restart to take effect.
sudo systemctl restart [email protected]
7. Create encrypted tables¶
The
USING tde_heapclause ensures the table is encrypted using pg_tde.
8. Verify if a table is encrypted¶
9. Insert and query data transparently¶
Encryption is transparent; use standard SQL commands:
Data is stored encrypted on disk but returned in plaintext when queried.
10. Check current encryption settings¶
Troubleshooting & Notes¶
- shared_preload_librariesmust include at least- 'pg_tde'.
- 
To change pg_tde,pg_tde.wal_encryptorpg_tde.enforce_encryption, a percona server restart is mandatory.
- 
Ensure SSL certificates are properly secured with appropriate file permissions 
- Store certificate files in a secure location accessible only to the PostgreSQL service
- Regularly rotate encryption keys as per your security policy
- Monitor KMS connectivity and have appropriate failover procedures
Common issues and solutions:
| Function | Description | 
|---|---|
| pg_tde_add_global_key_provider_kmip(...) | Add KMIP key provider | 
| pg_tde_set_default_key_using_global_key_provider(key, provider) | Set default encryption key using a provider | 
| pg_tde_is_encrypted(regclass) | Check if table is encrypted | 
| pg_tde_default_key_info() | Show info about default encryption key | 
For more detailed information, refer to the official pg_tde documentation.