efficient model history using database triggers
Project description
Django Chronicle is an implementation of the slowly changing dimensions type 4 which uses database triggers.
How to use?
-
Create a custom revision model. e.g.
from chronicle.models import AbstractRevision class Revision(AbstractRevision): user = models.ForeignKey(settings.AUTH_USER_MODEL) created = models.DateTimeField(auto_now_add=True)
-
Set
settings.REVISION_MODELto point to your revision model. e.g.REVISION_MODEL = 'revision.Revision'
-
Let your models inherit from
HistoryMixine.g.from chronicle.models import HistoryMixin from django.db import models class Food(HistoryMixin, models.Model): name = models.CharField(max_length=50)
-
Create all the migrations and run them:
$ manage.py makemigrations $ manage.py migrate
That should create all the
_historytables for your models that inherit from theHistoryMixin. -
Create the database triggers
$ manage.py create_history_triggers
Now every change to your models should be logged in the _history tables
and you can access the model history via the History model which becomes
a field of the original class.
Example usage:
# create
food = Food('Carot')
food.save()
assert(Food.History.objects.filter(id=food.id).count() == 1)
# update
food.name = 'Carrot'
food.save()
assert(Food.History.objects.filter(id=food.id).count() == 2)
# delete
food.delete()
assert(Food.History.objects.filter(id=food.id).count() == 3)
Why database triggers?
The obvious choice to implement model history would be to connect a signal
handler to the post_save and post_delete signal. This has some rather huge
downsides:
1.) QuerySet.update() and a lot of other QuerySet methods do not emit any
signals. Having to limit the code to only use save() can be a rather huge
performance problem depending on the type of application.
2.) There is a rather large performance impact when creating the history via
the Django ORM. A single QuerySet.update() call could result in hundreds
or thousands of inserts. While this could mostly be solved using the
Manager.bulk_create method a database trigger is a lot faster as there is no
extra database roundtrip required.
3.) This works for any kind of raw query - even outside of the Django ORM - as
long as the chronicle.revision_id session variable is properly set.
The only real downside is the DB compatibility. Right now this package only supports the PostgreSQL database engine.
How to issue queries without the Django ORM?
Create a revision by inserting a row into the revision table and set the
chronicle.revision_id session variable like so:
SET chronicle.revision_id = 42; -- replace 42 by the actual revision id
Once you have made all changes to your models don't forget to reset the session variable. Otherwise you might reuse the same revision by accident in the same DB session:
SET chronicle.revision_id TO DEFAULT;
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_chronicle-0.2.0.tar.gz.
File metadata
- Download URL: django_chronicle-0.2.0.tar.gz
- Upload date:
- Size: 12.7 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/1.7.1 CPython/3.10.12 Linux/5.15.0-97-generic
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
05d8370741493d01587ef6384844c4fecccb13f91079385b07d6453c0ec11dd1
|
|
| MD5 |
223a6d3524cfa4748dc25339d154cd8a
|
|
| BLAKE2b-256 |
ec120ec9869d45ef988aaa4f678b1ef251f6310c8c6103cee8fae1cb201561f1
|
File details
Details for the file django_chronicle-0.2.0-py3-none-any.whl.
File metadata
- Download URL: django_chronicle-0.2.0-py3-none-any.whl
- Upload date:
- Size: 10.7 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/1.7.1 CPython/3.10.12 Linux/5.15.0-97-generic
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
8b1cab138e7cc1e148d246a80f278650af26dcff93feedee78cb08342954c1f9
|
|
| MD5 |
2840b47af1c493feea66da33572bf3e7
|
|
| BLAKE2b-256 |
f2dd3369a8aeb927132fea0cc00c4eae893748de282189f3ff0c9d3d8025df96
|