Refs #32948, Refs #32946 -- Used Q.create() internally for dynamic Q() objects.

Node.create() which has a compatible signature with Node.__init__()
takes in a single `children` argument rather than relying in unpacking
*args in Q.__init__() which calls Node.__init__().

In addition, we were often needing to unpack iterables into *args and
can instead pass a list direct to Node.create().
This commit is contained in:
Nick Pope 2021-09-21 00:49:16 +01:00 committed by Mariusz Felisiak
parent 845667f2d1
commit 9dff316be4
9 changed files with 40 additions and 26 deletions

View File

@ -527,7 +527,7 @@ class EmptyFieldListFilter(FieldListFilter):
lookup_conditions.append((self.field_path, "")) lookup_conditions.append((self.field_path, ""))
if self.field.null: if self.field.null:
lookup_conditions.append((f"{self.field_path}__isnull", True)) lookup_conditions.append((f"{self.field_path}__isnull", True))
lookup_condition = models.Q(*lookup_conditions, _connector=models.Q.OR) lookup_condition = models.Q.create(lookup_conditions, connector=models.Q.OR)
if self.lookup_val == "1": if self.lookup_val == "1":
return queryset.filter(lookup_condition) return queryset.filter(lookup_condition)
return queryset.exclude(lookup_condition) return queryset.exclude(lookup_condition)

View File

@ -1146,12 +1146,12 @@ class ModelAdmin(BaseModelAdmin):
for bit in smart_split(search_term): for bit in smart_split(search_term):
if bit.startswith(('"', "'")) and bit[0] == bit[-1]: if bit.startswith(('"', "'")) and bit[0] == bit[-1]:
bit = unescape_string_literal(bit) bit = unescape_string_literal(bit)
or_queries = models.Q( or_queries = models.Q.create(
*((orm_lookup, bit) for orm_lookup in orm_lookups), [(orm_lookup, bit) for orm_lookup in orm_lookups],
_connector=models.Q.OR, connector=models.Q.OR,
) )
term_queries.append(or_queries) term_queries.append(or_queries)
queryset = queryset.filter(models.Q(*term_queries)) queryset = queryset.filter(models.Q.create(term_queries))
may_have_duplicates |= any( may_have_duplicates |= any(
lookup_spawns_duplicates(self.opts, search_spec) lookup_spawns_duplicates(self.opts, search_spec)
for search_spec in orm_lookups for search_spec in orm_lookups

View File

@ -627,17 +627,19 @@ def create_generic_related_manager(superclass, rel):
queryset._add_hints(instance=instances[0]) queryset._add_hints(instance=instances[0])
queryset = queryset.using(queryset._db or self._db) queryset = queryset.using(queryset._db or self._db)
# Group instances by content types. # Group instances by content types.
content_type_queries = ( content_type_queries = [
models.Q( models.Q.create(
(f"{self.content_type_field_name}__pk", content_type_id), [
(f"{self.object_id_field_name}__in", {obj.pk for obj in objs}), (f"{self.content_type_field_name}__pk", content_type_id),
(f"{self.object_id_field_name}__in", {obj.pk for obj in objs}),
]
) )
for content_type_id, objs in itertools.groupby( for content_type_id, objs in itertools.groupby(
sorted(instances, key=lambda obj: self.get_content_type(obj).pk), sorted(instances, key=lambda obj: self.get_content_type(obj).pk),
lambda obj: self.get_content_type(obj).pk, lambda obj: self.get_content_type(obj).pk,
) )
) ]
query = models.Q(*content_type_queries, _connector=models.Q.OR) query = models.Q.create(content_type_queries, connector=models.Q.OR)
# We (possibly) need to convert object IDs to the type of the # We (possibly) need to convert object IDs to the type of the
# instances' PK in order to match up instances: # instances' PK in order to match up instances:
object_id_converter = instances[0]._meta.pk.to_python object_id_converter = instances[0]._meta.pk.to_python

View File

@ -1152,8 +1152,8 @@ class Model(metaclass=ModelBase):
op = "gt" if is_next else "lt" op = "gt" if is_next else "lt"
order = "" if is_next else "-" order = "" if is_next else "-"
param = getattr(self, field.attname) param = getattr(self, field.attname)
q = Q((field.name, param), (f"pk__{op}", self.pk), _connector=Q.AND) q = Q.create([(field.name, param), (f"pk__{op}", self.pk)], connector=Q.AND)
q = Q(q, (f"{field.name}__{op}", param), _connector=Q.OR) q = Q.create([q, (f"{field.name}__{op}", param)], connector=Q.OR)
qs = ( qs = (
self.__class__._default_manager.using(self._state.db) self.__class__._default_manager.using(self._state.db)
.filter(**kwargs) .filter(**kwargs)

View File

@ -399,9 +399,9 @@ class Collector:
""" """
Get a QuerySet of the related model to objs via related fields. Get a QuerySet of the related model to objs via related fields.
""" """
predicate = query_utils.Q( predicate = query_utils.Q.create(
*((f"{related_field.name}__in", objs) for related_field in related_fields), [(f"{related_field.name}__in", objs) for related_field in related_fields],
_connector=query_utils.Q.OR, connector=query_utils.Q.OR,
) )
return related_model._base_manager.using(self.using).filter(predicate) return related_model._base_manager.using(self.using).filter(predicate)

View File

@ -407,12 +407,13 @@ class RelatedField(FieldCacheMixin, Field):
select all instances of self.related_field.model related through select all instances of self.related_field.model related through
this field to obj. obj is an instance of self.model. this field to obj. obj is an instance of self.model.
""" """
base_filter = ( base_q = Q.create(
(rh_field.attname, getattr(obj, lh_field.attname)) [
for lh_field, rh_field in self.related_fields (rh_field.attname, getattr(obj, lh_field.attname))
for lh_field, rh_field in self.related_fields
]
) )
descriptor_filter = self.get_extra_descriptor_filter(obj) descriptor_filter = self.get_extra_descriptor_filter(obj)
base_q = Q(*base_filter)
if isinstance(descriptor_filter, dict): if isinstance(descriptor_filter, dict):
return base_q & Q(**descriptor_filter) return base_q & Q(**descriptor_filter)
elif descriptor_filter: elif descriptor_filter:

View File

@ -1002,19 +1002,21 @@ def create_forward_many_to_many_manager(superclass, rel, reverse):
do_not_call_in_templates = True do_not_call_in_templates = True
def _build_remove_filters(self, removed_vals): def _build_remove_filters(self, removed_vals):
filters = Q((self.source_field_name, self.related_val)) filters = Q.create([(self.source_field_name, self.related_val)])
# No need to add a subquery condition if removed_vals is a QuerySet without # No need to add a subquery condition if removed_vals is a QuerySet without
# filters. # filters.
removed_vals_filters = ( removed_vals_filters = (
not isinstance(removed_vals, QuerySet) or removed_vals._has_filters() not isinstance(removed_vals, QuerySet) or removed_vals._has_filters()
) )
if removed_vals_filters: if removed_vals_filters:
filters &= Q((f"{self.target_field_name}__in", removed_vals)) filters &= Q.create([(f"{self.target_field_name}__in", removed_vals)])
if self.symmetrical: if self.symmetrical:
symmetrical_filters = Q((self.target_field_name, self.related_val)) symmetrical_filters = Q.create(
[(self.target_field_name, self.related_val)]
)
if removed_vals_filters: if removed_vals_filters:
symmetrical_filters &= Q( symmetrical_filters &= Q.create(
(f"{self.source_field_name}__in", removed_vals) [(f"{self.source_field_name}__in", removed_vals)]
) )
filters |= symmetrical_filters filters |= symmetrical_filters
return filters return filters

View File

@ -33,7 +33,7 @@ class Node:
__init__() with a signature that conflicts with the one defined in __init__() with a signature that conflicts with the one defined in
Node.__init__(). Node.__init__().
""" """
obj = Node(children, connector, negated) obj = Node(children, connector or cls.default, negated)
obj.__class__ = cls obj.__class__ = cls
return obj return obj

View File

@ -216,6 +216,15 @@ class QTests(SimpleTestCase):
flatten = list(q.flatten()) flatten = list(q.flatten())
self.assertEqual(len(flatten), 7) self.assertEqual(len(flatten), 7)
def test_create_helper(self):
items = [("a", 1), ("b", 2), ("c", 3)]
for connector in [Q.AND, Q.OR, Q.XOR]:
with self.subTest(connector=connector):
self.assertEqual(
Q.create(items, connector=connector),
Q(*items, _connector=connector),
)
class QCheckTests(TestCase): class QCheckTests(TestCase):
def test_basic(self): def test_basic(self):