Skip to content

Commit e93f56b

Browse files
committed
Fixed #13227 -- Ensure that the query cache is flushed when a QuerySet is deepcopied, avoiding problems when an evaluated queryset is used as a subquery. Thanks to claudep for the report.
git-svn-id: http://code.djangoproject.com/svn/django/trunk@12970 bcc190cf-cafb-0310-a4f2-bffc1f526a37
1 parent b3390fe commit e93f56b

File tree

2 files changed

+24
-5
lines changed

2 files changed

+24
-5
lines changed

django/db/models/query.py

Lines changed: 5 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -45,11 +45,12 @@ def __deepcopy__(self, memo):
4545
"""
4646
Deep copy of a QuerySet doesn't populate the cache
4747
"""
48-
obj_dict = deepcopy(self.__dict__, memo)
49-
obj_dict['_iter'] = None
50-
5148
obj = self.__class__()
52-
obj.__dict__.update(obj_dict)
49+
for k,v in self.__dict__.items():
50+
if k in ('_iter','_result_cache'):
51+
obj.__dict__[k] = None
52+
else:
53+
obj.__dict__[k] = deepcopy(v, memo)
5354
return obj
5455

5556
def __getstate__(self):

tests/regressiontests/queries/tests.py

Lines changed: 19 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@
44
from django.db.models import Count
55
from django.test import TestCase
66

7-
from models import Tag, Annotation, DumbCategory
7+
from models import Tag, Annotation, DumbCategory, Note, ExtraInfo
88

99
class QuerysetOrderedTests(unittest.TestCase):
1010
"""
@@ -63,3 +63,21 @@ def test_sliced_delete(self):
6363
# This prevents us from even evaluating this test case at all.
6464
# Refs #10099
6565
self.assertFalse(connections[DEFAULT_DB_ALIAS].features.allow_sliced_subqueries)
66+
67+
class CloneTests(TestCase):
68+
def test_evaluated_queryset_as_argument(self):
69+
"#13227 -- If a queryset is already evaluated, it can still be used as a query arg"
70+
n = Note(note='Test1', misc='misc')
71+
n.save()
72+
e = ExtraInfo(info='good', note=n)
73+
e.save()
74+
75+
n_list = Note.objects.all()
76+
# Evaluate the Note queryset, populating the query cache
77+
list(n_list)
78+
# Use the note queryset in a query, and evalute
79+
# that query in a way that involves cloning.
80+
try:
81+
self.assertEquals(ExtraInfo.objects.filter(note__in=n_list)[0].info, 'good')
82+
except:
83+
self.fail('Query should be clonable')

0 commit comments

Comments
 (0)