嵌套属性不允许的参数


127

我有一个Bill对象,其中有很多Due对象。该Due对象也属于Person。我想要一种可以在一个页面中全部创建Bill及其子项的表单Dues。我正在尝试使用嵌套的属性创建表单,类似于此Railscast中的属性。

相关代码如下:

due.rb

class Due < ActiveRecord::Base
    belongs_to :person
    belongs_to :bill
end

bill.rb

class Bill < ActiveRecord::Base
    has_many :dues, :dependent => :destroy 
    accepts_nested_attributes_for :dues, :allow_destroy => true
end

bills_controller.rb

  # GET /bills/new
  def new
      @bill = Bill.new
      3.times { @bill.dues.build }
  end

bills / _form.html.erb

  <%= form_for(@bill) do |f| %>
    <div class="field">
        <%= f.label :company %><br />
        <%= f.text_field :company %>
    </div>
    <div class="field">
        <%= f.label :month %><br />
        <%= f.text_field :month %>
    </div>
    <div class="field">
        <%= f.label :year %><br />
        <%= f.number_field :year %>
    </div>
    <div class="actions">
        <%= f.submit %>
    </div>
    <%= f.fields_for :dues do |builder| %>
        <%= render 'due_fields', :f => builder %>
    <% end %>
  <% end %>

bills / _due_fields.html.erb

<div>
    <%= f.label :amount, "Amount" %>        
    <%= f.text_field :amount %>
    <br>
    <%= f.label :person_id, "Renter" %>
    <%= f.text_field :person_id %>
</div>

更新到bills_controller.rb可行

def bill_params 
  params
  .require(:bill)
  .permit(:company, :month, :year, dues_attributes: [:amount, :person_id]) 
end

正确的字段将呈现在页面上(尽管还没有下拉列表Person),并且提交成功。但是,所有子项费用都不会保存到数据库,并且服务器日志中会引发错误:

Unpermitted parameters: dues_attributes

在错误之前,日志显示如下:

Started POST "/bills" for 127.0.0.1 at 2013-04-10 00:16:37 -0700
Processing by BillsController#create as HTML<br>
Parameters: {"utf8"=>"✓", 
"authenticity_token"=>"ipxBOLOjx68fwvfmsMG3FecV/q/hPqUHsluBCPN2BeU=",
 "bill"=>{"company"=>"Comcast", "month"=>"April ", 
"year"=>"2013", "dues_attributes"=>{
"0"=>{"amount"=>"30", "person_id"=>"1"}, 
"1"=>{"amount"=>"30", "person_id"=>"2"},
 "2"=>{"amount"=>"30", "person_id"=>"3"}}}, "commit"=>"Create Bill"}

Rails 4发生了一些变化吗?


5
修正格式:params.require(:bill).permit(:company,:month,:year,:dues_attributes => [:amount,:person_id])
Andy Copley

Answers:


187

似乎属性保护的处理方式发生了变化,现在您必须在控制器中将参数列入白名单(而不是模型中的attr_accessible),因为以前的可选gem strong_parameters成为Rails Core的一部分。

看起来应该像这样:

class PeopleController < ActionController::Base
  def create
    Person.create(person_params)
  end

private
  def person_params
    params.require(:person).permit(:name, :age)
  end
end

所以params.require(:model).permit(:fields)会用

对于嵌套属性,例如

params.require(:person).permit(:name, :age, pets_attributes: [:id, :name, :category])

可以在github此处Ruby edge API文档strong_parameters中找到更多详细信息


1
我将BillController更改为如下所示: def bill_params params.require(:bill).permit(:company, :month, :year, :dues_attributes[:amount, :person_id]) end 我现在收到此错误: 没有将Symbol隐式转换为Integer
jcanipar 2013年

2
好吧,这有助于将结肠放置在正确的位置……这正是需要做的事情。谢谢@ thorsten-muller!
jcanipar

88
不要忘记ID !!!!pets_attributes: [:id, :name, :category]否则,当您编辑时,将再次创建每个宠物。
Arcolye

8
您需要这样做Person.create(person_params),否则它将不会调用该方法。相反,您会得到ActiveModel::ForbiddenAttributesError
andorov 2013年

16
另外,如果要销毁表单中的项目,还需要将隐藏:_destroy参数列入白名单。即pets_attributes: [:id, :name, :category, :_destroy]
病原

21

来自文档

To whitelist an entire hash of parameters, the permit! method can be used

params.require(:log_entry).permit!

嵌套属性采用哈希形式。在我的应用程序中,我有一个Question.rb模型,可以接受Answer.rb模型的嵌套属性(用户在其中为自己创建的问题创建答案选项)。在questions_controller中,我这样做

  def question_params

      params.require(:question).permit!

  end

允许问题哈希中的所有内容,包括嵌套的答案属性。如果嵌套属性为数组形式,这也适用。

话虽如此,我想知道这种方法是否存在安全性问题,因为它基本上允许散列内的任何内容而无需指定确切的含义,这似乎与强参数的目的背道而驰。


太棒了,我无法明确允许范围参数,这为我节省了几个小时。
Bartek Skwira 2013年

3
是的,使用.permit!通常被视为潜在的安全隐患。如果用户是管理员,则只想真正使用它,但是即使那样,我也要警惕它的使用。
8bithero

6
我的嵌套属性也位于数组中。是.permit!唯一的选择吗?即使允许所有模型的属性,我也无法使其正常工作,因为它会阻塞数组。
Clifton Labrum 2013年


12

实际上,有一种方法可以将所有嵌套参数列入白名单。

params.require(:widget).permit(:name, :description).tap do |whitelisted|
  whitelisted[:position] = params[:widget][:position]
  whitelisted[:properties] = params[:widget][:properties]
end

该方法比其他解决方案具有优势。它允许允许深度嵌套的参数。

而其他解决方案例如:

params.require(:person).permit(:name, :age, pets_attributes: [:id, :name, :category])

别。


资源:

https://github.com/rails/rails/issues/9454#issuecomment-14167664


3

今天,我在处理Rails 4时遇到了同样的问题,通过将我的fields_for构造为:

<%= f.select :tag_ids, Tag.all.collect {|t| [t.name, t.id]}, {}, :multiple => true %>

然后在我的控制器中,我有很强的参数:

private
def post_params
    params.require(:post).permit(:id, :title, :content, :publish, tag_ids: [])
end

所有作品!


嗨,谢谢@KingsleyIjomah-如果您想将孩子的特定属性列入白名单该怎么办?
BKSpurgeon '16

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.