为什么Rails 5没有保存嵌套属性,因为父模型没有先保存

时间:2017-03-13 01:45:29

标签: ruby-on-rails activerecord ruby-on-rails-5 nested-forms nested-attributes

我正在使用Rails 5以及最新稳定版本的所有东西。所以我得到以下内容:

  

您已将关联设置为必需但已丢失。   默认情况下,在rails 5中将关联设置为必需,如果需要   要保持一个空,你需要在你的关联中设置optional:true   模式

这很好,我理解发生了什么然而对于我的生活我无法弄清楚如何让父模型先保存,以便user_id被翻译为嵌套模型记录。我在上面看到了相同的答案,但除了将初始化程序中的默认值从true变为false之外,没有人解释其他工作。这不能解决问题,因为记录肯定会保存,但不包括user_id。

下面是我的代码库,我会问,而不是回应上面的引用,有人可以启发我如何在保存时将USER_ID字段放入嵌套属性。我拒绝禁用验证并手动处理插入,因为这不是红宝石的方式而且违反了标准! 提前感谢任何能够直接回答这个问题的人,而且没有模糊的解释,从红宝石的方式中脱离出来!

###Models
#Users
class User < ApplicationRecord
  has_one :profile, inverse_of: :user
  accepts_nested_attributes_for :profile, allow_destroy: true
end

#Profiles
class Profile < ApplicationRecord
  belongs_to :user, inverse_of: :profile
end

###Controller
class UsersController < ApplicationController
  before_action :set_user, only: [:show, :edit, :update, :destroy]

  # GET /users
  # GET /users.json
  def index
    @users = User.all
  end

  # GET /users/1
  # GET /users/1.json
  def show
  end

  # GET /users/new
  def new
    @user = User.new
    @user.build_profile
  end

  # GET /users/1/edit
  def edit
    @user.build_profile
  end

  # POST /users
  # POST /users.json
  def create
    @user = User.new(user_params)

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

  # PATCH/PUT /users/1
  # PATCH/PUT /users/1.json
  def update
    respond_to do |format|
      if @user.update(user_params)
        format.html { redirect_to @user, notice: 'User was successfully updated.' }
        format.json { render :show, status: :ok, location: @user }
      else
        format.html { render :edit }
        format.json { render json: @user.errors, status: :unprocessable_entity }
      end
    end
  end

  # DELETE /users/1
  # DELETE /users/1.json
  def destroy
    @user.destroy
    respond_to do |format|
      format.html { redirect_to users_url, notice: 'User was successfully destroyed.' }
      format.json { head :no_content }
    end
  end

  private
    # Use callbacks to share common setup or constraints between actions.
    def set_user
      @user = User.find(params[:id])
    end

    # Never trust parameters from the scary internet, only allow the white list through.
    def user_params
      params.require(:user).permit(:username, :password, :user_type_id, profile_attributes: [:user_id, :first_name, :middle_name, :last_name, :phone_number, :cell_number, :email])
    end
end

##View
<%= form_for(@user) do |f| %>
  <% if user.errors.any? %>
    <div id="error_explanation">
      <h2><%= pluralize(user.errors.count, "error") %> prohibited this user from being saved:</h2>

      <ul>
      <% user.errors.full_messages.each do |message| %>
        <li><%= message %></li>
      <% end %>
        <!--<li><%= debug f %></li>-->
      </ul>
    </div>
  <% end %>

  <div class="field">
    <%= f.label :username %>
    <%= f.text_field :username %>
  </div>

  <div class="field">
    <%= f.label :password %>
    <%= f.text_field :password %>
  </div>

  <div class="field">
    <% if params[:trainer] == "true" %>
      <%= f.label :user_type_id %>
      <%= f.text_field :user_type_id, :readonly => true, :value => '2' %>
    <% else %>
      <%= f.label :user_type_id %>
      <%= f.text_field :user_type_id, :readonly => true, :value => '1'  %>
    <% end %>
  </div>
    <h2>Account Profile</h2>
    <%= f.fields_for :profile do |profile| %>
      <%#= profile.inspect %>
        <div>
          <%= profile.label :first_name %>
          <%= profile.text_field :first_name %>
        </div>
        <div>
          <%= profile.label :middle_name %>
          <%= profile.text_field :middle_name %>
        </div>
        <div>
          <%= profile.label :last_name %>
          <%= profile.text_field :last_name %>
        </div>
        <div>
          <%= profile.label :email %>
          <%= profile.text_field :email %>
        </div>
        <div>
          <%= profile.label :phone_number %>
          <%= profile.telephone_field :phone_number %>
        </div>
        <div>
          <%= profile.label :cell_phone %>
          <%= profile.telephone_field :cell_number %>
        </div>
    <% end %>
  <div class="actions">
    <%= f.submit %>
  </div>
    <%= debug params %>
    <%= debug user %>
    <%= debug user.profile %>
<% end %>

更新 对于初学者,我已经发现你需要将 autosave:true 包含在这样的关系中

class User < ApplicationRecord
  has_one :profile, inverse_of: :user, autosave: true
  accepts_nested_attributes_for :profile, allow_destroy: true
end

然后父记录在孩子之前保存。现在出现了另一个问题,我只是不确定并且在表单提交时很奇怪你会在我粘贴的控制台输出中注意到 INSERT INTO profiles 语句包含 user_id 列和 1 的值。它通过验证并看起来从输出中正常运行,但配置文件表中的 user_id 列仍为空。我将继续挖掘,希望我的其中一个rubyiests将会看到这个,并有一些关于如何完成修复的想法。到目前为止,我喜欢Rails 5的改进,但如果没有小有趣的陷阱,它就不会是ROR!提前再次感谢!

Started POST "/users" for 192.168.0.31 at 2017-03-12 22:28:14 -0400
Cannot render console from 192.168.0.31! Allowed networks: 127.0.0.1, ::1, 127.0.0.0/127.255.255.255
Processing by UsersController#create as HTML
  Parameters: {"utf8"=>"✓", "authenticity_token"=>"YA7kQnScvlIBy5OiT+BmOQ2bR7J00ANXId38FqNwX37Cejd+6faUyD3rMF4y0qJNKBUYGaxrRZqcLrXonL6ymA==", "user"=>{"username"=>"john", "password"=>"[FILTERED]", "user_type_id"=>"1", "profile_attributes"=>{"first_name"=>"john", "middle_name"=>"r", "last_name"=>"tung", "email"=>"thegugaru@gmail.com", "phone_number"=>"8033207677", "cell_number"=>"8033207677"}}, "commit"=>"Create User"}
   (0.1ms)  BEGIN
  SQL (0.3ms)  INSERT INTO `users` (`username`, `password`, `user_type_id`, `created_at`, `updated_at`) VALUES ('john', '0000', 1, '2017-03-13 02:28:14', '2017-03-13 02:28:14')
  SQL (0.4ms)  INSERT INTO `profiles` (`user_id`, `email`, `first_name`, `middle_name`, `last_name`, `phone_number`, `cell_number`, `created_at`, `updated_at`) VALUES (1, 'thegu@gmail.com', 'john', 'r', 'tung', '8033207677', '8033207677', '2017-03-13 02:28:14', '2017-03-13 02:28:14')
   (10.8ms)  COMMIT
Redirected to http://192.168.0.51:3000/users/1
Completed 302 Found in 24ms (ActiveRecord: 11.5ms)

1 个答案:

答案 0 :(得分:1)

好的,我正在回答我自己的问题,因为我知道很多人都在努力解决这个问题,而且我实际上有答案,而不是对文档做出模糊回应。

首先,我们将在这个例子中使用一对一的关系。创建关系时,需要确保父模型具有以下

  1. inverse_of:
  2. autosave:true
  3. accepts_nested_attributes_for:model,allow_destroy:true
  4. 以下是用户模型,然后我会解释,

    class User < ApplicationRecord
      has_one :profile, inverse_of: :user, autosave: true
      accepts_nested_attributes_for :profile, allow_destroy: true
    end
    

    在Rails 5中你需要inverse_of:因为这告诉Rails通过外键存在关系,并且在保存表单数据时需要在嵌套模型上设置它。现在,如果您要从关系线中离开自动保存:真实,那么您将使用 user_id 而不保存到配置文件表,只保留其他列,除非您有验证关闭然后它不会出错它只会在没有 user_id 的情况下保存它。这里发生的是 autosave:true 确保首先保存用户记录,以便 user_id 存储在的嵌套属性中个人资料模型。 简而言之,为什么 user_id 没有遍历到孩子并且它正在回滚而不是提交。 还有一个最后的问题是,有一些帖子在你的控制器中告诉你编辑路线你应该像我在我的帖子中添加 @ user.build_profile 。在评估控制台输出后,不要做它们是错误的结果

    Started GET "/users/1/edit" for 192.168.0.31 at 2017-03-12 22:38:17 -0400
    Cannot render console from 192.168.0.31! Allowed networks: 127.0.0.1, ::1, 127.0.0.0/127.255.255.255
    Processing by UsersController#edit as HTML
      Parameters: {"id"=>"1"}
      User Load (0.4ms)  SELECT  `users`.* FROM `users` WHERE `users`.`id` = 1 LIMIT 1
      Profile Load (0.5ms)  SELECT  `profiles`.* FROM `profiles` WHERE `profiles`.`user_id` = 1 LIMIT 1
       (0.1ms)  BEGIN
      SQL (0.5ms)  UPDATE `profiles` SET `user_id` = NULL, `updated_at` = '2017-03-13 02:38:17' WHERE `profiles`.`id` = 1
       (59.5ms)  COMMIT
      Rendering users/edit.html.erb within layouts/application
      Rendered users/_form.html.erb (44.8ms)
      Rendered users/edit.html.erb within layouts/application (50.2ms)
    Completed 200 OK in 174ms (Views: 98.6ms | ActiveRecord: 61.1ms)
    

    如果您看起来正在从头开始重建配置文件,并将user_id重置为null,以便与您正在编辑的当前用户匹配。所以要非常小心,因为我看到很多帖子提出了这个建议,并且花了我很多时间研究才找到解决方案!