New to KubeDB? Please start here.
Reconfigure PgBouncer
This guide will show you how to use KubeDB
Ops-manager operator to reconfigure a PgBouncer.
Before You Begin
At first, you need to have a Kubernetes cluster, and the
kubectl
command-line tool must be configured to communicate with your cluster.Install
KubeDB
Provisioner and Ops-manager operator in your cluster following the steps here.You should be familiar with the following
KubeDB
concepts:
To keep everything isolated, we are going to use a separate namespace called demo
throughout this tutorial.
$ kubectl create ns demo
namespace/demo created
Note: YAML files used in this tutorial are stored in docs/examples/pgbouncer directory of kubedb/docs repository.
Prepare Postgres
For a PgBouncer surely we will need a Postgres server so, prepare a KubeDB Postgres cluster using this tutorial, or you can use any externally managed postgres but in that case you need to create an appbinding yourself. In this tutorial we will use 3 node Postgres cluster named ha-postgres
.
Now, we are going to deploy a PgBouncer
using a supported version by KubeDB
operator. Then we are going to apply PgBouncerOpsRequest
to reconfigure its configuration.
Prepare PgBouncer
Now, we are going to deploy a PgBouncer
with version 1.18.0
.
Deploy PgBouncer
At first, we will create pgbouncer.ini
file containing required configuration settings.
$ cat pgbouncer.ini
[pgbouncer]
auth_type = scram-sha-256
Here, auth_type
is set to scram-sha-256
, whereas the default value is md5
.
Now, we will create a secret with this configuration file.
$ kubectl create secret generic -n demo pb-custom-config --from-file=./pgbouncer.ini
secret/pb-custom-config created
In this section, we are going to create a PgBouncer object specifying spec.configSecret
field to apply this custom configuration. Below is the YAML of the PgBouncer
CR that we are going to create,
apiVersion: kubedb.com/v1
kind: PgBouncer
metadata:
name: pb-custom
namespace: demo
spec:
replicas: 1
version: "1.18.0"
database:
syncUsers: true
databaseName: "postgres"
databaseRef:
name: "ha-postgres"
namespace: demo
connectionPool:
poolMode: session
port: 5432
reservePoolSize: 5
maxClientConnections: 87
defaultPoolSize: 2
minPoolSize: 1
deletionPolicy: WipeOut
Let’s create the PgBouncer
CR we have shown above,
$ kubectl create -f https://github.com/kubedb/docs/raw/v2025.1.9/docs/examples/pgbouncer/reconfigure/pb-custom-config.yaml
pgbouncer.kubedb.com/pb-custom created
Now, wait until pb-custom
has status Ready
. i.e,
$ kubectl get pb -n demo
NAME TYPE VERSION STATUS AGE
pb-custom kubedb.com/v1 1.18.0 Ready 112s
Now, we will check if the pgbouncer has started with the custom configuration we have provided.
Now, you can exec into the pgbouncer pod and find if the custom configuration is there,
$ kubectl exec -it -n demo pb-custom-0 -- /bin/sh
pb-custom-0:/$ cat etc/config/pgbouncer.ini
[databases]
postgres= host=ha-postgres.demo.svc port=5432 dbname=postgres
[pgbouncer]
max_client_conn = 87
default_pool_size = 2
min_pool_size = 1
max_db_connections = 1
logfile = /tmp/pgbouncer.log
listen_port = 5432
ignore_startup_parameters = extra_float_digits
pidfile = /tmp/pgbouncer.pid
listen_addr = *
reserve_pool_size = 5
reserve_pool_timeout = 5
auth_type = scram-sha-256
auth_file = /var/run/pgbouncer/secret/userlist
admin_users = pgbouncer
pool_mode = session
max_user_connections = 2
stats_period = 60
pb-custom-0:/$ exit
exit
As we can see from the configuration of running pgbouncer, the value of auth_type
has been set to scram-sha-256
.
Reconfigure using new secret
Now we will reconfigure this pgbouncer to set auth_type
to md5
.
Now, we will edit the pgbouncer.ini
file containing required configuration settings.
$ cat pgbouncer.ini
auth_type=md5
Then, we will create a new secret with this configuration file.
$ kubectl create secret generic -n demo new-custom-config --from-file=./pgbouncer.ini
secret/new-custom-config created
Create PgBouncerOpsRequest
Now, we will use this secret to replace the previous secret using a PgBouncerOpsRequest
CR. The PgBouncerOpsRequest
yaml is given below,
apiVersion: ops.kubedb.com/v1alpha1
kind: PgBouncerOpsRequest
metadata:
name: pbops-reconfigure
namespace: demo
spec:
type: Reconfigure
databaseRef:
name: pb-custom
configuration:
pgbouncer:
configSecret:
name: new-custom-config
timeout: 5m
apply: IfReady
Here,
spec.databaseRef.name
specifies that we are reconfiguringpb-csutom
pgbouncer.spec.type
specifies that we are performingReconfigure
on our pgbouncer.spec.configuration.pgbouncer.configSecret.name
specifies the name of the new secret.- Have a look here on the respective sections to understand the
timeout
&apply
fields.
Let’s create the PgBouncerOpsRequest
CR we have shown above,
$ kubectl apply -f https://github.com/kubedb/docs/raw/v2025.1.9/docs/examples/pgbouncer/reconfigure/pbops-reconfigure.yaml
pgbounceropsrequest.ops.kubedb.com/pbops-reconfigure created
Verify the new configuration is working
If everything goes well, KubeDB
Ops-manager operator will update the configSecret
of PgBouncer
object.
Let’s wait for PgBouncerOpsRequest
to be Successful
. Run the following command to watch PgBouncerOpsRequest
CR,
$ watch kubectl get pgbounceropsrequest -n demo
Every 2.0s: kubectl get pgbounceropsrequest -n demo
NAME TYPE STATUS AGE
pbops-reconfigure Reconfigure Successful 63s
We can see from the above output that the PgBouncerOpsRequest
has succeeded. If we describe the PgBouncerOpsRequest
we will get an overview of the steps that were followed to reconfigure the pgbouncer.
$ kubectl describe pgbounceropsrequest -n demo pbops-reconfigure
Name: pbops-reconfigure
Namespace: demo
Labels: <none>
Annotations: <none>
API Version: ops.kubedb.com/v1alpha1
Kind: PgBouncerOpsRequest
Metadata:
Creation Timestamp: 2024-11-28T10:06:23Z
Generation: 1
Resource Version: 86377
UID: f96d088e-a32b-40eb-bd9b-ca15a8370548
Spec:
Apply: IfReady
Configuration:
Pgbouncer:
Config Secret:
Name: new-custom-config
Database Ref:
Name: pb-custom
Timeout: 5m
Type: Reconfigure
Status:
Conditions:
Last Transition Time: 2024-11-28T10:06:23Z
Message: Controller has started to Progress with Reconfigure of PgBouncerOpsRequest: demo/pbops-reconfigure
Observed Generation: 1
Reason: Running
Status: True
Type: Running
Last Transition Time: 2024-11-28T10:06:26Z
Message: paused pgbouncer database
Observed Generation: 1
Reason: Paused
Status: True
Type: Paused
Last Transition Time: 2024-11-28T10:06:36Z
Message: Successfully updated PgBouncer
Observed Generation: 1
Reason: UpdateDatabase
Status: True
Type: UpdateDatabase
Last Transition Time: 2024-11-28T10:06:36Z
Message: Successfully updated PgBouncer backend secret
Observed Generation: 1
Reason: UpdateBackendSecret
Status: True
Type: UpdateBackendSecret
Last Transition Time: 2024-11-28T10:06:41Z
Message: get pod; ConditionStatus:True; PodName:pb-custom-0
Observed Generation: 1
Status: True
Type: GetPod--pb-custom-0
Last Transition Time: 2024-11-28T10:07:16Z
Message: volume mount check; ConditionStatus:True; PodName:pb-custom-0
Observed Generation: 1
Status: True
Type: VolumeMountCheck--pb-custom-0
Last Transition Time: 2024-11-28T10:07:21Z
Message: reload config; ConditionStatus:True; PodName:pb-custom-0
Observed Generation: 1
Status: True
Type: ReloadConfig--pb-custom-0
Last Transition Time: 2024-11-28T10:07:21Z
Message: Reloading performed successfully in PgBouncer: demo/pb-custom for PgBouncerOpsRequest: pbops-reconfigure
Observed Generation: 1
Reason: ReloadPodsSucceeded
Status: True
Type: ReloadPods
Last Transition Time: 2024-11-28T10:07:21Z
Message: Successfully Reconfigured
Observed Generation: 1
Reason: Reconfigure
Status: True
Type: Reconfigure
Last Transition Time: 2024-11-28T10:07:21Z
Message: Controller has successfully completed with Reconfigure of PgBouncerOpsRequest: demo/pbops-reconfigure
Observed Generation: 1
Reason: Successful
Status: True
Type: Successful
Observed Generation: 1
Phase: Successful
Events:
Type Reason Age From Message
---- ------ ---- ---- -------
Normal Starting 70s KubeDB Ops-manager Operator Start processing for PgBouncerOpsRequest: demo/pbops-reconfigure
Normal Starting 70s KubeDB Ops-manager Operator Pausing PgBouncer databse: demo/pb-custom
Normal Successful 70s KubeDB Ops-manager Operator Successfully paused PgBouncer database: demo/pb-custom for PgBouncerOpsRequest: pbops-reconfigure
Warning get pod; ConditionStatus:True; PodName:pb-custom-0 52s KubeDB Ops-manager Operator get pod; ConditionStatus:True; PodName:pb-custom-0
Warning volume mount check; ConditionStatus:False; PodName:pb-custom-0 52s KubeDB Ops-manager Operator volume mount check; ConditionStatus:False; PodName:pb-custom-0
Warning get pod; ConditionStatus:True; PodName:pb-custom-0 47s KubeDB Ops-manager Operator get pod; ConditionStatus:True; PodName:pb-custom-0
Warning get pod; ConditionStatus:True; PodName:pb-custom-0 42s KubeDB Ops-manager Operator get pod; ConditionStatus:True; PodName:pb-custom-0
Warning get pod; ConditionStatus:True; PodName:pb-custom-0 37s KubeDB Ops-manager Operator get pod; ConditionStatus:True; PodName:pb-custom-0
Warning get pod; ConditionStatus:True; PodName:pb-custom-0 32s KubeDB Ops-manager Operator get pod; ConditionStatus:True; PodName:pb-custom-0
Warning get pod; ConditionStatus:True; PodName:pb-custom-0 27s KubeDB Ops-manager Operator get pod; ConditionStatus:True; PodName:pb-custom-0
Warning get pod; ConditionStatus:True; PodName:pb-custom-0 22s KubeDB Ops-manager Operator get pod; ConditionStatus:True; PodName:pb-custom-0
Warning get pod; ConditionStatus:True; PodName:pb-custom-0 17s KubeDB Ops-manager Operator get pod; ConditionStatus:True; PodName:pb-custom-0
Warning volume mount check; ConditionStatus:True; PodName:pb-custom-0 17s KubeDB Ops-manager Operator volume mount check; ConditionStatus:True; PodName:pb-custom-0
Warning reload config; ConditionStatus:True; PodName:pb-custom-0 12s KubeDB Ops-manager Operator reload config; ConditionStatus:True; PodName:pb-custom-0
Warning reload config; ConditionStatus:True; PodName:pb-custom-0 12s KubeDB Ops-manager Operator reload config; ConditionStatus:True; PodName:pb-custom-0
Normal Successful 12s KubeDB Ops-manager Operator Reloading performed successfully in PgBouncer: demo/pb-custom for PgBouncerOpsRequest: pbops-reconfigure
Normal Starting 12s KubeDB Ops-manager Operator Resuming PgBouncer database: demo/pb-custom
Normal Successful 12s KubeDB Ops-manager Operator Successfully resumed PgBouncer database: demo/pb-custom
Normal Successful 12s KubeDB Ops-manager Operator Controller has Successfully Reconfigured PgBouncer databases: demo/pb-custom
Now let’s exec into the pgbouncer pod and check the new configuration we have provided.
$ kubectl exec -it -n demo pb-custom-0 -- /bin/sh
pb-custom-0:/$ cat etc/config/pgbouncer.ini
[databases]
postgres= host=ha-postgres.demo.svc port=5432 dbname=postgres
[pgbouncer]
max_db_connections = 1
logfile = /tmp/pgbouncer.log
listen_addr = *
admin_users = pgbouncer
pool_mode = session
max_client_conn = 87
listen_port = 5432
ignore_startup_parameters = extra_float_digits
auth_file = /var/run/pgbouncer/secret/userlist
default_pool_size = 2
min_pool_size = 1
max_user_connections = 2
stats_period = 60
auth_type = md5
pidfile = /tmp/pgbouncer.pid
reserve_pool_size = 5
reserve_pool_timeout = 5
pb-custom-0:/$ exit
exit
As we can see from the configuration of running pgbouncer, the value of auth_type
has been changed from scram-sha-256
to md5
. So the reconfiguration of the pgbouncer is successful.
Reconfigure using apply config
Now we will reconfigure this pgbouncer again to set auth_type
to scram-sha-256
. This time we won’t use a new secret. We will use the applyConfig
field of the PgBouncerOpsRequest
. This will merge the new config in the existing secret.
Create PgBouncerOpsRequest
Now, we will use the new configuration in the data
field in the PgBouncerOpsRequest
CR. The PgBouncerOpsRequest
yaml is given below,
apiVersion: ops.kubedb.com/v1alpha1
kind: PgBouncerOpsRequest
metadata:
name: pbops-reconfigure-apply
namespace: demo
spec:
type: Reconfigure
databaseRef:
name: pb-custom
configuration:
pgbouncer:
applyConfig:
pgbouncer.ini: |-
[pgbouncer]
auth_type=scram-sha-256
timeout: 5m
apply: IfReady
Here,
spec.databaseRef.name
specifies that we are reconfiguringpb-custom
pgbouncer.spec.type
specifies that we are performingReconfigure
on our pgbouncer.spec.configuration.pgbouncer.applyConfig
specifies the new configuration that will be merged in the existing secret.
Let’s create the PgBouncerOpsRequest
CR we have shown above,
$ kubectl apply -f https://github.com/kubedb/docs/raw/v2025.1.9/docs/examples/pgbouncer/reconfigure/pbops-reconfigure-apply.yaml
pgbounceropsrequest.ops.kubedb.com/pbops-reconfigure-apply created
Verify the new configuration is working
If everything goes well, KubeDB
Ops-manager operator will merge this new config with the existing configuration.
Let’s wait for PgBouncerOpsRequest
to be Successful
. Run the following command to watch PgBouncerOpsRequest
CR,
$ watch kubectl get pgbounceropsrequest -n demo
Every 2.0s: kubectl get pgbounceropsrequest -n demo
NAME TYPE STATUS AGE
pbops-reconfigure Reconfigure Successful 9m15s
pbops-reconfigure-apply Reconfigure Successful 53s
We can see from the above output that the PgBouncerOpsRequest
has succeeded. If we describe the PgBouncerOpsRequest
we will get an overview of the steps that were followed to reconfigure the pgbouncer.
$ kubectl describe pgbounceropsrequest -n demo pbops-reconfigure-apply
Name: pbops-reconfigure-apply
Namespace: demo
Labels: <none>
Annotations: <none>
API Version: ops.kubedb.com/v1alpha1
Kind: PgBouncerOpsRequest
Metadata:
Creation Timestamp: 2024-11-28T10:11:52Z
Generation: 1
Resource Version: 86774
UID: a4b8e8b5-0b82-4391-a8fe-66911aa5bee6
Spec:
Apply: IfReady
Configuration:
Pgbouncer:
Apply Config:
pgbouncer.ini: [pgbouncer]
auth_type=scram-sha-256
Database Ref:
Name: pb-custom
Timeout: 5m
Type: Reconfigure
Status:
Conditions:
Last Transition Time: 2024-11-28T10:11:52Z
Message: Controller has started to Progress with Reconfigure of PgBouncerOpsRequest: demo/pbops-reconfigure-apply
Observed Generation: 1
Reason: Running
Status: True
Type: Running
Last Transition Time: 2024-11-28T10:11:55Z
Message: paused pgbouncer database
Observed Generation: 1
Reason: Paused
Status: True
Type: Paused
Last Transition Time: 2024-11-28T10:11:55Z
Message: Successfully updated PgBouncer
Observed Generation: 1
Reason: UpdateDatabase
Status: True
Type: UpdateDatabase
Last Transition Time: 2024-11-28T10:11:55Z
Message: Successfully updated PgBouncer backend secret
Observed Generation: 1
Reason: UpdateBackendSecret
Status: True
Type: UpdateBackendSecret
Last Transition Time: 2024-11-28T10:12:00Z
Message: get pod; ConditionStatus:True; PodName:pb-custom-0
Observed Generation: 1
Status: True
Type: GetPod--pb-custom-0
Last Transition Time: 2024-11-28T10:12:00Z
Message: volume mount check; ConditionStatus:True; PodName:pb-custom-0
Observed Generation: 1
Status: True
Type: VolumeMountCheck--pb-custom-0
Last Transition Time: 2024-11-28T10:12:05Z
Message: reload config; ConditionStatus:True; PodName:pb-custom-0
Observed Generation: 1
Status: True
Type: ReloadConfig--pb-custom-0
Last Transition Time: 2024-11-28T10:12:05Z
Message: Reloading performed successfully in PgBouncer: demo/pb-custom for PgBouncerOpsRequest: pbops-reconfigure-apply
Observed Generation: 1
Reason: ReloadPodsSucceeded
Status: True
Type: ReloadPods
Last Transition Time: 2024-11-28T10:12:05Z
Message: Successfully Reconfigured
Observed Generation: 1
Reason: Reconfigure
Status: True
Type: Reconfigure
Last Transition Time: 2024-11-28T10:12:05Z
Message: Controller has successfully completed with Reconfigure of PgBouncerOpsRequest: demo/pbops-reconfigure-apply
Observed Generation: 1
Reason: Successful
Status: True
Type: Successful
Observed Generation: 1
Phase: Successful
Events:
Type Reason Age From Message
---- ------ ---- ---- -------
Normal Starting 54s KubeDB Ops-manager Operator Start processing for PgBouncerOpsRequest: demo/pbops-reconfigure-apply
Normal Starting 54s KubeDB Ops-manager Operator Pausing PgBouncer databse: demo/pb-custom
Normal Successful 54s KubeDB Ops-manager Operator Successfully paused PgBouncer database: demo/pb-custom for PgBouncerOpsRequest: pbops-reconfigure-apply
Warning get pod; ConditionStatus:True; PodName:pb-custom-0 46s KubeDB Ops-manager Operator get pod; ConditionStatus:True; PodName:pb-custom-0
Warning volume mount check; ConditionStatus:True; PodName:pb-custom-0 46s KubeDB Ops-manager Operator volume mount check; ConditionStatus:True; PodName:pb-custom-0
Warning reload config; ConditionStatus:True; PodName:pb-custom-0 41s KubeDB Ops-manager Operator reload config; ConditionStatus:True; PodName:pb-custom-0
Warning reload config; ConditionStatus:True; PodName:pb-custom-0 41s KubeDB Ops-manager Operator reload config; ConditionStatus:True; PodName:pb-custom-0
Normal Successful 41s KubeDB Ops-manager Operator Reloading performed successfully in PgBouncer: demo/pb-custom for PgBouncerOpsRequest: pbops-reconfigure-apply
Normal Starting 41s KubeDB Ops-manager Operator Resuming PgBouncer database: demo/pb-custom
Normal Successful 41s KubeDB Ops-manager Operator Successfully resumed PgBouncer database: demo/pb-custom
Normal Successful 41s KubeDB Ops-manager Operator Controller has Successfully Reconfigured PgBouncer databases: demo/pb-custom
Normal Starting 41s KubeDB Ops-manager Operator Resuming PgBouncer database: demo/pb-custom
Normal Successful 41s KubeDB Ops-manager Operator Successfully resumed PgBouncer database: demo/pb-custom
Normal Successful 41s KubeDB Ops-manager Operator Controller has Successfully Reconfigured PgBouncer databases: demo/pb-custom
Now let’s exec into the pgbouncer pod and check the new configuration we have provided.
$ kubectl exec -it -n demo pb-custom-0 -- /bin/sh
pb-custom-0:/$ cat etc/config/pgbouncer.ini
[databases]
postgres= host=ha-postgres.demo.svc port=5432 dbname=postgres
[pgbouncer]
stats_period = 60
pidfile = /tmp/pgbouncer.pid
pool_mode = session
reserve_pool_timeout = 5
max_client_conn = 87
min_pool_size = 1
default_pool_size = 2
listen_addr = *
max_db_connections = 1
max_user_connections = 2
auth_type=scram-sha-256
ignore_startup_parameters = extra_float_digits
admin_users = pgbouncer
auth_file = /var/run/pgbouncer/secret/userlist
logfile = /tmp/pgbouncer.log
listen_port = 5432
reserve_pool_size = 5
pb-custom-0:/$ exit
exit
As we can see from the configuration of running pgbouncer, the value of auth_type
has been changed from md5
to scram-sha-256
. So the reconfiguration of the pgbouncer using the applyConfig
field is successful.
Remove config
This will remove all the custom config previously provided. After this Ops-manager operator will merge the new given config with the default config and apply this.
spec.databaseRef.name
specifies that we are reconfiguringpb-custom
pgbouncer.spec.type
specifies that we are performingReconfigure
on our pgbouncer.spec.configuration.pgbouncer.removeCustomConfig
specifies for boolean values to remove previous custom configuration.
Cleaning Up
To clean up the Kubernetes resources created by this tutorial, run:
kubectl delete -n demo pb/pb-custom
kubectl delete pgbounceropsrequest -n demo pbops-reconfigure pbops-reconfigure-apply
kubectl delete pg -n demo ha-postgres
kubectl delete ns demo