Migrations
All migrations live in host/migrations/versions/ — not in module packages. Alembic runs from the repo root via host/alembic.ini and shares the app's .env / SM_DATABASE_URL.
Why centralized?
- Dependency ordering is global. If
invoicesdepends onorders_order.id, their migrations must order correctly. One linear Alembic history enforces this. - Autogenerate sees everything.
host/migrations/env.pycallsbuild_module_metadata()to union every installed module'sMetaData. Autogenerate diffs the DB against that union and writes one migration covering all changes. - Operators run one command.
make migrateis the only target. No "did you also runorders/migrate?" footgun.
Each module's first migration sets branch_labels = ("<module_name>",) so you can still downgrade one module at a time with alembic downgrade <module>@base.
Day-to-day workflow
Create a migration
make migration msg="add orders tables"This runs alembic -c host/alembic.ini revision --autogenerate from the repo root. The resulting file lands in host/migrations/versions/XXXX_add_orders_tables.py.
Always open and read the generated file before committing. Autogenerate is good but not perfect:
- It detects column additions, drops, type changes, index/constraint changes, new tables.
- It misses semantic constraints (e.g. "this ENUM now has one more value"), and it sometimes produces drops that should be renames.
- Constraint naming needs to be stable — provide explicit
name="..."kwargs when creating indexes/constraints in your models.
Apply migrations
make migrateRuns alembic -c host/alembic.ini upgrade heads. Idempotent.
Downgrade
make downgrade # back one revision
uv run --project host alembic -c host/alembic.ini downgrade <revision_id> # to a specific revision
uv run --project host alembic -c host/alembic.ini downgrade orders@base # back to the state before the orders module existedorders@base uses the branch_labels marker from the module's first migration. Module-level downgrade is the mechanism for uninstalling a module cleanly.
First migration of a new module
When you scaffold a module with smpy create-module, the first autogenerate revision produces a file that needs this marker added by hand:
# host/migrations/versions/XXXX_add_orders_tables.py
revision = "..."
down_revision = "..."
branch_labels = ("orders",) # ← add this
depends_on = NoneOnce the marker is in place, all future orders migrations inherit the branch.
Alembic environment setup
host/migrations/env.py looks roughly like:
from simple_module_db import (
build_module_metadata,
make_include_object,
make_process_revision_directives,
render_item,
)
target_metadata = build_module_metadata()
include_object = make_include_object(target_metadata)
process_revision_directives = make_process_revision_directives(target_metadata)
context.configure(
target_metadata=target_metadata,
include_object=include_object,
render_item=render_item,
process_revision_directives=process_revision_directives,
)target_metadata— union of every module'sMetaData.include_object— accepts only tables present intarget_metadata, so autogenerate never diffs system tables (alembic_version) or any host-owned tables outside the module system.render_item— collapses SQLModel'sAutoStringtosa.Stringand rendersStrEnumcolumns withvalues_callableso generated migrations are importable.process_revision_directives— re-emits expression-based (functional) indexes that autogenerate silently drops under SQLite.
Boot-time migration check
On startup, the framework compares alembic_version to the migration head:
- Development: prints a warning if the DB is behind. Result stored on
app.state.migration. - Production: fails boot with
SM010. Don't ship a web process that's pointed at an unmigrated DB.
Fix locally with make migrate. In production, run migrations before rolling over the web tier.
Diagnostic: SM011
Fires as a warning when a module's model declares a table that doesn't appear in any Alembic migration. Typical causes:
- You added a model but haven't generated a migration for it yet.
- You renamed a table but the old migration still references the old name.
- You used
__abstract__ = Truesomewhere it shouldn't be.
The dev-mode boot log prints the offending table names. Resolution: run make migration msg="...", review, make migrate.
Cross-module foreign keys
If invoices has an FK to orders_order.id:
- Alembic will emit
ADD CONSTRAINTin the invoices table's migration. - The migration that creates
invoices_invoicemust come after the one that createsorders_orderin linear history. smpy create-module+make migrationhandle this naturally as long asdepends_onis correct inModuleMeta.
All tables share the host's single schema on both Postgres and SQLite, so a cross-module FK is just an ordinary same-schema reference (orders_order.id ← invoices_invoice.order_id).
On SQLite, FKs are off by default but the test suite enables them; in production SQLite use (rare), set PRAGMA foreign_keys = ON.
Data migrations
Alembic supports ad-hoc op.execute("UPDATE ...") inside a migration. Use for:
- Backfilling new NOT NULL columns — add as nullable, backfill, alter to NOT NULL.
- Renaming columns —
op.alter_columnwithnew_column_name=...preserves data. - Migrating enum values.
Data migrations run as part of alembic upgrade. If they fail mid-way, you're left at an intermediate state — design them to be idempotent (use ON CONFLICT, IF EXISTS, UPDATE ... WHERE x IS NULL).
For long-running backfills on big tables, break into batches and run outside Alembic:
# scripts/backfill_orders_total.py
async def backfill(batch_size=1000):
...Migration drift in monorepos
git pull introducing two parallel branches of Alembic revisions:
revision A ← you created
revision B ← teammate's branch createdBoth have down_revision = <previous>. Alembic will refuse upgrade head on a tree with multiple heads. Resolve by:
uv run alembic heads— list them.uv run alembic merge -m "merge A and B" <rev-A> <rev-B>— creates a merge revision with both as parents.- Commit the merge revision.
Keep merges small; a merge revision with its own op.* logic is a code smell.
Initial-migration gotchas
When you autogenerate a migration for a freshly-added module, autogenerate writes op.create_table(...) for every table in the module's MetaData. Inspect:
- Are the table names right (the module-prefixed
orders_order, identical on Postgres and SQLite)? - Do indexes and constraints have stable names? Rename via
name=...on the model if not. - Did autogenerate also pick up any other module's tables? That means you forgot
make migrateafter the last scaffold. Squash the file down to just this module's changes.
Testing migrations
The db_session fixture stamps alembic_version at head on a fresh in-memory DB, which validates that build_module_metadata() + your migrations produce a schema the tables can operate against. That said, integration tests don't run every Alembic op — if you want to exercise the migrations explicitly:
@pytest.mark.asyncio
async def test_migration_up_then_down(tmp_path):
from alembic.config import Config
from alembic import command
db_url = f"sqlite:///{tmp_path}/migrate_test.db"
cfg = Config("host/alembic.ini")
cfg.set_main_option("sqlalchemy.url", db_url)
command.upgrade(cfg, "head")
command.downgrade(cfg, "base")That catches the common "autogenerate wrote a broken downgrade" class of bugs.