A Django application to track deleted records.
Project description
Django Deletion Records
This simple Django app contains a mechanism to track deleted model records. Unlike the soft-deletion mechanism, the deleted content is saved to a separate table.
The deleted records are inserted via database triggers, so you don't have to write any additional code or change your base model classes. In fact, you only need to perform related migration operations, and you are good to go; the deleted records will start to appear in Django admin site.
Comparison with soft delete
Pros
-
No overly-complicated application code. To make soft-deletion work, you need to override model managers so that deleted records are kept out. Even then, the abstraction is very leaky when managers are not available such as doing aggregations or joins.
-
Database constraints are easier since you don't need to consider deleted records.
-
Managing relationships are easier, for example you don't need to worry about
ForeignKey
cascades. -
Deleted records do not affect undeleted records at all. More often than not, soft-delete create problems with new or existing rows because someone forgets that soft-deleted records are actually there.
Cons
-
Reversing deletions are pretty difficult. In soft-delete, you can just flip a column (most of the time). For deletion records, you'll have to manually re-insert the data (and its dependencies).
-
Data lookup is relatively difficult especially if you're maintaining complex relationships spanning multiple tables. In soft-delete, you can just do regular SQL lookups. However, deletion records require custom resolution for
ForeignKey
's and such. -
Deletion operations are relatively slower since soft-delete updates are faster and there is no trigger penalty.
Soft-deleting is convenient when you need to restore data but creates complexity in application code. Deletion records are much easier manage, however it is difficult to restore data.
You may also use soft-deletes alongside deletion records if you want to keep some tables soft-deleted and others recorded (for example, as audit logs).
Installation
django-deletion-records
only works for PostgreSQL, other database backends are not supported.
-
Install from PyPI:
pip install django-deletion-records
-
Add
deletion_records
to yourINSTALLED_APPS
, such as:INSTALLED_APPS = [ 'django.contrib.admin', 'django.contrib.auth', 'django.contrib.contenttypes', 'django.contrib.sessions', 'django.contrib.messages', 'django.contrib.staticfiles', # ... 'deletion_records', ]
-
Run migrate for
deletion_records
:python manage.py migrate deletion_records Operations to perform: Apply all migrations: deletion_records Running migrations: Applying deletion_records.0001_initial... OK
Marking a model for deletion
To start recording deletions for a given model, you'll need to perform a
RunSQL
operation (that will create the related trigger) via Django
migrations.
To start, create an empty migration file in your related app:
python manage.py makemigrations --empty your-app
This will create an empty migration with no operations. To get the related
operations we'll use the deletion_migration_operations
management command:
python manage.py deletion_migration_operations yourapp.User
# yourapp.User
operations = [
migrations.RunSQL(
"""
CREATE TRIGGER deleted_record_insert
AFTER DELETE
ON yourapp_user
FOR EACH ROW
EXECUTE FUNCTION deletion_record_insert();
""",
reverse_sql="DROP TRIGGER IF EXISTS deleted_record_insert ON yourapp_user CASCADE;",
)
]
Copy the given operations to your empty migration file, and apply the migrations. Deletion records will start to appear, that's it!
Tips
-
You can supply multiple models for
deletion_migration_operations
to get multiple operations at once. -
You can have multiple operations in one migration file if you want to. It just depends on how you want to handle the migrations.
-
If you want to enable deletion records for a third party model, you can still employ the steps above. You just need to figure out a proper application (that'll hold the migrations), people generally use "core" apps to do that kind of stuff.
-
Reversing the migration created above will drop the related trigger, thus stopping the deletion recording. The formerly deleted records will be still available.
If you want to hard-delete some records (or manage deleted records for any other reason), you can use the provided model as such:
from deletion_records.models import DeletedRecord
# Deletes User objects with provided ids.
DeletedRecord.objects.filter(
table_name="yourapp_user", object_id__in=[2, 3, 4]
).delete()
from deletion_records.models import DeletedRecord
# Fetch a deleted record and view the data.
user = DeletedRecord.objects.get(table_name="account_user", object_id=276833)
assert user.data == {
"id": 276833,
"email": "lauren62@example.com",
"is_staff": False,
"password": "pbkdf2_sha256$600000$4X48PbRIeemb2ECW1pIlO4$jFuePmugUuTE4D6nIbP9TxGKcYxLBut81bR4JbshU8I=",
"username": "qwSyeamDqT",
"is_active": True,
"last_name": "Jackson",
"first_name": "Cynthia",
"last_login": None,
"date_joined": "2024-01-16T19:35:32.331011+00:00",
"is_superuser": False,
}
You can also browse deleted records via Django admin.
Credits
I primarily read the following posts to implement this in Django:
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
File details
Details for the file django_deletion_records-0.2.0.tar.gz
.
File metadata
- Download URL: django_deletion_records-0.2.0.tar.gz
- Upload date:
- Size: 6.3 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/5.0.0 CPython/3.11.7
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 5028dbbf03b6e69fb0da057956bd947901b476fcc0f14e927ec7278981e8a1bb |
|
MD5 | f99c21e2e6fab72efa7c513fdd0e9a4b |
|
BLAKE2b-256 | 17c26a44d82d1251c8d92f7ef46ad9f6fcb0ab09fb5ae351693394871fa0b477 |
File details
Details for the file django_deletion_records-0.2.0-py3-none-any.whl
.
File metadata
- Download URL: django_deletion_records-0.2.0-py3-none-any.whl
- Upload date:
- Size: 9.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/5.0.0 CPython/3.11.7
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 9cf22c4b8a2ebaed88758763363714ea8eb2497a1a73d1015a4cab49a30e7d29 |
|
MD5 | 5f882b38c58e8565697eaa05a7b4c549 |
|
BLAKE2b-256 | df897e8e7772c14bbe96c0c0da68ab92b5376fb21444b5df669b2d3489a2c70f |