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

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

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

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

jQuery

jQueryは、JavaScriptライブラリのひとつです。 簡単な記述で、JavaScriptコードを実行できるように設計されています。 2006年1月に、ジョン・レシグが発表しました。 jQueryは独特の記述法を用いており、機能のほとんどは「$関数」や「jQueryオブジェクト」のメソッドとして定義されています。

Ajax

Ajaxとは、Webブラウザ内で搭載されているJavaScriptのHTTP通信機能を使って非同期通信を利用し、インターフェイスの構築などを行う技術の総称です。XMLドキュメントを指定したURLから読み込み、画面描画やユーザの操作などと並行してサーバと非同期に通信するWebアプリケーションを実現することができます。

Q&A

解決済

2回答

3431閲覧

【jQuery・Ajax】ActionController::UnknownFormat

kumamin

総合スコア12

JavaScript

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

jQuery

jQueryは、JavaScriptライブラリのひとつです。 簡単な記述で、JavaScriptコードを実行できるように設計されています。 2006年1月に、ジョン・レシグが発表しました。 jQueryは独特の記述法を用いており、機能のほとんどは「$関数」や「jQueryオブジェクト」のメソッドとして定義されています。

Ajax

Ajaxとは、Webブラウザ内で搭載されているJavaScriptのHTTP通信機能を使って非同期通信を利用し、インターフェイスの構築などを行う技術の総称です。XMLドキュメントを指定したURLから読み込み、画面描画やユーザの操作などと並行してサーバと非同期に通信するWebアプリケーションを実現することができます。

0グッド

0クリップ

投稿2020/07/20 00:37

編集2020/07/21 12:50

前提・実現したいこと

今jQueryのAjaxを使い、レビューの投稿画面を実装しています。
フォームの中にレビューを書き、送信ボタンを押すとその内容が画面に非同期で表示されるようにしたいです。
レビュー投稿画面はreviews/show.html.hamlで設定しています。

そこでreview.jsのファイルにAjaxを記述したのですが、
フォームの中に文字を入力して送信ボタンを押すと、エラーが出てきてしまいました。
解決方法を教えていただけますと幸いです。

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

ActionController::UnknownFormat in ReviewsController#create ActionController::UnknownFormat

イメージ説明

json形式でコントローラーにデータを送りたいのですが、
ターミナルのログにはHTMLとして送信されたと書かれてしまっています。

該当のソースコード

▼app/assets/javascripts/review.js

ruby

1$(function(){ 2 function buildHTML(review){ 3 let html = `<div class="Message_box"> 4 <img class="Message_box__User_image" src="/attachments/1aab43602b1af4375a5bdc1eb3ad4ab5756602bd/store/${review.profile_image}/profile_image"> 5 <div class="Message_box__Content"> 6 <a class="Link Text Message_box__Content__Name" href="/posts_g/8">${review.user_name}</a> 7 <div class="Message_box__Text Text"> 8 ${review.content} 9 </div> 10 </div> 11 </div>` 12 return html; 13 } 14 $('.New_r').on('submit', function(e){ 15 e.preventDefault(); 16 let formData = new FormData(this); 17 let url = $(this).attr('action'); 18 $.ajax({ 19 url: url, 20 type: 'POST', 21 data: formData, 22 dataType: 'json', 23 format: 'json', 24 processData: false, 25 contentType: false 26 }) 27 .done(function(data){ 28 let html = buildHTML(data); 29 $('.Reviews').append(html); 30 $('.Textbox_r').val(''); 31 $('.Submit_r').prop('disabled', false); 32 }) 33 .fail(function() { 34 console.log('error'); 35 }); 36 }) 37})

▼app/controllers/reviews_controller.rb

ruby

1class ReviewsController < ApplicationController 2 3 def index 4 @user = User.find(params[:user_id]) 5 end 6 7 def create 8 @user = User.find(params[:user_id]) 9 @review = @user.reviews.new(review_params) 10 if @review.save 11 respond_to do |format| 12 format.json 13 end 14 else 15 @reviews = @user.reviews.includes(:user) 16 render :index 17 end 18 end 19 20 private 21 22 def review_params 23 params.require(:review).permit(:content).merge(user_id: current_user.id) 24 end 25 26end

▼app/views/reviews/show.html.haml

ruby

1.Header_r 2 .Title_r 3 = "#{@user.name}さんのレビュー" 4 .Main__content 5 .Wrapper_r 6 .Reviews 7 = render "review" 8 .Main__form 9 .Wrapper_r 10 = form_with model: [@user, @review], html: {class: "New_r"}, local: true do |f| 11 .Main__form__contents 12 .Main__form__contents__input 13 = f.text_field :content, class: 'Main__form__contents__input__text Textbox_r' 14 = f.submit '送信', class: 'Main__form__contents__btn Submit_r'

▼app/views/reviews/create.json.jbuilder

ruby

1json.content @review.content 2json.user_id @review.user.id 3json.user_name @review.user.name 4json.profile_image @review.user.profile_image_id

▼routes.rb

ruby

1 resources :users do 2 resources :reviews, only: [:index, :create] 3 end

試したこと

以下の内容を確認したところ、間違いは見つかりませんでした。

・AjaxのdataTypeに間違いがないか?
・create.json.jbuilderがちゃんと対応するコントローラー名のディレクトリに入っていて、アクション名と一致するファイル名になっているか?

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

バージョンは、Ruby 2.6.5、Rails 6.0.3.2です。

▼rails routesの結果(ターミナル)
イメージ説明

▼サーバーのログ(ターミナル)
イメージ説明

▼Fiddlerの検証結果
イメージ説明

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

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

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

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

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

退会済みユーザー

退会済みユーザー

2020/07/20 00:55

> let formData = new FormData(this); そのコードで this は間違いないのでしょうか? form 要素に enctype="multipart/form-data" は設定されてますか? Fiddler などのキャプチャツールを使って要求が期待通りの形で出ているか調べてみましたか?
kumamin

2020/07/20 23:18

ご回答ありがとうございました。 form要素にenctype="multipart/form-data"と記述してもエラーは解決しませんでした。 Fiddlerは使い方がわからないのですが、e.preventDefault();の後にconsole.log(this)を挿入しthisがformを指定できているのは確認しました。
退会済みユーザー

退会済みユーザー

2020/07/21 00:30

ファイルをアップロードするのだろうと思ってましたが、違ったら失礼しました。(そう思ってましたので、「form 要素に enctype="multipart/form-data" は設定されてますか?」とお聞きしました) 何にせよ、キャプチャツールで要求が期待通り出ているかを調べれば、問題がクライアント側にあるのかサーバー側にあるのか分かると思うのですが。Fiddler でなくても Chrome のディベロッパーツールでも調べられると思いますので、調べてみてはいかがでしょう。ちなみに Fiddler では以下の記事の一番下の画像のような結果が得られます。 http://surferonwww.info/BlogEngine/post/2020/07/09/uploading-multiple-files-in-different-folders.aspx jQuery ajax を使ってファイルその他を含む formData を enctype="multipart/form-data" で送信したものです。
kumamin

2020/07/21 12:48

ご丁寧に回答していただき、誠にありがとうございます。 フォームに入力するのは文字だけで、ファイルはアップロードはしないようになっています。 こちらこそ説明不足で失礼いたしました。 Fidderインストールしてみたんですが、あまり見方がわからなかったです...せっかく紹介してくださったのにすみません。 ちなみに前回の回答で書いたconsole.log(this)というのはChromeのデベロッパーツールで確認したもので、formのデータは取れていることはわかっています。 また、質問時に載せた画像のサーバーのログから判断すると、ajaxでjsonのフォーマットを指定しているはずなのに"as HTML"として送られてしまっていることがわかります。 jsonのフォーマットしか用意しておらずhtmlに対応するフォーマットがないためこのようなエラーが起こってしまっているんだと思います。 なのでjsonで確実にデータを送れるようにしたいのですが、そのやり方がわからず困っています。 一応Fiddlerの結果の画像を追記しておきますね。
guest

回答2

0

自己解決

原因はわかりませんでしたが、解決しました。

投稿2020/08/15 03:58

kumamin

総合スコア12

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

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

0

エラーメッセージのとおりです。format.jsonしか実装していないので、JSON指定(リクエストのURLに.jsonを付ける)以外のリクエストはこのようなエラーで失敗します。

対応策としては、「リクエストのURLを.jsonで終わるようにする(この場合POST /reviews/8.jsonのようになる)」のが手っ取り早いです。

投稿2020/07/20 01:04

maisumakun

総合スコア146018

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

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

maisumakun

2020/07/20 01:04

なお、$.ajaxの「dataType: 'json'」は、レスポンス結果の処理を指定するためのもので、リクエストに影響するものではありません。
kumamin

2020/07/20 23:12

ご回答ありがとうございました。 すみません、昨日ルーティングを変えて解決したと思ったんですがまた同じエラーが起こってしまいました。 review.jsで let url = $(this).attr('action'); の部分を let url = "/users/8/reviews.json"; のように変えてもエラー画面は変わりませんでした。 ルーティング変更後の編集したコードを投稿に反映させましたので、再度ご回答いただけると大変助かります。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.35%

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

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

質問する

関連した質問