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

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

ただいまの
回答率

88.92%

Ruby 3つの異なるテーブルデータを一度に保存したい

解決済

回答 1

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 154

Malson

score 10

前提・実現したいこと

rubyで、3つの異なるテーブルのデータを一度に保存したい。
モデルFolderのcreateアクション実行時に以下の3つの情報を格納したい

  • Folderテーブルに格納すべき情報、
  • 1:多数の関係にある親子関係のモデルCardの情報、
  • 1:多数の関係にあるモデルFolderUsersの情報

発生している問題・エラーメッセージ

エラーは表示されていないが、controllerで@folderが保存できない。

パラメータを確認したところidが全てnilになっている。何故idが生成されないかの理由が判然とせず、またcardsに関しては何一つデータを取得することもできていない

[1] pry(#<FoldersController>)> @folder
=> #<Folder:0x00007f9d009ce7e0 id: nil, name: "aaaa">
[2] pry(#<FoldersController>)> @folder.cards
=> [#<Card:0x00007f9d00a2c2a0 id: nil, omote: nil, ura: nil, folder_id: nil>]
[3] pry(#<FoldersController>)> @folder.folder_users
=> [#<FolderUser:0x00007faa9d074f78 id: nil, folder_id: nil, user_id: 4>]

該当のソースコード

class FoldersController < ApplicationController

  def index
    @folders = Folder.all
  end

  def new
    @folder = Folder.new
    @folder.cards.build
  end

  def create
    @folder = Folder.create(folder_params)
    @folder.folder_users.new(user: current_user)
    if @folder.save
      redirect_to :root, notice: '新しいフォルダを作成しました'
    else
      render :new
    end
  end
  ~~~~~~~~~~~~~~~~~~~~~~~略~~~~~~~~~~~~~~~~~~~~~~~~~~~
  private
  def folder_params
    params.require(:folder).permit(
      :name, 
      cards_attributes:  {
        omote: [], 
        ura: [],
        folder_id: []
      }
    )
  end
end

試したこと

補足情報(FW/ツールのバージョンなど)

>> FolderUser
=> FolderUser(id: integer, folder_id: integer, user_id: integer)
>> Folder
=> Folder(id: integer, name: string)
>> Card
=> Card(id: integer, omote: string, ura: string, folder_id: integer)
  ~~~略~~~
  resources :folders, only: [:index, :new, :create, :edit, :update] do
    resources :cards, only: [:index, :new, :create, :edit]
  end
  ~~~略~~~

アドバイスに従い以下のファイルを追記いたします

= form_with model: folder, local: true do |f|
      = f.label :name, "フォルダ名"
      = f.text_field :name
  = f.fields_for :cards do |c|
        = c.label :omote, "カード(問題)"
        = c.text_area :omote
        = c.label :ura, "カード(解答)"
        = c.text_area :ura
      = f.submit
class Folder < ApplicationRecord
  has_many :folder_users
  has_many :users, through: :folder_users
  has_many :cards, dependent: :destroy
  accepts_nested_attributes_for :cards
  validates :name, presence: true
end
class Card < ApplicationRecord
  belongs_to :folder, optional: true
  validates :omote, presence: true
  validates :ura, presence: true
end


参考にした記事など
https://qiita.com/hmuronaka/items/580b977834d4b3010454
https://qiita.com/tseno/items/70d39f7c0e8e1c779b5e

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

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

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

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

質問への追記・修正、ベストアンサー選択の依頼

  • kojiro12345

    2020/07/10 11:45

    View, Modelのファイルと参考にした記事があれば記載した方がいいと思います。

    キャンセル

  • Malson

    2020/07/10 12:03

    kojiro12345様、
    アドバイスありがとうございます。補足事項に追記させていただきました。
    質問の仕方や、問題がありそうな部位のピックアップなどまだハッキリとはできないため、申し訳ございません。
    記載内容でわかりにくいところや、伝えるべきことがあれば御指南いただければと思います。
    よろしくお願いいたします。

    キャンセル

回答 1

checkベストアンサー

+1

foldersとcardsにはuser_idがないですが、folder_paramsでuser_idを与えており、Folder.create(folder_params)で、パラメーターをセットしているときにfolderにはuser_idがないというエラーかと思います。
.merge(user_id: current_user.id)を外せば、foldersとcardsの登録できると思います。

folder_usersも登録したいということであれば、
それはそもそも3つのテーブルを同時に登録するということになります。(もしこれがやりたかったことであれば、質問を正しくしましょう。そうでなければ、スルーしてください。)

コントローラーファットになっていくので、
いい方法ではないですが、
手っ取り早くfolder_usersも保存するのは、下記のようにすることでしょうか。

  def create
    @folder = Folder.new(folder_params) #ここでエラー
    @folder.folder_users.new(user: current_user) #追加
    if @folder.save
      redirect_to :root, notice: '新しいフォルダを作成しました'
    else
      render :new
    end
  end

  private
  def folder_params
    params.require(:folder).permit(
      :name, 
      cards_attributes:  {
        omote: [], 
        ura: [],
        folder_id: []
      }
    )
    # .mergeを削除
  end

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2020/07/11 00:44

    kojiro12345様、
    ありがとうございます。

    精進が足りず、もはやどこがおかしいのかすらわかっていないのですが、子ストロングパラメータにfolder_idなどforeign_keyは記述は必要ないということですね。

    [ ]を付けたのは、ストロングパラメーターの入力が成功したあとで多数入力できる様に変更したいと考えていて配列を意識したのですが、一つの時は一つの要素しか持たない配列にならないのでしょうか?
    それとも属性の名前だけ記述するので、余計ってことでしょうか?

    キャンセル

  • 2020/07/11 00:51

    kojiro12345様、
    長々とお付き合いいただき、本当にありがとうございました。
    無事解決することができました。今回のことはしっかりと復習して身につけます。

    キャンセル

  • 2020/07/11 06:35

    そうですね。
    子モデルの同時登録は、少し時期尚早な気がしました。
    今回のこともそうですが、Railsについてもっと基本を学んだ方がいいと思います。

    > [ ]を付けたのは、ストロングパラメーターの入力が成功したあとで多数入力できる様に変更したいと考えていて配列を意識したのですが、

    やりとりしていて思いましたが、Railsはあなたの思い通りではなく、Railsの仕様通りに動きます。
    もちろん私のいう通りでもありません。
    思い込みや誤った理解を応用しているように見えます。
    思い込みや誤った理解は、そもそも推測であるはずなので、動作して確かめられていないものは動かして確かめる癖をつけた方がいいです。着実に事実に基づいて理解して実装するようにした方が結果として早く身につくと思います。

    >[ ]を付けたのは、ストロングパラメーターの入力が成功したあとで多数入力できる様に変更したいと考えていて配列を意識したのですが、一つの時は一つの要素しか持たない配列にならないのでしょうか?
    それとも属性の名前だけ記述するので、余計ってことでしょうか?

    cardsを複数作るようにして、paramsの中身を見れば解は得られると思います。
    が、cardsの数を動的に行うのは厳しいと思うので、2つ分のcardなど数を決めうちでやることをお勧めします。
    そうやってステップを作って、確実に仕様を理解していくことをお勧めします。Railsが動くのでRailsで動かして反応を見るのが何よりも信頼できるし、紛れもない事実です。
    頑張ってください。

    キャンセル

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

  • ただいまの回答率 88.92%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る