summaryrefslogtreecommitdiff
path: root/django/db/models/sql/query.py
diff options
context:
space:
mode:
authorSimon Charette <charette.s@gmail.com>2023-05-21 23:49:05 -0400
committerMariusz Felisiak <felisiak.mariusz@gmail.com>2023-05-23 06:25:27 +0200
commit2ee01747c32a7275a7a1a5f7862acba7db764921 (patch)
tree7a668fab0cc6996f7491fbe988500d0ee1f81992 /django/db/models/sql/query.py
parent89f10a80d7e681cd0cccf22d932e380f51bd3524 (diff)
Refs #34551 -- Fixed QuerySet.aggregate() crash on precending aggregation reference.
Regression in 1297c0d0d76a708017fe196b61a0ab324df76954. Refs #31679.
Diffstat (limited to 'django/db/models/sql/query.py')
-rw-r--r--django/db/models/sql/query.py16
1 files changed, 14 insertions, 2 deletions
diff --git a/django/db/models/sql/query.py b/django/db/models/sql/query.py
index 252e5e9fcc..c4b435df2d 100644
--- a/django/db/models/sql/query.py
+++ b/django/db/models/sql/query.py
@@ -400,7 +400,10 @@ class Query(BaseExpression):
"""
if not aggregate_exprs:
return {}
- aggregates = {}
+ # Store annotation mask prior to temporarily adding aggregations for
+ # resolving purpose to facilitate their subsequent removal.
+ replacements = {}
+ annotation_select_mask = self.annotation_select_mask
for alias, aggregate_expr in aggregate_exprs.items():
self.check_alias(alias)
aggregate = aggregate_expr.resolve_expression(
@@ -408,7 +411,16 @@ class Query(BaseExpression):
)
if not aggregate.contains_aggregate:
raise TypeError("%s is not an aggregate expression" % alias)
- aggregates[alias] = aggregate
+ # Temporarily add aggregate to annotations to allow remaining
+ # members of `aggregates` to resolve against each others.
+ self.append_annotation_mask([alias])
+ aggregate = aggregate.replace_expressions(replacements)
+ self.annotations[alias] = aggregate
+ replacements[Ref(alias, aggregate)] = aggregate
+ # Stash resolved aggregates now that they have been allowed to resolve
+ # against each other.
+ aggregates = {alias: self.annotations.pop(alias) for alias in aggregate_exprs}
+ self.set_annotation_mask(annotation_select_mask)
# Existing usage of aggregation can be determined by the presence of
# selected aggregates but also by filters against aliased aggregates.
_, having, qualify = self.where.split_having_qualify()