Дублирующая запись и это дети - но дети удаляются из старой записи
Моя проблема похожа на: Мой метод клонирования заключается в краже детей из оригинальной модели
Но я не могу найти решение для этого, чтобы работать с моим. Я пытаюсь создать форму обмена заказами, которая включает заполнение формы старыми данными записи. Поэтому, когда я сохраняю форму, она создает новую запись заказа, но кажется, что дети удаляются из старой записи заказа и перекачиваются в новую.
Вот код:
def new
@old_order = Order.includes(:line_items).find(params[:id])
@order = Order.new @old_order.attributes
@order.line_items = []
@old_order.line_items.each do |old|
new = old.dup # the line_item id is set before creation.
new.order_id = @order.id
new.save!
@order.line_items << new
@old_order.line_items << old # this was to see if the old line_items would reappend to the old order. Didn't help...
end
end
def create
@order = Order.new(exchange_order_params)
if @order.save
@order.update_attributes!(stage: 2, ordered_at: Date.today)
redirect_to admin_returns_url, notice: "Order moved to 'accepted' for processing"
else
flash.now[:alert] = "Please try again"
render :action => "new"
end
end
private
def exchange_order_params
params.require(:order).permit(:id, :user_id,
line_items_attributes: [:id, :order_id, :cart_id, :quantity, :_destroy,
product_attributes: [:id, :sku, :euro_price, :sterling_price, :product_group_id, :product_size_id, :product_waistband_id]])
end
schema.rb
create_table "orders", force: :cascade do |t|
t.datetime "created_at", null: false
t.datetime "updated_at", null: false
t.boolean "returned", default: false
t.date "date_sent"
t.date "ordered_at"
t.integer "user_id"
t.boolean "return_requested", default: false
t.integer "stage", default: 0
t.decimal "order_total", default: 0.0
t.string "transaction_secret"
t.string "token"
t.string "uuid"
t.string "currency"
t.float "discounted_by", default: 0.0
end
add_index "line_items", ["cart_id"], name: "index_line_items_on_cart_id", using: :btree
add_index "line_items", ["order_id"], name: "index_line_items_on_order_id", using: :btree
add_index "line_items", ["product_id"], name: "index_line_items_on_product_id", using: :btree
create_table "line_items", force: :cascade do |t|
t.integer "quantity"
t.integer "order_id"
t.integer "cart_id"
t.integer "product_id"
t.datetime "created_at", null: false
t.datetime "updated_at", null: false
t.float "unit_price"
t.string "currency"
end
create_table "product_groups", force: :cascade do |t|
t.string "name"
t.text "description"
t.datetime "created_at", null: false
t.datetime "updated_at", null: false
end
create_table "product_sizes", force: :cascade do |t|
t.string "specification"
t.datetime "created_at", null: false
t.datetime "updated_at", null: false
end
create_table "product_waistbands", force: :cascade do |t|
t.string "specification"
t.datetime "created_at", null: false
t.datetime "updated_at", null: false
end
create_table "products", force: :cascade do |t|
t.integer "sku"
t.integer "product_group_id"
t.integer "product_size_id"
t.integer "product_waistband_id"
t.decimal "euro_price"
t.decimal "sterling_price"
t.datetime "created_at", null: false
t.datetime "updated_at", null: false
t.integer "stock_level", default: 0
end
add_index "products", ["product_group_id"], name: "index_products_on_product_group_id", using: :btree
add_index "products", ["product_size_id"], name: "index_products_on_product_size_id", using: :btree
add_index "products", ["product_waistband_id"], name: "index_products_on_product_waistband_id", using: :btree
Также в модели Order я рандомизирую идентификатор before_create, чтобы, когда пользователь отправляет форму, он создавал двойную копию с другим идентификатором Order. Это то же самое для LineItems.
Order.rb (то же самое в LineItem.rb)
before_create :randomize_id
private
def randomize_id
begin
self.id = SecureRandom.random_number(1_000_000)
end while Order.where(id: self.id).exists?
end
1 ответ
Мой подход заключается в том, чтобы переопределить метод ActiveRecord::Base#dup в модели Order, чтобы он был рекурсивным, то есть он также дублировал коллекцию LineItem:
class Order < ActiveRecord::Base
def dup
duped_order = super
duped_order.line_items = line_items.map(&:dup)
duped_order
end
end
делая это таким образом, это легко тестируется. Теперь контроллер становится:
class OrderController < ApplicationController
def new
@order = Order.find(params[:id]).dup
end
def create
# not sure how your form populates the params hash
# here you need to new-up and then save the order and the line items
# with the attributes from the form
end
end
Пишите тесты, чтобы подтвердить, что вы делаете то, что намереваетесь. Это прекрасный пример того, где следует применять старую парадигму "толстая модель тощий контроллер".