我是初学SQLAlchemy用户,对广泛的文档感到沮丧。
我有一个新闻源,当对某些内容对象进行新的修订时会更新。 内容始终至少有一个修订。 内容通过关联表与一个或多个主题相关联。我给了一组 topic.id s T,并希望显示属于内容中一行的N个最近“已批准”的修订版,至少包含T.中的一个主题(“已批准”只是修订的枚举属性)
以下是模型和相关属性:
class Revision(Model):
__tablename__ = 'revision'
class Statuses(object): # enum
APPROVED = 'approved'
PROPOSED = 'proposed'
REJECTED = 'rejected'
values = [APPROVED, PROPOSED, REJECTED]
id = Column(Integer, primary_key=True)
data = Column(JSONB, default=[], nullable=False)
content_id = db.Column(db.Integer, db.ForeignKey('content.id'), nullable=False)
class Content(Model):
__tablename__ = 'content'
id = Column(Integer, primary_key=True)
topic_edges = relationship(
'TopicContentAssociation',
primaryjoin='Content.id == TopicContentAssociation.content_id',
backref='content',
lazy='dynamic',
cascade='all, delete-orphan'
)
revisions = relationship(
'Revision',
lazy='dynamic',
backref='content',
cascade='all, delete-orphan'
)
class TopicContentAssociation(Model):
__tablename__ = 'topic_content_association'
topic_id = Column(Integer, ForeignKey('topic.id'), primary_key=True)
content_id = Column(Integer, ForeignKey('content.id'), primary_key=True)
class Topic(Model):
__tablename__ = 'topic'
id = Column(Integer, primary_key=True)
这是我到目前为止所得到的:
revisions = session.query(Revision).outerjoin(Content).outerjoin(Topic).filter(
~exists().where(
and_(
Topic.id.in_(T),
Revision.status == Revision.Statuses.APPROVED
) )
).order_by(Revision.ts_created.desc()).limit(N)
并且发生了这个错误:
由于自相关,Select语句没有返回FROM子句;指定correlate(< tables>)以手动控制相关性。:
SELECT *
FROM topic, revision
WHERE topic.id IN (:id_1, :id_2, :id_3...)
AND revision.status = :status_1
有趣的是,如果我删除了and_运算符和第二个表达式(第3行,第5行和第6行),错误似乎就消失了。
奖励::)我还希望每行内容只显示一个修订版。如果有人点击了很多次,我不希望这些信息混乱。
同样,我对SQLAlchemy(实际上是关系数据库)非常陌生,因此非常感谢针对初学者的答案!
编辑:在.correlate(Revision)
子句修复后添加.where
,但我仍在努力弄清楚这里发生了什么。
答案 0 :(得分:0)
这是一个很晚的回复,但是如果有人找到这个主题,我会回信。
existing语句中的第二个表达式(Revision.status == Revision.Statuses.APPROVED)在查询中第二次调用表Revision。第一次调用此表是通过编写“ session.query(Revision)”
如果我们在PostgreSQL中“翻译”您的exist语句,则将是:
EXISTS (SELECT 1
FROM Revision
WHERE topic.id IN (:id_1, :id_2, :id_3...)
AND revision.status = :status_1
)
除非使用别名,否则不允许在同一查询中两次调用同一表(FROM修订版)。因此,您可以使用aliased()函数创建所需表的别名并解决您的问题。您的代码应该像这样:
from sqlalchemy.orm import aliased
aliasRev = aliased(Revision)
revisions = session.query(Revision).outerjoin(Content).outerjoin(Topic).filter(
~exists().where(
and_(
Topic.id.in_(T),
aliasRev.status == Revision.Statuses.APPROVED
) )
).order_by(Revision.ts_created.desc()).limit(N)