Как мне установить эту ассоциацию has_many? - PullRequest
0 голосов
/ 13 июня 2018

Итак, у меня есть модель PortStock, которая имеет атрибуты enum action: buy и action: sell.Каждый PortStock может быть либо position: open, либо position: closed (также значения enum).

Однако то, что закрывает одно PortStock, это другое PortStock (то есть PortStock.sell закрывает PortStock.buy).

Итак, я создал модель соединения под названием ClosedPositions, иПо сути, я хочу записать метаданные о закрытии одного PortStock другим.

Это схема и модель для моей ClosedPosition.rb модели:

# == Schema Information
#
# Table name: closed_positions
#
#  id             :bigint(8)        not null, primary key
#  closer_id      :integer
#  closed_id      :integer
#  num_units      :integer
#  closed_price   :float
#  dollar_change  :float
#  percent_change :float
#  ticker         :string
#  created_at     :datetime         not null
#  updated_at     :datetime         not null
#

class ClosedPosition < ApplicationRecord
  belongs_to :closer, class_name: "PortStock", foreign_key: "closer_id"
  belongs_to :closed, class_name: "PortStock", foreign_key: "closed_id" 
end

На моей PortStock модели у меня простое has_many :closed_positions.

Однако, когда я пытаюсь добавить closed_position к существующему port_stock объекту, я получаю ошибку unknown attribute port_stock_id.

[27] pry(main)> closer = PortStock.find(8)
  PortStock Load (0.4ms)  SELECT  "port_stocks".* FROM "port_stocks" WHERE "port_stocks"."id" = $1 LIMIT $2  [["id", 8], ["LIMIT", 1]]
=> #<PortStock:0x00007fc951a47c30
 id: 8,
 portfolio_id: 1,
 stock_id: 84,
 volume: 250,
 purchase_price: 4.1,
 current_price: 4.0,
 percent_change: -2.43902439024389,
 created_at: Tue, 12 Jun 2018 00:46:09 UTC +00:00,
 updated_at: Tue, 12 Jun 2018 00:46:09 UTC +00:00,
 current_value: 1000.0,
 dollar_change: -24.9999999999999,
 total_spend: 1025.0,
 bought_on: Sat, 09 Jun 2018 00:00:00 UTC +00:00,
 action: "sell",
 position: "open",
 ticker: "KEY">

[28] pry(main)> closed = PortStock.find(5)
  PortStock Load (0.7ms)  SELECT  "port_stocks".* FROM "port_stocks" WHERE "port_stocks"."id" = $1 LIMIT $2  [["id", 5], ["LIMIT", 1]]
=> #<PortStock:0x00007fc951a55858
 id: 5,
 portfolio_id: 1,
 stock_id: 84,
 volume: 250,
 purchase_price: 4.1,
 current_price: 4.0,
 percent_change: -2.43902439024389,
 created_at: Mon, 11 Jun 2018 03:11:10 UTC +00:00,
 updated_at: Mon, 11 Jun 2018 23:28:30 UTC +00:00,
 current_value: 1000.0,
 dollar_change: -24.9999999999999,
 total_spend: 1025.0,
 bought_on: Sat, 09 Jun 2018 00:00:00 UTC +00:00,
 action: "buy",
 position: "open",
 ticker: "KEY">

[34] pry(main)> closer.closed_positions.create(closed: closed, num_units: closer.volume, closed_price: closed.stock.price, ticker: closed.ticker)
  Stock Load (1.7ms)  SELECT  "stocks".* FROM "stocks" WHERE "stocks"."id" = $1 LIMIT $2  [["id", 84], ["LIMIT", 1]]
   (0.2ms)  BEGIN
   (0.3ms)  ROLLBACK
ActiveModel::UnknownAttributeError: unknown attribute 'port_stock_id' for ClosedPosition.
from /.rvm/rubies/ruby-2.5.1/lib/ruby/gems/2.5.0/gems/activemodel-5.2.0/lib/active_model/attribute_assignment.rb:53:in `_assign_attribute'
[35] pry(main)> closer.closed_positions.create(closed_id: closed.id, num_units: closer.volume, closed_price: closed.stock.price, ticker: closed.ticker)
   (0.2ms)  BEGIN
   (0.3ms)  ROLLBACK
ActiveModel::UnknownAttributeError: unknown attribute 'port_stock_id' for ClosedPosition.
from /.rvm/rubies/ruby-2.5.1/lib/ruby/gems/2.5.0/gems/activemodel-5.2.0/lib/active_model/attribute_assignment.rb:53:in `_assign_attribute'

Как это исправить?

Подхожу ли я к этому правильно?Я могу испортить синтаксис и логику ассоциации.

Ответы [ 2 ]

0 голосов
/ 13 июня 2018

ActiveRecord жаловался на то, что в моей ClosedPosition модели не было столбца port_stock_id.

Итак, я просто переименовал столбец closed_id в port_stock_id, а затем изменил свойассоциация в моей ClosedPosition модели от:

class ClosedPosition < ApplicationRecord
  belongs_to :closed, class_name: "PortStock", foreign_key: "closed_id" 
end

До:

class ClosedPosition < ApplicationRecord
  belongs_to :closed, class_name: "PortStock", foreign_key: "port_stock_id" 
end

Работает как шарм.

0 голосов
/ 13 июня 2018

Когда вы пытаетесь создать запись closed_positions, ActiveRecord предполагает наличие внешнего ключа port_stock_id, которого в этом случае нет.Поэтому попробуйте решение ниже, добавьте ассоциативные ассоциации.Я добавил оба (я не знаю его актуальность с точки зрения текущего контекста).Таким образом, вы сможете создавать связанные записи.

class PortStock < ApplicationRecord has_many :closed_stock_positions, foreign_key: 'closed_id', class_name: 'ClosedPosition' has_many :closer_stock_positions, foreign_key: 'closer_id', class_name: 'ClosedPosition' end

...