Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

RecordNotFound with accepts_nested_attributes_for and belongs_to

I get

ActiveRecord::RecordNotFound: Couldn't find Client with ID=3 for Order with ID=

when trying to submit an Order form for an existing client. This happens through the form or the console by typing:

Order.new(:client_attributes => { :id => 3 })

payment_form.html.erb:

<%= semantic_form_for @order, :url => checkout_purchase_url(:secure => true) do |f| %>

        <%= f.inputs "Personal Information" do %>

            <%= f.semantic_fields_for :client do |ff| %>
                <%= ff.input :first_name %>
                <%= ff.input :last_name %>              
                <!-- looks like semantic_fields_for auto-inserts a hidden field for client ID -->
            <% end %>

        <% end %>
<% end %>

Order.rb:

class Order < ActiveRecord::Base
  belongs_to :client
  accepts_nested_attributes_for :client, :reject_if => :check_client

  def check_client(client_attr)
    if _client = Client.find(client_attr['id'])
      self.client = _client
      return true
    else
      return false
    end    
  end
end

The reject_if idea came from here but I logged the method and it's not even being called! It doesn't matter what its name is!

like image 870
manafire Avatar asked Mar 25 '12 22:03

manafire


3 Answers

Note: Feb 2020

Since I'm starting to get downvotes on this 8 years later, adding this note. While this was the original solution I went with 8 years ago, a better one has been proposed by MatayoshiMariano (5 years after my OP).

My Original Fix

Fixed the issue by overloading the client_attributes= method, as described here:

  def client_attributes=(client_attrs)    
    self.client = Client.find_or_initialize_by_id(client_attrs.delete(:id))
    self.client.attributes = client_attrs
  end
like image 69
manafire Avatar answered Nov 03 '22 17:11

manafire


If you have has_many relationship, this will work. Tested on Rails 6.0.2

  def clients_attributes =(attributes)
    # Get IDs for any clients that already exist.
    client_ids = attributes.values.map { |a| a[:id] }.compact

    # Now find them all and move them to this section.
    clients << Client.find(client_ids)

    # Update them with standard `accepts_nested_attributes_for` behaviour.
    super attributes
  end
like image 31
Henry Jacob Avatar answered Nov 03 '22 18:11

Henry Jacob


If you only want a new Order with an existing client, without modifying the client, you need to assign the id.

Order.new(client_id: 3)

This is another way to do this without overloading the client_attributes= method and cleanest

The new Order now has the client with ID 3

If you also want to update ant client's attributes you must add the client_attributes, for example:

Order.new(client_id: 3, client_attributes: { id: 3, last_order_at: Time.current })

See https://github.com/rails/rails/issues/7256 from 2012.

like image 43
MatayoshiMariano Avatar answered Nov 03 '22 19:11

MatayoshiMariano