There already exists many tools to manage database schema versions, such as
sqitch, alembic
or pgroll.
Please consider them first to check whether they fit your needs before
considering this one.
In contrast to these tools, pg-schema-version emphasizes a simple approach
based on a single plain psql SQL scripts and no configuration, to provide
limited but useful features with safety in mind.
The application schema status is maintained in one table to detect reruns,
including checking patch signatures.
Several application can share the same setup.
Here is a typical use case for pg-schema-version:
Install from PyPi, e.g. with pip:
pip install pg-schema-version
Write a sequence of incremental postgres SQL data definition scripts.
The -- psv: comment header is mandatory to declare the application name,
version and optional description.
initial schema creation create_000.sql
-- psv: acme +1 Acme Schema v1.0
CREATE TABLE AcmeData(aid SERIAL PRIMARY KEY, data TEXT UNIQUE NOT NULL);
first schema upgrade create_001.sql
-- psv: acme +2 Acme Schema v1.1
CREATE TABLE AcmeType(atid SERIAL PRIMARY KEY, atype TEXT UNIQUE NOT NULL);
INSERT INTO AcmeType(atype) VALUES ('great'), ('super');
ALTER TABLE AcmeData ADD COLUMN atid INT NOT NULL DEFAULT 1 REFERENCES AcmeType;
second schema upgrade create_002.sql
-- psv: acme +3 Acme Schema v2.0
INSERT INTO AcmeType(atype) VALUES ('wow'), ('incredible');
Generate a psql-script from these for the target application:
pg-schema-version create_*.sql > acme.sql
Execute the script against a database to bring its schema up to date. By default the script runs in dry mode and reports the proposed changes to be applied by setting it in wet mode.
# first time can use command create to init the setup and register the app.
psql -v psv=create acme < acme.sql
# psv for application acme
# psv dry create for acme on acme, enable with -v psv=create:latest:wet
# psv will create infra, register acme and execute all steps
psql -v psv=create:wet acme < acme.sql
# psv for application acme
# psv wet create for acme on acme
# psv creating infra
# psv registering acme
# psv considering applying steps
# psv acme version: 0
# psv applying acme 1
# psv applying acme 2
# psv applying acme 3
# psv acme version: 3
# psv wet create for acme done
# on rerun, do nothing
psql -v psv=wet acme < acme.sql
# psv for application acme
# psv wet apply for acme on acme
# psv skipping acme registration
# psv considering applying steps
# psv acme version: 3
# psv skipping acme 1
# psv skipping acme 2
# psv skipping acme 3
# psv acme version: 3
# psv wet apply for acme done
# show current status
psql -v psv=status acme < acme.sql
# …
app version description acme 3 Acme Schema v2.0 psv 0 •
See pg-schema-version --help for a synopsis and explanations of all available
options.
The python command generates a reasonably safe re-entrant idempotent psql
script driven by variable psv with value command:version:moist
apply):
init just initialize an empty psv infrastructure.register add new application to psv versioning.apply execute required steps on an already registered application.reverse execute scripts to reverse steps.create do the 3 phases above: init, register and apply.unregister remove application from psv versioning.remove drop psv infrastructure.help show some help.status show version status of applications.history show history of application changes.catchup update application version status without actually executing steps
(imply init and register).latest.dry):
dry meaning that no changes are applied.wet to trigger actual changes.Each provided script must contain a special -- psv: name +5432 description
header with:
name the application name, which must be consistent accross all scripts.+5432 the version for apply (+) or reverse (-) a schema step, which
will be checked for inconsistencies such as repeated or missing versions.description an optional description of the resulting application status,
eg the corresponding application version.Beware that reversing may help you lose precious data, and that it is your responsability that the provided reverse scripts undo what was done by the forward scripts.
Other options at the psql script level:
-v psv_debug=1 to set debug mode.-v psv_app=foo to change the application registration name.
Probably a bad idea.Always:
There is no magic involved, you can still shot yourself in the foot, although with an effort. For safety, SQL schema creation scripts must not:
Imperfect checks are performed to try to detect the above issues.
They can be circumvented with option --trust-scripts or -T.
foo =n …? so what?psv_appreverse command to allow going backwards, and testshistory command to show application history of changes--version optionpsv command parsing-- psv: foo +1 …) mandatory,
including many sanity checks about names, versions…run to apply--partial option to allow partial scripts (i.e. missing versions)--app to check script consistencyThis code is Public Domain.
See online documentation. Sources and issues are on GitHub. Packages are distributed from PyPi.
All software has bug, this is software, hence… Beware that you may lose your hairs or your friends because of it. If you like it, feel free to send a postcard to the author.