summaryrefslogtreecommitdiff
path: root/django/db/models/sql
diff options
context:
space:
mode:
Diffstat (limited to 'django/db/models/sql')
-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()