From 3297f9e1ad97da1e63ca827239663d6b33949037 Mon Sep 17 00:00:00 2001 From: Akis Kesoglou Date: Fri, 29 Aug 2014 17:01:21 +0300 Subject: [PATCH] [1.7.x] Fixed #23370 -- defer() + select_related() crashed with inherited models. Backport of 6613ea6e3f from master --- AUTHORS | 1 + django/db/models/query.py | 6 +++--- docs/releases/1.6.7.txt | 7 ++++-- tests/model_inheritance/tests.py | 37 ++++++++++++++++++++++++++++++++ 4 files changed, 46 insertions(+), 5 deletions(-) diff --git a/AUTHORS b/AUTHORS index 40196ef3b72..37fd21c3a4d 100644 --- a/AUTHORS +++ b/AUTHORS @@ -61,6 +61,7 @@ answer newbie questions, and generally made Django that much better: Mathieu Agopian Roberto Aguilar ajs + Akis Kesoglou alang@bright-green.com A S Alam Andi Albrecht diff --git a/django/db/models/query.py b/django/db/models/query.py index 30fd786a336..22919e504e0 100644 --- a/django/db/models/query.py +++ b/django/db/models/query.py @@ -1332,12 +1332,12 @@ def get_klass_info(klass, max_depth=0, cur_depth=0, requested=None, init_list = [] # Build the list of fields that *haven't* been requested for field, model in klass._meta.get_concrete_fields_with_model(): - if field.name not in load_fields: - skip.add(field.attname) - elif from_parent and issubclass(from_parent, model.__class__): + if from_parent and model and issubclass(from_parent, model): # Avoid loading fields already loaded for parent model for # child models. continue + elif field.name not in load_fields: + skip.add(field.attname) else: init_list.append(field.attname) # Retrieve all the requested fields diff --git a/docs/releases/1.6.7.txt b/docs/releases/1.6.7.txt index 5bb8f0f3528..479a0b9636f 100644 --- a/docs/releases/1.6.7.txt +++ b/docs/releases/1.6.7.txt @@ -4,10 +4,13 @@ Django 1.6.7 release notes *Under development* -Django 1.6.7 fixes a regression in the 1.6.6 security release. +Django 1.6.7 fixes several bugs in 1.6.6, including a regression related to +a security fix in that release. Bugfixes ======== * Allowed inherited and m2m fields to be referenced in the admin - :ticket:`23329` + (:ticket:`23329`). +* Fixed a crash when using ``QuerySet.defer()`` with ``select_related()`` + (:ticket:`23370`). diff --git a/tests/model_inheritance/tests.py b/tests/model_inheritance/tests.py index 0af96b820d0..77210174a0c 100644 --- a/tests/model_inheritance/tests.py +++ b/tests/model_inheritance/tests.py @@ -255,6 +255,43 @@ class ModelInheritanceTests(TestCase): 1, lambda: ItalianRestaurant.objects.select_related("chef")[0].chef ) + def test_select_related_defer(self): + """ + #23370 - Should be able to defer child fields when using + select_related() from parent to child. + """ + Restaurant.objects.create( + name="Demon Dogs", + address="944 W. Fullerton", + serves_hot_dogs=True, + serves_pizza=False, + rating=2, + ) + ItalianRestaurant.objects.create( + name="Ristorante Miron", + address="1234 W. Ash", + serves_hot_dogs=False, + serves_pizza=False, + serves_gnocchi=True, + rating=4, + ) + + qs = (Restaurant.objects + .select_related("italianrestaurant") + .defer("italianrestaurant__serves_gnocchi") + .order_by("rating")) + + # Test that the field was actually defered + with self.assertNumQueries(2): + objs = list(qs.all()) + self.assertTrue(objs[1].italianrestaurant.serves_gnocchi) + + # Test that model fields where assigned correct values + self.assertEqual(qs[0].name, 'Demon Dogs') + self.assertEqual(qs[0].rating, 2) + self.assertEqual(qs[1].italianrestaurant.name, 'Ristorante Miron') + self.assertEqual(qs[1].italianrestaurant.rating, 4) + def test_mixin_init(self): m = MixinModel() self.assertEqual(m.other_attr, 1)