SQLAlChemy:直接从一对多关系中删除对象,而不使用会话。删除()

2022-04-17 00:00:00 python sqlalchemy session one-to-many mysql

我有以下SQLAlChemy安装程序:

Base = declarative_base()

class Post(Base):
    __tablename__ = 'post'
    id = Column(Integer, primary_key=True)
    title = Column(String(30))
    comments = relationship('Comment', cascade='all')

class Comment(Base):
    __tablename__ = 'comment'
    id = Column(Integer, primary_key=True)
    post_id = Column(Integer, ForeignKey(Post.id, ondelete='CASCADE'), nullable=False)
    text = Column(Text)

有了它,我就可以创建与评论具有一对多关系的POST对象。我想在不引用会话的情况下处理帖子评论的创建和删除。在帖子中添加评论效果很好:

post = Post(title='some post')
comment = Comment(text='some comment')
post.comments.append(comment)

我的会话处理程序只知道POST,所以它将执行session.add(post),并自动将评论放入会话中,并在下一个session.commit()时与数据库同步。然而,删除评论就不是这样了。我希望只需执行以下操作即可删除评论:

post.comments.remove(comment)

但是,这会在下一个session.commit()上产生以下错误:

sqlalchemy.exc.OperationalError: (OperationalError) (1048, "Column 'post_id' cannot be null") 'UPDATE comment SET post_id=%s WHERE comment.id = %s' (None, 1L)
如何告诉SQLAlChemy不要使用post_id的NULL值更新注释(由于列上的NOT NULL约束,这是不允许的),而是删除该注释?我知道我可以这样做session.delete(comment),但由于我不需要将注释显式添加到会话中,因此我看不出为什么必须从会话中显式删除它。

我找到了几种用于级联删除相关对象的解决方案,但由于我从未对会话发出任何显式删除(帖子仍在那里),我认为这不适用。

编辑:我调整了示例以包括从帖子中删除的级联。现在它可以执行session.delete(post),并且所有评论都被删除。但我只想自动删除我从关系中删除的评论,而不是删除包含所有评论的整个帖子。

tl;dr:当我从一对多关系的关系列表中删除条目时,如何告诉SQLAlChemy发出DELETE语句而不是UPDATE语句?


解决方案

有关详细信息,请阅读文档的Configuring delete/delete-orphan Cascade部分,但基本上您还需要delete-orphan以及relationshipcascade选项:

class Post(Base):
    # ...
    comments = relationship('Comment', cascade="all, delete-orphan")

相关文章