diff --git a/django/db/models/expressions.py b/django/db/models/expressions.py index c6959f87b8..43061933e6 100644 --- a/django/db/models/expressions.py +++ b/django/db/models/expressions.py @@ -780,6 +780,11 @@ class Case(Expression): c.default = c.default.resolve_expression(query, allow_joins, reuse, summarize, for_save) return c + def copy(self): + c = super(Case, self).copy() + c.cases = c.cases[:] + return c + def as_sql(self, compiler, connection, template=None, extra=None): connection.ops.check_expression_support(self) if not self.cases: diff --git a/docs/releases/1.8.2.txt b/docs/releases/1.8.2.txt index c66539c9b7..a0c18e78bb 100644 --- a/docs/releases/1.8.2.txt +++ b/docs/releases/1.8.2.txt @@ -10,3 +10,6 @@ Bugfixes ======== * Fixed check for template engine alias uniqueness (:ticket:`24685`). + +* Fixed crash when reusing the same ``Case`` instance in a query + (:ticket:`24752`). diff --git a/tests/expressions_case/tests.py b/tests/expressions_case/tests.py index 1ebaf6d220..5eacd1cf9f 100644 --- a/tests/expressions_case/tests.py +++ b/tests/expressions_case/tests.py @@ -274,6 +274,18 @@ class CaseExpressionTests(TestCase): transform=attrgetter('integer', 'integer2') ) + def test_case_reuse(self): + SOME_CASE = Case( + When(pk=0, then=Value('0')), + default=Value('1'), + output_field=models.CharField(), + ) + self.assertQuerysetEqual( + CaseTestModel.objects.annotate(somecase=SOME_CASE).order_by('pk'), + CaseTestModel.objects.annotate(somecase=SOME_CASE).order_by('pk').values_list('pk', 'somecase'), + lambda x: (x.pk, x.somecase) + ) + def test_aggregate(self): self.assertEqual( CaseTestModel.objects.aggregate(