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

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

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

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

Ruby on Rails

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

Q&A

解決済

2回答

3914閲覧

1つのフォームで親子関係(1:多)のモデルを更新したい

ohtakazuki

総合スコア31

Ruby

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

Ruby on Rails

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

0グッド

0クリップ

投稿2017/09/26 06:50

お世話になります。

Ruby on Rails(5.0.6)で、1つのフォームで親子関係(1:多)のモデルの項目を展開し、POSTによって親と子両方のレコードを追加(または更新)しようとしています。
親項目
子1
子2
子3
・・・

画面表示、strong parameterによる受け取りなど、とても苦労してやっと出来ました。。。

ところが、save時に子->親の参照キー(当例では「job_id」)に値が入らないせいか、Columns job must exist とエラーになってしまいます><

あと一歩で完成なのですが、どう調べても分かりません。。。
アドバイス頂けると嬉しいです。よろしくお願いします!

models

job.rb

class Job < ApplicationRecord has_many :columns, :dependent => :destroy accepts_nested_attributes_for :columns, :allow_destroy => true end

column.rb

class Column < ApplicationRecord belongs_to :job end

migrate files

class CreateJobs < ActiveRecord::Migration[5.0] def change create_table :jobs do |t| t.string :name t.timestamps end end end
class CreateColumns < ActiveRecord::Migration[5.0] def change create_table :columns do |t| t.references :job, foreign_key: true t.string :name t.timestamps end end end

controller

jobs_controller.rb

class JobsController < ApplicationController def new @job = Job.new 3.times {|i| @job.columns.build(name:"name#{i}") } end def create @job = Job.new(job_params) if @job.save # 成功時の処理 else render 'new' end end private def job_params params.require(:job).permit(:name, :format, :char_code, columns_attributes: [:id, :name, :data_type]) end end

view

new.html.erb

<div class="row"> <%= form_for(@job, html: {class: 'form-horizontal'}) do |f| %> <div class="form-group"> <%= f.label :name, class: 'col-md-2 control-label' %> <div class="col-md-10"> <%= f.text_field :name, class: 'form-control' %> </div> </div> <div class="form-group"> <div class="col-md-10 col-md-offset-2"> <%= f.submit "save", class: "btn btn-primary" %> </div> </div> <div class="form-group"> <%= f.fields_for :columns do |builder| %> <%= render 'column_fields', :f => builder %> <% end %> </div> <% end %> </div>

_column_fields.html.erb

<div> <%= f.text_field :id %> <%= f.text_field :job_id %> <%= f.text_field :name %> </div>

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

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

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

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

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

guest

回答2

0

params.require(:job).permit(:name, :format, :char_code, columns_attributes: [:id, :name, :data_type])

params.require(:job).permit(:name, :format, :char_code, columns_attributes: [:id, :name, :data_type, :job_id])

にしてみてください。

投稿2017/09/26 07:27

y_y3

総合スコア179

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

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

ohtakazuki

2017/09/26 07:54

ご回答ありがとうございます。残念ながらエラーメッセージ変わらず「Columns job must exist」でした。 debuggerを入れて確認しましたが、 create アクション の if @job.save は false となってしまいます。 @job.columns.first.job_idはnilでした(これはsaveが失敗してrollbackされたから、ですよね) (byebug) @job #<Job id: nil, name: "teste", format: "CSV", char_code: "utf-8", created_at: nil, updated_at: nil> (byebug) @job.columns.first #<Column id: nil, job_id: nil, name: "name0", data_type: "Faker", data_key: nil, created_at: nil, updated_at: nil>
y_y3

2017/09/26 08:14

外部キー制約がかかってしまっているので t.references :job, foreign_key: true ではなく  t.integer :job_id とすれば保存できるようになると思います。
ohtakazuki

2017/09/26 08:40

何度もご回答ありがとうございます! 教えて頂いたように修正し、rails db:migrate:reset し、再度実行してみましたが、エラーメッセージは変わりませんでした。 また、親子関係を保って保存したいです(親レコードをsaveしたときに自動取得されるidを子のjob_idにも設定したい)
y_y3

2017/09/26 09:25

外部キー制約が無い場合に「Columns job must exist」は出ないと思いますが。job_idにvalidate掛かってないですか?それかrailsの再起動をしていないかですね。 この方法でやれば親子関係を保って保存することができると思います
guest

0

自己解決

自己解決しました。belongs_to の場合、optional: true を付けないと、外部キーがnil不可になり保存できないようです(上記エラーはoptional:trueにより出ていたようです)。
https://github.com/rails/rails/issues/18233
親から子へjob_idが設定される前に、外部キーのoptionalの検査が行われるのかな?
column.rbの「belongs_to :jobを「belongs_to :job, optional: true
」に変更して解決しました。
コメントありがとうございました!

投稿2017/09/27 06:01

ohtakazuki

総合スコア31

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

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

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.50%

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

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

質問する

関連した質問