mirror of
https://github.com/django/django.git
synced 2025-11-29 06:51:10 +00:00
Fixed #22429 -- Incorrect SQL when using ~Q and F
This commit is contained in:
parent
13ec89f267
commit
5e1f4656b9
4 changed files with 50 additions and 10 deletions
|
|
@ -1159,6 +1159,9 @@ class Query(object):
|
|||
try:
|
||||
field, sources, opts, join_list, path = self.setup_joins(
|
||||
parts, opts, alias, can_reuse, allow_many)
|
||||
# split_exclude() needs to know which joins were generated for the
|
||||
# lookup parts
|
||||
self._lookup_joins = join_list
|
||||
except MultiJoin as e:
|
||||
return self.split_exclude(filter_expr, LOOKUP_SEP.join(parts[:e.level]),
|
||||
can_reuse, e.names_with_path)
|
||||
|
|
@ -1899,17 +1902,21 @@ class Query(object):
|
|||
for _, paths in names_with_path:
|
||||
all_paths.extend(paths)
|
||||
contains_louter = False
|
||||
for pos, path in enumerate(all_paths):
|
||||
# Trim and operate only on tables that were generated for
|
||||
# the lookup part of the query. That is, avoid trimming
|
||||
# joins generated for F() expressions.
|
||||
lookup_tables = [t for t in self.tables if t in self._lookup_joins or t == self.tables[0]]
|
||||
for trimmed_paths, path in enumerate(all_paths):
|
||||
if path.m2m:
|
||||
break
|
||||
if self.alias_map[self.tables[pos + 1]].join_type == self.LOUTER:
|
||||
if self.alias_map[lookup_tables[trimmed_paths + 1]].join_type == self.LOUTER:
|
||||
contains_louter = True
|
||||
self.unref_alias(self.tables[pos])
|
||||
self.unref_alias(lookup_tables[trimmed_paths])
|
||||
# The path.join_field is a Rel, lets get the other side's field
|
||||
join_field = path.join_field.field
|
||||
# Build the filter prefix.
|
||||
paths_in_prefix = trimmed_paths
|
||||
trimmed_prefix = []
|
||||
paths_in_prefix = pos
|
||||
for name, path in names_with_path:
|
||||
if paths_in_prefix - len(path) < 0:
|
||||
break
|
||||
|
|
@ -1921,12 +1928,12 @@ class Query(object):
|
|||
# Lets still see if we can trim the first join from the inner query
|
||||
# (that is, self). We can't do this for LEFT JOINs because we would
|
||||
# miss those rows that have nothing on the outer side.
|
||||
if self.alias_map[self.tables[pos + 1]].join_type != self.LOUTER:
|
||||
if self.alias_map[lookup_tables[trimmed_paths + 1]].join_type != self.LOUTER:
|
||||
select_fields = [r[0] for r in join_field.related_fields]
|
||||
select_alias = self.tables[pos + 1]
|
||||
self.unref_alias(self.tables[pos])
|
||||
select_alias = lookup_tables[trimmed_paths + 1]
|
||||
self.unref_alias(lookup_tables[trimmed_paths])
|
||||
extra_restriction = join_field.get_extra_restriction(
|
||||
self.where_class, None, self.tables[pos + 1])
|
||||
self.where_class, None, lookup_tables[trimmed_paths + 1])
|
||||
if extra_restriction:
|
||||
self.where.add(extra_restriction, AND)
|
||||
else:
|
||||
|
|
@ -1934,7 +1941,7 @@ class Query(object):
|
|||
# inner query if it happens to have a longer join chain containing the
|
||||
# values in select_fields. Lets punt this one for now.
|
||||
select_fields = [r[1] for r in join_field.related_fields]
|
||||
select_alias = self.tables[pos]
|
||||
select_alias = lookup_tables[trimmed_paths]
|
||||
self.select = [SelectInfo((select_alias, f.column), f) for f in select_fields]
|
||||
return trimmed_prefix, contains_louter
|
||||
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue