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

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

詳細はこちら
Ruby on Rails

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

Q&A

解決済

2回答

1653閲覧

railsで投稿一覧にユーザー名を表示する際に発生したNoMethodError in Posts#indexの原因を知りたい

Yuukou

総合スコア5

Ruby on Rails

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

0グッド

0クリップ

投稿2019/12/06 07:29

編集2019/12/09 03:16

前提・実現したいこと

railsにて簡単なSNSサービスを作っています。
その中の機能の一つとして投稿一覧にユーザー名の画像とユーザー名を表示させる機能を実装中に以下のエラーメッセージが発生しました。

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

NoMethodError in Posts#index /app/views/posts/index.html.erb where line #7 raised: undefined method `image_name' for nil:NilClass

該当のソースコード

erb

1 2 <img src="<%= "/user_images/#{post.user.image_name}" %>">

試したこと

コードの解法がある教材(Progate)を進めていたので、該当するファイルの答えをコピーして見ましたがエラーになってしまいました。

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

問題のposts/index.html.erbのコードはこちらです。

erb

1<div class="main posts-index"> 2 <div class="container"> 3 <% @posts.each do |post| %> 4 <div class="posts-index-item"> 5 <div class="post-left"> 6 7 <img src="<%= "/user_images/#{post.user.image_name}" %>"> 8 </div> 9 <div class="post-right"> 10 <div class="post-user-name"> 11 12 <%= link_to(post.user.name, "/users/#{post.user.id}") %> 13 </div> 14 <%= link_to(post.content, "/posts/#{post.id}") %> 15 </div> 16 </div> 17 <% end %> 18 </div> 19</div>

post.rbは以下のようになります。

ruby

1class Post < ApplicationRecord 2 validates :content, {presence: true, length: {maximum: 140}} 3 validates :user_id, {presence: true} 4 belongs_to :user 5 def user 6 return User.find_by(id: self.user_id) 7 end 8 # self.user_idのユーザーを戻り値として返す 9end

コントローラー内でのアクションは以下のようになります。

ruby

1class PostsController < ApplicationController 2 before_action :authenticate_user 3 4 def index 5 @posts = Post.all.order(created_at: :desc) 6 end 7 8 def show 9 @post = Post.find_by(id: params[:id]) 10 @user = @post.user 11 end 12 13 def new 14 @post = Post.new 15 end 16 17 def create 18 @post = Post.new( 19 content: params[:content], 20 user_id: @current_user.id 21 ) 22 if @post.save 23 flash[:notice] = "投稿を作成しました" 24 redirect_to("/posts/index") 25 else 26 render("posts/new") 27 end 28 end 29 30 def edit 31 @post = Post.find_by(id: params[:id]) 32 end 33 34 def update 35 @post = Post.find_by(id: params[:id]) 36 @post.content = params[:content] 37 if @post.save 38 flash[:notice] = "投稿を編集しました" 39 redirect_to("/posts/index") 40 else 41 render("posts/edit") 42 end 43 end 44 45 def destroy 46 @post = Post.find_by(id: params[:id]) 47 @post.destroy 48 flash[:notice] = "投稿を削除しました" 49 redirect_to("/posts/index") 50 end 51 52end 53

usersのコントローラーはこちらになります。

class UsersController < ApplicationController before_action :authenticate_user,{only: [:index,:show,:edit,:update]} before_action :forbid_login_user,{only:[:new,:create,:login_form,:login]} before_action :ensure_correct_user, {only: [:edit, :update]} def index @users = User.all end def show @user = User.find_by(id: params[:id]) end def new @user = User.new end def create @user = User.new( name: params[:name], email: params[:email], image_name: "default_user.jpg", password: params[:password] ) if @user.save session[:user_id] = @user.id # 登録されたユーザーのidを変数sessionに代入 flash[:notice] = "新規登録しました。" redirect_to("/users/#{@user.id}") else render("users/new") end end def edit @user = User.find_by(id: params[:id]) end def update @user = User.find_by(id: params[:id]) @user.name = params[:name] @user.email = params[:email] if params[:image] @user.image_name = "#{@user.id}.jpg" image = params[:image] File.binwrite("public/user_images/#{@user.image_name}", image.read) end if @user.save flash[:notice] = "登録内容を編集しました。" redirect_to("/users/#{@user.id}") else render("users/edit") end end def login_form end def login @user = User.find_by( email: params[:email], password: params[:password] ) #フォームに入力されたメールアドレスとパスワードを取得 if @user session[:user_id] = @user.id # 変数sessionに、ログインに成功したユーザーのidを代入 flash[:notice] ="ログインしました" redirect_to("/posts/index") else @error_message = "メールアドレスまたはパスワードが間違っています" @email = params[:email] @password = params[:password] render("users/login_form") end #@userが存在するかどうかを判定するif文作成 end def logout session[:user_id] = nil flash[:notice] = "ログアウトしました" redirect_to("/login") end def ensure_correct_user if @current_user.id != params[:id].to_i flash[:notice] = "権限がありません" redirect_to("/posts/index") end end # ログイン中のユーザー」と「編集しようとしているユーザー」が正しくない場合の処理 end

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

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

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

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

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

NCC1701

2019/12/06 11:25

コントローラーは全て記載しましょう
Yuukou

2019/12/06 12:26

ご指摘いただきありがとうございます。コントローラを追記しました。 もしまだ記載の足りない箇所がありましたらよろしくお願いします。
NCC1701

2019/12/06 13:47

`NoMethodError in Posts#index'のエラーはpost_controller.rb(PostsController)に`index’メソッドが無い、とのことです。掲載されたコントローラーはUsersContorollerなので、分析できません。
Yuukou

2019/12/06 14:32

申し訳ありません。コントローラーを変更しました。
guest

回答2

0

class Post に
belongs_to :user
を追加して下さい。

それでも駄目だった時は、Yuukouさんの指摘です。
user_idにvalidatinが入ってるので、そこに値は入っていることは保証されますが、belongs_toがないので、そのidにあたるUserがいないということかも。

belongs_toをつけた後、rails c にて
Post.first.user
してみるとわかります

投稿2019/12/08 22:48

winterboum

総合スコア23567

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

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

0

ベストアンサー

エラーメッセージがすべて掲載されていないので、想像で答えます。が、ほぼ合っているはずです。
エラー発生箇所の
<img src="<%= "/user_images/#{post.user.image_name}" %>">
で、post.user.image_nameが原因でしょう。
すなわち、
「postインスタンスのuserにはimage_nameというメソッドがないよ」のはずです。
多くの場合、これはpostインスタンスのuserが「nil」のケースが多いです。
(理屈上では、userにはimage_nameという属性(カラム)が無いよというパターンもありえますが、モデル設計の問題で発見しやすいので稀でしょう))

つまり、postとuserの関連付けのモデル設計はしたものの、具体的なインスタンスにおいて、保存がされていないケースです。
コンソール等で@post.userを確かめるとnil(か空配列)が返ってくると思います。
なので、post.userにuserインスタンス(@user)を代入しているか、代入ごsaveされているかを確認してみてください。

投稿2019/12/07 03:41

NCC1701

総合スコア1680

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

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

Yuukou

2019/12/07 14:08

ありがとうございます。 コンソールで確認してみたところ ``` Traceback (most recent call last): 1: from (irb):4 NoMethodError (undefined method `user' for nil:NilClass) Did you mean? super ``` という表示をされました。 そのため、post.userにuserインスタンス(@user)を代入しているか、代入後saveされているか、usersのコントローラーを確認したのですが、異常はないと思われます...。(見落としがあったら、すみません...。) usersのコントローラーを新たに追加しましたので、よろしければご覧ください。
winterboum

2019/12/08 00:47

それは、post=Post,なにがし を行わずにpost.user.image_nameをやってるからです。 で、 正しい答えを得るには正しい質問が必要です。 まず、エラーメッセージを全文載せましょう
Yuukou

2019/12/08 12:27

申し訳ございません。 エラーメッセージを編集しました。よろしければご覧ください。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.36%

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

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

質問する

関連した質問