mirror of
https://github.com/we-promise/sure.git
synced 2026-06-01 00:39:01 +00:00
* refactor: rename `raw_investments_payload` to `raw_holdings_payload` - Update references and models to use consistent naming. - Adjust migrations, tests, and encryption setup accordingly. * fix: improve safety when accessing raw_holdings_payload keys - Use `dig` with safe navigation to prevent potential nil errors. - Add support for decryption from the old column name `raw_investments_payload`. - Adjust related methods and calculations for consistency. --------- Co-authored-by: luckyPipewrench <luckypipewrench@proton.me>
93 lines
2.5 KiB
Ruby
93 lines
2.5 KiB
Ruby
class PlaidAccount::Investments::HoldingsProcessor
|
|
def initialize(plaid_account, security_resolver:)
|
|
@plaid_account = plaid_account
|
|
@security_resolver = security_resolver
|
|
end
|
|
|
|
def process
|
|
holdings.each do |plaid_holding|
|
|
resolved_security_result = security_resolver.resolve(plaid_security_id: plaid_holding["security_id"])
|
|
|
|
next unless resolved_security_result.security.present?
|
|
|
|
security = resolved_security_result.security
|
|
|
|
# Parse quantity and price into BigDecimal for proper arithmetic
|
|
quantity_bd = parse_decimal(plaid_holding["quantity"])
|
|
price_bd = parse_decimal(plaid_holding["institution_price"])
|
|
|
|
# Skip if essential values are missing
|
|
next if quantity_bd.nil? || price_bd.nil?
|
|
|
|
# Compute amount using BigDecimal arithmetic to avoid floating point drift
|
|
amount_bd = quantity_bd * price_bd
|
|
|
|
# Normalize date - handle string, Date, or nil
|
|
holding_date = parse_date(plaid_holding["institution_price_as_of"]) || Date.current
|
|
|
|
import_adapter.import_holding(
|
|
security: security,
|
|
quantity: quantity_bd,
|
|
amount: amount_bd,
|
|
currency: plaid_holding["iso_currency_code"] || account.currency,
|
|
date: holding_date,
|
|
price: price_bd,
|
|
account_provider_id: plaid_account.account_provider&.id,
|
|
source: "plaid",
|
|
delete_future_holdings: false # Plaid doesn't allow holdings deletion
|
|
)
|
|
end
|
|
end
|
|
|
|
private
|
|
attr_reader :plaid_account, :security_resolver
|
|
|
|
def import_adapter
|
|
@import_adapter ||= Account::ProviderImportAdapter.new(account)
|
|
end
|
|
|
|
def account
|
|
plaid_account.current_account
|
|
end
|
|
|
|
def holdings
|
|
plaid_account.raw_holdings_payload&.[]("holdings") || []
|
|
end
|
|
|
|
def parse_decimal(value)
|
|
return nil if value.nil?
|
|
|
|
case value
|
|
when BigDecimal
|
|
value
|
|
when String
|
|
BigDecimal(value)
|
|
when Numeric
|
|
BigDecimal(value.to_s)
|
|
else
|
|
nil
|
|
end
|
|
rescue ArgumentError => e
|
|
Rails.logger.error("Failed to parse Plaid holding decimal value: #{value.inspect} - #{e.message}")
|
|
nil
|
|
end
|
|
|
|
def parse_date(date_value)
|
|
return nil if date_value.nil?
|
|
|
|
case date_value
|
|
when Date
|
|
date_value
|
|
when String
|
|
Date.parse(date_value)
|
|
when Time, DateTime
|
|
date_value.to_date
|
|
else
|
|
nil
|
|
end
|
|
rescue ArgumentError, TypeError => e
|
|
Rails.logger.error("Failed to parse Plaid holding date: #{date_value.inspect} - #{e.message}")
|
|
nil
|
|
end
|
|
end
|