🎄teratailクリスマスプレゼントキャンペーン2024🎄』開催中!

\teratail特別グッズやAmazonギフトカード最大2,000円分が当たる!/

詳細はこちら
Ruby on Rails 5

Ruby on Rails 5は、オープンソースのWebアプリケーションフレームワークです。「同じことを繰り返さない」というRailsの基本理念のもと、他のフレームワークより少ないコードで簡単に開発できるよう設計されています。

SQLite

SQLiteはリレーショナルデータベース管理システムの1つで、サーバーではなくライブラリとして使用されている。

Model

MVCモデルの一部であるModelはアプリケーションで扱うデータとその動作を管理するために扱います。

Q&A

解決済

2回答

642閲覧

rails5のmodel内のアソシエーションのforeign_keyの書き方

raitehu

総合スコア13

Ruby on Rails 5

Ruby on Rails 5は、オープンソースのWebアプリケーションフレームワークです。「同じことを繰り返さない」というRailsの基本理念のもと、他のフレームワークより少ないコードで簡単に開発できるよう設計されています。

SQLite

SQLiteはリレーショナルデータベース管理システムの1つで、サーバーではなくライブラリとして使用されている。

Model

MVCモデルの一部であるModelはアプリケーションで扱うデータとその動作を管理するために扱います。

0グッド

2クリップ

投稿2019/09/19 15:07

編集2019/09/19 15:54

versionなど

  • ruby 2.6.4
  • rails 5.2.3
  • sqlite3 3.29.0

やりたいこと

お金の受け渡し記録のようなものを作りたい。
Membersテーブルにidとユーザ名。
MoneyRecordsテーブルにidと差出人、受取人、金額が記録されている。
参考にしたのは以下の記事:
[Rails] 同じmodelを参照する外部キーを一つのmodelでもつ方法

スキーマ

schema.rb

1# 一部抜粋 2 create_table 'members', force: :cascade do |t| 3 t.string 'member_name' 4 t.string 'email' 5 t.string 'password_digest' 6 t.string 'remember_token' 7 t.datetime 'created_at', null: false 8 t.datetime 'updated_at', null: false 9 end 10 11 create_table 'money_records', force: :cascade do |t| 12 t.float 'price' 13 t.integer 'receiver_id' 14 t.integer 'sender_id' 15 t.datetime 'created_at', null: false 16 t.datetime 'updated_at', null: false 17 t.index ['receiver_id'], name: 'index_money_records_on_receiver_id' 18 t.index ['sender_id'], name: 'index_money_records_on_sender_id' 19 end

モデル

money_record.rb

1class MoneyRecord < ApplicationRecord 2 belongs_to :receiver , class_name: 'Member', foreign_key: 'receiver_id' 3 belongs_to :sender , class_name: 'Member', foreign_key: 'sender_id' 4end

member.rb

1class Member < ApplicationRecord 2 has_many :receiver_records , class_name: 'MoneyRecord', foreign_key: 'receiver_records_id' 3 has_many :sender_records , class_name: 'MoneyRecord', foreign_key: 'sender_records_id' 4end

できたこと/できなかったこと

rails consoleで

bash

1member_1 = Member.new(member_name: "hoge") 2member_1.save 3member_2 = Member.new(member_name: "fuga") 4member_2.save 5 6mr = MoneyRecord.new 7mr.sender = member_1 8mr.receiver = member_2

でsender, receiverにMemberを割り当てることはできましたが、
mr.save! を実行すると

bash

1ActiveRecord::StatementInvalid: SQLite3::SQLException: no such table: main.sender

というエラーが出てdbに登録できません。
おそらくモデル内のアソシエーションの記述がまずいのだと思いますが、
どのようにすればこの問題が解消するでしょうか?

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

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

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

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

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

guest

回答2

0

自己解決

結局あのあといろいろと調べた結果、
Railsが参照先のテーブルを解決できない場合はマイグレーションファイルに
table_to オプションを使用して明示的に教えてあげる必要があるということが分かりました。

詳しくはQiitaにまとめましたのでそちらをご覧ください↓
1つのモデルに別の1つのモデルを複数個持たせる方法[Rails5]

winterboumさん、ご回答ありがとうございました。

投稿2019/09/26 13:14

raitehu

総合スコア13

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

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

0

関連の記述は問題ないように思えます。

no such table: main.sender というのがなんとも、、、

class MoneyRecord の定義は2行で全てですか?
出さなかった所に何か有るかも

あと migationかdb/schema.rbかをみせて下さい。

投稿2019/09/20 07:54

winterboum

総合スコア23567

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

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

raitehu

2019/09/20 14:11

ご回答ありがとうございます。 試しに簡易版を作成してrails consoleでデータを登録しようとしてみましたが、同様の `no such table` で登録できず、、、 以下にmigrationファイルとschemaファイルを貼ります。 ```create_money.rb class CreateMoney < ActiveRecord::Migration[5.2] def change create_table :money do |t| t.references :receiver, foreign_key: true t.references :sender, foreign_key: true t.timestamps end end end ``` ```create_members.rb class CreateMembers < ActiveRecord::Migration[5.2] def change create_table :members do |t| t.string :name t.timestamps end end end ``` ```db/schema.rb ActiveRecord::Schema.define(version: 2019_09_19_160323) do create_table "members", force: :cascade do |t| t.string "name" t.datetime "created_at", null: false t.datetime "updated_at", null: false end create_table "money", force: :cascade do |t| t.integer "receiver_id" t.integer "sender_id" t.datetime "created_at", null: false t.datetime "updated_at", null: false t.index ["receiver_id"], name: "index_money_on_receiver_id" t.index ["sender_id"], name: "index_money_on_sender_id" end end ``` ```member.rb class Member < ApplicationRecord has_many :sender_records, class_name: 'Money', foreign_key: 'sender_id' has_many :receiver_records, class_name: 'Money', foreign_key: 'receiver_id' end ``` ```money.rb class Money < ApplicationRecord belongs_to :receiver, class_name: 'Member', foreign_key: 'receiver_id' belongs_to :sender, class_name: 'Member', foreign_key: 'sender_id' end ```
winterboum

2019/09/20 21:04

エラーメッセージで no such table: main.sender となる所は解せませんが、一つ問題が有るのは判りました。 モデル MoneyRecord を定義してありますが、DBにそれに対応するtableがありません。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.36%

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

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

質問する

関連した質問