The guide shows you how to use the Zalando Postgres operator to deploy Postgres clusters to Google Kubernetes Engine (GKE).
PostgreSQL is a powerful, open source object-relational database system with several decades of active development that has earned it a strong reputation for reliability, feature robustness, and performance.
This guide is intended for platform administrators, cloud architects, and operations professionals interested in running PostgreSQL as a database application on GKE instead of using Cloud SQL for PostgreSQL.
Set up your environment
To set up your environment, follow these steps
- Set environment variables: - export PROJECT_ID=PROJECT_ID export KUBERNETES_CLUSTER_PREFIX=postgres export REGION=us-central1- Replace - PROJECT_IDwith your Google Cloud project ID.
- Clone the GitHub repository: - git clone https://github.com/GoogleCloudPlatform/kubernetes-engine-samples
- Change to the working directory: - cd kubernetes-engine-samples/databases/postgres-zalando
Create your cluster infrastructure
In this section, you run a Terraform script to create a private, highly-available, regional GKE cluster.
You can install the operator using a Standard or Autopilot cluster.
Standard
The following diagram shows a private regional Standard GKE cluster deployed across three different zones:
Deploy this infrastructure:
export GOOGLE_OAUTH_ACCESS_TOKEN=$(gcloud auth print-access-token)
terraform -chdir=terraform/gke-standard init
terraform -chdir=terraform/gke-standard apply \
  -var project_id=${PROJECT_ID} \
  -var region=${REGION} \
  -var cluster_prefix=${KUBERNETES_CLUSTER_PREFIX}
When prompted, type yes. It might take several minutes for this command to
complete and for the cluster to show a ready status.
Terraform creates the following resources:
- A VPC network and private subnet for the Kubernetes nodes
- A router to access the internet through NAT
- A private GKE cluster in the us-central1region
- A node pools with auto scaling enabled (one to two nodes per zone, one node per zone minimum)
- A ServiceAccountwith logging and monitoring permissions
- Backup for GKE for disaster recovery
- Google Cloud Managed Service for Prometheus for cluster monitoring
The output is similar to the following:
...
Apply complete! Resources: 14 added, 0 changed, 0 destroyed.
...
Autopilot
The following diagram shows a private regional Autopilot GKE cluster:
Deploy the infrastructure:
export GOOGLE_OAUTH_ACCESS_TOKEN=$(gcloud auth print-access-token)
terraform -chdir=terraform/gke-autopilot init
terraform -chdir=terraform/gke-autopilot apply \
  -var project_id=${PROJECT_ID} \
  -var region=${REGION} \
  -var cluster_prefix=${KUBERNETES_CLUSTER_PREFIX}
When prompted, type yes. It might take several minutes for this command to
complete and for the cluster to show a ready status.
Terraform creates the following resources:
- A VPC network and private subnet for the Kubernetes nodes
- A router to access the internet through NAT
- A private GKE cluster in the us-central1region
- A ServiceAccountwith logging and monitoring permission
- Google Cloud Managed Service for Prometheus for cluster monitoring
The output is similar to the following:
...
Apply complete! Resources: 12 added, 0 changed, 0 destroyed.
...
Connect to the cluster
Configure kubectl to communicate with the cluster:
gcloud container clusters get-credentials ${KUBERNETES_CLUSTER_PREFIX}-cluster --location ${REGION}
Deploy the Zalando operator to your cluster
Deploy the Zalando operator to your Kubernetes cluster using a Helm chart.
- Add the Zalando operator Helm Chart repository: - helm repo add postgres-operator-charts https://opensource.zalando.com/postgres-operator/charts/postgres-operator
- Create a namespace for the Zalando operator and the Postgres cluster: - kubectl create ns postgres kubectl create ns zalando
- Deploy the Zalando operator using the Helm command-line tool: - helm install postgres-operator postgres-operator-charts/postgres-operator -n zalando \ --set configKubernetes.enable_pod_antiaffinity=true \ --set configKubernetes.pod_antiaffinity_preferred_during_scheduling=true \ --set configKubernetes.pod_antiaffinity_topology_key="topology.kubernetes.io/zone" \ --set configKubernetes.spilo_fsgroup="103"- You can't configure - podAntiAffinitysettings directly on the custom resource representing the Postgres cluster. Instead, set- podAntiAffinitysettings globally for all Postgres clusters in the operator settings.
- Check the deployment status of the Zalando operator using Helm: - helm ls -n zalando- The output is similar to the following: - NAME NAMESPACE REVISION UPDATED STATUS CHART APP VERSION postgres-operator zalando 1 2023-10-13 16:04:13.945614 +0200 CEST deployed postgres-operator-1.10.1 1.10.1
Deploy Postgres
The basic configuration for the Postgres cluster instance includes the following components:
- Three Postgres replicas: one leader and two standby replicas.
- CPU resource allocation of one CPU request and two CPU limits, with 4 GB memory requests and limits.
- Tolerations, nodeAffinities, andtopologySpreadConstraintsconfigured for each workload, ensuring proper distribution across Kubernetes nodes, utilizing their respective node pools and different availability zones.
This configuration represents the minimal setup required to create a production-ready Postgres cluster.
The following manifest describes a Postgres cluster:
This manifest has the following fields:
- spec.teamId: a prefix for the cluster objects that you choose
- spec.numberOfInstances: the total number of instances for a cluster
- spec.users: the user list with privileges
- spec.databases: the database list in the format- dbname: ownername
- spec.postgresql: postgres parameters
- spec.volume: Persistent Disk parameters
- spec.tolerations: the tolerations Pod template that allows cluster Pods to be scheduled on- pool-postgresnodes
- spec.nodeAffinity: the- nodeAffinityPod template that tells GKE that cluster Pods prefer to be scheduled on- pool-postgresnodes.
- spec.resources: requests and limits for cluster Pods
- spec.sidecars: a list of sidecar containers, which contains- postgres-exporter
For more information, see Cluster manifest reference in the Postgres documentation.
Create a basic Postgres cluster
- Create a new Postgres cluster using the basic configuration: - kubectl apply -n postgres -f manifests/01-basic-cluster/my-cluster.yaml- This command creates a PostgreSQL Custom Resource of the Zalando operator with: - CPU and memory requests and limits
- Taints and affinities to distribute the provisioned Pod replicas across GKE nodes.
- A database
- Two users with database owner permissions
- A user with no permissions
 
- Wait for GKE to start the required workloads: - kubectl wait pods -l cluster-name=my-cluster --for condition=Ready --timeout=300s -n postgres- This command might take a few minutes to complete. 
- Verify that GKE created the Postgres workloads: - kubectl get pod,svc,statefulset,deploy,pdb,secret -n postgres- The output is similar to the following: - NAME READY STATUS RESTARTS AGE pod/my-cluster-0 1/1 Running 0 6m41s pod/my-cluster-1 1/1 Running 0 5m56s pod/my-cluster-2 1/1 Running 0 5m16s pod/postgres-operator-db9667d4d-rgcs8 1/1 Running 0 12m NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE service/my-cluster ClusterIP 10.52.12.109 <none> 5432/TCP 6m43s service/my-cluster-config ClusterIP None <none> <none> 5m55s service/my-cluster-repl ClusterIP 10.52.6.152 <none> 5432/TCP 6m43s service/postgres-operator ClusterIP 10.52.8.176 <none> 8080/TCP 12m NAME READY AGE statefulset.apps/my-cluster 3/3 6m43s NAME READY UP-TO-DATE AVAILABLE AGE deployment.apps/postgres-operator 1/1 1 1 12m NAME MIN AVAILABLE MAX UNAVAILABLE ALLOWED DISRUPTIONS AGE poddisruptionbudget.policy/postgres-my-cluster-pdb 1 N/A 0 6m44s NAME TYPE DATA AGE secret/my-user.my-cluster.credentials.postgresql.acid.zalan.do Opaque 2 6m45s secret/postgres.my-cluster.credentials.postgresql.acid.zalan.do Opaque 2 6m44s secret/sh.helm.release.v1.postgres-operator.v1 helm.sh/release.v1 1 12m secret/standby.my-cluster.credentials.postgresql.acid.zalan.do Opaque 2 6m44s secret/zalando.my-cluster.credentials.postgresql.acid.zalan.do Opaque 2 6m44s
The operator creates the following resources:
- A Postgres StatefulSet, which controls three Pod replicas for Postgres
- A PodDisruptionBudgets, ensuring a minimum of one available replica
- The my-clusterService, which targets the leader replica only
- The my-cluster-replService, which exposes the Postgres port for incoming connections and for replication between Postgres replicas
- The my-cluster-configheadless Service, to get the list of running Postgres Pod replicas
- Secrets with user credentials for accessing the database and replication between Postgres nodes
Authenticate to Postgres
You can create Postgres users and assign them database permissions. For example, the following manifest describes a custom resource that assigns users and roles:
apiVersion: "acid.zalan.do/v1"
kind: postgresql
metadata:
  name: my-cluster
spec:
  ...
  users:
    mydatabaseowner:
    - superuser
    - createdb
    myuser: []
  databases:
    mydatabase: mydatabaseowner
In this manifest:
- The mydatabaseowneruser has theSUPERUSERandCREATEDBroles, which permit full administrator rights (i.e. manage Postgres configuration, create new databases, tables, and users). You shouldn't share this user with clients. For example Cloud SQL doesn't allow customers to have access to users with theSUPERUSERrole.
- The myuseruser has no roles assigned. This follows the best practice of using theSUPERUSERto create users with least privileges. Granular rights are granted tomyuserbymydatabaseowner. To maintain security, you should only sharemyusercredentials with client applications.
Store passwords
You should use the scram-sha-256
recommended method for storing passwords. For example, the following manifest describes a custom
resource that specifies scram-sha-256 encryption using the
postgresql.parameters.password_encryption field:
apiVersion: "acid.zalan.do/v1"
kind: postgresql
metadata:
  name: my-cluster
spec:
  ...
  postgresql:
    parameters:
      password_encryption: scram-sha-256
Rotate user credentials
You can
rotate user credentials
that are stored in Kubernetes Secrets with Zalando. For example, the following
manifest describes a custom resource that defines user credential rotation using
the usersWithSecretRotation field:
apiVersion: "acid.zalan.do/v1"
kind: postgresql
metadata:
  name: my-cluster
spec:
  ...
  usersWithSecretRotation:
  - myuser
  - myanotheruser
  - ...
Authentication example: connect to Postgres
This section shows you how to deploy an example Postgres client and connect to the database using the password from a Kubernetes Secret.
- Run the client Pod to interact with your Postgres cluster: - kubectl apply -n postgres -f manifests/02-auth/client-pod.yaml- The credentials of the - myuserand- mydatabaseownerusers are taken from the related Secrets and mounted as environment variables to the Pod.
- Connect to the Pod when it is ready: - kubectl wait pod postgres-client --for=condition=Ready --timeout=300s -n postgres kubectl exec -it postgres-client -n postgres -- /bin/bash
- Connect to Postgres and attempt to create a new table using - myusercredentials:- PGPASSWORD=$CLIENTPASSWORD psql \ -h my-cluster \ -U $CLIENTUSERNAME \ -d mydatabase \ -c "CREATE TABLE test (id serial PRIMARY KEY, randomdata VARCHAR ( 50 ) NOT NULL);"- The command should fail with an error similar to the following: - ERROR: permission denied for schema public LINE 1: CREATE TABLE test (id serial PRIMARY KEY, randomdata VARCHAR...- The command fails because users without assigned privileges by default can only login to Postgres and list databases. 
- Create a table with - mydatabaseownercredentials and grant all privileges on the table to- myuser:- PGPASSWORD=$OWNERPASSWORD psql \ -h my-cluster \ -U $OWNERUSERNAME \ -d mydatabase \ -c "CREATE TABLE test (id serial PRIMARY KEY, randomdata VARCHAR ( 50 ) NOT NULL);GRANT ALL ON test TO myuser;GRANT ALL ON SEQUENCE test_id_seq TO myuser;"- The output is similar to the following: - CREATE TABLE GRANT GRANT
- Insert random data into the table using - myusercredentials:- for i in {1..10}; do DATA=$(tr -dc A-Za-z0-9 </dev/urandom | head -c 13) PGPASSWORD=$CLIENTPASSWORD psql \ -h my-cluster \ -U $CLIENTUSERNAME \ -d mydatabase \ -c "INSERT INTO test(randomdata) VALUES ('$DATA');" done- The output is similar to the following: - INSERT 0 1 INSERT 0 1 INSERT 0 1 INSERT 0 1 INSERT 0 1 INSERT 0 1 INSERT 0 1 INSERT 0 1 INSERT 0 1 INSERT 0 1
- Get the values that you inserted: - PGPASSWORD=$CLIENTPASSWORD psql \ -h my-cluster \ -U $CLIENTUSERNAME \ -d mydatabase \ -c "SELECT * FROM test;"- The output is similar to the following: - id | randomdata ----+--------------- 1 | jup9HYsAjwtW4 2 | 9rLAyBlcpLgNT 3 | wcXSqxb5Yz75g 4 | KoDRSrx3muD6T 5 | b9atC7RPai7En 6 | 20d7kC8E6Vt1V 7 | GmgNxaWbkevGq 8 | BkTwFWH6hWC7r 9 | nkLXHclkaqkqy 10 | HEebZ9Lp71Nm3 (10 rows)
- Exit the Pod shell: - exit
Understand how Prometheus collects metrics for your Postgres cluster
The following diagram shows how Prometheus metrics collecting works:
In the diagram, a GKE private cluster contains:
- A Postgres Pod that gathers metrics on path /and port9187
- Prometheus-based collectors that process the metrics from the Postgres Pod
- A PodMonitoringresource that sends metrics to Cloud Monitoring
Google Cloud Managed Service for Prometheus supports metrics collection in the Prometheus format. Cloud Monitoring uses an integrated dashboard for Postgres metrics.
Zalando exposes cluster metrics in the Prometheus format using the postgres_exporter component as a sidecar container.
- Create the - PodMonitoringresource to scrape metrics by- labelSelector:- kubectl apply -n postgres -f manifests/03-prometheus-metrics/pod-monitoring.yaml
- In the Google Cloud console, go to the GKE Clusters Dashboard page. - The dashboard shows a non-zero metrics ingestion rate. 
- In the Google Cloud console, go to the Dashboards page. 
- Open the PostgreSQL Prometheus Overview dashboard. The dashboard shows the number of fetched rows. It might take several minutes for the dashboard to auto-provision. 
- Connect to the client Pod: - kubectl exec -it postgres-client -n postgres -- /bin/bash
- Insert random data: - for i in {1..100}; do DATA=$(tr -dc A-Za-z0-9 </dev/urandom | head -c 13) PGPASSWORD=$CLIENTPASSWORD psql \ -h my-cluster \ -U $CLIENTUSERNAME \ -d mydatabase \ -c "INSERT INTO test(randomdata) VALUES ('$DATA');" done
- Refresh the page. The Rows and Blocks graphs update to show the actual database state. 
- Exit the Pod shell: - exit