django-bulk-hooks 0.1.54__tar.gz → 0.1.56__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of django-bulk-hooks might be problematic. Click here for more details.

Files changed (18) hide show
  1. {django_bulk_hooks-0.1.54 → django_bulk_hooks-0.1.56}/PKG-INFO +1 -1
  2. {django_bulk_hooks-0.1.54 → django_bulk_hooks-0.1.56}/django_bulk_hooks/handler.py +60 -5
  3. django_bulk_hooks-0.1.56/django_bulk_hooks/queryset.py +43 -0
  4. {django_bulk_hooks-0.1.54 → django_bulk_hooks-0.1.56}/pyproject.toml +1 -1
  5. django_bulk_hooks-0.1.54/django_bulk_hooks/queryset.py +0 -27
  6. {django_bulk_hooks-0.1.54 → django_bulk_hooks-0.1.56}/LICENSE +0 -0
  7. {django_bulk_hooks-0.1.54 → django_bulk_hooks-0.1.56}/README.md +0 -0
  8. {django_bulk_hooks-0.1.54 → django_bulk_hooks-0.1.56}/django_bulk_hooks/__init__.py +0 -0
  9. {django_bulk_hooks-0.1.54 → django_bulk_hooks-0.1.56}/django_bulk_hooks/conditions.py +0 -0
  10. {django_bulk_hooks-0.1.54 → django_bulk_hooks-0.1.56}/django_bulk_hooks/constants.py +0 -0
  11. {django_bulk_hooks-0.1.54 → django_bulk_hooks-0.1.56}/django_bulk_hooks/context.py +0 -0
  12. {django_bulk_hooks-0.1.54 → django_bulk_hooks-0.1.56}/django_bulk_hooks/decorators.py +0 -0
  13. {django_bulk_hooks-0.1.54 → django_bulk_hooks-0.1.56}/django_bulk_hooks/engine.py +0 -0
  14. {django_bulk_hooks-0.1.54 → django_bulk_hooks-0.1.56}/django_bulk_hooks/enums.py +0 -0
  15. {django_bulk_hooks-0.1.54 → django_bulk_hooks-0.1.56}/django_bulk_hooks/manager.py +0 -0
  16. {django_bulk_hooks-0.1.54 → django_bulk_hooks-0.1.56}/django_bulk_hooks/models.py +0 -0
  17. {django_bulk_hooks-0.1.54 → django_bulk_hooks-0.1.56}/django_bulk_hooks/priority.py +0 -0
  18. {django_bulk_hooks-0.1.54 → django_bulk_hooks-0.1.56}/django_bulk_hooks/registry.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.1
2
2
  Name: django-bulk-hooks
3
- Version: 0.1.54
3
+ Version: 0.1.56
4
4
  Summary: Lifecycle-style hooks for Django bulk operations like bulk_create and bulk_update.
5
5
  Home-page: https://github.com/AugendLimited/django-bulk-hooks
6
6
  License: MIT
@@ -7,7 +7,18 @@ from django_bulk_hooks.registry import get_hooks, register_hook
7
7
 
8
8
  logger = logging.getLogger(__name__)
9
9
 
10
- # Thread-local hook queue context
10
+ # Thread-local hook context and trigger state
11
+ class TriggerVars(threading.local):
12
+ def __init__(self):
13
+ self.new = None
14
+ self.old = None
15
+ self.event = None
16
+ self.model = None
17
+ self.depth = 0
18
+
19
+ trigger = TriggerVars()
20
+
21
+ # Hook queue per thread
11
22
  _hook_context = threading.local()
12
23
 
13
24
  def get_hook_queue():
@@ -15,6 +26,37 @@ def get_hook_queue():
15
26
  _hook_context.queue = deque()
16
27
  return _hook_context.queue
17
28
 
29
+ class TriggerContextState:
30
+ @property
31
+ def is_before(self):
32
+ return trigger.event.startswith("before_") if trigger.event else False
33
+
34
+ @property
35
+ def is_after(self):
36
+ return trigger.event.startswith("after_") if trigger.event else False
37
+
38
+ @property
39
+ def is_create(self):
40
+ return "create" in trigger.event if trigger.event else False
41
+
42
+ @property
43
+ def is_update(self):
44
+ return "update" in trigger.event if trigger.event else False
45
+
46
+ @property
47
+ def new(self):
48
+ return trigger.new
49
+
50
+ @property
51
+ def old(self):
52
+ return trigger.old
53
+
54
+ @property
55
+ def model(self):
56
+ return trigger.model
57
+
58
+ Trigger = TriggerContextState()
59
+
18
60
  class TriggerHandlerMeta(type):
19
61
  _registered = set()
20
62
 
@@ -67,6 +109,12 @@ class TriggerHandler(metaclass=TriggerHandlerMeta):
67
109
  old_records,
68
110
  **kwargs,
69
111
  ):
112
+ trigger.depth += 1
113
+ trigger.new = new_records
114
+ trigger.old = old_records
115
+ trigger.event = event
116
+ trigger.model = model
117
+
70
118
  hooks = sorted(get_hooks(model, event), key=lambda x: x[3])
71
119
  logger.debug("Processing %d hooks for %s.%s", len(hooks), model.__name__, event)
72
120
 
@@ -96,7 +144,14 @@ class TriggerHandler(metaclass=TriggerHandlerMeta):
96
144
  logger.exception("Error in hook %s.%s", handler_cls.__name__, method_name)
97
145
 
98
146
  conn = transaction.get_connection()
99
- if conn.in_atomic_block and event.startswith("after_"):
100
- transaction.on_commit(_execute)
101
- else:
102
- _execute()
147
+ try:
148
+ if conn.in_atomic_block and event.startswith("after_"):
149
+ transaction.on_commit(_execute)
150
+ else:
151
+ _execute()
152
+ finally:
153
+ trigger.new = None
154
+ trigger.old = None
155
+ trigger.event = None
156
+ trigger.model = None
157
+ trigger.depth -= 1
@@ -0,0 +1,43 @@
1
+ from django.db import models, transaction
2
+
3
+
4
+ class LifecycleQuerySet(models.QuerySet):
5
+ @transaction.atomic
6
+ def delete(self):
7
+ objs = list(self)
8
+ if not objs:
9
+ return 0
10
+ return self.model.objects.bulk_delete(objs)
11
+
12
+ @transaction.atomic
13
+ def update(self, **kwargs):
14
+ instances = list(self)
15
+ if not instances:
16
+ return 0
17
+
18
+ model_cls = self.model
19
+ pks = [obj.pk for obj in instances]
20
+
21
+ # Load originals for hook comparison
22
+ originals = list(model_cls.objects.filter(pk__in=pks))
23
+ originals_by_pk = {obj.pk: obj for obj in originals}
24
+
25
+ # Apply field updates to instances
26
+ for obj in instances:
27
+ for field, value in kwargs.items():
28
+ setattr(obj, field, value)
29
+
30
+ # Run BEFORE_UPDATE hooks
31
+ from django_bulk_hooks.context import TriggerContext
32
+ from django_bulk_hooks import engine
33
+ ctx = TriggerContext(model_cls)
34
+ engine.run(model_cls, "before_update", instances, originals, ctx=ctx)
35
+
36
+ # Use Django's built-in update logic directly
37
+ update_count = super().update(**kwargs)
38
+
39
+ # Run AFTER_UPDATE hooks
40
+ engine.run(model_cls, "after_update", instances, originals, ctx=ctx)
41
+
42
+ return update_count
43
+
@@ -1,6 +1,6 @@
1
1
  [tool.poetry]
2
2
  name = "django-bulk-hooks"
3
- version = "0.1.54"
3
+ version = "0.1.56"
4
4
  description = "Lifecycle-style hooks for Django bulk operations like bulk_create and bulk_update."
5
5
  authors = ["Konrad Beck <konrad.beck@merchantcapital.co.za>"]
6
6
  readme = "README.md"
@@ -1,27 +0,0 @@
1
- from django.db import models, transaction
2
-
3
-
4
- class LifecycleQuerySet(models.QuerySet):
5
- @transaction.atomic
6
- def delete(self):
7
- objs = list(self)
8
- if not objs:
9
- return 0
10
- return self.model.objects.bulk_delete(objs)
11
-
12
- @transaction.atomic
13
- def update(self, **kwargs):
14
- instances = list(self)
15
- if not instances:
16
- return 0
17
-
18
- for obj in instances:
19
- for field, value in kwargs.items():
20
- setattr(obj, field, value)
21
-
22
- self.model.objects.bulk_update(
23
- instances,
24
- fields=list(kwargs.keys()),
25
- )
26
-
27
- return len(instances)