Fixed #25095 -- Fixed annotate() + values() group by bug

Thanks Josh Smeaton for help on the tests.
This commit is contained in:
Mitchell Kotler 2015-07-15 17:29:33 -04:00 committed by Tim Graham
parent 199a02d1e2
commit 6024fd5dc2
2 changed files with 33 additions and 2 deletions

View File

@ -1679,8 +1679,8 @@ class Query(object):
for col in self.select: for col in self.select:
self.group_by.append(col) self.group_by.append(col)
if self._annotations: if self.annotation_select:
for alias, annotation in six.iteritems(self.annotations): for alias, annotation in six.iteritems(self.annotation_select):
for col in annotation.get_group_by_cols(): for col in annotation.get_group_by_cols():
self.group_by.append(col) self.group_by.append(col)

View File

@ -395,6 +395,37 @@ class AggregateTestCase(TestCase):
vals = Book.objects.aggregate(Count("rating", distinct=True)) vals = Book.objects.aggregate(Count("rating", distinct=True))
self.assertEqual(vals, {"rating__count": 4}) self.assertEqual(vals, {"rating__count": 4})
def test_non_grouped_annotation_not_in_group_by(self):
"""
An annotation not included in values() before an aggregate should be
excluded from the group by clause.
"""
qs = (
Book.objects.annotate(xprice=F('price')).filter(rating=4.0).values('rating')
.annotate(count=Count('publisher_id', distinct=True)).values('count', 'rating').order_by('count')
)
self.assertEqual(
list(qs), [
{'rating': 4.0, 'count': 2},
]
)
def test_grouped_annotation_in_group_by(self):
"""
An annotation included in values() before an aggregate should be
included in the group by clause.
"""
qs = (
Book.objects.annotate(xprice=F('price')).filter(rating=4.0).values('rating', 'xprice')
.annotate(count=Count('publisher_id', distinct=True)).values('count', 'rating').order_by('count')
)
self.assertEqual(
list(qs), [
{'rating': 4.0, 'count': 1},
{'rating': 4.0, 'count': 2},
]
)
def test_fkey_aggregate(self): def test_fkey_aggregate(self):
explicit = list(Author.objects.annotate(Count('book__id'))) explicit = list(Author.objects.annotate(Count('book__id')))
implicit = list(Author.objects.annotate(Count('book'))) implicit = list(Author.objects.annotate(Count('book')))