mirror of
				https://github.com/django/django.git
				synced 2025-10-25 06:36:07 +00:00 
			
		
		
		
	Fixed #10432 -- Handle all kinds of iterators in queryset filters.
Only consumes the iterators once and works with Python 2.3. git-svn-id: http://code.djangoproject.com/svn/django/trunk@9986 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
		| @@ -47,6 +47,10 @@ class WhereNode(tree.Node): | |||||||
|             return |             return | ||||||
|  |  | ||||||
|         obj, lookup_type, value = data |         obj, lookup_type, value = data | ||||||
|  |         if hasattr(value, '__iter__') and hasattr(value, 'next'): | ||||||
|  |             # Consume any generators immediately, so that we can determine | ||||||
|  |             # emptiness and transform any non-empty values correctly. | ||||||
|  |             value = list(value) | ||||||
|         if hasattr(obj, "process"): |         if hasattr(obj, "process"): | ||||||
|             try: |             try: | ||||||
|                 obj, params = obj.process(lookup_type, value) |                 obj, params = obj.process(lookup_type, value) | ||||||
|   | |||||||
| @@ -1090,6 +1090,19 @@ to set things up correctly internally so that subqueries can continue properly. | |||||||
| >>> Tag.objects.filter(name__in=()).update(name="foo") | >>> Tag.objects.filter(name__in=()).update(name="foo") | ||||||
| 0 | 0 | ||||||
|  |  | ||||||
|  | Bug #10432 (see also the Python 2.4+ tests for this, below). Testing an empty | ||||||
|  | "__in" filter with a generator as the value. | ||||||
|  | >>> def f(): | ||||||
|  | ...     return iter([]) | ||||||
|  | >>> n_obj = Note.objects.all()[0] | ||||||
|  | >>> def g(): | ||||||
|  | ...     for i in [n_obj.pk]: | ||||||
|  | ...         yield i | ||||||
|  | >>> Note.objects.filter(pk__in=f()) | ||||||
|  | [] | ||||||
|  | >>> list(Note.objects.filter(pk__in=g())) == [n_obj] | ||||||
|  | True | ||||||
|  |  | ||||||
| """} | """} | ||||||
|  |  | ||||||
| # In Python 2.3 and the Python 2.6 beta releases, exceptions raised in __len__ | # In Python 2.3 and the Python 2.6 beta releases, exceptions raised in __len__ | ||||||
| @@ -1140,3 +1153,13 @@ True | |||||||
| True | True | ||||||
|  |  | ||||||
| """ | """ | ||||||
|  |  | ||||||
|  | # Generator expressions are only in Python 2.4 and later. | ||||||
|  | if sys.version_info >= (2, 4): | ||||||
|  |     __test__["API_TESTS"] += """ | ||||||
|  | Using an empty generator expression as the rvalue for an "__in" lookup is legal | ||||||
|  | (regression for #10432). | ||||||
|  | >>> Note.objects.filter(pk__in=(x for x in ())) | ||||||
|  | [] | ||||||
|  |  | ||||||
|  | """ | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user