mirror of
https://github.com/we-promise/sure.git
synced 2026-04-17 19:14:11 +00:00
191 lines
6.9 KiB
Ruby
191 lines
6.9 KiB
Ruby
require "test_helper"
|
|
|
|
class InvestmentStatementTest < ActiveSupport::TestCase
|
|
setup do
|
|
@family = families(:empty)
|
|
# families(:empty) defaults to currency "USD"
|
|
@statement = InvestmentStatement.new(@family, user: nil)
|
|
end
|
|
|
|
test "portfolio_value and cash_balance with a single-currency family" do
|
|
create_investment_account(balance: 1000, cash_balance: 100)
|
|
|
|
assert_equal 1000, @statement.portfolio_value
|
|
assert_equal 100, @statement.cash_balance
|
|
assert_equal 900, @statement.holdings_value
|
|
end
|
|
|
|
test "portfolio_value converts foreign-currency accounts to family currency" do
|
|
create_investment_account(balance: 1921.92, cash_balance: -162, currency: "USD")
|
|
create_investment_account(balance: 1000, cash_balance: 1000, currency: "EUR")
|
|
|
|
ExchangeRate.create!(
|
|
from_currency: "EUR",
|
|
to_currency: "USD",
|
|
date: Date.current,
|
|
rate: 1.1
|
|
)
|
|
|
|
# 1921.92 + 1000 * 1.1 = 3021.92
|
|
assert_in_delta 3021.92, @statement.portfolio_value, 0.001
|
|
# -162 + 1000 * 1.1 = 938
|
|
assert_in_delta 938, @statement.cash_balance, 0.001
|
|
# 3021.92 - 938 = 2083.92
|
|
assert_in_delta 2083.92, @statement.holdings_value, 0.001
|
|
end
|
|
|
|
test "portfolio_value falls back to 1:1 when FX rate is missing" do
|
|
create_investment_account(balance: 1921.92, currency: "USD")
|
|
create_investment_account(balance: 1000, currency: "EUR")
|
|
|
|
# No ExchangeRate row → rates_for defaults to 1
|
|
assert_in_delta 2921.92, @statement.portfolio_value, 0.001
|
|
end
|
|
|
|
test "current_holdings includes holdings from every investment account regardless of currency" do
|
|
usd_account = create_investment_account(balance: 2100, currency: "USD")
|
|
eur_account = create_investment_account(balance: 2000, currency: "EUR")
|
|
|
|
usd_security = Security.create!(ticker: "AAPL", name: "Apple")
|
|
eur_security = Security.create!(ticker: "ASML", name: "ASML")
|
|
|
|
Holding.create!(
|
|
account: usd_account, security: usd_security, date: Date.current,
|
|
qty: 10, price: 210, amount: 2100, currency: "USD"
|
|
)
|
|
Holding.create!(
|
|
account: eur_account, security: eur_security, date: Date.current,
|
|
qty: 4, price: 500, amount: 2000, currency: "EUR"
|
|
)
|
|
|
|
assert_equal 2, @statement.current_holdings.count
|
|
end
|
|
|
|
test "top_holdings ranks by family-currency value across currencies" do
|
|
usd_account = create_investment_account(balance: 2100, currency: "USD")
|
|
eur_account = create_investment_account(balance: 2000, currency: "EUR")
|
|
|
|
usd_security = Security.create!(ticker: "AAPL", name: "Apple")
|
|
eur_security = Security.create!(ticker: "ASML", name: "ASML")
|
|
|
|
Holding.create!(
|
|
account: usd_account, security: usd_security, date: Date.current,
|
|
qty: 10, price: 210, amount: 2100, currency: "USD"
|
|
)
|
|
Holding.create!(
|
|
account: eur_account, security: eur_security, date: Date.current,
|
|
qty: 4, price: 500, amount: 2000, currency: "EUR"
|
|
)
|
|
|
|
ExchangeRate.create!(
|
|
from_currency: "EUR", to_currency: "USD",
|
|
date: Date.current, rate: 1.1
|
|
)
|
|
|
|
# 2000 EUR = 2200 USD > 2100 USD, so ASML outranks AAPL in family currency
|
|
top = @statement.top_holdings(limit: 2)
|
|
assert_equal %w[ASML AAPL], top.map(&:ticker)
|
|
end
|
|
|
|
test "allocation weights sum to 100% with mixed currencies" do
|
|
usd_account = create_investment_account(balance: 2100, currency: "USD")
|
|
eur_account = create_investment_account(balance: 2000, currency: "EUR")
|
|
|
|
usd_security = Security.create!(ticker: "AAPL", name: "Apple")
|
|
eur_security = Security.create!(ticker: "ASML", name: "ASML")
|
|
|
|
Holding.create!(
|
|
account: usd_account, security: usd_security, date: Date.current,
|
|
qty: 10, price: 210, amount: 2100, currency: "USD"
|
|
)
|
|
Holding.create!(
|
|
account: eur_account, security: eur_security, date: Date.current,
|
|
qty: 4, price: 500, amount: 2000, currency: "EUR"
|
|
)
|
|
|
|
ExchangeRate.create!(
|
|
from_currency: "EUR", to_currency: "USD",
|
|
date: Date.current, rate: 1.1
|
|
)
|
|
|
|
allocation = @statement.allocation
|
|
assert_equal 2, allocation.size
|
|
assert_in_delta 100.0, allocation.sum(&:weight), 0.01
|
|
# Every row is labeled in family currency
|
|
assert allocation.all? { |a| a.amount.currency.iso_code == "USD" }
|
|
end
|
|
|
|
test "unrealized_gains sums in family currency with mixed-currency holdings" do
|
|
usd_account = create_investment_account(balance: 2100, currency: "USD")
|
|
eur_account = create_investment_account(balance: 2000, currency: "EUR")
|
|
|
|
usd_security = Security.create!(ticker: "AAPL", name: "Apple")
|
|
eur_security = Security.create!(ticker: "ASML", name: "ASML")
|
|
|
|
Holding.create!(
|
|
account: usd_account, security: usd_security, date: Date.current,
|
|
qty: 10, price: 210, amount: 2100, currency: "USD",
|
|
cost_basis: 200, cost_basis_locked: true
|
|
)
|
|
Holding.create!(
|
|
account: eur_account, security: eur_security, date: Date.current,
|
|
qty: 4, price: 500, amount: 2000, currency: "EUR",
|
|
cost_basis: 450, cost_basis_locked: true
|
|
)
|
|
|
|
ExchangeRate.create!(
|
|
from_currency: "EUR", to_currency: "USD",
|
|
date: Date.current, rate: 1.1
|
|
)
|
|
|
|
# AAPL unrealized = 2100 - (10 * 200) = 100 USD
|
|
# ASML unrealized = 2000 - (4 * 450) = 200 EUR → 220 USD @ 1.1
|
|
# Total = 320 USD
|
|
assert_in_delta 320, @statement.unrealized_gains, 0.001
|
|
assert_equal "USD", @statement.unrealized_gains_money.currency.iso_code
|
|
end
|
|
|
|
test "unrealized_gains_trend is denominated in family currency" do
|
|
usd_account = create_investment_account(balance: 2100, currency: "USD")
|
|
eur_account = create_investment_account(balance: 2000, currency: "EUR")
|
|
|
|
usd_security = Security.create!(ticker: "AAPL", name: "Apple")
|
|
eur_security = Security.create!(ticker: "ASML", name: "ASML")
|
|
|
|
Holding.create!(
|
|
account: usd_account, security: usd_security, date: Date.current,
|
|
qty: 10, price: 210, amount: 2100, currency: "USD",
|
|
cost_basis: 200, cost_basis_locked: true
|
|
)
|
|
Holding.create!(
|
|
account: eur_account, security: eur_security, date: Date.current,
|
|
qty: 4, price: 500, amount: 2000, currency: "EUR",
|
|
cost_basis: 450, cost_basis_locked: true
|
|
)
|
|
|
|
ExchangeRate.create!(
|
|
from_currency: "EUR", to_currency: "USD",
|
|
date: Date.current, rate: 1.1
|
|
)
|
|
|
|
trend = @statement.unrealized_gains_trend
|
|
assert_equal "USD", trend.current.currency.iso_code
|
|
assert_equal "USD", trend.previous.currency.iso_code
|
|
# current = 2100 USD + (2000 EUR * 1.1) = 4300 USD
|
|
assert_in_delta 4300, trend.current.amount, 0.001
|
|
# previous (cost basis) = (10 * 200) USD + (4 * 450 * 1.1) EUR→USD = 2000 + 1980 = 3980 USD
|
|
assert_in_delta 3980, trend.previous.amount, 0.001
|
|
end
|
|
|
|
private
|
|
def create_investment_account(balance:, cash_balance: 0, currency: "USD")
|
|
@family.accounts.create!(
|
|
name: "Investment #{SecureRandom.hex(3)}",
|
|
balance: balance,
|
|
cash_balance: cash_balance,
|
|
currency: currency,
|
|
accountable: Investment.new
|
|
)
|
|
end
|
|
end
|