--- /dev/null
+.. 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.
\ No newline at end of file
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):
)
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()
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()
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)
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()