Fixed #32478 -- Included nested columns referenced by subqueries in GROUP BY on aggregations.
Regression in fb3f034f1c
.
Refs #31094, #31150.
Thanks Igor Pejic for the report.
This commit is contained in:
parent
3aa545281e
commit
277eea8fcc
|
@ -1127,6 +1127,9 @@ class Subquery(Expression):
|
||||||
def external_aliases(self):
|
def external_aliases(self):
|
||||||
return self.query.external_aliases
|
return self.query.external_aliases
|
||||||
|
|
||||||
|
def get_external_cols(self):
|
||||||
|
return self.query.get_external_cols()
|
||||||
|
|
||||||
def as_sql(self, compiler, connection, template=None, query=None, **extra_context):
|
def as_sql(self, compiler, connection, template=None, query=None, **extra_context):
|
||||||
connection.ops.check_expression_support(self)
|
connection.ops.check_expression_support(self)
|
||||||
template_params = {**self.extra, **extra_context}
|
template_params = {**self.extra, **extra_context}
|
||||||
|
@ -1141,7 +1144,7 @@ class Subquery(Expression):
|
||||||
def get_group_by_cols(self, alias=None):
|
def get_group_by_cols(self, alias=None):
|
||||||
if alias:
|
if alias:
|
||||||
return [Ref(alias, self)]
|
return [Ref(alias, self)]
|
||||||
external_cols = self.query.get_external_cols()
|
external_cols = self.get_external_cols()
|
||||||
if any(col.possibly_multivalued for col in external_cols):
|
if any(col.possibly_multivalued for col in external_cols):
|
||||||
return [self]
|
return [self]
|
||||||
return external_cols
|
return external_cols
|
||||||
|
|
|
@ -1072,7 +1072,7 @@ class Query(BaseExpression):
|
||||||
def get_external_cols(self):
|
def get_external_cols(self):
|
||||||
exprs = chain(self.annotations.values(), self.where.children)
|
exprs = chain(self.annotations.values(), self.where.children)
|
||||||
return [
|
return [
|
||||||
col for col in self._gen_cols(exprs)
|
col for col in self._gen_cols(exprs, include_external=True)
|
||||||
if col.alias in self.external_aliases
|
if col.alias in self.external_aliases
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@ -1707,12 +1707,17 @@ class Query(BaseExpression):
|
||||||
return targets, joins[-1], joins
|
return targets, joins[-1], joins
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def _gen_cols(cls, exprs):
|
def _gen_cols(cls, exprs, include_external=False):
|
||||||
for expr in exprs:
|
for expr in exprs:
|
||||||
if isinstance(expr, Col):
|
if isinstance(expr, Col):
|
||||||
yield expr
|
yield expr
|
||||||
|
elif include_external and callable(getattr(expr, 'get_external_cols', None)):
|
||||||
|
yield from expr.get_external_cols()
|
||||||
else:
|
else:
|
||||||
yield from cls._gen_cols(expr.get_source_expressions())
|
yield from cls._gen_cols(
|
||||||
|
expr.get_source_expressions(),
|
||||||
|
include_external=include_external,
|
||||||
|
)
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def _gen_col_aliases(cls, exprs):
|
def _gen_col_aliases(cls, exprs):
|
||||||
|
|
|
@ -1311,6 +1311,21 @@ class AggregateTestCase(TestCase):
|
||||||
# self.assertSequenceEqual(books_qs, [book])
|
# self.assertSequenceEqual(books_qs, [book])
|
||||||
# self.assertEqual(ctx[0]['sql'].count('SELECT'), 2)
|
# self.assertEqual(ctx[0]['sql'].count('SELECT'), 2)
|
||||||
|
|
||||||
|
@skipUnlessDBFeature('supports_subqueries_in_group_by')
|
||||||
|
def test_aggregation_nested_subquery_outerref(self):
|
||||||
|
publisher_with_same_name = Publisher.objects.filter(
|
||||||
|
id__in=Subquery(
|
||||||
|
Publisher.objects.filter(
|
||||||
|
name=OuterRef(OuterRef('publisher__name')),
|
||||||
|
).values('id'),
|
||||||
|
),
|
||||||
|
).values(publisher_count=Count('id'))[:1]
|
||||||
|
books_breakdown = Book.objects.annotate(
|
||||||
|
publisher_count=Subquery(publisher_with_same_name),
|
||||||
|
authors_count=Count('authors'),
|
||||||
|
).values_list('publisher_count', flat=True)
|
||||||
|
self.assertSequenceEqual(books_breakdown, [1] * 6)
|
||||||
|
|
||||||
def test_aggregation_random_ordering(self):
|
def test_aggregation_random_ordering(self):
|
||||||
"""Random() is not included in the GROUP BY when used for ordering."""
|
"""Random() is not included in the GROUP BY when used for ordering."""
|
||||||
authors = Author.objects.annotate(contact_count=Count('book')).order_by('?')
|
authors = Author.objects.annotate(contact_count=Count('book')).order_by('?')
|
||||||
|
|
Loading…
Reference in New Issue