我的情况与Railscast 196-197: Nested Model Form中的情况非常相似。但是,我遇到了这种方法和强参数之间的冲突。我无法想出一个很好的方法来填充子对象上的父记录id字段,因为我不希望通过表单分配它(以防止用户将子记录与他们不拥有的父记录相关联) )。我有一个解决方案(见下面的代码),但这似乎是Rails可能有一个聪明,简单的方法为我做的事情。
这是代码......
有一个父对象(称之为Survey)has_many子对象(称之为问题):
# app/models/survey.rb
class Survey
belongs_to :user
has_many :questions
accepts_nested_attributes_for :questions
end
# app/models/question.rb
class Question
validates :survey_id, :presence => true
belongs_to :survey
end
有一个表单允许用户同时创建调查和调查问题(为简单起见,下面的代码将调查视为只有问题):
# app/views/surveys/edit.html.erb
<%= form_for @survey do |f| %>
<%= f.label :name %>
<%= f.text_field :name %><br />
<%= f.fields_for :questions do |builder| %>
<%= builder.label :content, "Question" %>
<%= builder.text_area :content, :rows => 3 %><br />
<% end %>
<%= f.submit "Submit" %>
<% end %>
问题是控制器。我想通过强参数保护问题记录中的survey_id字段,但这样做的问题不会通过验证,因为survey_id是必填字段。
# app/controllers/surveys_controller.rb
class SurveysController
def edit
@survey = Survey.new
Survey.questions.build
end
def create
@survey = current_user.surveys.build(survey_params)
if @survey.save
redirect_to @survey
else
render :new
end
end
private
def survey_params
params.require(:survey).permit(:name, :questions_attributes => [:content])
end
end
我能想到解决这个问题的唯一方法是将问题与调查分开建立:
def create
@survey = current_user.surveys.build(survey_params)
if @survey.save
if params[:survey][:questions_attributes]
params[:survey][:questions_attributes].each_value do |q|
question_params = ActionController::Parameters.new(q)
@survey.questions.build(question_params.permit(:content))
end
end
redirect_to @survey
else
render :new
end
end
private
def survey_params
params.require(:survey).permit(:name)
end
(Rails 4 beta 1,Ruby 2)
更新
处理此问题的最佳方法可能是按照this Code Climate blog post中的建议分解出“表单对象”。我打开这个问题,因为我对其他观点感到好奇
答案 0 :(得分:69)
所以你遇到的问题是子对象没有通过验证,对吧?当子对象与父对象同时创建时,子对象不可能知道其父对象的id以通过验证,这是真的。
以下是解决该问题的方法。按如下方式更改模型:
# app/models/survey.rb
class Survey
belongs_to :user
has_many :questions, :inverse_of => :survey
accepts_nested_attributes_for :questions
end
# app/models/question.rb
class Question
validates :survey, :presence => true
belongs_to :survey
end
此处的差异是传递给:inverse_of
关联的has_many
,而问题现在仅在:survey
而不是:survey_id
上验证。
:inverse_of
使得当使用关联创建或构建子对象时,它还会接收对创建它的父级的反向引用。这似乎应该是自动发生的事情,但遗憾的是,除非您指定此选项。
在:survey
而非:survey_id
上进行验证是一种妥协。验证不再只是在survey_id字段中检查是否存在非空白;它现在实际上检查关联是否存在父对象。在这种情况下,由于:inverse_of
,它有助于知道,但在其他情况下,它实际上必须使用id从数据库加载关联以进行验证。这也意味着不匹配数据库中任何内容的ID将无法通过验证。
希望有所帮助。