1. ホーム
  2. ruby-on-rails

[解決済み】強いパラメータを持つ配列を許可する方法

2022-04-01 02:38:36

質問

has_many :through アソシエーションを使用したRails 3アプリが機能していますが、Rails 4アプリとして作り直すと、Rails 4バージョンで関連付けられたモデルからIDを保存することができません。

これら3つの関連するモデルは、2つのバージョンで同じものです。

カテゴリ分類.rb

class Categorization < ActiveRecord::Base

  belongs_to :question
  belongs_to :category
end

質問.rb

has_many :categorizations
has_many :categories, through: :categorizations

カテゴリ.rb

has_many :categorizations
has_many :questions, through: :categorizations

どちらのアプリでも、カテゴリIDは次のようにcreateアクションに渡されています。

  "question"=>{"question_content"=>"How do you spell car?", "question_details"=>"blah ", "category_ids"=>["", "2"],

Rails 3アプリでは、新しい質問を作成すると、質問テーブルに挿入され、次に分類テーブルに挿入されます。

 SQL (82.1ms)  INSERT INTO "questions" ("accepted_answer_id", "city", "created_at", "details", "province", "province_id", "question", "updated_at", "user_id") VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)  [["accepted_answer_id", nil], ["city", "dd"], ["created_at", Tue, 14 May 2013 17:10:25 UTC +00:00], ["details", "greyound?"], ["province", nil], ["province_id", 2], ["question", "Whos' the biggest dog in the world"], ["updated_at", Tue, 14 May 2013 17:10:25 UTC +00:00], ["user_id", 53]]
  SQL (0.4ms)  INSERT INTO "categorizations" ("category_id", "created_at", "question_id", "updated_at") VALUES (?, ?, ?, ?)  [["category_id", 2], ["created_at", Tue, 14 May 2013 17:10:25 UTC +00:00], ["question_id", 66], ["updated_at", Tue, 14 May 2013 17:10:25 UTC +00:00]]

rails 4アプリで、QuestionController#createのパラメータを処理した後、サーバーログに次のようなエラーが表示されます。

Unpermitted parameters: category_ids

で、質問は質問テーブルに挿入されるだけです。

 (0.2ms)  BEGIN
  SQL (67.6ms)  INSERT INTO "questions" ("city", "created_at", "province_id", "question_content", "question_details", "updated_at", "user_id") VALUES ($1, $2, $3, $4, $5, $6, $7) RETURNING "id"  [["city", "dd"], ["created_at", Tue, 14 May 2013 17:17:53 UTC +00:00], ["province_id", 3], ["question_content", "How's your car?"], ["question_details", "is it runnign"], ["updated_at", Tue, 14 May 2013 17:17:53 UTC +00:00], ["user_id", 12]]
   (31.9ms)  COMMIT

質問モデルにはcategory_idを保存していませんが、questions_controllerの許可パラメータとしてcategory_idを設定しました。

   def question_params

      params.require(:question).permit(:question_details, :question_content, :user_id, :accepted_answer_id, :province_id, :city, :category_ids)
    end

category_idをどのように保存すればいいのか、どなたか教えてください。 なお、どちらのアプリのcategories_controller.rbにもcreateアクションはありません。

両アプリで同じになっているのは、以下の3つのテーブルです。

 create_table "questions", force: true do |t|
    t.text     "question_details"
    t.string   "question_content"
    t.integer  "user_id"
    t.integer  "accepted_answer_id"
    t.datetime "created_at"
    t.datetime "updated_at"
    t.integer  "province_id"
    t.string   "city"
  end

 create_table "categories", force: true do |t|
    t.string   "name"
    t.datetime "created_at"
    t.datetime "updated_at"
  end

  create_table "categorizations", force: true do |t|
    t.integer  "category_id"
    t.integer  "question_id"
    t.datetime "created_at"
    t.datetime "updated_at"
  end

更新情報

これはRails 3アプリのcreateアクションです。

  def create
      @question = Question.new(params[:question])
      respond_to do |format|
      if @question.save
        format.html { redirect_to @question, notice: 'Question was successfully created.' }
        format.json { render json: @question, status: :created, location: @question }
      else
        format.html { render action: "new" }
        format.json { render json: @question.errors, status: :unprocessable_entity }
      end
    end
end

これはRails 4アプリのcreateアクションです。

   def create
      @question = Question.new(question_params)

       respond_to do |format|
      if @question.save
        format.html { redirect_to @question, notice: 'Question was successfully created.' }
        format.json { render json: @question, status: :created, location: @question }
      else
        format.html { render action: "new" }
        format.json { render json: @question.errors, status: :unprocessable_entity }
      end
    end
    end

これは、question_paramsメソッドです。

 private
    def question_params 
      params.require(:question).permit(:question_details, :question_content, :user_id, :accepted_answer_id, :province_id, :city, :category_ids)
    end

解決方法は?

これは https://github.com/rails/strong_parameters は、docsの該当箇所のようです。

<ブロッククオート

許可されるスカラー型は、String, Symbol, NilClass, Numeric, TrueClass, FalseClass, Date, Time, DateTime, StringIO, IO.FalseClassです。 ActionDispatch::Http::UploadedFile と Rack::Test::UploadedFile です。

params の値が、許可されたスカラー値の配列でなければならないことを宣言するには、 key に空の配列をマップします。

params.permit(:id => [])

私のアプリでは、category_idsは配列でcreateアクションに渡されます。

"category_ids"=>["", "2"],

そのため、strongパラメータを宣言する際に、明示的にcategory_idsを配列に設定しています。

params.require(:question).permit(:question_details, :question_content, :user_id, :accepted_answer_id, :province_id, :city, :category_ids => [])

これで完璧に動作するようになりました。

( 重要なことです。 コメントで @Lenart が指摘しているように、配列の宣言は必ず 最後に を指定しないと、シンタックス・エラーが発生します)。