前提・実現したいこと
メッセージの送受信機能を実装しました。
現段階ではUser
とRoom
の中間テーブルであるEntry
にid
が作成された時間順でメッセージ一覧が表示されています。
これを、メッセージが送信された時間順で一覧を表示したいです。
リレーション関連
user.rb
ruby
1class User < ApplicationRecord 2has_many :messages, dependent: :destroy 3has_many :entries, dependent: :destroy 4has_many :rooms, through: :entries 5end
room.rb
ruby
1class Room < ApplicationRecord 2has_many :messages, dependent: :destroy 3has_many :entries, dependent: :destroy 4has_many :users, through: :entries 5end
entry.rb
ruby
1class Entry < ApplicationRecord 2belongs_to :user 3belongs_to :room 4end
message.rb
ruby
1class Message < ApplicationRecord 2belongs_to :user 3belongs_to :room 4end
rooms_controller.rb
このコントローラ内で定義したEntry
をビュー側で繰り返し処理し、表示しています。
ruby
1class RoomsController < ApplicationController 2before_action :set_another_entries, only: [:index, :show] 3private 4def set_another_entries 5my_room_ids = @current_user.entries.pluck(:room_id) # ログイン中のユーザーの持つEntryを配列にする。 6# 上で定義した「ログイン中のユーザーの持つEntry」の内、自分以外の相手一覧を表示する。 7@another_entries = Entry.includes(:user, :room).where(room_id: my_room_ids).where('user_id != ?', @current_user.id).order(created_at: :desc) 8end 9end
試したこと
まず過去の質問を参考にしました。こちらの質問者さまは、私の場合で言うRoom
を繰り返し表示されているので、Message
テーブルを結合(join
)する事で解決されたようです。
ただ私の場合はEntry
を繰り返し表示させており、message
テーブルとのリレーションが出来ていません。
ですのでそれを明示しようと考えました。
entry.rb
ruby
1class Entry < ApplicationRecord 2belongs_to :user 3belongs_to :room 4has_many :messages, through: :room #この行を追加。 5end
message.rb
ruby
1class Message < ApplicationRecord 2belongs_to :user 3belongs_to :room 4has_one :entry, through: :room #この行を追加。 5end
その上でMessage
テーブルの結合(join
)を試みました。
rooms_controller.rb
ruby
1class RoomsController < ApplicationController 2before_action :set_another_entries, only: [:index, :show] 3private 4def set_another_entries 5my_room_ids = @current_user.entries.pluck(:room_id) 6@another_entries = Entry.includes(:user, :room).where(room_id: my_room_ids).where('user_id != ?', @current_user.id) 7@another_entries = @another_entries.joins(:messages).order("messages.created_at DESC") 8end
しかし以下のエラーが発生しました。
ActiveRecord::StatementInvalid in Rooms#index Showing C:/hoge/app/views/rooms/index.html.erb where line #12 raised: SQLite3::SQLException: ambiguous column name: user_id: SELECT "entries".* FROM "entries" INNER JOIN "rooms" ON "rooms"."id" = "entries"."room_id" INNER JOIN "messages" ON "messages"."room_id" = "rooms"."id" WHERE "entries"."room_id" IN (?, ?, ?, ?) AND (user_id != 20) ORDER BY messages.created_at DESC Extracted source (around line #12): 10 <div class="message-items"> 11 12 <% @another_entries.each do |e| %> 13 <div class="message-item"> 14 15 <div class="message-left">
恐らくリレーションを与えた事でEntry
とMessage
の両テーブルにuser_id
が存在する事になり、どちらのテーブルの物かを判別する事が出来ない、と言った内容なのかと思います。
scope
を使ってそれを明示する(?)と言った記事を見つけたので試してみましたが、また同様にambiguous column name
のエラーが表示され上手くいきませんでした。
room.rb
ruby
1class Room < ApplicationRecord 2has_many :messages, dependent: :destroy 3has_many :entries, dependent: :destroy 4has_many :users, through: :entries 5scope :select_entry, -> (entry) { where('user_id = ?', entry) } #この行を追加。 6end
実はこの後で「message
の中身が存在するEntry
のみを表示する」と言った記述もしていきたいので、出来ましたらEntry
とMessage
の両テーブルはリンクさせた上で実装出来れば一番有難いです。
どなたかご助言を頂けませんでしょうか。
補足情報(FW/ツールのバージョンなど)
ruby 2.6.4p104
RubyGems 3.0.3
Rails 5.2.3
追記
その後、以下の2点を無事実装する事が出来ました。
- メッセージが送信された時間順でトークルーム一覧を表示する。
message
の中身が存在しないEntry
は表示しない。
こちらの質問をご参考頂ければと思います。
回答1件
あなたの回答
tips
プレビュー
バッドをするには、ログインかつ
こちらの条件を満たす必要があります。
2020/12/24 04:15
2020/12/24 04:43
2020/12/24 05:58
2020/12/24 06:15
2020/12/24 06:55
2020/12/27 08:01
2020/12/27 23:36
2020/12/28 06:57