Safely run migrations before deployment
Project description
django-safemigrate adds a safemigrate command to Django to allow for safely running a migration command when deploying.
Usage
Install django-safemigrate, then add this to the INSTALLED_APPS in the settings file:
INSTALLED_APPS = [
# ...
"django_safemigrate",
]
Then mark any migration that may be run during a pre-deployment stage, such as a migration to add a column.
from django_safemigrate import Safe
class Migration(migrations.Migration):
safe = Safe.before_deploy
At this point you can run the safemigrate Django command to run the migrations, and only these migrations will run. However, if migrations that are not safe to run before the code is deployed are dependencies of this migration, then these migrations will be blocked, and the safemigrate command will fail with an error.
When the code is fully deployed, just run the normal migrate Django command, which still functions normally. For example, you could add the command to the release phase for your Heroku app, and the safe migrations will be run automatically when the new release is promoted.
Safety Options
There are three options for the value of the safe property of the migration.
Safe.before_deploy
This migration is only safe to run before the code change is deployed. For example, a migration that adds a new field to a model.
Safe.after_deploy
This migration is only safe to run after the code change is deployed. For example, a migration that removes a field from a model.
By specifying a delay parameter, you can specify when a Safe.after_deploy migration can be run with the safemigrate command. For example, if it’s desired to wait a week before applying a migration, you can specify Safe.after_deploy(delay=timedelta(days=7)).
The delay is used with the datetime when the migration is first detected. The detection datetime is when the safemigrate command detects the migration in a plan that successfully runs. If the migration plan is blocked, such when a Safe.after_deploy is in front of a Safe.before_deploy, no migrations are marked as detected.
Note that a Safe.after_deploy migration will not run the first time it’s encountered.
Safe.always
This migration is safe to run before and after the code change is deployed. This is the default that is applied if no safe property is given. For example, a migration that changes the help_text of a field.
Pre-commit Hook
To get the most from django-safemigrate, it is important to make sure that all migrations are marked with the appropriate safe value. To help with this, we provide a hook for use with pre-commit. Install and configure pre-commit, then add this to the repos key of your .pre-commit-config.yaml:
repos:
- repo: https://github.com/ryanhiebert/django-safemigrate
rev: "5.2"
hooks:
- id: check
Nonstrict Mode
Under normal operation, if there are migrations that must run before the deployment that depend on any migration that is marked to run after deployment, the command will raise an error to indicate that there are protected migrations that should have already been run, but have not been, and are blocking migrations that are expected to run.
In development, however, it is common that these would accumulate between developers, and since it is acceptable for there to be downtime during the transitional period in development, it is better to allow the command to continue without raising.
To enable nonstrict mode, add the SAFEMIGRATE setting:
SAFEMIGRATE = "nonstrict"
In this mode safemigrate will run all the migrations that are not blocked by any unsafe migrations. Any remaining migrations can be run after the fact using the normal migrate Django command.
Disabled Mode
To disable the protections of safemigrate entirely, add the SAFEMIGRATE setting:
SAFEMIGRATE = "disabled"
In this mode safemigrate will migrations as if they were using the normal migrate Django command.
Contributing
To get started contributing, you’ll want to clone the repository, install dependencies with uv, and set up pre-commit.
git clone git@github.com:ryanhiebert/django-safemigrate.git
cd django-safemigrate
uv sync
pre-commit install
To run the tests use:
uvx --with tox-uv tox
To publish a new version:
Find and replace all instances of the previous version with the new version.
Commit and push that to origin.
Tag the commit with the new version git tag 1.0 and push that to origin.
Create the new release on GitHub. It will be published to PyPI automatically.
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 django_safemigrate-5.2.tar.gz.
File metadata
- Download URL: django_safemigrate-5.2.tar.gz
- Upload date:
- Size: 15.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.12.9
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
374631b609743cafa77e69c5eac04e9a19bc678af110e02603f8929214902bb4
|
|
| MD5 |
7230363c08f4a109b78ff5acf18289e3
|
|
| BLAKE2b-256 |
c187c57fd0a1780aeb81defa86a38a5c007c5c38cc49a5d2dc8fdeb020a649bd
|
Provenance
The following attestation bundles were made for django_safemigrate-5.2.tar.gz:
Publisher:
publish.yml on ryanhiebert/django-safemigrate
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
django_safemigrate-5.2.tar.gz -
Subject digest:
374631b609743cafa77e69c5eac04e9a19bc678af110e02603f8929214902bb4 - Sigstore transparency entry: 191557984
- Sigstore integration time:
-
Permalink:
ryanhiebert/django-safemigrate@e9004df44ac877d87bd3b39d6ce2f538c21c68e8 -
Branch / Tag:
refs/tags/5.2 - Owner: https://github.com/ryanhiebert
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@e9004df44ac877d87bd3b39d6ce2f538c21c68e8 -
Trigger Event:
release
-
Statement type:
File details
Details for the file django_safemigrate-5.2-py3-none-any.whl.
File metadata
- Download URL: django_safemigrate-5.2-py3-none-any.whl
- Upload date:
- Size: 11.3 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.12.9
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
a8c1e4c0687b4b9c65017477cce4231f695c6685b02de2154b9580909c6099da
|
|
| MD5 |
cc0362c4aa3b9db8a99bb1ec6065e4d5
|
|
| BLAKE2b-256 |
a8b74eb4eb54c0097a9010e56036eede9e4e3fe5bc575fcbfdb6923c2c6c5ed1
|
Provenance
The following attestation bundles were made for django_safemigrate-5.2-py3-none-any.whl:
Publisher:
publish.yml on ryanhiebert/django-safemigrate
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
django_safemigrate-5.2-py3-none-any.whl -
Subject digest:
a8c1e4c0687b4b9c65017477cce4231f695c6685b02de2154b9580909c6099da - Sigstore transparency entry: 191557986
- Sigstore integration time:
-
Permalink:
ryanhiebert/django-safemigrate@e9004df44ac877d87bd3b39d6ce2f538c21c68e8 -
Branch / Tag:
refs/tags/5.2 - Owner: https://github.com/ryanhiebert
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@e9004df44ac877d87bd3b39d6ce2f538c21c68e8 -
Trigger Event:
release
-
Statement type: