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

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

新規登録して質問してみよう
ただいま回答率
85.35%
Vue.js

Vue.jsは、Webアプリケーションのインターフェースを構築するためのオープンソースJavaScriptフレームワークです。

Ruby

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

Ruby on Rails

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

JavaScript

JavaScriptは、プログラミング言語のひとつです。ネットスケープコミュニケーションズで開発されました。 開発当初はLiveScriptと呼ばれていましたが、業務提携していたサン・マイクロシステムズが開発したJavaが脚光を浴びていたことから、JavaScriptと改名されました。 動きのあるWebページを作ることを目的に開発されたもので、主要なWebブラウザのほとんどに搭載されています。

非同期処理

非同期処理とは一部のコードを別々のスレッドで実行させる手法です。アプリケーションのパフォーマンスを向上させる目的でこの手法を用います。

Q&A

解決済

1回答

1198閲覧

railsの非同期処理でCouldn't find Upload without an IDと表示される

yuu0000

総合スコア4

Vue.js

Vue.jsは、Webアプリケーションのインターフェースを構築するためのオープンソースJavaScriptフレームワークです。

Ruby

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

Ruby on Rails

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

JavaScript

JavaScriptは、プログラミング言語のひとつです。ネットスケープコミュニケーションズで開発されました。 開発当初はLiveScriptと呼ばれていましたが、業務提携していたサン・マイクロシステムズが開発したJavaが脚光を浴びていたことから、JavaScriptと改名されました。 動きのあるWebページを作ることを目的に開発されたもので、主要なWebブラウザのほとんどに搭載されています。

非同期処理

非同期処理とは一部のコードを別々のスレッドで実行させる手法です。アプリケーションのパフォーマンスを向上させる目的でこの手法を用います。

0グッド

0クリップ

投稿2021/06/15 10:01

編集2021/06/17 15:02

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)

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

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

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

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

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

guest

回答1

0

自己解決

<div id="favorite"> ~~省略~~ <div class="favorite-content"> <% if user_signed_in? %> <favorite-button :user-id="<%= current_user.id %>" :uplaod-id="<%= upload.id %>"></favorite-button> <% end %> </div> </div>

こちら正しくは、

<favorite-button :user-id="<%= current_user.id %>" :upload-id="<%= upload.id %>"></favorite-button> :uploadの記述ミスでした。

投稿2021/07/08 03:05

yuu0000

総合スコア4

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

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

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.35%

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

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

質問する

関連した質問