RailsにVue.jsを使用して非同期処理をさせたい
現在、RailsアプリケーションにVue.jsを用いてお気に入り機能の実装をしているのですが、クリック時にActiveRecord::RecordNotFound (Couldn't find Upload without an ID):
と表示される原因を解決したくご相談させて頂きました。
具体的な実装イメージとしては、クリック時にデータベースにidが保存され、クリック数に応じてカウント数が変動する実装になっております。
発生している問題・エラーメッセージ
クリック時のエラー
ActiveRecord::RecordNotFound in Api::FavoritesController#create Couldn't find Upload without an ID Extracted source (around line #12): 10 11 12 13 14 15 def create @upload = Upload.find(params[:upload_id])←該当箇所 current_user.favorites.create!(favorites_params) head :created end
エラー内容としては、お気に入りをする際にupload_idがnull
のためにお気に入りの保存ができないようになっているのが現状です。
今回は非同期処理で実装を進めているため、投稿内容などが保存されるレイアウトの処理はcomponentsディレクトリ
配下にあるFavoriteButton
に記述しております。
こちらの箇所でupload_idがnullと表示される原因があると考えております。
該当のソースコード
app/javascript/components/Favorite/FavoriteButton.vue
vue
1<template> 2 <div> 3 <div v-if="isFavorited" @click="deleteFavorite()"> 4 いいねを取り消す {{ count }} 5 </div> 6 <div v-else @click="registerFavorite()"> 7 いいねする {{ count }} 8 </div> 9 </div> 10</template> 11 12<script> 13import axios from 'axios' 14import { csrfToken } from 'rails-ujs' 15axios.defaults.headers.common['X-CSRF-TOKEN'] = csrfToken() 16 17export default { 18 props: ['userId', 'uploadId'], 19 data() { 20 return { 21 favoriteList: [] 22 } 23 }, 24 computed: { 25 count() { 26 return this.favoriteList.length 27 }, 28 isFavorited() { 29 if (this.favoriteList.length === 0) { return false } 30 return Boolean(this.findFavoriteId()) 31 } 32 }, 33 created: function() { 34 this.fetchFavoriteByUploadId().then(result => { 35 this.favoriteList = result 36 }) 37 }, 38 methods: { 39 fetchFavoriteByUploadId: async function() { 40 const res = await axios.get(`/api/favorites/?upload_id=${this.uploadId}`) 41 if (res.status !== 200) { process.exit() } 42 return res.data 43 }, 44 registerFavorite: async function() { 45 const res = await axios.post('/api/favorites', { upload_id: this.uploadId }) 46 if (res.status !== 201) { process.exit() } 47 this.fetchFavoriteByUploadId().then(result => { 48 this.favoriteList = result 49 }) 50 }, 51 deleteFavorite: async function() { 52 const favoriteId = this.findFavoriteId() 53 const res = await axios.delete(`/api/favorites/${favoriteId}`) 54 if (res.status !== 200) { process.exit() } 55 this.favoriteList = this.favoriteList.filter(n => n.id !== favoriteId) 56 }, 57 findFavoriteId: function() { 58 const favorite = this.favoriteList.find((favorite) => { 59 return (favorite.user_id === this.userId) 60 }) 61 if (favorite) { return favorite.id } 62 } 63 } 64} 65</script>
app/views/uploads/index.html.erb
erb
1 2<div id="favorite"> 3~~省略~~ 4 <div class="favorite-content"> 5 <% if user_signed_in? %> 6 <favorite-button :user-id="<%= current_user.id %>" :uplaod-id="<%= upload.id %>"></favorite-button> 7 <% end %> 8 </div> 9</div>
app/controllers/api/favorites_controller
controller
1# frozen_string_literal: true 2 3module Api 4 class FavoritesController < ActionController::API 5 before_action :authenticate_user! 6 7 def index 8 render json: Favorite.filter_by_upload(params[:upload_id]).select(:id, :user_id, :upload_id) 9 end 10 11 def create 12 @upload = Upload.find(params[:upload_id]) 13 current_user.favorites.create!(favorites_params) 14 head :created 15 end 16 17 def destroy 18 current_user.favorites.find(params[:id]).destroy! 19 head :ok 20 end 21 22 private 23 24 def favorites_params 25 params.require(:favorite).permit(:upload_id) 26 end 27 end 28end
app/models/favorite.rb
model
1# frozen_string_literal: true 2 3class Favorite < ApplicationRecord 4 belongs_to :user 5 belongs_to :upload 6 7 validates :user_id, presence: true 8 validates :upload_id, presence: true 9 10 scope :filter_by_upload, ->(upload_id) { where(upload_id: upload_id) if upload_id } 11end 12
/config/routes
# frozen_string_literal: true Rails.application.routes.draw do namespace :api, { format: 'json' } do resources :favorites, only: [:index, :create, :destroy] end devise_for :users get 'uploads/index' root to: "uploads#index" resources :uploads do resource :favorites, only: [:create, :destroy] end resources :users, only: [:show, :edit, :update] end
試したこと
・favorites_controllerに@upload = Upload.find(params[upload_id])
と記述してupload_idを取得できるようにしてみましたが、エラーは変わらずidがnullと表示されたままでした。
・ルーティングに原因がある可能性を考え、
resources :uploads do resource :favorites, only: [:create, :destroy] end
と記述してuploadのルーティングにネストして記述しましたが、こちらもエラー解決にはなりませんでした。
・Vue.jsのファイルで、apiが想定しているparamsに対して異なる値があるために正常に処理されない可能性があると考えておりますが、どの部分を修正すればいいのか分からない状態です。
const res = await axios.post('/api/favorites', { upload_id: this.uploadId }) { favorite: { upload_id: 実際の値 } } ==こちらにparamsとして処理される値が、apiが想定している値とは違うものが送られている可能性がある
補足情報(FW/ツールのバージョンなど)
開発環境
・rubymine
・ruby(3.0.1)
・Ruby on rails (6.1.3.1)
回答1件
あなたの回答
tips
プレビュー
バッドをするには、ログインかつ
こちらの条件を満たす必要があります。