質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.48%
ORM

ORM(オブジェクト関係マッピング)はオブジェクト指向のシステムとリレーショナルデータベースの間でマッピングを行う技術です。

Python 3.x

Python 3はPythonプログラミング言語の最新バージョンであり、2008年12月3日にリリースされました。

SQLAlchemy

SQLAlchemyとはPython 用のORMライブラリです。MIT Licenceのオープンソースとして提供されています。

Python

Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

Q&A

0回答

1046閲覧

【SQLAlchemy】relationshipでmany to manyを定義しているテーブルにinsertするとduplicateエラーになる

退会済みユーザー

退会済みユーザー

総合スコア0

ORM

ORM(オブジェクト関係マッピング)はオブジェクト指向のシステムとリレーショナルデータベースの間でマッピングを行う技術です。

Python 3.x

Python 3はPythonプログラミング言語の最新バージョンであり、2008年12月3日にリリースされました。

SQLAlchemy

SQLAlchemyとはPython 用のORMライブラリです。MIT Licenceのオープンソースとして提供されています。

Python

Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

0グッド

0クリップ

投稿2020/01/18 15:44

前提・実現したいこと

以下のようなテーブル構成で、Many to Many のリレーションを作っています。

productsテーブル

Columndescription
product_id商品ID
title商品名

genresテーブル

Columndescription
genre_idジャンルID
genre_nameジャンル名

products_in_genresテーブル(中間テーブル)

Columndescription
product_id商品ID
genre_idジャンル名

そして、DBへの登録処理を下記のようなコードで行おうとしています。

  1. IDからデータを取得(商品に紐づいた、ジャンル一覧も取得できる)
  2. 商品マスタ情報の設定
  3. ジャンルマスタ情報の設定
  4. 商品マスタとジャンルマスタ、中間テーブルを一度に登録

python

1class ProductGenre(db.DbBase): 2 __tablename__ = 'products_in_genres' 3 4 product_id = Column(String(20), 5 ForeignKey('dsc_products.product_id', ondelete="CASCADE", onupdate="CASCADE"), 6 primary_key=True, comment='商品ID') 7 genre_id = Column(String(10), 8 ForeignKey('dsc_genres.genre_id', ondelete="CASCADE", onupdate="CASCADE"), 9 primary_key=True, comment='ジャンルID') 10 11 12class Products(db.DbBase): 13 __tablename__ = "products" 14 15 product_id = Column(String(20), primary_key=True, comment='商品ID') 16 title = Column(String(300), nullable=False, comment='タイトル名') 17 genres = relationship('Genres', secondary=ProductGenre.__tablename__) 18 19 20class Genres(db.DbBase): 21 22 __tablename__ = 'genres' 23 24 genre_id = Column(String(10), primary_key=True, comment='ジャンルID') 25 genre_name = Column(String(50), nullable=False, comment='ジャンル名') 26 products = relationship('Products', secondary=ProductGenre.__tablename__) 27 28 29def product_register(info: dict) -> None: 30 """ 31 登録処理 32 """ 33 # 2.商品マスタ情報の設定 34 product = Products() 35 product.product_id = info.get('product_id') 36 product.title = info.get('title') 37 38 # 3.ジャンルマスタ情報の設定 39 genres_list = list() 40 for genre_id, genre_name in info.get('genres').items(): 41 genre = Genres() 42 genre.genre_id = genre_id 43 genre.genre_name = genre_name 44 45 genres_list.append(genre) 46 # 4.商品マスタとジャンルマスタ、中間テーブルを一度に登録 47 product.genres = genres_list 48 db.session.add(product) 49 db.session.commit() 50 51if __name__ == '__main__': 52 info = get_product_info("AAAA") #1 53 product_register(info) 54 info = get_product_info("BBBB") #1 55 product_register(info) 56 57

発生している問題

当然、同一のジャンルIDが複数の商品IDに紐づきます。

1つ目のデータでは問題なく3テーブルに一度に登録できます。
2つ目以降で、登録済みのジャンルIDを登録しようとするとキー重複エラーとなってしまします。

もちろん、
ジャンルマスタ

商品マスタ

中間テーブル
の順番にinsertすることでは実現できます。

しかし、上記コードのようにリレーションを定義しており、1レコード目のように3テーブルを一括で登録できれば便利です。
キーがすでにあれば、重複したデータのみ関連マスタにはinsertしないというようなことはできないのでしょうか?

よろしくお願いいたします。

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

まだ回答がついていません

会員登録して回答してみよう

アカウントをお持ちの方は

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問