あんパン

こしあん派

Ridgepoleでfk設定するときはnameが必要

Railsで

$ rails g model Hoge fuga:references
$ rake db:migrate
$ rm -rf db/migrate

みたいな感じでモデル新規作成時だけ一旦マイグレーションファイル作ってからRidgepoleのSchemafileにdumpして… みたいな運用してたら詰まった.

ActiveRecordのreferencesはadd_foreign_key "hoge", "fugas"みたいな感じで名前を設定しないので,このままdumpするとSchemafileにも名前なしの外部キーが出力される.これをDBにapplyしようとすると

Apply `db/schemas/Schemafile`
[ERROR] Foreign key name in `hoge` is undefined
rake aborted!
Command failed with status (1): [bundle exec ridgepole -Edevelopment -c con...]
...

みたいな感じでエラーを吐く.修正するには

  1. db/schema.rbのfkに名前を設定
  2. Schemafileのfkに名前を設定
  3. $ rake db:resetとか実行しておく

みたいなことすればよさそう.

ちゃんと名前を設定しようという話でした.


外部キー詳しくないまま使ってるのだけど,Ridgepoleで名前の設定を強制させる理由ってなんだろう.Rails側が無名で出力できるんだからわざわざraiseすることなくてwarnとかでとどめておけば良いのに.

ActiveRecordでSTIとrelationを組み合わせる

Rails(ActiveRecord)にはSTI(単一継承テーブル)があって,テーブルは1つだけどtypeというカラムでモデルを出し分けることができる.例えば,アンケートの項目には単一/複数選択があり,これらを保存する先のテーブルは1つにまとめたいがプログラム上では別のモデルとして扱いたい場合にSTIを使うと良い.

継承したモデルと別のテーブルとの間にリレーションを作成するとkeyが見つからないと怒られることがある.

has_manyforeign_keyを指定すれば良い.以下のように書ける.はず.

class Answer < ActiveRecord::Base
  has_many :item_answer_relation
  has_many :checks, through: :item_answer_relation
  has_many :radios, through: :item_answer_relation
end

class Item < ActiveRecord::Base
end

class Check < Item
  has_many :item_answer_relation, foreign_key: 'item_id'
  has_many :answers, through: :item_answer_relation
end

class Radio < Item
  has_many :item_answer_relation, foreign_key: 'item_id'
  has_many :answers, through: :item_answer_relation
end

参考: - ruby on rails - Single-table inheritance with relationships? - Stack Overflow


サンプルコード書いてて若干自信なくなってきたので間違ってたら教えてください