Subscription model for a django model instance.
Project description
django-model-subscription
Table of contents
Motivation
- Using Observer Pattern notify subscribers about changes to a django model.
- Decouple Business logic from Models.save
- Support for bulk actions (Not available using django signals.)
- Use noop subscribers when
settings.SUBSCRIPTION_RUN_EXTERNALisFalsewhich prevents having to mock subscribers that call external services in testing, local development environments. - Show changes to the instance after it has been updated i.e diff's the initial state and the current state.
Installation
$ pip install django-model-subscription
Add model_subscription to your INSTALLED_APPS
INSTALLED_APPS = [
...,
'model_subscription',
...
]
Usage
Using the SubscriptionModelMixin and SubscriptionQuerySet
from model_subscription.mixin import SubscriptionModelMixin
from model_subscription.model import SubscriptionQuerySet
class TestModel(SubscriptionModelMixin, models.Model):
name = models.CharField(max_length=255)
objects = SubscriptionQuerySet.as_manager()
Subclassing the SubscriptionModel base class.
from model_subscription.model import SubscriptionModel
class TestModel(SubscriptionModel):
name = models.CharField(max_length=255)
Creating subscribers.
- Using
OperationType
import logging
from model_subscription.decorators import subscribe
from model_subscription.constants import OperationType
log = logging.getLogger(__name__)
@subscribe(OperationType.CREATE, TestModel)
def handle_create(instance):
log.debug('Created {}'.format(instance.name))
- Using
create_subscriptiondirectly (succinct version).
import logging
from model_subscription.decorators import create_subscription
log = logging.getLogger(__name__)
@create_subscription(TestModel)
def handle_create(instance):
log.debug('Created {}'.format(instance.name))
Decorators
subscribe: Explicit (Requires a valid OperationType).
(Create, Update, Delete) operations.
create_subscription: Subscribes to create operation i.e a new instance.
@create_subscription(TestModel)
def handle_create(instance):
log.debug('1. Created {}'.format(instance.name))
update_subscription: Subscribes to updates also includes (changed_data).
@update_subscription(TestModel)
def handle_update(instance, changed_data):
log.debug('Updated {} {}'.format(instance.name, changed_data))
delete_subscription: Subscribes to delete operation:
NOTE: The instance.pk is already set to None.
@delete_subscription(TestModel)
def handle_delete(instance):
log.debug('Deleted {}'.format(instance.name))
(Bulk Create, Bulk Update, Bulk Delete) operations.
bulk_create_subscription: Subscribe to bulk create operations.
@bulk_create_subscription(TestModel)
def handle_bulk_create(instances):
for instance in instances:
log.debug('Bulk Created {}'.format(instance.name))
bulk_update_subscription: Subscribe to bulk update operations.
@bulk_update_subscription(TestModel)
def handle_bulk_update(instances):
for instance in instances:
log.debug('Updated {}'.format(instance.name))
bulk_delete_subscription: Subscribe to bulk delete operations.
@bulk_delete_subscription(TestModel)
def handle_bulk_delete(instances):
for instance in instances:
log.debug('Deleted {}'.format(instance.name))
Setup Subscribers using AppConfig.ready (Recomended).
Update you apps.py
from django.apps import AppConfig
class MyAppConfig(AppConfig):
name = 'myapp'
def ready(self):
from myapp import subscriptions
Setup Subscribers using auto discovery.
By default the settings.SUBSCRIPTION_AUTO_DISCOVER is set to False.
To use auto discovery this is not recommended as it would notify the subscribers wherever the model is used i.e IPython notebook, external scripts.
In your settings.py add
SUBSCRIPTION_AUTO_DISCOVER = True
Setting up the SUBSCRIPTION_MODULE
NOTE: This is only required when
SUBSCRIPTION_AUTO_DISCOVER = True
SUBSCRIPTION_MODULE = 'subscription'
Credits
If you feel generous and want to show some extra appreciation:
Resources
- https://python-3-patterns-idioms-test.readthedocs.io/en/latest/Observer.html
- https://refactoring.guru/design-patterns/observer
- https://hackernoon.com/observer-vs-pub-sub-pattern-50d3b27f838c
TODO's
- Supporting field level subscriptions.
- Support class based subscribers which implements
__call__ - Extend to include custom OperationType.
- Add support for using a single class to manage multiple actions i.e MyClass.update, MyClass.create.
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-model-subscription-0.2.2.tar.gz.
File metadata
- Download URL: django-model-subscription-0.2.2.tar.gz
- Upload date:
- Size: 13.3 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/1.1.9 CPython/3.7.12 Darwin/20.5.0
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
a874ae344f094e9a0b800e4035f8abc4272f21c3c0596b69f0b34a01b2747eac
|
|
| MD5 |
e42575a6566bad19cfef119c23d5d6d4
|
|
| BLAKE2b-256 |
84fc9943f9997527c946dc714b8d03e313b7da40f8b930608baa5cf1ea72328d
|
File details
Details for the file django_model_subscription-0.2.2-py3-none-any.whl.
File metadata
- Download URL: django_model_subscription-0.2.2-py3-none-any.whl
- Upload date:
- Size: 12.5 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/1.1.9 CPython/3.7.12 Darwin/20.5.0
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
c21bb53decbb20658c9e659306ad40bb296ab16336d3b5834387c2e5b0e130c4
|
|
| MD5 |
1723072f8d0a02072fa76b4f4e33b561
|
|
| BLAKE2b-256 |
2a721717a6d6becac83064b5d9d855e6e137d110a16653a1edffe02d818740b0
|