CYBERTEC PostgreSQL Logo

Monitoring PostgreSQL replication

09.2023 / Category: / Tags:

PostgreSQL replication is not just a way to scale your database to run ever larger workloads: it's also a way to make your database infrastructure redundant, more reliable and resilient. There is, however, a potential for replication lag, which needs to be monitored. How can you monitor replication lag in PostgreSQL? What is replication lag? And how can you monitor PostgreSQL replication in general?
 
Let's dive in and find out.

Checking replication lag while monitoring PostgreSQL replication

Streaming replication

For the sake of this example, I have set up a database server (PostgreSQL 16) and a single replica.
 
monitoring postgresql replication primary and secondary database 

When monitoring replication delay and replication lag, look at the system view called pg_stat_replication. It contains all the information you’ll need to identify and diagnose replication problems. Here’s what the view looks like:

Technically, this is information about the “WAL sender”. What does that mean? When two PostgreSQL servers communicate, the sending machine undergoes a “WAL sender” process, while the receiving machine undergoes a “WAL receiver” process. The pg_stat_replication view tells us all we need to know about the “WAL senders” (= 1 per destination). The view contains data when ask for information from a primary - but it also contains information in case of cascading replication when you ask a replica to pass data to other replicas.

Cascading replication

In the case of cascading replication, a machine in the middle will have one or more WAL sender(s) as well as a WAL receiver:
 
monitoring postgresql replication cascading primary secondary
 

In this case, the setup is easy: A single server will stream to a secondary one on the same machine. Let's take a look and see what happens here:

The fact that there is one entry in the system view tells us that there is one ACTIVE stream. Note that we are talking about active streams - in case a stream that is supposed to be there is not active, there is no entry. In other words: Check for the existence of the row to validate that streaming is indeed active.

What is also important is the state of the stream. If all is normal you can identify the stream as “streaming”. However, you might also see “catchup” or some other state in case your server is still syncing.

The basic check is therefore: Ensure that the right processes are correctly aimed at the right replication target.

What is replication lag?

Once this is done, we need to take a deeper look at the *_lsn columns. There are four, and it’s important to understand them:

  • sent_lsn
  • write_lsn
  • flush_lsn
  • replay_lsn

*_lsn columns and the flow of data

To understand what's going on here, we need to inspect the flow of data in the first place:
 
monitoring postgresql replication graphic showing lsn columns and flow of data
 
When data flows from one server to another, it reaches the second server through the WAL-receiver. This is the “sent_lsn” (or “sent_location” as it used to be called in older versions of PostgreSQL).

Once data has reached the second server it will be processed and sent to the operating system. PostgreSQL will ask the kernel to write the data. However, this does not mean that the data has actually hit the disk - it only means that we have sent data to the file system which might end up caching things. In this case, we are talking about the “write_lsn” - it is the position in the transaction log stream which has made it to the disk already.

The next relevant number is “flush_lsn”.

What is the difference between write_lsn and flush_lsn?

write_lsn tells us how much data has been sent to the kernel - flush_lsn tells us how much data has been flushed already. Writing to a file does not mean that data has indeed reached the storage device. Flushing ensures that data will survive a power outage.

replay_lsn

Finally, there is replay_lsn. This value causes a lot of confusion. It can happen that a transaction has made it to disk on the replica but it might not be visible to end users yet. This is even true in case of synchronous replication. Yes, you read that correctly. Synchronous replication usually does NOT guarantee that you can see data on the replica which was committed on the primary. How can that happen? The answer is: During a replication conflict the replica might write and flush a transaction to disk, but not apply the change yet before the conflict ends (check this link out for details). In a nutshell, replay_lsn will tell you how much data is already visible.

Using pg_stat_wal_receiver

So far we have inspected what we can do on the WAL sender side. There is also a view named pg_stat_wal_receiver. Not surprisingly, it covers the receiving end of things:

However, this one is far harder to read. I prefer using pg_stat_replication to make monitoring easier.

Monitoring PostgreSQL replication slots

A replication slot ensures that the WAL will not go away if the replica lags behind. Without a replication slot, a primary will recycle its WAL as soon as it doesn’t need it on its own anymore.

The system view to check for stale replication slots is as follows:

Make sure that there are no pending replication slots which are stale and not needed anymore. Any of those should be dropped.

Finally …

 


In order to receive regular updates on important changes in PostgreSQL, subscribe to our newsletter, or follow us on Facebook or LinkedIn.

One response to “Monitoring PostgreSQL replication”

  1. Hi Hans, nice blog! Thank you for sharing it.
    The first item in the "Table Of Contents" appeared with text from the blog paragraph in my FF browser 🙂

Leave a Reply

Your email address will not be published. Required fields are marked *

CYBERTEC Logo white
Get the newest PostgreSQL Info & Tools


    This site is protected by reCAPTCHA and the Google Privacy Policy & Terms of Service apply.

    ©
    2025
    CYBERTEC PostgreSQL International GmbH
    phone-handsetmagnifiercrosscross-circle
    linkedin facebook pinterest youtube rss twitter instagram facebook-blank rss-blank linkedin-blank pinterest youtube twitter instagram