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:
Russell Keith-Magee 2011-01-16 08:03:25 +00:00
parent b4f0921463
commit ae8ff5f476
2 changed files with 27 additions and 4 deletions

View File

@ -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())

View File

@ -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'))