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

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

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

Rubyはプログラミング言語のひとつで、オープンソース、オブジェクト指向のプログラミング開発に対応しています。

Active Record

Active Recordは、一つのオブジェクトに対しドメインのロジックとストレージの抽象性を結合するデザインパターンです。

Q&A

解決済

1回答

3289閲覧

activerecord-import(multi-level)の使い方

kanedayo

総合スコア8

Ruby

Rubyはプログラミング言語のひとつで、オープンソース、オブジェクト指向のプログラミング開発に対応しています。

Active Record

Active Recordは、一つのオブジェクトに対しドメインのロジックとストレージの抽象性を結合するデザインパターンです。

1グッド

0クリップ

投稿2016/02/23 06:53

編集2016/02/24 01:37

activerecord-import(multi-level)の使い方で悩んでいます。
実験用のコードを書いてみましたが、思うようにデータベースが更新されません。
コードの書き方や考え方で間違っているところをご指摘いただけると幸いです。

リンク : activerecord-import(multi-level)

$ ruby -v
ruby 1.9.3p551 (2014-11-13 revision 48407) [x86_64-darwin14.0.0]

ruby

1# https://github.com/zdennis/activerecord-import/wiki#multi-level-example 2 3require "pp" 4require "rubygems" 5require "active_record" 6require "activerecord-import" 7 8# 環境設定 9ActiveRecord::Base.establish_connection( 10 adapter: "sqlite3", 11 database: "main.db" 12) 13 14# クラス宣言 15class Book < ActiveRecord::Base 16 has_many :reviews 17end 18class Review < ActiveRecord::Base 19 belongs_to :book 20end 21 22# テーブルの作成 23class CreateBook < ActiveRecord::Migration 24 def self.up 25 create_table :books, force: true do|t| 26 t.string :name 27 end 28 end 29end 30CreateBook.new.up 31class CreateReview < ActiveRecord::Migration 32 def self.up 33 create_table :reviews, force: true do|t| 34 t.integer :book_id 35 t.string :title 36 end 37 end 38end 39CreateReview.new.up 40 41# エントリーの追加 42books = [] 4310.times do |i| 44 book = Book.new(:name => "book #{i}") 45 book.reviews.build(:title => "Excellent") 46 books << book 47end 48Book.import books, recursive: true 49 50# レコード取得 51pp Book.all # OK : [#<Book id: ... 52pp Review.all # NG : []
miyamiya👍を押しています

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

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

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

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

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

guest

回答1

0

ベストアンサー

んー、この機能は、sqliteやMySQLはサポートしてないようです。PostgresSQLでは有効です。
おそらく、primary_keyがNULLの状態でもセットできるようなデータベース側でのサポートが必要です。
機能を持っているかは、アダプターの
support_setting_primary_key_of_imported_objects?
で判断しています。これを以下で無理やりtrueにしても、動かないので対応していないのかと思います。
importは、オブジェクトごとにbulk-insertすれば同じことはできそうです。

Ruby

1#無理やり機能をtrueにしたが失敗記録、正しくないかもしれません。 2# 環境設定 3require "pp" 4require "rubygems" 5require "active_record" 6require "activerecord-import" 7 8module SQLite3Adapter 9 def support_setting_primary_key_of_imported_objects? 10 true 11 end 12end 13 14con = ActiveRecord::Base.establish_connection( 15 adapter: "sqlite3", 16 database: "main.db" 17) 18p con.extend(SQLite3Adapter) 19p con.respond_to?(:support_setting_primary_key_of_imported_objects?) 20 21# クラス宣言 22class Book < ActiveRecord::Base 23 has_many :reviews 24end 25class Review < ActiveRecord::Base 26 belongs_to :book 27end 28 29# テーブルの作成 30class CreateBook < ActiveRecord::Migration 31 def self.up 32 create_table :books, force: true do|t| 33 t.string :name 34 end 35 end 36end 37CreateBook.new.up 38class CreateReview < ActiveRecord::Migration 39 def self.up 40 create_table :reviews, force: true do|t| 41 t.integer :book_id 42 t.string :title 43 end 44 end 45end 46CreateReview.new.up 47 48# エントリーの追加 49books = [] 5010.times do |i| 51 book = Book.new(:name => "book #{i}") 52 book.reviews.build(:title => "Excellent") 53 books << book 54end 55pp books 56pp books.first.reviews 57Book.import books, recursive: true 58 59# レコード取得 60pp Book.all # OK : [#<Book id: ... 61pp Book.first.reviews 62pp Review.all # NG : []

投稿2016/02/23 15:27

編集2016/02/23 15:39
miyamiya

総合スコア691

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

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

kanedayo

2016/02/24 02:31

ありがとうございます。 PostgressSQLでは動くという情報は貴重でした。 ご指摘の通り、bulk-insertを検討したところ、 そこから「activerecord-import」に辿り着きました。 > importは、オブジェクトごとにbulk-insertすれば同じことはできそうです。 この場合、オブジェクトの関連(この例だとbook_id)の管理で悩みが生じます。 uniqなbook_idをアプリ側で管理できれば、book_idを事前に詰めこんでbulk-insertできそうだと気付いたのですが、適切な実現方法がわかりません。
kanedayo

2016/02/25 11:19

その後、postgres特有のreturning句というのを知り、 「データベースに依存する」という意味を理解できた気がします。 また、その解決方法として、 採番テーブルやLAST_INSERT_ID()を用いた uniqueなIDの管理方法があることを知りました。 気付きを与えてくださりありがとうございました。 Postgresで利用できるRETURNING句の例: INSERT INTO books(name) VALUES('~') RETURNING id;
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.49%

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

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

質問する

関連した質問