We are excited to ship the first version of pgroll, a command line tool that offers safe and reversible schema migrations for PostgreSQL
Written by
Carlos Pérez-Aradros Herce
Published on
October 3, 2023
Database schema migrations can be a double-edged sword. They are essential for keeping our systems up to date and in sync with evolving application requirements, but often come bundled with a set of challenges that can leave even the most seasoned developers and database administrators scratching their heads (or banging them on the keyboard).
Due to these issues, many developers choose to avoid complex migrations and only make additive changes. This leads to the accumulation of technical debt in the database schema, such as orphaned columns or missing constraints.
At Xata, we use Postgres for our internal systems and to host our users' data. As a result, we need to perform migrations for both our internal development and from the Xata product itself.
We believe evolving your Postgres schema can be a considerably better experience:
This is why we created pgroll
: https://github.com/xataio/pgroll
pgroll
is an open source command-line tool for performing schema migrations against Postgres databases. Designed on these principles, pgroll
allows users to define schema migrations using a high-level JSON format and then takes care of executing them. These are some of the key features:
CREATE
, ALTER
, etc.)pgroll
uses the expand and contract pattern to evolve the database schema, automating its whole lifecycle behind an easy-to-use command line interface.
Previous and new versions of the schema are made available as “virtual” schemas on top of the Postgres physical one. By leveraging table views pointing to the right columns, pgroll
is able to expose new parts of the schema and hide the old parts before safely removing them after the migration is completed.
As discussed in a previous blog post, ensuring Postgres locks on tables (i.e. ACCESS EXCLUSIVE
) while executing DDL statements don’t end up in data access blocking is possible. pgroll
implements all migration operations using the right techniques to avoid this situation, so you don’t need to think about it.
Backfilling data is also a big part of performing backwards-compatible schema changes. pgroll
takes care of performing automatic backfills when they are needed, abstracting the problem away while keeping things transparent.
Let’s show how pgroll
works using an example. For instance, a typically complex migration would be to update a column to add a NOT NULL
constraint to an existing column, while still allowing existing client applications to work without changes, providing time for devs to update them after the migration as part of their normal workflow. Other similar migrations that would typically result in breaking changes on the schema could be renaming a column, adding or removing constraints, setting UNIQUE
... all of them supported by pgroll
.
This defines the migration for setting a column as NOT NULL
using pgroll
. Additional operations can be included in the same migration, but we'll focus on this one for simplicity:
{
"name": "review_not_null",
"operations": [
{
"alter_column": {
"table": "reviews",
"column": "review",
"not_null": true,
"up": "(SELECT CASE WHEN review IS NULL THEN product || ' is good' ELSE review END)",
"down": "review"
}
}
]
}
This is executed by running this simple command:
pgroll start review_not_null.json
pgroll
will perform all the necessary steps to make this change available without disrupting existing client applications (the expand step from the expand/contract pattern). In this particular example, it will:
review
column that respects the NOT NULL
constraint.NOT NULL
constraint to the new column, using the NOT VALID
clause to ensure that data access doesn’t block due to this statement, avoiding unexpected downtime.NULL
, based on the user-defined function. Of course, backfill happens in batches, avoiding excessively large updates that could also block the database.Typically, all of these steps would require manual execution to achieve a complex update like this. However, pgroll
streamlines the process by performing all of them with a single command ✨.
Once you are happy with the new version of the schema and all clients have been updated to use it, the old schema will no longer be accessed. It’s time to complete the migration (contract the schema) to get rid of the old column & triggers:
pgroll complete
For more complex examples, there is a wide range of schema migration samples in our examples and docs.
Today, we are rolling out the first version of pgroll
and we are looking forward to your feedback! We plan to continue developing pgroll
and exploring how it can make it easier and safer for Xata users to evolve their schema.
If you have any suggestions or questions, please open an issue in our GitHub repo, reach out to us on Discord or follow us on X / Twitter. We'd love to hear from you and keep you up to date with the latest progress on pgroll
.
Let’s start rolling!
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.
Copyright © 2025 Xatabase Inc.
All rights reserved.