これは本当に問題ではありません、私はただ理解したいです。次のコードを検討します。
from sqlalchemy.ext.declarative import declarative_base
from sqlalchemy import *
from sqlalchemy.orm import sessionmaker, relationship
Base = declarative_base()
class AB(Base):
__tablename__= 'ab'
id_a = Column(Integer, ForeignKey('a.id', ondelete='CASCADE'), primary_key=True)
id_b = Column(Integer, ForeignKey('b.id', ondelete='CASCADE'), primary_key=True)
rel = Column(Unicode)
class A(Base):
__tablename__ = 'a'
id = Column(Integer, primary_key=True)
class B(Base):
__tablename__ = 'b'
id = Column(Integer, primary_key=True)
#1: doesn’t work try to set id_b to null
rel_a = relationship('AB')
# Works, but cascade='all' seems uneeded to me
rel_a = relationship('AB', cascade='all')
# Works
rel_a = relationship('AB', passive_deletes=True)
engine = create_engine('sqlite://', echo=True)
import logging
logger = logging.getLogger('sqlalchemy.engine.base.Engine')
logger.setLevel(logging.DEBUG)
handler = logger.handlers[0]
handler.setLevel(logging.DEBUG)
handler.setFormatter(logging.Formatter('%(levelname)s %(message)s', ''))
Base.metadata.create_all(engine)
sess = sessionmaker(engine)()
a1 = A()
b1 = B()
ab = AB()
sess.add_all([a1,b1])
sess.flush()
ab.id_a = a1.id
ab.id_b = b1.id
ab.rel = u'truite'
sess.add(ab)
sess.flush()
sess.delete(b1)
sess.flush()
AB
からの関連レコードが削除されたときに、テーブルからのレコードを削除したいB
。私は3種類の関係を試しました(Bテーブルをチェックしてください):
1:機能しません(AssertionError:依存関係ルールがインスタンス''の主キー列'ab.id_b'を空白にしようとしました)が、データベースで直接削除しようとすると、制約が正しく使用され、ABからのレコード削除されます。
2:動作します。生成されたデータベースは同一であるため、なぜこれが必要なのかわかりません(出力の差分を確認できます)
3:動作し、DB制約が動作します。
(3)を離しておくと、(2)が必要な理由がわかりません。これは、ondelete='cascade'
がすでに設定されており、生成されたDBが同一であるためです。私の推測では(1)で、SQLAlchemyには正しい動作をするのに十分な情報があります。
私は何かが足りないのですか?ありがとう。