Skip to main content

Drop-in versioning for Django models — releases, data status workflow, and CI integration

Project description

django-versioned-models

PyPI version Python Development Status Maintenance PyPI License


django-versioned-models

Drop-in versioning for Django models. Every model that inherits from VersionedModel gets full release management, data status workflow, and CI integration — automatically.


🚀 Features

  • Release management - every row in every table is tagged to a release. Branch, lock, patch, and deprecate with simple commands.
  • Data status workflow - DRAFT → FUTURE → APPROVED. CI only sees approved rows. Live edits by architects never break tests.
  • Lock enforcement - locked releases are immutable at the model level. No edits, no deletes, from anywhere — Admin, API, or shell.
  • Auto-discovery - inherit VersionedModel and your model is versioned. No registration needed, no matter how many models.
  • Topological FK sort - when copying a release, models are duplicated in the correct dependency order automatically.
  • Soft deprecation - old releases are hidden by default but data is always preserved and fully reversible.
  • Standalone releases - create a release with no source for bootstrapping a new project or parallel versioning.
  • CI-ready commands - create_release, approve_release, lock_release and more, ready to plug into any pipeline.

Installation

pip install django-versioned-models

Quick Start

1. Add to INSTALLED_APPS

INSTALLED_APPS = [
    ...
    'django_versioned_models',
]

2. Run migrations

python manage.py migrate

3. Create your models

from django_versioned_models.mixins import VersionedModel

class MyModel(VersionedModel):
    name = models.CharField(max_length=255)

    class Meta:
        unique_together = [('release', 'name')]  # unique per release, not globally

4. Run migrations for your models

python manage.py makemigrations
python manage.py migrate

5. Create the first release

python manage.py create_release --release-version v1.0.0

6. Add data, then lock

# Add data via Admin or API...

python manage.py lock_release --release-version v1.0.0

Ongoing Flow

# Create a new release branched from the previous one
python manage.py create_release --release-version v1.1.0 --based-on v1.0.0

# Architects edit data (status=DRAFT by default)

# CI approves stable rows
python manage.py approve_release --release-version v1.1.0

# Run tests against approved data only
pytest --release-version v1.1.0

# Lock and ship
python manage.py lock_release --release-version v1.1.0

# Bug found after deployment? Create a patch — never modify a locked release
python manage.py create_release --release-version v1.1.1 --based-on v1.1.0

How It Works

Every model that inherits from VersionedModel gets two fields added automatically:

  • release — which version this row belongs to
  • status — data readiness (draft / future / approved)

Data Status Workflow

DRAFT <-> FUTURE -> APPROVED  (one-way, CI only)
Status Who Meaning
DRAFT Architects Being worked on
FUTURE Architects Planned for a future release
APPROVED CI only Stable — what tests run against

CI always queries approved rows. DRAFT and FUTURE are invisible to CI — live edits never break tests.

MyModel.objects.approved(release)     # CI — stable rows only
MyModel.objects.for_release(release)  # everyone — all statuses

Lock Enforcement

Locked releases are immutable. Any attempt to save or delete a row in a locked release raises a ValidationError — from the Admin, the API, the shell, anywhere.

# This will raise ValidationError if release is locked
my_instance.save()
my_instance.delete()

Auto-Discovery

All models that inherit from VersionedModel are discovered automatically on every create_release. No manual registration needed. FK dependencies are resolved via topological sort — no ordering required.


Management Commands

Command Description
create_release --release-version v1.0.0 Create a standalone release (unlocked)
create_release --release-version v1.1.0 --based-on v1.0.0 Branch from a locked release
approve_release --release-version v1.1.0 Approve all DRAFT rows (CI only — FUTURE untouched)
lock_release --release-version v1.1.0 Lock a release (immutable)
unlock_release --release-version v1.1.0 Unlock (only before deployment)
deprecate_release --release-version v1.0.0 Soft-delete (data preserved, hidden by default)
deprecate_release --release-version v1.0.0 --undo Restore a deprecated release

Querying

from django_versioned_models.models import Release

release = Release.objects.get(version='v1.1.0')

# All rows for a release
MyModel.objects.for_release(release)

# Approved rows only (CI)
MyModel.objects.approved(release)

# Filter by status
MyModel.objects.filter(release=release, status='future')

Imports Reference

from django_versioned_models.mixins import VersionedModel, DataStatus
from django_versioned_models.models import Release
from django_versioned_models.services import (
    get_versioned_models,
    get_versioned_models_ordered,
    create_release,
    lock_release,
)

🤝 Contributing

Contributions are welcome! Please fork the repo, create a branch, and submit a pull request.


📄 License

MIT License — see LICENSE for details.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

django_versioned_models-0.1.1.tar.gz (10.0 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

django_versioned_models-0.1.1-py3-none-any.whl (13.5 kB view details)

Uploaded Python 3

File details

Details for the file django_versioned_models-0.1.1.tar.gz.

File metadata

  • Download URL: django_versioned_models-0.1.1.tar.gz
  • Upload date:
  • Size: 10.0 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for django_versioned_models-0.1.1.tar.gz
Algorithm Hash digest
SHA256 d6ff227db25c0ef06bd0c9b20936a4383e3fe53b79907fa2c066f0ff45b6181e
MD5 3a08a9c4f1cfb9bc10ab4c24e6572a70
BLAKE2b-256 5fdf65900ee7960743dd4fa79c0640afff3bf27919796c52e59ac0aad7e5c5fa

See more details on using hashes here.

File details

Details for the file django_versioned_models-0.1.1-py3-none-any.whl.

File metadata

File hashes

Hashes for django_versioned_models-0.1.1-py3-none-any.whl
Algorithm Hash digest
SHA256 c08c8e1d1d6faa5eb4a03c2f2c1feb063332f5fa9cad7c8497f8f9ca7b9a6ab1
MD5 582805527278d93ed808becb5d32eee7
BLAKE2b-256 67512afffc329cf1c578e424eff4eeb73467729bd1f847e49974df17221a373d

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page