mirror of
				https://github.com/django/django.git
				synced 2025-10-24 22:26:08 +00:00 
			
		
		
		
	Fixed #14707 -- Relax the protections on aggregate naming collisions when a ValuesQuerySet removes the colliding name. Thanks to Andy McKay for the report.
git-svn-id: http://code.djangoproject.com/svn/django/trunk@15223 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
		| @@ -620,18 +620,19 @@ class QuerySet(object): | |||||||
|         """ |         """ | ||||||
|         for arg in args: |         for arg in args: | ||||||
|             if arg.default_alias in kwargs: |             if arg.default_alias in kwargs: | ||||||
|                 raise ValueError("The %s named annotation conflicts with the " |                 raise ValueError("The named annotation '%s' conflicts with the " | ||||||
|                                  "default name for another annotation." |                                  "default name for another annotation." | ||||||
|                                  % arg.default_alias) |                                  % arg.default_alias) | ||||||
|             kwargs[arg.default_alias] = arg |             kwargs[arg.default_alias] = arg | ||||||
|  |  | ||||||
|         names = set(self.model._meta.get_all_field_names()) |         names = getattr(self, '_fields', None) | ||||||
|  |         if names is None: | ||||||
|  |             names = set(self.model._meta.get_all_field_names()) | ||||||
|         for aggregate in kwargs: |         for aggregate in kwargs: | ||||||
|             if aggregate in names: |             if aggregate in names: | ||||||
|                 raise ValueError("The %s annotation conflicts with a field on " |                 raise ValueError("The annotation '%s' conflicts with a field on " | ||||||
|                     "the model." % aggregate) |                     "the model." % aggregate) | ||||||
|  |  | ||||||
|  |  | ||||||
|         obj = self._clone() |         obj = self._clone() | ||||||
|  |  | ||||||
|         obj._setup_aggregate_query(kwargs.keys()) |         obj._setup_aggregate_query(kwargs.keys()) | ||||||
|   | |||||||
| @@ -474,6 +474,28 @@ class AggregationTests(TestCase): | |||||||
|         # Regression for #11256 - providing an aggregate name that conflicts with an m2m name on the model raises ValueError |         # Regression for #11256 - providing an aggregate name that conflicts with an m2m name on the model raises ValueError | ||||||
|         self.assertRaises(ValueError, Author.objects.annotate, friends=Count('friends')) |         self.assertRaises(ValueError, Author.objects.annotate, friends=Count('friends')) | ||||||
|  |  | ||||||
|  |     def test_values_queryset_non_conflict(self): | ||||||
|  |         # Regression for #14707 -- If you're using a values query set, some potential conflicts are avoided. | ||||||
|  |  | ||||||
|  |         # age is a field on Author, so it shouldn't be allowed as an aggregate. | ||||||
|  |         # But age isn't included in the ValuesQuerySet, so it is. | ||||||
|  |         results = Author.objects.values('name').annotate(age=Count('book_contact_set')) | ||||||
|  |         self.assertEquals(len(results), 9) | ||||||
|  |         self.assertEquals(results[0]['name'], u'Adrian Holovaty') | ||||||
|  |         self.assertEquals(results[0]['age'], 1) | ||||||
|  |  | ||||||
|  |         # Same problem, but aggregating over m2m fields | ||||||
|  |         results = Author.objects.values('name').annotate(age=Avg('friends__age')) | ||||||
|  |         self.assertEquals(len(results), 9) | ||||||
|  |         self.assertEquals(results[0]['name'], u'Adrian Holovaty') | ||||||
|  |         self.assertEquals(results[0]['age'], 32.0) | ||||||
|  |  | ||||||
|  |         # Same problem, but colliding with an m2m field | ||||||
|  |         results = Author.objects.values('name').annotate(friends=Count('friends')) | ||||||
|  |         self.assertEquals(len(results), 9) | ||||||
|  |         self.assertEquals(results[0]['name'], u'Adrian Holovaty') | ||||||
|  |         self.assertEquals(results[0]['friends'], 2) | ||||||
|  |  | ||||||
|     def test_reverse_relation_name_conflict(self): |     def test_reverse_relation_name_conflict(self): | ||||||
|         # Regression for #11256 - providing an aggregate name that conflicts with a reverse-related name on the model raises ValueError |         # Regression for #11256 - providing an aggregate name that conflicts with a reverse-related name on the model raises ValueError | ||||||
|         self.assertRaises(ValueError, Author.objects.annotate, book_contact_set=Avg('friends__age')) |         self.assertRaises(ValueError, Author.objects.annotate, book_contact_set=Avg('friends__age')) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user