Skip to content

Commit

Permalink
feat(db): Iterate on Alembic
Browse files Browse the repository at this point in the history
Support custom schema alembic migrations.

Update make targets.

Deprecate `init` command from service CLI.
  • Loading branch information
tcjennings committed Dec 17, 2024
1 parent ae63d8d commit a685f0a
Show file tree
Hide file tree
Showing 5 changed files with 108 additions and 29 deletions.
24 changes: 17 additions & 7 deletions Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -100,7 +100,7 @@ test: export CM_DATABASE_URL=postgresql://cm-service@localhost:${CM_DATABASE_POR
test: export CM_DATABASE_PASSWORD=INSECURE-PASSWORD
test: export CM_DATABASE_SCHEMA=cm_service_test
test: run-compose
python3 -m lsst.cmservice.cli.server init
alembic upgrade head
pytest -vvv --asyncio-mode=auto --cov=lsst.cmservice --cov-branch --cov-report=term --cov-report=html ${PYTEST_ARGS}

.PHONY: run
Expand All @@ -109,7 +109,7 @@ run: export CM_DATABASE_URL=postgresql://cm-service@localhost:${CM_DATABASE_PORT
run: export CM_DATABASE_PASSWORD=INSECURE-PASSWORD
run: export CM_DATABASE_ECHO=true
run: run-compose
python3 -m lsst.cmservice.cli.server init
alembic upgrade head
python3 -m lsst.cmservice.cli.server run

.PHONY: run-worker
Expand All @@ -118,9 +118,19 @@ run-worker: export CM_DATABASE_URL=postgresql://cm-service@localhost:${CM_DATABA
run-worker: export CM_DATABASE_PASSWORD=INSECURE-PASSWORD
run-worker: export CM_DATABASE_ECHO=true
run-worker: run-compose
python3 -m lsst.cmservice.cli.server init
alembic upgrade head
python3 -m lsst.cmservice.daemon

.PHONY: migrate
migrate: export PGUSER=cm-service
migrate: export PGDATABASE=cm-service
migrate: export PGHOST=localhost
migrate: export CM_DATABASE_PORT=$(shell docker compose port postgresql 5432 | cut -d: -f2)
migrate: export CM_DATABASE_PASSWORD=INSECURE-PASSWORD
migrate: export CM_DATABASE_URL=postgresql://${PGHOST}/${PGDATABASE}
migrate: run-compose
alembic upgrade head


#------------------------------------------------------------------------------
# Targets for developers to debug running against local sqlite. Can be used on
Expand All @@ -130,21 +140,21 @@ run-worker: run-compose
.PHONY: test-sqlite
test-sqlite: export CM_DATABASE_URL=sqlite+aiosqlite://///test_cm.db
test-sqlite:
python3 -m lsst.cmservice.cli.server init
alembic -x cm_database_url=sqlite:///test_cm.db upgrade head
pytest -vvv --asyncio-mode=auto --cov=lsst.cmservice --cov-branch --cov-report=term --cov-report=html ${PYTEST_ARGS}

.PHONY: run-sqlite
run-sqlite: export CM_DATABASE_URL=sqlite+aiosqlite://///test_cm.db
run-sqlite: export CM_DATABASE_ECHO=true
run-sqlite:
python3 -m lsst.cmservice.cli.server init
alembic -x cm_database_url=sqlite:///test_cm.db upgrade head
python3 -m lsst.cmservice.cli.server run

.PHONY: run-worker-sqlite
run-worker-sqlite: export CM_DATABASE_URL=sqlite+aiosqlite://///test_cm.db
run-worker-sqlite: export CM_DATABASE_ECHO=true
run-worker-sqlite:
python3 -m lsst.cmservice.cli.server init
alembic -x cm_database_url=sqlite:///test_cm.db upgrade head
python3 -m lsst.cmservice.daemon


Expand Down Expand Up @@ -180,7 +190,7 @@ run-usdf-dev: export CM_DATABASE_URL=postgresql://cm-service@${CM_DATABASE_HOST}
run-usdf-dev: export CM_DATABASE_PASSWORD=$(shell kubectl --cluster=usdf-cm-dev -n cm-service get secret/cm-pg-app -o jsonpath='{.data.password}' | base64 --decode)
run-usdf-dev: export CM_DATABASE_ECHO=true
run-usdf-dev:
python3 -m lsst.cmservice.cli.server init
alembic upgrade head
python3 -m lsst.cmservice.cli.server run

.PHONY: run-worker-usdf-dev
Expand Down
41 changes: 36 additions & 5 deletions alembic/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -16,13 +16,13 @@ The default database connection URL used by Alembic to create a SQLAlchemy engin
a `:memory:` instance of SQLite, which is effectively a no-op. In descending order
of preference, the database connection URL can be specified by:

- the `CM_DATABASE_URL` environment variable.
- any `-x cm_database_url=...` argument passed to `alembic`.
- the `CM_DATABASE_URL` environment variable.
- the value of the `sqlalchemy.url` configuration value in `alembic.ini` (which should
be a path to a sqlite database file).

With the project's virtual environment installed and activated (`source .venv/bin/activate`),
`alembic` is availble to invoke at the command line.
`alembic` is available to invoke at the command line.

## Migrating a Database

Expand All @@ -31,12 +31,23 @@ most recent migration, and `alembic downgrade base` will do the opposite, effect
destroying all Alembic-managed database resources.

Alembic can also run in "offline" mode, which instead of using a connection engine to
affect change in a database generates SQL files that can be manually-applied to a database.
In offline mode, the connection URL is used to flavor the dialect of the generated SQL.
affect change in a database generates SQL files that can be manually applied to a database.
In offline mode, the connection URL is used to flavor the dialect of the generated SQL
instead of creating a database connection.

To use offline mode, pass `--sql` as an argument to the `alembic` command; optionally
pipe the command's output to a file: `alembic upgrade head --sql > migration.sql`.

The database revisions are applied to a database schema identified by one of these
parameters:

- `-x schema=...` Alembic command parameter
- `CM_DATABASE_SCHEMA` environment variable (via app config object)
- `"public"`

The Alembic versions table is stored in the same schema as the database revisions
unless the `-x alembic_schema=...` command parameter is used.

### Incremental Migrations

In development or testing, it may be useful to "step" through the migrations one or more at
Expand All @@ -54,9 +65,29 @@ existing base model with the Alembic `--autogenerate` option against an empty Po
database.

Before an autogenerated revision is attempted, an empty Alembic revision is created and applied
to the (empty) database. This initial empty base revision is then updated to contain
to the (empty) database. This initial empty base revision is then backfilled to contain
the definitions of any Enum column types detected by Alembic, since these cannot be created
automatically.

Finally, the instances of Enum columns in the auto-generated revision are modified to
include a `create_type=False` parameter.

## Creating Additional Revisions

A new revision is created when Alembic is invoked with the `revision` command. The
template file `script.py.mako` is used as a blueprint for the new revision's Python
script.

A new migration can be created one of two ways:

1. Make code changes to affect the SQLAlchemy `DeclarativeBase` and use `--autogenerate`
with Alembic.
2. Write Alembic revision and then change code to match.

In the either case, a new Alembic revision will be most often created by calling

```
alembic revision -m MESSAGE [--autogenerate]
```

Where `MESSAGE` is a commit-style message describing the revision.
48 changes: 44 additions & 4 deletions alembic/env.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,8 @@
import os
from logging import getLogger
from logging.config import fileConfig

from sqlalchemy import create_engine, pool
from sqlalchemy import create_engine, pool, text

import lsst.cmservice.db
from alembic import context
Expand All @@ -24,6 +25,8 @@
# my_important_option = config.get_main_option("my_important_option")
# ... etc.

logger = getLogger("alembic")


def run_migrations_offline() -> None:
"""Run migrations in 'offline' mode.
Expand Down Expand Up @@ -67,28 +70,65 @@ def run_migrations_online() -> None:
and associate a connection with the context.
The database engine URI is consumed in descending order from:
- A `CM_DATABASE_URL` environment variable
- An `-x cm_database_url=...` CLI argument
- A `CM_DATABASE_URL` environment variable
- The value of `sqlalchemy.url` specified in the alembic.ini
- A sqlite :memory: database
"""
url = next(
(
u
for u in [
os.getenv("CM_DATABASE_URL"),
context.get_x_argument(as_dictionary=True).get("cm_database_url"),
os.getenv("CM_DATABASE_URL"),
config.get_main_option("sqlalchemy.url", default=None),
"sqlite://",
]
if u is not None
),
)

# Set PG* environment variables from CM_* environment variables to capture
# any that are not part of the URL
os.environ["PGPASSWORD"] = os.getenv("CM_DATABASE_PASSWORD", "")
os.environ["PGPORT"] = os.getenv("CM_DATABASE_PORT", "5432")

connectable = create_engine(url, poolclass=pool.NullPool)

# Build the migration in the schema given on the command line or default to
# the schema built into the metadata. This should be the app config value
# derived from env:CM_DATABASE_SCHEMA.
target_schema = (
context.get_x_argument(as_dictionary=True).get("schema") or target_metadata.schema or "public"
)
alembic_schema = context.get_x_argument(as_dictionary=True).get("alembic_schema") or target_schema

logger.info(f"Using schema {alembic_schema} for alembic revision table")
logger.info(f"Using schema {target_schema} for database revisions")

with connectable.connect() as connection:
context.configure(connection=connection, target_metadata=target_metadata)
if connection.dialect.name == "postgresql":
# Explicit pre-migration schema creation, needed for the schema
# in which the alembic version_table is being created, if different
# the target schema.
connection.execute(text(f"CREATE SCHEMA IF NOT EXISTS {alembic_schema}"))
connection.execute(text(f"CREATE SCHEMA IF NOT EXISTS {target_schema}"))
connection.commit()
# Operations are performed in the first schema on the search_path
connection.execute(text(f"set search_path to {target_schema}"))
connection.commit()

elif connection.dialect.name == "sqlite":
# SQLite does not care about schema
alembic_schema = None
target_schema = None

context.configure(
connection=connection,
target_metadata=target_metadata,
version_table_schema=alembic_schema,
include_schemas=False,
)

with context.begin_transaction():
context.run_migrations()
Expand Down
6 changes: 3 additions & 3 deletions alembic/script.py.mako
Original file line number Diff line number Diff line change
Expand Up @@ -14,9 +14,9 @@ ${imports if imports else ""}

# revision identifiers, used by Alembic.
revision: str = ${repr(up_revision)}
down_revision: Union[str, None] = ${repr(down_revision)}
branch_labels: Union[str, Sequence[str], None] = ${repr(branch_labels)}
depends_on: Union[str, Sequence[str], None] = ${repr(depends_on)}
down_revision: str | None = ${repr(down_revision)}
branch_labels: str | Sequence[str] | None = ${repr(branch_labels)}
depends_on: str | Sequence[str] | None = ${repr(depends_on)}


def upgrade() -> None:
Expand Down
18 changes: 8 additions & 10 deletions src/lsst/cmservice/cli/server.py
Original file line number Diff line number Diff line change
@@ -1,11 +1,8 @@
import click
import structlog
import uvicorn
from safir.asyncio import run_with_asyncio
from safir.database import create_database_engine, initialize_database

from .. import __version__, db
from ..config import config
from .. import __version__


# build the server CLI
Expand All @@ -15,15 +12,16 @@ def server() -> None:
"""Administrative command-line interface for cm-service."""


@server.command()
@server.command(deprecated=True)
@click.option("--reset", is_flag=True, help="Delete all existing database data.")
@run_with_asyncio
async def init(*, reset: bool) -> None: # pragma: no cover
"""Initialize the service database."""
logger = structlog.get_logger(config.logger_name)
engine = create_database_engine(config.database_url, config.database_password)
await initialize_database(engine, logger, schema=db.Base.metadata, reset=reset)
await engine.dispose()
"""Initialize the service database.
.. deprecated:: v1.5.0
The `init` command is deprecated in v0.2.0; it is replaced by alembic.
"""
print("Use `alembic upgrade head` instead.")


@server.command()
Expand Down

0 comments on commit a685f0a

Please sign in to comment.