2013-07-29 19:19:04 +02:00
|
|
|
from __future__ import unicode_literals
|
2011-10-13 21:34:56 +00:00
|
|
|
|
2013-12-24 12:25:17 +01:00
|
|
|
from django.apps import apps
|
2012-10-06 14:21:57 +08:00
|
|
|
from django.db import models
|
2013-09-30 10:21:21 +07:00
|
|
|
from django.template import Context, Template
|
2013-12-23 16:01:13 +01:00
|
|
|
from django.test import TestCase, override_settings
|
2013-09-30 10:21:21 +07:00
|
|
|
from django.utils.encoding import force_text
|
2010-10-14 01:17:14 +00:00
|
|
|
|
2012-10-06 14:21:57 +08:00
|
|
|
from .models import (
|
2015-02-09 13:19:34 -05:00
|
|
|
AbstractBase1, AbstractBase2, AbstractBase3, Child1, Child2, Child3,
|
|
|
|
Child4, Child5, Child6, Child7, RelatedModel, RelationModel,
|
2012-10-06 14:21:57 +08:00
|
|
|
)
|
2010-10-14 01:17:14 +00:00
|
|
|
|
|
|
|
|
|
|
|
class ManagersRegressionTests(TestCase):
|
|
|
|
def test_managers(self):
|
2012-10-06 14:21:57 +08:00
|
|
|
Child1.objects.create(name='fred', data='a1')
|
|
|
|
Child1.objects.create(name='barney', data='a2')
|
|
|
|
Child2.objects.create(name='fred', data='b1', value=1)
|
|
|
|
Child2.objects.create(name='barney', data='b2', value=42)
|
|
|
|
Child3.objects.create(name='fred', data='c1', comment='yes')
|
|
|
|
Child3.objects.create(name='barney', data='c2', comment='no')
|
|
|
|
Child4.objects.create(name='fred', data='d1')
|
|
|
|
Child4.objects.create(name='barney', data='d2')
|
|
|
|
Child5.objects.create(name='fred', comment='yes')
|
|
|
|
Child5.objects.create(name='barney', comment='no')
|
|
|
|
Child6.objects.create(name='fred', data='f1', value=42)
|
|
|
|
Child6.objects.create(name='barney', data='f2', value=42)
|
|
|
|
Child7.objects.create(name='fred')
|
|
|
|
Child7.objects.create(name='barney')
|
2010-10-14 01:17:14 +00:00
|
|
|
|
|
|
|
self.assertQuerysetEqual(Child1.manager1.all(), ["<Child1: a1>"])
|
|
|
|
self.assertQuerysetEqual(Child1.manager2.all(), ["<Child1: a2>"])
|
|
|
|
self.assertQuerysetEqual(Child1._default_manager.all(), ["<Child1: a1>"])
|
|
|
|
|
|
|
|
self.assertQuerysetEqual(Child2._default_manager.all(), ["<Child2: b1>"])
|
|
|
|
self.assertQuerysetEqual(Child2.restricted.all(), ["<Child2: b2>"])
|
|
|
|
|
|
|
|
self.assertQuerysetEqual(Child3._default_manager.all(), ["<Child3: c1>"])
|
|
|
|
self.assertQuerysetEqual(Child3.manager1.all(), ["<Child3: c1>"])
|
|
|
|
self.assertQuerysetEqual(Child3.manager2.all(), ["<Child3: c2>"])
|
|
|
|
|
|
|
|
# Since Child6 inherits from Child4, the corresponding rows from f1 and
|
|
|
|
# f2 also appear here. This is the expected result.
|
|
|
|
self.assertQuerysetEqual(Child4._default_manager.order_by('data'), [
|
2013-10-20 00:33:10 +01:00
|
|
|
"<Child4: d1>",
|
|
|
|
"<Child4: d2>",
|
|
|
|
"<Child4: f1>",
|
|
|
|
"<Child4: f2>"
|
2013-10-18 10:02:43 +01:00
|
|
|
]
|
2010-10-14 01:17:14 +00:00
|
|
|
)
|
|
|
|
self.assertQuerysetEqual(Child4.manager1.all(), [
|
2013-10-20 00:33:10 +01:00
|
|
|
"<Child4: d1>",
|
|
|
|
"<Child4: f1>"
|
2013-10-18 10:02:43 +01:00
|
|
|
],
|
2012-12-13 13:33:11 +02:00
|
|
|
ordered=False
|
2010-10-14 01:17:14 +00:00
|
|
|
)
|
|
|
|
self.assertQuerysetEqual(Child5._default_manager.all(), ["<Child5: fred>"])
|
|
|
|
self.assertQuerysetEqual(Child6._default_manager.all(), ["<Child6: f1>"])
|
|
|
|
self.assertQuerysetEqual(Child7._default_manager.order_by('name'), [
|
2013-10-20 00:33:10 +01:00
|
|
|
"<Child7: barney>",
|
|
|
|
"<Child7: fred>"
|
2013-10-18 10:02:43 +01:00
|
|
|
]
|
2010-10-14 01:17:14 +00:00
|
|
|
)
|
2012-10-06 14:21:57 +08:00
|
|
|
|
|
|
|
def test_abstract_manager(self):
|
|
|
|
# Accessing the manager on an abstract model should
|
|
|
|
# raise an attribute error with an appropriate message.
|
|
|
|
try:
|
|
|
|
AbstractBase3.objects.all()
|
|
|
|
self.fail('Should raise an AttributeError')
|
|
|
|
except AttributeError as e:
|
|
|
|
# This error message isn't ideal, but if the model is abstract and
|
|
|
|
# a lot of the class instantiation logic isn't invoked; if the
|
|
|
|
# manager is implied, then we don't get a hook to install the
|
|
|
|
# error-raising manager.
|
|
|
|
self.assertEqual(str(e), "type object 'AbstractBase3' has no attribute 'objects'")
|
|
|
|
|
|
|
|
def test_custom_abstract_manager(self):
|
|
|
|
# Accessing the manager on an abstract model with an custom
|
|
|
|
# manager should raise an attribute error with an appropriate
|
|
|
|
# message.
|
|
|
|
try:
|
|
|
|
AbstractBase2.restricted.all()
|
|
|
|
self.fail('Should raise an AttributeError')
|
|
|
|
except AttributeError as e:
|
|
|
|
self.assertEqual(str(e), "Manager isn't available; AbstractBase2 is abstract")
|
|
|
|
|
|
|
|
def test_explicit_abstract_manager(self):
|
|
|
|
# Accessing the manager on an abstract model with an explicit
|
|
|
|
# manager should raise an attribute error with an appropriate
|
|
|
|
# message.
|
|
|
|
try:
|
|
|
|
AbstractBase1.objects.all()
|
|
|
|
self.fail('Should raise an AttributeError')
|
|
|
|
except AttributeError as e:
|
|
|
|
self.assertEqual(str(e), "Manager isn't available; AbstractBase1 is abstract")
|
|
|
|
|
2013-10-14 15:14:17 +02:00
|
|
|
@override_settings(TEST_SWAPPABLE_MODEL='managers_regress.Parent')
|
2012-10-06 14:21:57 +08:00
|
|
|
def test_swappable_manager(self):
|
2013-12-12 21:34:39 +01:00
|
|
|
# The models need to be removed after the test in order to prevent bad
|
|
|
|
# interactions with the flush operation in other tests.
|
2013-12-24 12:25:17 +01:00
|
|
|
_old_models = apps.app_configs['managers_regress'].models.copy()
|
2012-10-06 14:21:57 +08:00
|
|
|
|
2013-12-12 21:34:39 +01:00
|
|
|
try:
|
2012-10-06 14:21:57 +08:00
|
|
|
class SwappableModel(models.Model):
|
|
|
|
class Meta:
|
|
|
|
swappable = 'TEST_SWAPPABLE_MODEL'
|
|
|
|
|
|
|
|
# Accessing the manager on a swappable model should
|
|
|
|
# raise an attribute error with a helpful message
|
|
|
|
try:
|
|
|
|
SwappableModel.objects.all()
|
|
|
|
self.fail('Should raise an AttributeError')
|
|
|
|
except AttributeError as e:
|
|
|
|
self.assertEqual(str(e), "Manager isn't available; SwappableModel has been swapped for 'managers_regress.Parent'")
|
|
|
|
|
|
|
|
finally:
|
2013-12-24 12:25:17 +01:00
|
|
|
apps.app_configs['managers_regress'].models = _old_models
|
|
|
|
apps.all_models['managers_regress'] = _old_models
|
2013-12-29 21:24:40 +01:00
|
|
|
apps.clear_cache()
|
2012-10-06 14:21:57 +08:00
|
|
|
|
2013-10-14 15:14:17 +02:00
|
|
|
@override_settings(TEST_SWAPPABLE_MODEL='managers_regress.Parent')
|
2012-10-06 14:21:57 +08:00
|
|
|
def test_custom_swappable_manager(self):
|
2013-12-12 21:34:39 +01:00
|
|
|
# The models need to be removed after the test in order to prevent bad
|
|
|
|
# interactions with the flush operation in other tests.
|
2013-12-24 12:25:17 +01:00
|
|
|
_old_models = apps.app_configs['managers_regress'].models.copy()
|
2012-10-06 14:21:57 +08:00
|
|
|
|
2013-12-12 21:34:39 +01:00
|
|
|
try:
|
2012-10-06 14:21:57 +08:00
|
|
|
class SwappableModel(models.Model):
|
|
|
|
|
|
|
|
stuff = models.Manager()
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
swappable = 'TEST_SWAPPABLE_MODEL'
|
|
|
|
|
|
|
|
# Accessing the manager on a swappable model with an
|
|
|
|
# explicit manager should raise an attribute error with a
|
|
|
|
# helpful message
|
|
|
|
try:
|
|
|
|
SwappableModel.stuff.all()
|
|
|
|
self.fail('Should raise an AttributeError')
|
|
|
|
except AttributeError as e:
|
|
|
|
self.assertEqual(str(e), "Manager isn't available; SwappableModel has been swapped for 'managers_regress.Parent'")
|
|
|
|
|
|
|
|
finally:
|
2013-12-24 12:25:17 +01:00
|
|
|
apps.app_configs['managers_regress'].models = _old_models
|
|
|
|
apps.all_models['managers_regress'] = _old_models
|
2013-12-29 21:24:40 +01:00
|
|
|
apps.clear_cache()
|
2012-10-06 14:21:57 +08:00
|
|
|
|
2013-10-14 15:14:17 +02:00
|
|
|
@override_settings(TEST_SWAPPABLE_MODEL='managers_regress.Parent')
|
2012-10-06 14:21:57 +08:00
|
|
|
def test_explicit_swappable_manager(self):
|
2013-12-12 21:34:39 +01:00
|
|
|
# The models need to be removed after the test in order to prevent bad
|
|
|
|
# interactions with the flush operation in other tests.
|
2013-12-24 12:25:17 +01:00
|
|
|
_old_models = apps.app_configs['managers_regress'].models.copy()
|
2012-10-06 14:21:57 +08:00
|
|
|
|
2013-12-12 21:34:39 +01:00
|
|
|
try:
|
2012-10-06 14:21:57 +08:00
|
|
|
class SwappableModel(models.Model):
|
|
|
|
|
|
|
|
objects = models.Manager()
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
swappable = 'TEST_SWAPPABLE_MODEL'
|
|
|
|
|
|
|
|
# Accessing the manager on a swappable model with an
|
|
|
|
# explicit manager should raise an attribute error with a
|
|
|
|
# helpful message
|
|
|
|
try:
|
|
|
|
SwappableModel.objects.all()
|
|
|
|
self.fail('Should raise an AttributeError')
|
|
|
|
except AttributeError as e:
|
|
|
|
self.assertEqual(str(e), "Manager isn't available; SwappableModel has been swapped for 'managers_regress.Parent'")
|
|
|
|
|
|
|
|
finally:
|
2013-12-24 12:25:17 +01:00
|
|
|
apps.app_configs['managers_regress'].models = _old_models
|
|
|
|
apps.all_models['managers_regress'] = _old_models
|
2013-12-29 21:24:40 +01:00
|
|
|
apps.clear_cache()
|
2013-09-30 10:21:21 +07:00
|
|
|
|
|
|
|
def test_regress_3871(self):
|
|
|
|
related = RelatedModel.objects.create()
|
|
|
|
|
|
|
|
relation = RelationModel()
|
|
|
|
relation.fk = related
|
|
|
|
relation.gfk = related
|
|
|
|
relation.save()
|
|
|
|
relation.m2m.add(related)
|
|
|
|
|
|
|
|
t = Template('{{ related.test_fk.all.0 }}{{ related.test_gfk.all.0 }}{{ related.test_m2m.all.0 }}')
|
|
|
|
|
|
|
|
self.assertEqual(
|
|
|
|
t.render(Context({'related': related})),
|
|
|
|
''.join([force_text(relation.pk)] * 3),
|
|
|
|
)
|
2015-01-22 01:43:49 -05:00
|
|
|
|
|
|
|
def test_field_can_be_called_exact(self):
|
|
|
|
# Make sure related managers core filters don't include an
|
|
|
|
# explicit `__exact` lookup that could be interpreted as a
|
|
|
|
# reference to a foreign `exact` field. refs #23940.
|
|
|
|
related = RelatedModel.objects.create(exact=False)
|
|
|
|
relation = related.test_fk.create()
|
|
|
|
self.assertEqual(related.test_fk.get(), relation)
|