From be9d645346a20a6c394bf70d47b1b1d5c81ff530 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Anssi=20K=C3=A4=C3=A4ri=C3=A4inen?= Date: Mon, 11 May 2015 10:02:41 +0300 Subject: [PATCH] Fixed #24766 -- Added join promotion for Case expressions --- django/db/models/query_utils.py | 7 ++++++- docs/releases/1.8.2.txt | 4 ++++ tests/expressions_case/tests.py | 32 ++++++++++++++++++++++++++++++++ 3 files changed, 42 insertions(+), 1 deletion(-) diff --git a/django/db/models/query_utils.py b/django/db/models/query_utils.py index 1c7ed08106..b1eac58bd9 100644 --- a/django/db/models/query_utils.py +++ b/django/db/models/query_utils.py @@ -87,7 +87,12 @@ class Q(tree.Node): return clone def resolve_expression(self, query=None, allow_joins=True, reuse=None, summarize=False, for_save=False): - clause, _ = query._add_q(self, reuse, allow_joins=allow_joins) + # We must promote any new joins to left outer joins so that when Q is + # used as an expression, rows aren't filtered due to joins. + joins_before = query.tables[:] + clause, joins = query._add_q(self, reuse, allow_joins=allow_joins) + joins_to_promote = [j for j in joins if j not in joins_before] + query.promote_joins(joins_to_promote) return clause @classmethod diff --git a/docs/releases/1.8.2.txt b/docs/releases/1.8.2.txt index a0c18e78bb..f430eb6cb0 100644 --- a/docs/releases/1.8.2.txt +++ b/docs/releases/1.8.2.txt @@ -13,3 +13,7 @@ Bugfixes * Fixed crash when reusing the same ``Case`` instance in a query (:ticket:`24752`). + +* Corrected join promotion for ``Case`` expressions. For example, annotating a + query with a ``Case`` expression could unexpectedly filter out results + (:ticket:`24766`). diff --git a/tests/expressions_case/tests.py b/tests/expressions_case/tests.py index 2c3186f10e..3479932865 100644 --- a/tests/expressions_case/tests.py +++ b/tests/expressions_case/tests.py @@ -1016,6 +1016,38 @@ class CaseExpressionTests(TestCase): transform=attrgetter('integer', 'test') ) + def test_join_promotion(self): + o = CaseTestModel.objects.create(integer=1, integer2=1, string='1') + # Testing that: + # 1. There isn't any object on the remote side of the fk_rel + # relation. If the query used inner joins, then the join to fk_rel + # would remove o from the results. So, in effect we are testing that + # we are promoting the fk_rel join to a left outer join here. + # 2. The default value of 3 is generated for the case expression. + self.assertQuerysetEqual( + CaseTestModel.objects.filter(pk=o.pk).annotate( + foo=Case( + When(fk_rel__pk=1, then=2), + default=3, + output_field=models.IntegerField() + ), + ), + [(o, 3)], + lambda x: (x, x.foo) + ) + # Now 2 should be generated, as the fk_rel is null. + self.assertQuerysetEqual( + CaseTestModel.objects.filter(pk=o.pk).annotate( + foo=Case( + When(fk_rel__isnull=True, then=2), + default=3, + output_field=models.IntegerField() + ), + ), + [(o, 2)], + lambda x: (x, x.foo) + ) + class CaseDocumentationExamples(TestCase): @classmethod