Is pg_dump a backup tool?

I was invited to the Postgres.fm podcast together with Robert Haas, hosted by Michael Christofides and Nikolay Samokhvalov, to discuss the technical aspects of the question if pg_dump is a backup tool, and in what way it is or is not.

Written by

Gulcin Yildirim Jelinek

Published on

December 5, 2024

#

Podcast Appearance: Postgres.fm 🎙

I was invited to the Postgres.fm podcast together with Robert Haas, hosted by Michael Christofides and Nikolay Samokhvalov, to discuss the technical aspects of the question: Is pg_dump a backup tool, and in what way is it or is not? The podcast experience was much better for me this time, as I was joined by Robert, compared to solo episodes I’ve done before. Robert and I had already been exploring this topic through our blogs, and it became a topic we liked to chat about.

That familiarity carried over into the podcast, making the discussion flow naturally. We had a great time recording this episode and hope listeners enjoy it too!

Is pg_dump a backup tool?
Is pg_dump a backup tool?

I will share some of the highlights of our conversation in this blog post. 🎧 If you prefer listening, you can find the episode on postgres.fm.

The podcast started with high level thoughts on pg_dump and Michael asked what prompted us to blog about it. And for me, it was the CVE-2024-7348 vulnerability.

Why you should upgrade PostgreSQL today
Why you should upgrade PostgreSQL today

I joined Xata in August, and I was checking the discussions going around in our Discord channel. I noticed a user reporting a problem with a pg_dump query:

pg_dump: error: query failed: couldn't get first string argument
pg_dump: detail: Query was: SELECT set_config(name, 'view, foreign-table', false) FROM pg_settings WHERE name = 'restrict_nonsystem_relation_kind'

I did not recognize the restrict_nonsystem_relation_kind setting and this led me to discover about the CVE-2024-7348 vulnerability that affects pg_dump. In the blog I used the sentence below and mentioned pg_dump is a utility used for backing up PostgreSQL databases, the way it is explained in the Postgres documentation.

CVE-2024-7348 is a Time-of-Check Time-of-Use (TOCTOU) race condition vulnerability in pg_dump, a utility used for backing up PostgreSQL databases.

pg_dump, a utility used for backing up PostgreSQL databases
pg_dump, a utility used for backing up PostgreSQL databases

Some people disagreed with me calling pg_dump a backup tool, as Postgres documentation is being changed for PostgreSQL 18. It will be updated to say that pg_dump is a utility for exporting a PostgreSQL database. Referring to the output as dump instead of backup. This led to a discussion and my writing another blog: If pg_dump is not a backup tool, what is?.

If pg_dump is not a backup tool, what is?
If pg_dump is not a backup tool, what is?

I understand where people are coming from when they say pg_dump is not a backup tool. What they likely mean is that pg_dump alone isn't sufficient for a comprehensive backup solution. I agree with this view, but I also believe that pg_dump is a valuable tool and can play a role in a backup strategy if used appropriately.

What I don't find helpful is when Postgres users ask for a question related to pg_dump and the answer is that pg_dump is not a backup tool.

For example, these are the conversations I have seen in the past in Postgres groups:

User: pg_dump can limit a backup by schema.. Answer: But pg_dump is not a backup.

User: Can someone send me the command to take backup of partial database.. Answer: There is no such command. The standard backup tools take backups of the entire database cluster.

I think these answers are not helpful and we should try to help users understand the nuance of the tools they are using without being dismissive.

Robert has two great blogs on this topic:

I recommend reading both blogs but I have a few points I'd like to mention here.

One of the most common arguments against pg_dump is that it takes a long time to create dumps and restore them, especially for large databases. pg_dump holds an MVCC snapshot while it's running, which can lead to database bloat and that is not fun to deal with afterwards. We should be aware of this and plan for VACUUM or VACUUM FULL after running pg_dump.

The process is error-prone, users need to back up each database individually (using pg_dump) and also globals (roles, tablespaces using pg_dumpall -g) and configuration files (pg_hba.conf, postgresql.conf) separately. In multiple database setups, this can get cumbersome and potentially lead to inconsistencies during restoration.

Using a specialized backup tool (for physical backups) can help with management of the backup process, especially in large-scale or mission-critical environments. They also have additional features like backup retention, continuous backups, point-in-time recovery, orchestration, etc.

But logical backups (or dumps) have their own set of benefits, if you treat them as complimentary to your physical backup strategy. Robert has a great example of a scenario where there is a database corruption:

Converting to and from text format can also be used as a way of eliminating corruption in the internal binary format. For example, imagine that you have a database table containing rows where the xmin of a tuple is less than the relation's relfrozenxid but, sadly, the tuple is not frozen. In other words, you've got a corrupted database. Hopefully this won't happen to you, and if it does, hopefully you will be able to fail over to an unaffected replica or restore from an uncorrupted backup. However, it's easy for this kind of corruption to go undetected for a relatively long period of time because the database is still readable despite the corruption, so it's possible that by the time the problem is realized, all of your replicas and backups will be affected. One day, though, the system will remove old clog files that it things are no longer needed, and suddenly that data will become unreadable. Obviously, I've chosen this example to prove a point, because even if you're not familiar with the details of how PostgreSQL represents data on disk, you can perhaps see that the solution here is to dump and restore the affected table at a time when the data is still readable. In fact, this shows how database dumps and physical backups can work hand in glove to solve real-world problems. When one day the table abruptly becomes unreadable, you can restore from a physical backup where the data still was readable, and then use pg_dump to get the data out, and then reload it wherever you like.

One of the least talked features of pg_dump is how it handles dependencies:

Have you ever tried to manually recreate a database schema from one server on another server? If you have, you've probably discovered that it's pretty much no fun at all. For instance, you might try to create a table and find that it because that table has a foreign key to some other table that doesn't exist yet. You need to recreate things in the right order for it to work but you don't know what the right order is. Sometimes you might even have circular dependencies, where no matter which table you create first, it fails for some reason. Then, you might need to solve the dependency problem by first creating the table without subsidiary objects like foreign keys or constraints or indexes, and adding those later, after you've set up some other database objects. And the great thing is that instead of doing all this yourself, you can just run pg_dump and it will figure it out for you. If you're dumping the data and not just the schema, pg_dump will even arrange the dump so that the data restoration is as efficient as possible, as by postponing the creation of indexes until after the data load. All of this can save enormous amounts of DBA time and energy.

I like to go deeper into a particular topic related to Postgres, this time it was pg_dump. I am thankful there are podcasts like Postgres.fm that give me the opportunity to share my thoughts on these topics.

If you read this far, I would love to hear your thoughts on pg_dump and your experiences with it.

🎧 You can listen our opinion on the Postgres.fm episode.

Start free,
pay as you grow

Xata provides the best free plan in the industry. It is production ready by default and doesn't pause or cool-down. Take your time to build your business and upgrade when you're ready to scale.

Free plan includes
  • Single team member
  • 10 database branches
  • High availability
  • 15 GB data storage
Start freeExplore all plans
Free plan includes
  • Single team member
  • 10 database branches
  • High availability
  • 15 GB data storage

Sign up to our newsletter

By subscribing, you agree with Xata’s Terms of Service and Privacy Policy.

Copyright © 2025 Xatabase Inc.
All rights reserved.

Product

Feature requestsPricingStatusAI solutions