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

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

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

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

Ruby on Rails

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

Q&A

解決済

2回答

366閲覧

form_forでデータが保存されない

taishiaaaaa

総合スコア30

Ruby

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

Ruby on Rails

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

0グッド

0クリップ

投稿2018/12/06 12:49

編集2018/12/13 03:43

form_forで@inviteをデータベースに保存することができません。
下記にrouting、controller,viewのコードをまとめました。
恐らくnewとcreate内の@inviteの書き方が間違っているとは思うのですが正解が導き出せません。

urlに関しては:idを:usernameに変化させ、そのあとinviteをネストさせた形になっています。

privateでparamsをリファクタリングしていない点はご容赦ください。

回答いただけたら幸いです。

routes

1new_user_invite GET /:user_username/invites/new(.:format) invites#new

InvitesController

1class InvitesController < ApplicationController 2 3 def new 4 @user = User.find_by(username: params[:user_username]) 5 @invite = Invite.new 6 end 7 8 def create 9 @invite = Invite.new(fromid: params[:user_id], content: params[:content], title: params[:title]) 10 @invite.save 11 redirect_to user_invite_path(id: @invite.id) 12 end 13 14 def show 15 @user = User.find_by(username: params[:user_username]) 16 end 17end

new

1<%= form_for([@user, @invite]) do |f| %> 2 <%= f.label :title %> 3 <%= f.text_field :title %> 4 5 <%= f.label :content %> 6 <%= f.text_field :content %> 7 8 <%= hidden_field_tag :user_id, @user.id %> 9 <%= f.submit "投稿する"%> 10<% end %>

submitを実行すると

ActionController::UrlGenerationError in InvitesController#create

No route matches {:action=>"show", :controller=>"invites", :id=>nil, :user_username=>"tarou"}, possible unmatched constraints: [:id]

parameters
{"utf8"=>"✓",
"authenticity_token"=>"*********************",
"invite"=>{"title"=>"タイトル", "content"=>"コンテンツ"},
"user_id"=>"4",
"commit"=>"投稿する",
"user_username"=>"tarou"}

と表示されました。
idが付与されていないのは恐らく@inviteが保存されていないことが原因だと思われます。

<-------------------------------------------追記------------------------------------------>

invite.rb

invite.rb

1class Invite < ApplicationRecord 2 has_many :messages, foreign_key: "fromid" 3 belongs_to :users 4end 5

user.rb

class User < ApplicationRecord # Include default devise modules. Others available are: # :confirmable, :lockable, :timeoutable, :trackable and :omniauthable has_many :messages, foreign_key: "toid" has_many :invites, foreign_key: "fromid" devise :database_authenticatable, :registerable, :recoverable, :rememberable, :trackable, :validatable, :lockable, :timeoutable, :omniauthable def self.find_for_oauth(auth) user = User.where(uid: auth.uid, provider: auth.provider).first unless user user = User.create( uid: auth.uid, provider: auth.provider, email: User.dummy_email(auth), username: auth.info.nickname, password: Devise.friendly_token[0, 20], image: auth.info.image ) end user end def to_param username end private def self.dummy_email(auth) "#{auth.uid}-#{auth.provider}@example.com" end end

schema.rb

# This file is auto-generated from the current state of the database. Instead # of editing this file, please use the migrations feature of Active Record to # incrementally modify your database, and then regenerate this schema definition. # # Note that this schema.rb definition is the authoritative source for your # database schema. If you need to create the application database on another # system, you should be using db:schema:load, not running all the migrations # from scratch. The latter is a flawed and unsustainable approach (the more migrations # you'll amass, the slower it'll run and the greater likelihood for issues). # # It's strongly recommended that you check this file into your version control system. ActiveRecord::Schema.define(version: 2018_12_06_044716) do create_table "invites", force: :cascade do |t| t.integer "fromid" t.text "content" t.text "title" t.datetime "created_at", null: false t.datetime "updated_at", null: false end create_table "messages", force: :cascade do |t| t.integer "fromid" t.integer "toid" t.text "content" t.text "title" t.datetime "created_at", null: false t.datetime "updated_at", null: false end create_table "users", force: :cascade do |t| t.string "email", default: "", null: false t.string "encrypted_password", default: "", null: false t.string "reset_password_token" t.datetime "reset_password_sent_at" t.datetime "remember_created_at" t.integer "sign_in_count", default: 0, null: false t.datetime "current_sign_in_at" t.datetime "last_sign_in_at" t.string "current_sign_in_ip" t.string "last_sign_in_ip" t.string "confirmation_token" t.datetime "confirmed_at" t.datetime "confirmation_sent_at" t.string "unconfirmed_email" t.integer "failed_attempts", default: 0, null: false t.string "unlock_token" t.datetime "locked_at" t.datetime "created_at", null: false t.datetime "updated_at", null: false t.string "provider" t.string "uid" t.string "username" t.string "image" t.index ["confirmation_token"], name: "index_users_on_confirmation_token", unique: true t.index ["email"], name: "index_users_on_email", unique: true t.index ["reset_password_token"], name: "index_users_on_reset_password_token", unique: true t.index ["unlock_token"], name: "index_users_on_unlock_token", unique: true end end

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

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

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

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

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

troch

2018/12/07 12:54

【form_forで@inviteをデータベースに保存することができません。】とありますが、もう少し具体的な状況を共有していただけますでしょうか(例えば何かしらのエラーが出ているのであればそのエラー文など)
taishiaaaaa

2018/12/11 04:54

エラー文と渡されているパラメーターを追記しました。お時間あるとき見ていただけたら嬉しいです。
troch

2018/12/11 08:11

確認しました。共有ありがとうございます。taishiaaaaaさんのおっしゃる通り、@invite.saveが失敗しているのかもしれません…
troch

2018/12/11 08:13

追加で申し訳ないのですが、app/models/invite.rbとapp/models/user.rbと、db/schema.rbの中身を見せていただけますでしょうか。
taishiaaaaa

2018/12/13 03:44

回答いただいているのに遅くなって申し訳ありません。 invite.rb user.rb schema.rbを追記しました。 よろしくお願いします。
troch

2018/12/13 10:15

確認しました。ありがとうございます。回答を記載いたしましたので、お時間あるときに見ていただけたら幸いです。
guest

回答2

0

ベストアンサー

invite.rb

class Invite < ApplicationRecord has_many :messages, foreign_key: "fromid" belongs_to :user, foreign_key: "fromid" end

としてみてはどうでしょうか。

外部キーがRailsの慣習であるuser_idではなくfromidになっているため、
モデルファイル側で明示的に指定しなければならないと思います。

投稿2018/12/13 10:14

troch

総合スコア349

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

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

taishiaaaaa

2018/12/13 15:49

ありがとうございます、解決しました! モデルの関連付けは調べても調べても慣習通りの記事や公式ガイドしか出てこなかったため、正しく書けているか不安でした。これからは慣習通りの設計やコーディングを心掛けたいと思います。 いつも頼ってばかりですみません、回答リクエストはほどほどにしてもう少し自重します。
troch

2018/12/14 01:31

解決してよかったです! > いつも頼ってばかりですみません、回答リクエストはほどほどにしてもう少し自重します。 とんでもないです。こちらも勉強になりました。 私は モデル名_id というRailsの慣習に沿ったやり方しか試したことがなかったので、 foreign_key:オプションを使った指定方法を学ぶ切っ掛けになりました。 ありがとうございます。 引き続き頑張ってください!
guest

0

@invite = Invite.new(fromid: params[:user_id], content: params[:content], title: params[:title])

ここでnewメソッドでInviteクラスのインスタンスを生成するとき、id:を指定していないので@invite.idにはデータが保存されていませんね。
@inviteに保存はできていますが、idにはデータが入っていない状態です。

ここで主キーを保存しません。解答欄汚し、失礼いたしました

投稿2018/12/11 09:32

編集2018/12/11 10:10
ymt4832

総合スコア17

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

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

troch

2018/12/11 09:55

Invite.new(id: ~~~)のようにInviteモデルの主キーが指定されていないからデータが保存できない、という指摘でしょうか。 ActiveRecordでレコードを追加する際に主キーを指定しなくても自動で連番が振られるのではないでしょうか。 例えば以下のサイトでRailsのモデル生成について解説がされていますが、主キーは指定していません。 https://ruby-rails.hatenadiary.com/entry/20140724/1406142120 また主キーのようにユーザ側の操作で恣意的に変えられたくないデータについては、 strong parametersで許可すべきではないのでは…と思います。 むしろ別の箇所(バリデーションやアソシエーション)に原因があるのでは、と思っています。
troch

2018/12/11 13:02

追記: いえ、こちらこそ配慮のない言い方になってしまったかもしれず、失礼致しました…; ymt4832さんが回答しようとしたことは良いことだと思うので、 あまりお気になさらず、頑張ってください。
taishiaaaaa

2018/12/13 15:49

回答ありがとうございます。 解決いたしました。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問