Fixed #11293 -- fixed using Q objects to generate ORs with aggregates.
git-svn-id: http://code.djangoproject.com/svn/django/trunk@15173 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
parent
1b90cdcf23
commit
29de7ee9cb
|
@ -922,6 +922,19 @@ class Query(object):
|
||||||
self.unref_alias(alias)
|
self.unref_alias(alias)
|
||||||
self.included_inherited_models = {}
|
self.included_inherited_models = {}
|
||||||
|
|
||||||
|
def need_force_having(self, q_object):
|
||||||
|
"""
|
||||||
|
Returns whether or not all elements of this q_object need to be put
|
||||||
|
together in the HAVING clause.
|
||||||
|
"""
|
||||||
|
for child in q_object.children:
|
||||||
|
if isinstance(child, Node):
|
||||||
|
if self.need_force_having(child):
|
||||||
|
return True
|
||||||
|
else:
|
||||||
|
if child[0].split(LOOKUP_SEP)[0] in self.aggregates:
|
||||||
|
return True
|
||||||
|
return False
|
||||||
|
|
||||||
def add_aggregate(self, aggregate, model, alias, is_summary):
|
def add_aggregate(self, aggregate, model, alias, is_summary):
|
||||||
"""
|
"""
|
||||||
|
@ -972,7 +985,7 @@ class Query(object):
|
||||||
aggregate.add_to_query(self, alias, col=col, source=source, is_summary=is_summary)
|
aggregate.add_to_query(self, alias, col=col, source=source, is_summary=is_summary)
|
||||||
|
|
||||||
def add_filter(self, filter_expr, connector=AND, negate=False, trim=False,
|
def add_filter(self, filter_expr, connector=AND, negate=False, trim=False,
|
||||||
can_reuse=None, process_extras=True):
|
can_reuse=None, process_extras=True, force_having=False):
|
||||||
"""
|
"""
|
||||||
Add a single filter to the query. The 'filter_expr' is a pair:
|
Add a single filter to the query. The 'filter_expr' is a pair:
|
||||||
(filter_string, value). E.g. ('name__contains', 'fred')
|
(filter_string, value). E.g. ('name__contains', 'fred')
|
||||||
|
@ -1026,14 +1039,14 @@ class Query(object):
|
||||||
value = SQLEvaluator(value, self)
|
value = SQLEvaluator(value, self)
|
||||||
having_clause = value.contains_aggregate
|
having_clause = value.contains_aggregate
|
||||||
|
|
||||||
for alias, aggregate in self.aggregates.items():
|
if parts[0] in self.aggregates:
|
||||||
if alias == parts[0]:
|
aggregate = self.aggregates[parts[0]]
|
||||||
entry = self.where_class()
|
entry = self.where_class()
|
||||||
entry.add((aggregate, lookup_type, value), AND)
|
entry.add((aggregate, lookup_type, value), AND)
|
||||||
if negate:
|
if negate:
|
||||||
entry.negate()
|
entry.negate()
|
||||||
self.having.add(entry, AND)
|
self.having.add(entry, connector)
|
||||||
return
|
return
|
||||||
|
|
||||||
opts = self.get_meta()
|
opts = self.get_meta()
|
||||||
alias = self.get_initial_alias()
|
alias = self.get_initial_alias()
|
||||||
|
@ -1082,7 +1095,7 @@ class Query(object):
|
||||||
self.promote_alias_chain(table_it, table_promote)
|
self.promote_alias_chain(table_it, table_promote)
|
||||||
|
|
||||||
|
|
||||||
if having_clause:
|
if having_clause or force_having:
|
||||||
if (alias, col) not in self.group_by:
|
if (alias, col) not in self.group_by:
|
||||||
self.group_by.append((alias, col))
|
self.group_by.append((alias, col))
|
||||||
self.having.add((Constraint(alias, col, field), lookup_type, value),
|
self.having.add((Constraint(alias, col, field), lookup_type, value),
|
||||||
|
@ -1123,7 +1136,7 @@ class Query(object):
|
||||||
self.add_filter(filter, negate=negate, can_reuse=can_reuse,
|
self.add_filter(filter, negate=negate, can_reuse=can_reuse,
|
||||||
process_extras=False)
|
process_extras=False)
|
||||||
|
|
||||||
def add_q(self, q_object, used_aliases=None):
|
def add_q(self, q_object, used_aliases=None, force_having=False):
|
||||||
"""
|
"""
|
||||||
Adds a Q-object to the current filter.
|
Adds a Q-object to the current filter.
|
||||||
|
|
||||||
|
@ -1141,16 +1154,25 @@ class Query(object):
|
||||||
else:
|
else:
|
||||||
subtree = False
|
subtree = False
|
||||||
connector = AND
|
connector = AND
|
||||||
|
if q_object.connector == OR and not force_having:
|
||||||
|
force_having = self.need_force_having(q_object)
|
||||||
for child in q_object.children:
|
for child in q_object.children:
|
||||||
if connector == OR:
|
if connector == OR:
|
||||||
refcounts_before = self.alias_refcount.copy()
|
refcounts_before = self.alias_refcount.copy()
|
||||||
self.where.start_subtree(connector)
|
if force_having:
|
||||||
|
self.having.start_subtree(connector)
|
||||||
|
else:
|
||||||
|
self.where.start_subtree(connector)
|
||||||
if isinstance(child, Node):
|
if isinstance(child, Node):
|
||||||
self.add_q(child, used_aliases)
|
self.add_q(child, used_aliases, force_having=force_having)
|
||||||
else:
|
else:
|
||||||
self.add_filter(child, connector, q_object.negated,
|
self.add_filter(child, connector, q_object.negated,
|
||||||
can_reuse=used_aliases)
|
can_reuse=used_aliases, force_having=force_having)
|
||||||
self.where.end_subtree()
|
if force_having:
|
||||||
|
self.having.end_subtree()
|
||||||
|
else:
|
||||||
|
self.where.end_subtree()
|
||||||
|
|
||||||
if connector == OR:
|
if connector == OR:
|
||||||
# Aliases that were newly added or not used at all need to
|
# Aliases that were newly added or not used at all need to
|
||||||
# be promoted to outer joins if they are nullable relations.
|
# be promoted to outer joins if they are nullable relations.
|
||||||
|
|
|
@ -4,8 +4,8 @@ from decimal import Decimal
|
||||||
from operator import attrgetter
|
from operator import attrgetter
|
||||||
|
|
||||||
from django.core.exceptions import FieldError
|
from django.core.exceptions import FieldError
|
||||||
|
from django.db.models import Count, Max, Avg, Sum, StdDev, Variance, F, Q
|
||||||
from django.test import TestCase, Approximate, skipUnlessDBFeature
|
from django.test import TestCase, Approximate, skipUnlessDBFeature
|
||||||
from django.db.models import Count, Max, Avg, Sum, StdDev, Variance, F
|
|
||||||
|
|
||||||
from models import Author, Book, Publisher, Clues, Entries, HardbackBook
|
from models import Author, Book, Publisher, Clues, Entries, HardbackBook
|
||||||
|
|
||||||
|
@ -673,6 +673,58 @@ class AggregationTests(TestCase):
|
||||||
list(qs), list(Book.objects.values_list("name", flat=True))
|
list(qs), list(Book.objects.values_list("name", flat=True))
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def test_annotation_disjunction(self):
|
||||||
|
qs = Book.objects.annotate(n_authors=Count("authors")).filter(
|
||||||
|
Q(n_authors=2) | Q(name="Python Web Development with Django")
|
||||||
|
)
|
||||||
|
self.assertQuerysetEqual(
|
||||||
|
qs, [
|
||||||
|
"Artificial Intelligence: A Modern Approach",
|
||||||
|
"Python Web Development with Django",
|
||||||
|
"The Definitive Guide to Django: Web Development Done Right",
|
||||||
|
],
|
||||||
|
attrgetter("name")
|
||||||
|
)
|
||||||
|
|
||||||
|
qs = Book.objects.annotate(n_authors=Count("authors")).filter(
|
||||||
|
Q(name="The Definitive Guide to Django: Web Development Done Right") | (Q(name="Artificial Intelligence: A Modern Approach") & Q(n_authors=3))
|
||||||
|
)
|
||||||
|
self.assertQuerysetEqual(
|
||||||
|
qs, [
|
||||||
|
"The Definitive Guide to Django: Web Development Done Right",
|
||||||
|
],
|
||||||
|
attrgetter("name")
|
||||||
|
)
|
||||||
|
|
||||||
|
qs = Publisher.objects.annotate(
|
||||||
|
rating_sum=Sum("book__rating"),
|
||||||
|
book_count=Count("book")
|
||||||
|
).filter(
|
||||||
|
Q(rating_sum__gt=5.5) | Q(rating_sum__isnull=True)
|
||||||
|
).order_by('pk')
|
||||||
|
self.assertQuerysetEqual(
|
||||||
|
qs, [
|
||||||
|
"Apress",
|
||||||
|
"Prentice Hall",
|
||||||
|
"Jonno's House of Books",
|
||||||
|
],
|
||||||
|
attrgetter("name")
|
||||||
|
)
|
||||||
|
|
||||||
|
qs = Publisher.objects.annotate(
|
||||||
|
rating_sum=Sum("book__rating"),
|
||||||
|
book_count=Count("book")
|
||||||
|
).filter(
|
||||||
|
Q(pk__lt=F("book_count")) | Q(rating_sum=None)
|
||||||
|
).order_by("pk")
|
||||||
|
self.assertQuerysetEqual(
|
||||||
|
qs, [
|
||||||
|
"Apress",
|
||||||
|
"Jonno's House of Books",
|
||||||
|
],
|
||||||
|
attrgetter("name")
|
||||||
|
)
|
||||||
|
|
||||||
@skipUnlessDBFeature('supports_stddev')
|
@skipUnlessDBFeature('supports_stddev')
|
||||||
def test_stddev(self):
|
def test_stddev(self):
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
|
|
Loading…
Reference in New Issue