State-based, cross-engine database deployment — git-driven, parser-assisted, SQL-first.
Project description
Declare your desired database state in git. dbly makes it real.
dbly deploys your database objects — tables, views, functions, procedures, packages, triggers, grants — from version control to PostgreSQL, SQL Server, Oracle and SQLite. You keep one SQL file per object, like normal source code; dbly works out what changed and brings the target database in sync. Think Terraform for your database: declarative, predictable, repeatable.
Why
- Declarative — your repo is the source of truth. No hand-written migration chains, no version-number collisions on parallel branches.
- Plan before apply — preview exactly what will run, then execute. No surprises.
- Idempotent & safe — only the necessary changes are applied. Additive changes go automatically; destructive ones (dropping columns, etc.) are flagged and never run unless you explicitly allow them.
- Multi-database — one workflow across all four engines.
Install
uv sync # PostgreSQL + SQLite work out of the box
uv sync --extra oracle # add the Oracle driver
uv sync --extra mssql # add the SQL Server driver
uv sync --extra all # both
Organize your repo
One file per object; folder names map to database schemas. Use any extension you like
(.sql, .tbl, .vw, .prc, …) — dbly reads the SQL to know what each object is.
db/
sales/
customer.tbl
v_open_orders.vw
grants.sql
init/ # optional: privileged greenfield groundwork (CREATE DATABASE/ROLE…)
migrations/ # optional: ordered, run-once ALTERs for renames / data backfills
hooks/pre/ hooks/post/ # optional: .sql or .py hooks (e.g. ArcGIS/ArcPy steps)
.dbignore # files in the repo that should not be deployed
Most changes are declarative — edit the object file, dbly figures out the additive diff. For
changes the diff can't do safely (renaming a column, moving data), drop an ordered script in
migrations/ (0001_…sql); it runs exactly once, is recorded in the ledger, and the table
it touches defers to it for that deploy. On a fresh database, migrations are baselined
(recorded, not run) since the object files already describe the end state.
Connect
A connection profile (reuses the familiar connection.properties format):
environment=postgres # postgres | sqlserver | oracle | sqlite
service=db.example.com:5432
username=app
password=${DB_PASSWORD} # ${ENV} placeholders → keep secrets out of the repo (CI/CD-safe)
database=appdb
Use
# preview the changes between the deployed state and a git ref
dbly plan --to main --target prod.connection.properties
# apply them
dbly apply --to main --target prod.connection.properties
# export a plain SQL script to run by hand (e.g. through a customer VPN, no dbly needed)
dbly plan --to main --target prod.connection.properties --sql deploy.sql
# what is currently deployed?
dbly status --target prod.connection.properties
# has the database drifted from the desired state?
dbly check --target prod.connection.properties
# greenfield only: run privileged groundwork once under a superuser profile
dbly init --init-target super.connection.properties
Typical workflow: edit your object files → commit → dbly plan to review → dbly apply.
Deploying a subset of features is just choosing the git ref you deploy (a release tag or
branch). Destructive steps require --allow-destructive.
Built for trunk-based development
Database teams are usually locked out of trunk-based development: migration scripts collide on parallel branches, and "merge" effectively means "deploy to the customer". dbly is designed to break that deadlock for teams who write their logic in SQL, in the database:
- No migration numbers, no collisions. Two developers edit different objects — git merges them like any other code. Integrate early, every day.
- Merge ≠ deploy. The trunk is your desired state;
dbly apply --to <tag>ships the ref you choose, in the maintenance window you choose. Release what you want, when you want. - One trunk, every customer. dbly reads each database's real state, so customers on different versions are no problem.
Integrate continuously, deploy on your own schedule — trunk-based development, finally practical for state-based database developers.
Status
Early alpha — all four engines are implemented and tested against live databases.
License
MIT
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file dbly-0.2.0.tar.gz.
File metadata
- Download URL: dbly-0.2.0.tar.gz
- Upload date:
- Size: 1.5 MB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
acdca11f4a1f686564d9bde4fe21a722fb763af6279989e959828934fb64845b
|
|
| MD5 |
60afdac7a5320eea130e931a7b409c78
|
|
| BLAKE2b-256 |
bbe6e025d6530b0b9480012d259085e9c955de9e7ffb4627d4bbaed3f66588eb
|
Provenance
The following attestation bundles were made for dbly-0.2.0.tar.gz:
Publisher:
publish.yml on angrydat/dbly
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
dbly-0.2.0.tar.gz -
Subject digest:
acdca11f4a1f686564d9bde4fe21a722fb763af6279989e959828934fb64845b - Sigstore transparency entry: 1999471069
- Sigstore integration time:
-
Permalink:
angrydat/dbly@e58472b70e1c971797fc488484565f74682b664a -
Branch / Tag:
refs/tags/v0.2.0 - Owner: https://github.com/angrydat
-
Access:
private
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@e58472b70e1c971797fc488484565f74682b664a -
Trigger Event:
release
-
Statement type:
File details
Details for the file dbly-0.2.0-py3-none-any.whl.
File metadata
- Download URL: dbly-0.2.0-py3-none-any.whl
- Upload date:
- Size: 39.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
943a58571c3fd194f26925e7e5beba3c615696ba27bf42b586ce78573a521ec0
|
|
| MD5 |
140b3a4d7f9b43e7416ba3fc188a4c7f
|
|
| BLAKE2b-256 |
04aa4102f1c3553b69ade4381923b14ab5585baef2bb7191f7b5c83290832d46
|
Provenance
The following attestation bundles were made for dbly-0.2.0-py3-none-any.whl:
Publisher:
publish.yml on angrydat/dbly
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
dbly-0.2.0-py3-none-any.whl -
Subject digest:
943a58571c3fd194f26925e7e5beba3c615696ba27bf42b586ce78573a521ec0 - Sigstore transparency entry: 1999471117
- Sigstore integration time:
-
Permalink:
angrydat/dbly@e58472b70e1c971797fc488484565f74682b664a -
Branch / Tag:
refs/tags/v0.2.0 - Owner: https://github.com/angrydat
-
Access:
private
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@e58472b70e1c971797fc488484565f74682b664a -
Trigger Event:
release
-
Statement type: