mirror of
				https://github.com/django/django.git
				synced 2025-10-24 22:26:08 +00:00 
			
		
		
		
	Fixed #11082 -- Ensured that subqueries used in an exclude(X__in=) clause aren't pre-evaluated. Thanks to Henry Andrews for the report, and clement for the fix.
git-svn-id: http://code.djangoproject.com/svn/django/trunk@10929 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
		| @@ -7,6 +7,8 @@ try: | ||||
| except NameError: | ||||
|     from sets import Set as set     # Python 2.3 fallback | ||||
|  | ||||
| from copy import deepcopy | ||||
|  | ||||
| from django.db import connection, transaction, IntegrityError | ||||
| from django.db.models.aggregates import Aggregate | ||||
| from django.db.models.fields import DateField | ||||
| @@ -40,6 +42,17 @@ class QuerySet(object): | ||||
|     # PYTHON MAGIC METHODS # | ||||
|     ######################## | ||||
|  | ||||
|     def __deepcopy__(self, memo): | ||||
|         """ | ||||
|         Deep copy of a QuerySet doesn't populate the cache | ||||
|         """ | ||||
|         obj_dict = deepcopy(self.__dict__, memo) | ||||
|         obj_dict['_iter'] = None | ||||
|  | ||||
|         obj = self.__class__() | ||||
|         obj.__dict__.update(obj_dict) | ||||
|         return obj | ||||
|  | ||||
|     def __getstate__(self): | ||||
|         """ | ||||
|         Allows the QuerySet to be pickled. | ||||
|   | ||||
| @@ -1625,10 +1625,14 @@ class BaseQuery(object): | ||||
|                             entry.negate() | ||||
|                             self.where.add(entry, AND) | ||||
|                             break | ||||
|                 elif not (lookup_type == 'in' and not value) and field.null: | ||||
|                 elif not (lookup_type == 'in' | ||||
|                             and not hasattr(value, 'as_sql') | ||||
|                             and not hasattr(value, '_as_sql') | ||||
|                             and not value) and field.null: | ||||
|                     # Leaky abstraction artifact: We have to specifically | ||||
|                     # exclude the "foo__in=[]" case from this handling, because | ||||
|                     # it's short-circuited in the Where class. | ||||
|                     # We also need to handle the case where a subquery is provided | ||||
|                     entry = self.where_class() | ||||
|                     entry.add((Constraint(alias, col, None), 'isnull', True), AND) | ||||
|                     entry.negate() | ||||
|   | ||||
| @@ -1143,6 +1143,33 @@ True | ||||
| >>> r.save() | ||||
| >>> Ranking.objects.all() | ||||
| [<Ranking: 3: a1>, <Ranking: 2: a2>, <Ranking: 1: a3>] | ||||
|  | ||||
| # Regression test for #10742: | ||||
| # Queries used in an __in clause don't execute subqueries | ||||
|  | ||||
| >>> subq = Author.objects.filter(num__lt=3000) | ||||
| >>> qs = Author.objects.filter(pk__in=subq) | ||||
| >>> list(qs) | ||||
| [<Author: a1>, <Author: a2>] | ||||
| # The subquery result cache should not be populated | ||||
| >>> subq._result_cache is None | ||||
| True | ||||
|  | ||||
| >>> subq = Author.objects.filter(num__lt=3000) | ||||
| >>> qs = Author.objects.exclude(pk__in=subq) | ||||
| >>> list(qs) | ||||
| [<Author: a3>, <Author: a4>] | ||||
| # The subquery result cache should not be populated | ||||
| >>> subq._result_cache is None | ||||
| True | ||||
|  | ||||
| >>> subq = Author.objects.filter(num__lt=3000) | ||||
| >>> list(Author.objects.filter(Q(pk__in=subq) & Q(name='a1'))) | ||||
| [<Author: a1>] | ||||
| # The subquery result cache should not be populated | ||||
| >>> subq._result_cache is None | ||||
| True | ||||
|  | ||||
| """} | ||||
|  | ||||
| # In Python 2.3 and the Python 2.6 beta releases, exceptions raised in __len__ | ||||
|   | ||||
		Reference in New Issue
	
	Block a user