RDS can create serious bottlenecks in engineering productivity — is this you? See how Neon can help
Docs/Migrate to Neon/Migrate from Postgres with pg_dump and pg_restore

Migrate data from Postgres with pg_dump and pg_restore

This topic describes migrating data from one Postgres database to another using the pg_dump and pg_restore.

important

Avoid using pg_dump over a pooled connection string (see PgBouncer issues 452 & 976 for details). Use an unpooled connection string instead.

Repeat the pg_dump and pg_restore process for each database you want to migrate.

If you are performing this procedure to migrate data from one Neon project to another to upgrade to a new Postgres version, read Upgrading your Postgres version first.

Before you begin

  • We recommended that you use the pg_dump and pg_restore programs from the latest version of Postgres, to take advantage of enhancements that might have been made in these programs. To check the version of pg_dump or pg_restore, use the -V option. For example: pg_dump -V.

  • Neon supports PostgreSQL 14, 15, 16, and 17. We recommend that clients are the same version as source Postgres instance.

  • Retrieve the connection parameters or connection string for your source Postgres database. This could be a Neon Postgres database or another Postgres database. The instructions below use a connection string, but you can use the connection format you prefer. If you are logged in to a local Postgres instance, you may only need to provide the database name. Refer to the pg_dump documentation for information about connection parameters.

  • Optionally, create a role in Neon to perform the restore operation. The role that performs the restore operation becomes the owner of restored database objects. For example, if you want role sally to own database objects, create role sally in Neon and perform the restore operation as sally.

  • If you have assigned database object ownership to different roles in your source database, read Database object ownership considerations. You may want to add the -O, --no-owner option to your pg_restore command to avoid errors.

  • Create the target database in Neon. For example, if you are migrating a database named pagila, create a database named pagila in Neon. For instructions, see Create a database.

  • Retrieve the connection string for the target Neon database. You can find it in the Connection Details widget on the Neon Dashboard. It will look something like this:

    postgresql://[user]:[password]@[neon_hostname]/[dbname]
  • Consider running a test migration first to ensure your actual migration goes smoothly. See Run a test migration.

  • If your database is small, you can pipe pg_dump output directly to pg_restore to save time. See Pipe pg_dump to pg_restore.

Export data with pg_dump

Export your data from the source database with pg_dump:

pg_dump -Fc -v -d <source_database_connection_string> -f <dump_file_name>

The pg_dump command above includes these arguments:

  • -Fc: Sends the output to a custom-format archive suitable for input into pg_restore.
  • -v: Runs pg_dump in verbose mode, allowing you to monitor what happens during the dump operation.
  • -d: Specifies the source database name or connection string.
  • -f: The dump file name. It can be any name you choose (mydumpfile.bak, for example).

For more command options, see Advanced pg_dump and pg_restore options.

Restore data to Neon with pg_restore

Restore your data to the target database in Neon with pg_restore.

note

If you assigned database object ownership to different roles in your source database, consider adding the -O, --no-owner option to your pg_restore command to avoid errors. See Database object ownership considerations.

pg_restore -v -d <neon_database_connection_string> <dump_file_name>

The example above includes these arguments:

  • -v: Runs pg_restore in verbose mode, allowing you to monitor what happens during the restore operation.
  • -d: Specifies the Neon database to connect to. The value is a Neon database connection string. See Before you begin.
  • <dump_file_name> is the name of the dump file you created with pg_dump.

For more command options, see Advanced pg_dump and pg_restore options.

pg_dump and pg_restore example

The following example shows how data from a pagila source database is dumped and restored to a pagila database in Neon using the commands described in the previous sections. (A database named pagila was created in Neon prior to running the restore operation.)

~$ cd mydump
~/mydump$ pg_dump -Fc -v -d postgresql://[user]:[password]@[neon_hostname]/pagila -f mydumpfile.bak

~/mydump$ ls
mydumpfile.bak

~/mydump$ pg_restore -v -d postgresql://[user]:[password]@[neon_hostname]/pagila mydumpfile.bak

Pipe pg_dump to pg_restore

For small databases, the standard output of pg_dump can be piped directly into a pg_restore command to minimize migration downtime:

pg_dump [args] | pg_restore [args]

For example:

pg_dump -Fc -v -d <source_database_connection_string> | pg_restore -v -d <neon-database-connection-string>

Piping is not recommended for large databases, as it is susceptible to failures during lengthy migration operations.

When piping pg_dump output directly to pg_restore, the custom output format (-Fc) is most efficient. The directory format (-Fd) format cannot be piped to pg_restore.

Post-migration steps

After migrating your data, update your applications to connect to your new database in Neon. You will need the database connection string that you used in your pg_restore command. If you run into any problems, see Connect from any application. After connecting your applications, test them thoroughly to ensure they function correctly with your new database.

Database object ownership considerations

Roles created in the Neon Console, including the default role created with your Neon project, are automatically granted membership in the neon_superuser role. This role can create roles and databases, select from all tables and views, and insert, update, or delete data in all tables. However, the neon_superuser is not a PostgreSQL superuser. It cannot run ALTER OWNER statements to grant ownership of database objects. As a result, if you granted ownership of database objects in your source database to different roles, your dump file will contain ALTER OWNER statements, and those statements will cause non-fatal errors when you restore data to your Neon database.

note

Regardless of ALTER OWNER statement errors, a restore operation still succeeds because assigning ownership is not necessary for the data itself to be restored. The restore operation will still create tables, import data, and create other objects.

To avoid the non-fatal errors, you can ignore database object ownership statements when restoring data by specifying the -O, --no-owner option in your pg_restore command:

pg_restore -v -O -d postgresql://[user]:[password]@[neon_hostname]/pagila mydumpfile.bak

The Neon role performing the restore operation becomes the owner of all database objects.

Advanced pg_dump and pg_restore options

The pg_dump and pg_restore commands provide numerous advanced options, some of which are described below. Full descriptions and more options are found in the PostgreSQL pg_dump and pg_restore documentation.

pg_dump options

  • -Z: Defines the compression level to use when using a compressible format. 0 means no compression, while 9 means maximum compression. In general, we recommend a setting of 1. A higher compression level slows the dump and restore process but also uses less disk space.
  • --lock-wait-timeout=20s: Error out early in the dump process instead of waiting for an unknown amount of time if there is lock contention. Do not wait forever to acquire shared table locks at the beginning of the dump. Instead fail if unable to lock a table within the specified timeout.`
  • -j <njobs>: Consider this option for large databases to dump tables in parallel. Set <njobs> to the number of available CPUs. Refer to the pg_dump documentation for more information. In Neon, this option only make sense for Neon paid plan users who can configure computes with >1 vCPU.
  • --no-blobs: Excludes large objects from your dump. See Data migration notes.

pg_restore options

  • -c --if-exists: Drop database objects before creating them if they already exist. If you had a failed migration, you can use these options to drop objects created by the previous migration to avoid errors when retrying the migration.
  • -j <njobs>: Consider this option for large databases to run the restore process in parallel. Set <njobs> to the number of available vCPUs. Refer to the pg_dump documentation for more information. In Neon, this option only makes sense for Neon paid plan users who can configure computes with >1 vCPU. It cannot be used together with --single-transaction.
  • --single-transaction: Forces the operation to run as an atomic transaction, which ensures that no data is left behind when a restore operation fails. Retrying an import operation after a failed attempt that leaves data behind may result in "duplicate key value" errors.
  • --no-tablespaces: Do not output commands to select tablespaces. See Data migration notes.
  • -t <table_name>: Allows you to restore individual tables from a custom-format database dump. Individual tables can also be imported from a CSV file. See Import from CSV.

Run a test migration

It is recommended that you run a test migration before migrating your production database. Make sure you can successfully migrate data to the new database and connect to it. Before starting the actual migration, create a database dump and address any issues that show up. In Neon, you can quickly create a test database, obtain the connection string, and delete the database when you are finished with it. See Create a database.

Other migration options

This section discusses migration options other than pg_dump and pg_restore.

Postgres GUI clients

Some Postgres clients offer backup and restore capabilities. These include pgAdmin and phppgadmin, among others. We have not tested migrations using these clients, but if you are uncomfortable using command-line utilities, they may provide an alternative.

Table-level data migration

Table-level data migration (using CSV files, for example) does not preserve database schemas, constraints, indexes, types, or other database features. You will have to create these separately. Table-level migration is simple but could result in significant downtime depending on the size of your data and the number of tables. For instructions, see Import data from CSV.

Data migration notes

  • You can load data using the psql utility, but it only supports plain-text SQL dumps, which you should only consider for small datasets or specific use cases. To create a plain-text SQL dump with pg_dump utility, leave out the -F format option. Plain-text SQL is the default pg_dump output format.
  • pg_dumpall is not supported.
  • pg_dump with the -C, --create option is not supported.
  • Some PostgreSQL features, such as tablespaces and large objects, which require access to the local file system are not supported by Neon. To exclude selecting tablespaces, specify the --no-tablespaces option with pg_restore. To exclude large objects, specify the --no-blobs option with pg_dump.

Reference

For information about the Postgres client utilities referred to in this topic, refer to the following topics in the Postgres documentation:

Need help?

Join our Discord Server to ask questions or see what others are doing with Neon. Users on paid plans can open a support ticket from the console. For more details, see Getting Support.

Last updated on

Was this page helpful?