Fixed #26686 -- Fixed crash when registering model signals with abstract senders.
This commit is contained in:
parent
2c90981c5f
commit
08014fe75b
|
@ -3,6 +3,7 @@ from functools import partial
|
||||||
|
|
||||||
from django.db.models.utils import make_model_tuple
|
from django.db.models.utils import make_model_tuple
|
||||||
from django.dispatch import Signal
|
from django.dispatch import Signal
|
||||||
|
from django.utils import six
|
||||||
from django.utils.deprecation import RemovedInDjango20Warning
|
from django.utils.deprecation import RemovedInDjango20Warning
|
||||||
|
|
||||||
|
|
||||||
|
@ -15,15 +16,15 @@ class ModelSignal(Signal):
|
||||||
of the `app_label.ModelName` form.
|
of the `app_label.ModelName` form.
|
||||||
"""
|
"""
|
||||||
def _lazy_method(self, method, apps, receiver, sender, **kwargs):
|
def _lazy_method(self, method, apps, receiver, sender, **kwargs):
|
||||||
|
from django.db.models.options import Options
|
||||||
|
|
||||||
# This partial takes a single optional argument named "sender".
|
# This partial takes a single optional argument named "sender".
|
||||||
partial_method = partial(method, receiver, **kwargs)
|
partial_method = partial(method, receiver, **kwargs)
|
||||||
# import models here to avoid a circular import
|
if isinstance(sender, six.string_types):
|
||||||
from django.db import models
|
apps = apps or Options.default_apps
|
||||||
if isinstance(sender, models.Model) or sender is None:
|
|
||||||
# Skip lazy_model_operation to get a return value for disconnect()
|
|
||||||
return partial_method(sender)
|
|
||||||
apps = apps or models.base.Options.default_apps
|
|
||||||
apps.lazy_model_operation(partial_method, make_model_tuple(sender))
|
apps.lazy_model_operation(partial_method, make_model_tuple(sender))
|
||||||
|
else:
|
||||||
|
return partial_method(sender)
|
||||||
|
|
||||||
def connect(self, receiver, sender=None, weak=True, dispatch_uid=None, apps=None):
|
def connect(self, receiver, sender=None, weak=True, dispatch_uid=None, apps=None):
|
||||||
self._lazy_method(super(ModelSignal, self).connect, apps, receiver, sender, dispatch_uid=dispatch_uid)
|
self._lazy_method(super(ModelSignal, self).connect, apps, receiver, sender, dispatch_uid=dispatch_uid)
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
from __future__ import unicode_literals
|
from __future__ import unicode_literals
|
||||||
|
|
||||||
|
from django.apps.registry import Apps
|
||||||
from django.db import models
|
from django.db import models
|
||||||
from django.db.models import signals
|
from django.db.models import signals
|
||||||
from django.dispatch import receiver
|
from django.dispatch import receiver
|
||||||
|
@ -317,3 +318,14 @@ class LazyModelRefTest(BaseSignalTest):
|
||||||
|
|
||||||
Created()
|
Created()
|
||||||
self.assertEqual(received, [])
|
self.assertEqual(received, [])
|
||||||
|
|
||||||
|
def test_register_model_class_senders_immediately(self):
|
||||||
|
"""
|
||||||
|
Model signals registered with model classes as senders don't use the
|
||||||
|
Apps.lazy_model_operation() mechanism.
|
||||||
|
"""
|
||||||
|
# Book isn't registered with apps2, so it will linger in
|
||||||
|
# apps2._pending_operations if ModelSignal does the wrong thing.
|
||||||
|
apps2 = Apps()
|
||||||
|
signals.post_init.connect(self.receiver, sender=Book, apps=apps2)
|
||||||
|
self.assertEqual(list(apps2._pending_operations), [])
|
||||||
|
|
Loading…
Reference in New Issue