123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159 |
- from django.db.models.lookups import (
- Exact, GreaterThan, GreaterThanOrEqual, In, IsNull, LessThan,
- LessThanOrEqual,
- )
- class MultiColSource:
- contains_aggregate = False
- def __init__(self, alias, targets, sources, field):
- self.targets, self.sources, self.field, self.alias = targets, sources, field, alias
- self.output_field = self.field
- def __repr__(self):
- return "{}({}, {})".format(
- self.__class__.__name__, self.alias, self.field)
- def relabeled_clone(self, relabels):
- return self.__class__(relabels.get(self.alias, self.alias),
- self.targets, self.sources, self.field)
- def get_lookup(self, lookup):
- return self.output_field.get_lookup(lookup)
- def resolve_expression(self, *args, **kwargs):
- return self
- def get_normalized_value(value, lhs):
- from django.db.models import Model
- if isinstance(value, Model):
- value_list = []
- sources = lhs.output_field.get_path_info()[-1].target_fields
- for source in sources:
- while not isinstance(value, source.model) and source.remote_field:
- source = source.remote_field.model._meta.get_field(source.remote_field.field_name)
- try:
- value_list.append(getattr(value, source.attname))
- except AttributeError:
-
-
- return (value.pk,)
- return tuple(value_list)
- if not isinstance(value, tuple):
- return (value,)
- return value
- class RelatedIn(In):
- def get_prep_lookup(self):
- if not isinstance(self.lhs, MultiColSource) and self.rhs_is_direct_value():
-
- self.rhs = [get_normalized_value(val, self.lhs)[0] for val in self.rhs]
-
-
-
-
- if hasattr(self.lhs.output_field, 'get_path_info'):
-
-
- target_field = self.lhs.output_field.get_path_info()[-1].target_fields[-1]
- self.rhs = [target_field.get_prep_value(v) for v in self.rhs]
- return super().get_prep_lookup()
- def as_sql(self, compiler, connection):
- if isinstance(self.lhs, MultiColSource):
-
-
-
- from django.db.models.sql.where import (
- AND, OR, SubqueryConstraint, WhereNode,
- )
- root_constraint = WhereNode(connector=OR)
- if self.rhs_is_direct_value():
- values = [get_normalized_value(value, self.lhs) for value in self.rhs]
- for value in values:
- value_constraint = WhereNode()
- for source, target, val in zip(self.lhs.sources, self.lhs.targets, value):
- lookup_class = target.get_lookup('exact')
- lookup = lookup_class(target.get_col(self.lhs.alias, source), val)
- value_constraint.add(lookup, AND)
- root_constraint.add(value_constraint, OR)
- else:
- root_constraint.add(
- SubqueryConstraint(
- self.lhs.alias, [target.column for target in self.lhs.targets],
- [source.name for source in self.lhs.sources], self.rhs),
- AND)
- return root_constraint.as_sql(compiler, connection)
- else:
- if (not getattr(self.rhs, 'has_select_fields', True) and
- not getattr(self.lhs.field.target_field, 'primary_key', False)):
- self.rhs.clear_select_clause()
- if (getattr(self.lhs.output_field, 'primary_key', False) and
- self.lhs.output_field.model == self.rhs.model):
-
-
-
- target_field = self.lhs.field.name
- else:
- target_field = self.lhs.field.target_field.name
- self.rhs.add_fields([target_field], True)
- return super().as_sql(compiler, connection)
- class RelatedLookupMixin:
- def get_prep_lookup(self):
- if not isinstance(self.lhs, MultiColSource) and not hasattr(self.rhs, 'resolve_expression'):
-
- self.rhs = get_normalized_value(self.rhs, self.lhs)[0]
-
-
-
-
- if self.prepare_rhs and hasattr(self.lhs.output_field, 'get_path_info'):
-
-
- target_field = self.lhs.output_field.get_path_info()[-1].target_fields[-1]
- self.rhs = target_field.get_prep_value(self.rhs)
- return super().get_prep_lookup()
- def as_sql(self, compiler, connection):
- if isinstance(self.lhs, MultiColSource):
- assert self.rhs_is_direct_value()
- self.rhs = get_normalized_value(self.rhs, self.lhs)
- from django.db.models.sql.where import AND, WhereNode
- root_constraint = WhereNode()
- for target, source, val in zip(self.lhs.targets, self.lhs.sources, self.rhs):
- lookup_class = target.get_lookup(self.lookup_name)
- root_constraint.add(
- lookup_class(target.get_col(self.lhs.alias, source), val), AND)
- return root_constraint.as_sql(compiler, connection)
- return super().as_sql(compiler, connection)
- class RelatedExact(RelatedLookupMixin, Exact):
- pass
- class RelatedLessThan(RelatedLookupMixin, LessThan):
- pass
- class RelatedGreaterThan(RelatedLookupMixin, GreaterThan):
- pass
- class RelatedGreaterThanOrEqual(RelatedLookupMixin, GreaterThanOrEqual):
- pass
- class RelatedLessThanOrEqual(RelatedLookupMixin, LessThanOrEqual):
- pass
- class RelatedIsNull(RelatedLookupMixin, IsNull):
- pass
|