1
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

SQLAlchemy 2.0 で複雑なテーブルをいろいろ作ったときのメモ

Last updated at Posted at 2024-06-07

書き足していきます。

table_args が sqlalchemy.exc.ArgumentError のエラーを吐く

テーブル定義のクラスに

    __table_args__ = (
        # データセットをまたがって同じグローバルIDをもつプロセスを扱うことがあるためデータセットIDとグローバルIDのセットでユニーク
        UniqueConstraint("hoge_id", "fuga_id", name="uq_samples_hoge_id_fuga_id")
    )

sqlalchemy.exc.ArgumentError: __table_args__ value must be a tuple, dict, or None

というエラーになります。正解は

    __table_args__ = 
    (
        # データセットをまたがって同じグローバルIDをもつプロセスを扱うことがあるためデータセットIDとグローバルIDのセットでユニーク
-        UniqueConstraint("hoge_id", "fuga_id", name="uq_samples_hoge_id_fuga_id")
+        UniqueConstraint("hoge_id", "fuga_id", name="uq_samples_hoge_id_fuga_id"),
    )

カンマが要ります。

単一継承テーブル (single table inheritance:STI) の子クラスで table_args が使えない

ちゃんとカンマを入れてもダメです。

Stackoverflow で同じことを聞いてるかたがいました( https://stackoverflow.com/questions/43832848/cant-define-table-args-on-a-child-class-in-a-single-table-inheritance-setup )古い質問ではありますが、最新版のSQLAlchemry 2.0 でもおなじワークアラウンドが使えました。

子クラスでは table_args が使えないので、クラスの外に定義を出します。クラスがSampleというクラスだった場合はクラス定義の外で

UniqueConstraint(Sammple.hoge_id, Sample.fuga_id, name="uq_samples_hoge_id_fuga_id")

と書くことで子クラスに(このケースでは複合ユニークキー制約)書くことが出来ます。

同じテーブルへの外部キーが二つあるときに AmbiguousForeignKeysError になる

こんなエラーが出ます。

sqlalchemy.exc.AmbiguousForeignKeysError: Could not determine join condition between parent/child tables on relationship ***.***** - there are multiple foreign key paths linking the tables.  Specify the 'foreign_keys' argument, providing a list of those columns which should be counted as containing a foreign key reference to the parent table.

片方だけで良いように思えますが。親子の relationship に両側とも foreign_keys パラメータがいります。

class Author(Base):
    __tablename__ = 'authors'
    id = Column(Integer, primary_key=True)
    name = Column(String, nullable=False)

    books = relationship('Book', foreign_keys='Book.author_id',     sub_books = relationship('Book', foreign_keys='Book.sub_author_id', back_populates='sub_author')

class Book(Base):
    __tablename__ = 'books'
    id = Column(Integer, primary_key=True)
    title = Column(String, nullable=False)
    author_id = Column(Integer, ForeignKey('authors.id'))
    sub_author_id = Column(Integer, ForeignKey('authors.id'))

    author = relationship('Author', foreign_keys=[author_id], back_populates='books')
    sub_author = relationship('Author', foreign_keys=[sub_author_id], back_populates='sub_books')

意図せず関係の無いテーブルへUPDATEやDELETEが発生する

どうも非同期利用時に relationship の cascade オプションで all を設定しているリレーションがあるとリレーション先が更新されたものとして扱われたりしてへんなクエリが発行されてしまうようです。公式のこのページの途中

The all cascade option implies the refresh-expire cascade setting which may not be desirable when using the Asynchronous I/O (asyncio) extension, as it will expire related objects more aggressively than is typically appropriate in an explicit IO context. See the notes at Preventing Implicit IO when Using AsyncSession for further background.

allでなく個別に指定してくれ、all に含まれる refresh-expire がわるさをするから、ということのようです。all という設定は save-update, merge, refresh-expire, expunge が含まれているとのことなので、save-update, merge, expunge を設定してみました。

リンク先にも記述がありました。

Avoid using the all cascade option documented at Cascades in favor of listing out the desired cascade features explicitly. The all cascade option implies among others the refresh-expire setting, which means that the AsyncSession.refresh() method will expire the attributes on related objects, but not necessarily refresh those related objects assuming eager loading is not configured within the relationship(), leaving them in an expired state.

詳細はわかりにくいのですが、関連オブジェクトが期限切れの状態になったものを明示的に削除したと勘違いしてしまうってことかな……。

1
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
1
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?