diff --git a/django/db/models/sql/where.py b/django/db/models/sql/where.py index 160d5733b84..50ff13be75e 100644 --- a/django/db/models/sql/where.py +++ b/django/db/models/sql/where.py @@ -148,7 +148,11 @@ class WhereNode(tree.Node): clone = self.__class__._new_instance( children=None, connector=self.connector, negated=self.negated, ) - clone.children = self.children[:] + for child in self.children: + if hasattr(child, 'clone'): + clone.children.append(child.clone()) + else: + clone.children.append(child) return clone def relabeled_clone(self, change_map): diff --git a/tests/queries/tests.py b/tests/queries/tests.py index fa87e7859c1..9c1b41a395d 100644 --- a/tests/queries/tests.py +++ b/tests/queries/tests.py @@ -1663,6 +1663,12 @@ class Queries5Tests(TestCase): 'bar %s' ) + def test_queryset_reuse(self): + # Using querysets doesn't mutate aliases. + authors = Author.objects.filter(Q(name='a1') | Q(name='nonexistent')) + self.assertEqual(Ranking.objects.filter(author__in=authors).get(), self.rank3) + self.assertEqual(authors.count(), 1) + class SelectRelatedTests(TestCase): def test_tickets_3045_3288(self):