|
7 | 7 | from __future__ import absolute_import |
8 | 8 |
|
9 | 9 | from django.db import models |
| 10 | +from django.apps import apps |
10 | 11 | from django.dispatch import Signal |
11 | | - |
12 | 12 | from .registries import registry |
13 | | - |
| 13 | +from django.core.exceptions import ObjectDoesNotExist |
| 14 | +from importlib import import_module |
| 15 | +# Sent after document indexing is completed |
| 16 | +post_index = Signal() |
14 | 17 |
|
15 | 18 | class BaseSignalProcessor(object): |
16 | 19 | """Base signal processor. |
@@ -96,6 +99,124 @@ def teardown(self): |
96 | 99 | models.signals.m2m_changed.disconnect(self.handle_m2m_changed) |
97 | 100 | models.signals.pre_delete.disconnect(self.handle_pre_delete) |
98 | 101 |
|
| 102 | +try: |
| 103 | + from celery import shared_task |
| 104 | +except ImportError: |
| 105 | + pass |
| 106 | +else: |
| 107 | + class CelerySignalProcessor(RealTimeSignalProcessor): |
| 108 | + """Celery signal processor. |
| 109 | +
|
| 110 | + Allows automatic updates on the index as delayed background tasks using |
| 111 | + Celery. |
| 112 | +
|
| 113 | + NB: We cannot process deletes as background tasks. |
| 114 | + By the time the Celery worker would pick up the delete job, the |
| 115 | + model instance would already deleted. We can get around this by |
| 116 | + setting Celery to use `pickle` and sending the object to the worker, |
| 117 | + but using `pickle` opens the application up to security concerns. |
| 118 | + """ |
99 | 119 |
|
100 | | -# Sent after document indexing is completed |
101 | | -post_index = Signal() |
| 120 | + def handle_save(self, sender, instance, **kwargs): |
| 121 | + """Handle save with a Celery task. |
| 122 | +
|
| 123 | + Given an individual model instance, update the object in the index. |
| 124 | + Update the related objects either. |
| 125 | + """ |
| 126 | + pk = instance.pk |
| 127 | + app_label = instance._meta.app_label |
| 128 | + model_name = instance.__class__.__name__ |
| 129 | + |
| 130 | + self.registry_update_task.delay(pk, app_label, model_name) |
| 131 | + self.registry_update_related_task.delay(pk, app_label, model_name) |
| 132 | + |
| 133 | + def handle_pre_delete(self, sender, instance, **kwargs): |
| 134 | + """Handle removing of instance object from related models instance. |
| 135 | + We need to do this before the real delete otherwise the relation |
| 136 | + doesn't exists anymore and we can't get the related models instance. |
| 137 | + """ |
| 138 | + self.prepare_registry_delete_related_task(instance) |
| 139 | + |
| 140 | + def handle_delete(self, sender, instance, **kwargs): |
| 141 | + """Handle delete. |
| 142 | +
|
| 143 | + Given an individual model instance, delete the object from index. |
| 144 | + """ |
| 145 | + self.prepare_registry_delete_task(instance) |
| 146 | + |
| 147 | + def prepare_registry_delete_related_task(self, instance): |
| 148 | + """ |
| 149 | + Select its related instance before this instance was deleted. |
| 150 | + And pass that to celery. |
| 151 | + """ |
| 152 | + action = 'index' |
| 153 | + for doc in registry._get_related_doc(instance): |
| 154 | + doc_instance = doc(related_instance_to_ignore=instance) |
| 155 | + try: |
| 156 | + related = doc_instance.get_instances_from_related(instance) |
| 157 | + except ObjectDoesNotExist: |
| 158 | + related = None |
| 159 | + if related is not None: |
| 160 | + doc_instance.update(related) |
| 161 | + if isinstance(related, models.Model): |
| 162 | + object_list = [related] |
| 163 | + else: |
| 164 | + object_list = related |
| 165 | + bulk_data = list(doc_instance._get_actions(object_list, action)), |
| 166 | + self.registry_delete_task.delay(doc_instance.__class__.__name__, bulk_data) |
| 167 | + |
| 168 | + @shared_task() |
| 169 | + def registry_delete_task(doc_label, data): |
| 170 | + """ |
| 171 | + Handle the bulk delete data on the registry as a Celery task. |
| 172 | + The different implementations used are due to the difference between delete and update operations. |
| 173 | + The update operation can re-read the updated data from the database to ensure eventual consistency, |
| 174 | + but the delete needs to be processed before the database record is deleted to obtain the associated data. |
| 175 | + """ |
| 176 | + doc_instance = import_module(doc_label) |
| 177 | + parallel = True |
| 178 | + doc_instance._bulk(bulk_data, parallel=parallel) |
| 179 | + |
| 180 | + def prepare_registry_delete_task(self, instance): |
| 181 | + """ |
| 182 | + Get the prepare did before database record deleted. |
| 183 | + """ |
| 184 | + action = 'delete' |
| 185 | + for doc in registry._get_related_doc(instance): |
| 186 | + doc_instance = doc(related_instance_to_ignore=instance) |
| 187 | + try: |
| 188 | + related = doc_instance.get_instances_from_related(instance) |
| 189 | + except ObjectDoesNotExist: |
| 190 | + related = None |
| 191 | + if related is not None: |
| 192 | + doc_instance.update(related) |
| 193 | + if isinstance(related, models.Model): |
| 194 | + object_list = [related] |
| 195 | + else: |
| 196 | + object_list = related |
| 197 | + bulk_data = list(doc_instance.get_actions(object_list, action)), |
| 198 | + self.registry_delete_task.delay(doc_instance.__class__.__name__, bulk_data) |
| 199 | + |
| 200 | + @shared_task() |
| 201 | + def registry_update_task(pk, app_label, model_name): |
| 202 | + """Handle the update on the registry as a Celery task.""" |
| 203 | + try: |
| 204 | + model = apps.get_model(app_label, model_name) |
| 205 | + except LookupError: |
| 206 | + pass |
| 207 | + else: |
| 208 | + registry.update( |
| 209 | + model.objects.get(pk=pk) |
| 210 | + ) |
| 211 | + |
| 212 | + @shared_task() |
| 213 | + def registry_update_related_task(pk, app_label, model_name): |
| 214 | + """Handle the related update on the registry as a Celery task.""" |
| 215 | + try: |
| 216 | + model = apps.get_model(app_label, model_name) |
| 217 | + except LookupError: |
| 218 | + pass |
| 219 | + else: |
| 220 | + registry.update_related( |
| 221 | + model.objects.get(pk=pk) |
| 222 | + ) |
0 commit comments