Add secondary selectable to FROM clauses for correlated exists

In continuing with a similar theme as that of very recent 🎫`4349`,
repaired issue with :meth:`.RelationshipProperty.Comparator.any` and
:meth:`.RelationshipProperty.Comparator.has` where the "secondary"
selectable needs to be explicitly part of the FROM clause in the
EXISTS subquery to suit the case where this "secondary" is a :class:`.Join`
object.

Fixes: #4366
Change-Id: Icd0d0c3871bbd0059f0c9256e2b980edc2c90551
This commit is contained in:
Mike Bayer
2018-11-14 11:02:40 -05:00
parent fd8f0044fe
commit fdfd168060
3 changed files with 38 additions and 2 deletions
+10
View File
@@ -0,0 +1,10 @@
.. change::
:tags: bug, orm
:tickets: 4366
In continuing with a similar theme as that of very recent :ticket:`4349`,
repaired issue with :meth:`.RelationshipProperty.Comparator.any` and
:meth:`.RelationshipProperty.Comparator.has` where the "secondary"
selectable needs to be explicitly part of the FROM clause in the
EXISTS subquery to suit the case where this "secondary" is a :class:`.Join`
object.
+5 -2
View File
@@ -1110,9 +1110,12 @@ class RelationshipProperty(StrategizedProperty):
crit = j & sql.True_._ifnone(criterion)
ex = sql.exists([1], crit, from_obj=dest).correlate_except(dest)
if secondary is not None:
ex = ex.correlate_except(secondary)
ex = sql.exists([1], crit, from_obj=[dest, secondary]).\
correlate_except(dest, secondary)
else:
ex = sql.exists([1], crit, from_obj=dest).\
correlate_except(dest)
return ex
def any(self, criterion=None, **kwargs):
+23
View File
@@ -2308,6 +2308,8 @@ class FilterTest(QueryTest, AssertsCompiledSQL):
)
def test_any(self):
# see also HasAnyTest, a newer suite which tests these at the level of
# SQL compilation
User, Address = self.classes.User, self.classes.Address
sess = create_session()
@@ -2344,6 +2346,8 @@ class FilterTest(QueryTest, AssertsCompiledSQL):
filter(~User.addresses.any()).all()
def test_any_doesnt_overcorrelate(self):
# see also HasAnyTest, a newer suite which tests these at the level of
# SQL compilation
User, Address = self.classes.User, self.classes.Address
sess = create_session()
@@ -2356,6 +2360,8 @@ class FilterTest(QueryTest, AssertsCompiledSQL):
Address.email_address == 'fred@fred.com')).all()
def test_has(self):
# see also HasAnyTest, a newer suite which tests these at the level of
# SQL compilation
Dingaling, User, Address = (
self.classes.Dingaling, self.classes.User, self.classes.Address)
@@ -2607,6 +2613,23 @@ class HasAnyTest(
id = Column(Integer, primary_key=True)
b_id = Column(ForeignKey(B.id))
d = relationship(
'D',
secondary="join(B, C)",
primaryjoin="A.b_id == B.id",
secondaryjoin="C.d_id == D.id",
uselist=False)
def test_has_composite_secondary(self):
A, D = self.classes("A", "D")
s = Session()
self.assert_compile(
s.query(A).filter(A.d.has(D.id == 1)),
"SELECT a.id AS a_id, a.b_id AS a_b_id FROM a WHERE EXISTS "
"(SELECT 1 FROM d, b JOIN c ON c.id = b.c_id "
"WHERE a.b_id = b.id AND c.d_id = d.id AND d.id = :id_1)"
)
def test_has_many_to_one(self):
B, C = self.classes("B", "C")
s = Session()