如何允许具有强参数的数组

Lea*_*cim 255 arrays ruby-on-rails strong-parameters

我有一个功能正常的Rails 3应用程序使用has_many:通过关联,当我将其重新设置为Rails 4应用程序时,让我从Rails 4版本中的相关模型中保存ID.

这两个版本的三个相关模型是相同的.

Categorization.rb

class Categorization < ActiveRecord::Base

  belongs_to :question
  belongs_to :category
end
Run Code Online (Sandbox Code Playgroud)

Question.rb

has_many :categorizations
has_many :categories, through: :categorizations
Run Code Online (Sandbox Code Playgroud)

Category.rb

has_many :categorizations
has_many :questions, through: :categorizations
Run Code Online (Sandbox Code Playgroud)

在这两个应用程序中,类别ID都会像这样传递到create动作中

  "question"=>{"question_content"=>"How do you spell car?", "question_details"=>"blah ", "category_ids"=>["", "2"],
Run Code Online (Sandbox Code Playgroud)

在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]]
Run Code Online (Sandbox Code Playgroud)

在rails 4应用程序中,在处理QuestionController #create中的参数之后,我在服务器日志中收到此错误

Unpermitted parameters: category_ids
Run Code Online (Sandbox Code Playgroud)

问题只是插入到问题表中

 (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
Run Code Online (Sandbox Code Playgroud)

虽然我没有在Questions模型中存储category_ids,但我将categories_ids设置为questions_controller中的允许参数

   def question_params

      params.require(:question).permit(:question_details, :question_content, :user_id, :accepted_answer_id, :province_id, :city, :category_ids)
    end
Run Code Online (Sandbox Code Playgroud)

任何人都可以解释我应该如何保存category_ids?请注意,任一应用程序的categories_controller.rb中都没有创建操作.

这两个应用程序中的三个表是相同的

 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
Run Code Online (Sandbox Code Playgroud)

更新

这是Rails 3应用程序的创建操作

  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
Run Code Online (Sandbox Code Playgroud)

这是Rails 4应用程序的创建操作

   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
Run Code Online (Sandbox Code Playgroud)

这是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
Run Code Online (Sandbox Code Playgroud)

Lea*_*cim 506

这个https://github.com/rails/strong_parameters看起来像文档的相关部分:

允许的标量类型是String,Symbol,NilClass,Numeric,TrueClass,FalseClass,Date,Time,DateTime,StringIO,IO,ActionDispatch :: Http :: UploadedFile和Rack :: Test :: UploadedFile.

要声明params中的值必须是允许的标量值数组,请将键映射到空数组:

params.permit(:id => [])
Run Code Online (Sandbox Code Playgroud)

在我的应用程序中,category_ids被传递给数组中的create动作

"category_ids"=>["", "2"],
Run Code Online (Sandbox Code Playgroud)

因此,在声明强参数时,我明确将category_ids设置为数组

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

现在完美运作!

  • 我还注意到数组的声明应该在属性列表的末尾.否则我得到语法错误`语法错误,意外')',期待=>` (97认同)
  • 数组声明(嵌套参数)最后的原因是ActionController :: Parameters.permit期望每个参数都是哈希或符号.Ruby magic会将方法调用的_end_处的所有键值对转换为一个哈希值,但如果在方法调用中将符号与键/值对混合,Ruby将不知道该怎么做. (43认同)
  • 如果```category_ids```不是一个数组(例如一个字符串被发送)那么rails完全疯了并引发一个```ERROR TypeError:期望数组(得到字符串)为param`category_ids'```这是一个铁轨中的错误?编辑:是的,它是:https://github.com/rails/rails/issues/11502 (7认同)
  • @Lenart谢谢,我以为我疯了.这至少应该出现在strong_params自述文件中 (4认同)
  • @Lenart如果想避免语法错误,可以将其添加为哈希。`params.permit(:foo,{bar:[]},:zoo)`。 (2认同)

Bri*_*ian 93

如果要允许哈希数组(或an array of objects从JSON的角度来看)

params.permit(:foo, array: [:key1, :key2])
Run Code Online (Sandbox Code Playgroud)

2点注意到这里:

  1. array应该是该permit方法的最后一个参数.
  2. 你应该在数组中指定哈希的键,否则你会得到一个错误Unpermitted parameter: array,在这种情况下很难调试.

  • `array`不需要在最后,但你需要确保你有有效的Ruby.`params.permit(:foo,array:[:key1,:key2] ,: bar)`将不是有效的ruby,因为解释器需要在第一个集合之后的散列键:值对.要拥有有效的Ruby,你需要`params.permit(:foo,{array:[:key1,:key2]},:bar)` (9认同)

Mat*_*uel 20

应该是这样的

params.permit(:id => [])
Run Code Online (Sandbox Code Playgroud)

也可以使用rails 4+以上版本:

params.permit(id: [])
Run Code Online (Sandbox Code Playgroud)

  • 请注意,数组应该是permit方法的最后一个参数 (11认同)
  • 您所说的适用于Rails v4 +的哈希语法实际上是Ruby 1.9及更高版本中可用的语法,而不是Rails框架(尽管Rails 4需要Ruby 1.9.3或更高版本) (2认同)

Fel*_*ger 11

如果你有这样的哈希结构:

Parameters: {"link"=>{"title"=>"Something", "time_span"=>[{"start"=>"2017-05-06T16:00:00.000Z", "end"=>"2017-05-06T17:00:00.000Z"}]}}
Run Code Online (Sandbox Code Playgroud)

然后我就是这样做的:

params.require(:link).permit(:title, time_span: [[:start, :end]])
Run Code Online (Sandbox Code Playgroud)

  • 我花了很长时间才找到这个 (3认同)

小智 6

当你想允许多个数组字段时,你必须在允许的情况下最后列出数组字段,如给定 -

params.require(:questions).permit(:question, :user_id, answers: [], selected_answer: [] )
Run Code Online (Sandbox Code Playgroud)

(这有效)


小智 5

我还不能发表评论,但是在Fellow Stranger解决方案之后,您还可以保留嵌套,以防您拥有键值是数组的情况。像这样:

filters: [{ name: 'test name', values: ['test value 1', 'test value 2'] }]
Run Code Online (Sandbox Code Playgroud)

这有效:

params.require(:model).permit(filters: [[:name, values: []]])
Run Code Online (Sandbox Code Playgroud)