# Django Unified Signals
[![Continuous Integration](https://github.com/ivellios/django-unified-signals/actions/workflows/ci.yaml/badge.svg)](https://github.com/ivellios/django-unified-signals/actions/workflows/ci.yaml)
This package extends behavior of the Django Signals, by unifying the data passed
when the signal is sent. That way both: the sender and the receiver can be sure
that the data passed is always of the same type.
Django Signals are not very strict on the data that can be passed with [send method](https://docs.djangoproject.com/en/4.2/topics/signals/#django.dispatch.Signal.send).
Basically anything can be passed as the arguments and receivers
need to be aware of that from the project documentation. It gets even worse when on one
side the signal send params change and the receivers' maintainers may not be aware of that.
It gets even worse, when using send_robust method, which ignores all exceptions
and any errors may pass without being noticed.
Using this package you have to define the message type class which object
is always expected to be passed when sending the signal. That way receiver
knows what type of message will be received. This package automates the process
of checking if the send message is following the contract.
## Installation
The package is [available on PyPI](https://pypi.org/project/django-unified-signals/):
```bash
pip install django-unified-signals
```
## Usage
Let's start by defining the message structure. It can be any class you want.
In the example we will use `dataclass`:
```python
import dataclasses
@dataclasses.dataclass
class UserMessage:
name: str
age: int
```
Now that we have the message structure defined, we can create the signal:
```python
from unified_signals import UnifiedSignal
user_deactivated_signal = UnifiedSignal(UserMessage)
```
It extends the standard `django.dispatch.Signal` class,
so it can be used in the same way.
```python
user_deactivated_signal.send(sender, UserMessage(name='John', age=30))
```
The receiver can be defined in the same way as for the standard Django Signal:
```python
@receiver(user_deactivated_signal)
def handle_user_deactivated(sender, message: UserMessage, **kwargs):
print(message.name)
print(message.age)
...
```
The difference is that the message is always of the same type, so the receiver
can be sure that the message is always of the same type. If the message is not
of the expected type when sending the signal, the `unified_signals.exceptions.UnifiedSignalMessageTypeError`
exception will be raised.
```python
user_deactivated_signal.send(sender, 'not a message') # raises UnifiedSignalMessageTypeError
```
Raw data
{
"_id": null,
"home_page": "https://github.com/ivellios/django-unified-signals",
"name": "django-unified-signals",
"maintainer": null,
"docs_url": null,
"requires_python": "<4.0,>=3.9",
"maintainer_email": null,
"keywords": "django, signals",
"author": "Janusz Kamie\u0144ski",
"author_email": "200957+ivellios@users.noreply.github.com",
"download_url": "https://files.pythonhosted.org/packages/1c/3f/8132c1e18dafa0f1b88d5738be1ad4197a1b19cb39cce84ea88c283b965b/django_unified_signals-0.2.1.tar.gz",
"platform": null,
"description": "# Django Unified Signals\n\n[![Continuous Integration](https://github.com/ivellios/django-unified-signals/actions/workflows/ci.yaml/badge.svg)](https://github.com/ivellios/django-unified-signals/actions/workflows/ci.yaml)\n\nThis package extends behavior of the Django Signals, by unifying the data passed\nwhen the signal is sent. That way both: the sender and the receiver can be sure \nthat the data passed is always of the same type.\n\nDjango Signals are not very strict on the data that can be passed with [send method](https://docs.djangoproject.com/en/4.2/topics/signals/#django.dispatch.Signal.send).\nBasically anything can be passed as the arguments and receivers \nneed to be aware of that from the project documentation. It gets even worse when on one\nside the signal send params change and the receivers' maintainers may not be aware of that. \nIt gets even worse, when using send_robust method, which ignores all exceptions \nand any errors may pass without being noticed.\n\nUsing this package you have to define the message type class which object \nis always expected to be passed when sending the signal. That way receiver \nknows what type of message will be received. This package automates the process\nof checking if the send message is following the contract.\n\n## Installation\n\nThe package is [available on PyPI](https://pypi.org/project/django-unified-signals/): \n\n```bash\npip install django-unified-signals\n```\n\n## Usage\n\nLet's start by defining the message structure. It can be any class you want.\nIn the example we will use `dataclass`:\n\n```python\nimport dataclasses\n\n@dataclasses.dataclass\nclass UserMessage:\n name: str\n age: int\n```\n\nNow that we have the message structure defined, we can create the signal:\n\n```python\n\nfrom unified_signals import UnifiedSignal\n\nuser_deactivated_signal = UnifiedSignal(UserMessage)\n```\n\nIt extends the standard `django.dispatch.Signal` class, \nso it can be used in the same way.\n\n```python\nuser_deactivated_signal.send(sender, UserMessage(name='John', age=30))\n```\n\nThe receiver can be defined in the same way as for the standard Django Signal:\n\n```python\n@receiver(user_deactivated_signal)\ndef handle_user_deactivated(sender, message: UserMessage, **kwargs):\n print(message.name)\n print(message.age)\n ...\n```\n\nThe difference is that the message is always of the same type, so the receiver\ncan be sure that the message is always of the same type. If the message is not\nof the expected type when sending the signal, the `unified_signals.exceptions.UnifiedSignalMessageTypeError` \nexception will be raised.\n\n```python\nuser_deactivated_signal.send(sender, 'not a message') # raises UnifiedSignalMessageTypeError\n```\n",
"bugtrack_url": null,
"license": "MIT",
"summary": "Django Signals extension for unifying data passed to receivers.",
"version": "0.2.1",
"project_urls": {
"Homepage": "https://github.com/ivellios/django-unified-signals",
"Repository": "https://github.com/ivellios/django-unified-signals"
},
"split_keywords": [
"django",
" signals"
],
"urls": [
{
"comment_text": "",
"digests": {
"blake2b_256": "52c4b0931453be880e391fd6d42d8f2207269458bf55e302d7d8c140c3b8ecb7",
"md5": "da80fa2e073854500ab9f049576697b0",
"sha256": "45a0c2b64e8a3db7f58cee55cd3ab75c40dcdbec09f118a4ee51ddf26ab50752"
},
"downloads": -1,
"filename": "django_unified_signals-0.2.1-py3-none-any.whl",
"has_sig": false,
"md5_digest": "da80fa2e073854500ab9f049576697b0",
"packagetype": "bdist_wheel",
"python_version": "py3",
"requires_python": "<4.0,>=3.9",
"size": 4306,
"upload_time": "2024-10-10T17:29:17",
"upload_time_iso_8601": "2024-10-10T17:29:17.681536Z",
"url": "https://files.pythonhosted.org/packages/52/c4/b0931453be880e391fd6d42d8f2207269458bf55e302d7d8c140c3b8ecb7/django_unified_signals-0.2.1-py3-none-any.whl",
"yanked": false,
"yanked_reason": null
},
{
"comment_text": "",
"digests": {
"blake2b_256": "1c3f8132c1e18dafa0f1b88d5738be1ad4197a1b19cb39cce84ea88c283b965b",
"md5": "98616e139004962031eb81ec0f604b54",
"sha256": "1350551e64662bbb3514464619b1d5808711aa60ad9f33f6bd0f3cb4e6e3fe5b"
},
"downloads": -1,
"filename": "django_unified_signals-0.2.1.tar.gz",
"has_sig": false,
"md5_digest": "98616e139004962031eb81ec0f604b54",
"packagetype": "sdist",
"python_version": "source",
"requires_python": "<4.0,>=3.9",
"size": 3781,
"upload_time": "2024-10-10T17:29:18",
"upload_time_iso_8601": "2024-10-10T17:29:18.673933Z",
"url": "https://files.pythonhosted.org/packages/1c/3f/8132c1e18dafa0f1b88d5738be1ad4197a1b19cb39cce84ea88c283b965b/django_unified_signals-0.2.1.tar.gz",
"yanked": false,
"yanked_reason": null
}
],
"upload_time": "2024-10-10 17:29:18",
"github": true,
"gitlab": false,
"bitbucket": false,
"codeberg": false,
"github_user": "ivellios",
"github_project": "django-unified-signals",
"travis_ci": false,
"coveralls": false,
"github_actions": true,
"tox": true,
"lcname": "django-unified-signals"
}