mirror of
https://github.com/maybe-finance/maybe.git
synced 2025-08-04 21:15:19 +02:00
Transaction transfers, payments, and matching (#883)
* Add transfer model and clean up family snapshot fixtures * Ignore transfers in income and expense snapshots * Add transfer validations * Implement basic transfer matching UI * Fix merge conflicts * Add missing translations * Tweak selection states for transfer types * Add missing i18n translation
This commit is contained in:
parent
b462bc8f8c
commit
ca39b26070
57 changed files with 991 additions and 427 deletions
|
@ -2,111 +2,82 @@ require "test_helper"
|
|||
require "csv"
|
||||
|
||||
class Account::Balance::CalculatorTest < ActiveSupport::TestCase
|
||||
# See: https://docs.google.com/spreadsheets/d/18LN5N-VLq4b49Mq1fNwF7_eBiHSQB46qQduRtdAEN98/edit?usp=sharing
|
||||
setup do
|
||||
@expected_balances = CSV.read("test/fixtures/account/expected_balances.csv", headers: true).map do |row|
|
||||
{
|
||||
"date" => (Date.current + row["date_offset"].to_i.days).to_date,
|
||||
"collectable" => row["collectable"],
|
||||
"checking" => row["checking"],
|
||||
"savings_with_valuation_overrides" => row["savings_with_valuation_overrides"],
|
||||
"credit_card" => row["credit_card"],
|
||||
"multi_currency" => row["multi_currency"],
|
||||
include FamilySnapshotTestHelper
|
||||
|
||||
# Balances should be calculated for all currencies of an account
|
||||
"eur_checking_eur" => row["eur_checking_eur"],
|
||||
"eur_checking_usd" => row["eur_checking_usd"]
|
||||
}
|
||||
test "syncs other asset balances" do
|
||||
expected_balances = get_expected_balances_for(:collectable)
|
||||
assert_account_balances calculated_balances_for(:collectable), expected_balances
|
||||
end
|
||||
|
||||
test "syncs other liability balances" do
|
||||
expected_balances = get_expected_balances_for(:iou)
|
||||
assert_account_balances calculated_balances_for(:iou), expected_balances
|
||||
end
|
||||
|
||||
test "syncs credit balances" do
|
||||
expected_balances = get_expected_balances_for :credit_card
|
||||
assert_account_balances calculated_balances_for(:credit_card), expected_balances
|
||||
end
|
||||
|
||||
test "syncs checking account balances" do
|
||||
expected_balances = get_expected_balances_for(:checking)
|
||||
assert_account_balances calculated_balances_for(:checking), expected_balances
|
||||
end
|
||||
|
||||
test "syncs foreign checking account balances" do
|
||||
# Foreign accounts will generate balances for all currencies
|
||||
expected_usd_balances = get_expected_balances_for(:eur_checking_usd)
|
||||
expected_eur_balances = get_expected_balances_for(:eur_checking_eur)
|
||||
|
||||
calculated_balances = calculated_balances_for(:eur_checking)
|
||||
calculated_usd_balances = calculated_balances.select { |b| b[:currency] == "USD" }
|
||||
calculated_eur_balances = calculated_balances.select { |b| b[:currency] == "EUR" }
|
||||
|
||||
assert_account_balances calculated_usd_balances, expected_usd_balances
|
||||
assert_account_balances calculated_eur_balances, expected_eur_balances
|
||||
end
|
||||
|
||||
test "syncs multi-currency checking account balances" do
|
||||
expected_balances = get_expected_balances_for(:multi_currency)
|
||||
assert_account_balances calculated_balances_for(:multi_currency), expected_balances
|
||||
end
|
||||
|
||||
test "syncs savings accounts balances" do
|
||||
expected_balances = get_expected_balances_for(:savings)
|
||||
assert_account_balances calculated_balances_for(:savings), expected_balances
|
||||
end
|
||||
|
||||
test "syncs investment account balances" do
|
||||
expected_balances = get_expected_balances_for(:brokerage)
|
||||
assert_account_balances calculated_balances_for(:brokerage), expected_balances
|
||||
end
|
||||
|
||||
test "syncs loan account balances" do
|
||||
expected_balances = get_expected_balances_for(:mortgage_loan)
|
||||
assert_account_balances calculated_balances_for(:mortgage_loan), expected_balances
|
||||
end
|
||||
|
||||
test "syncs property account balances" do
|
||||
expected_balances = get_expected_balances_for(:house)
|
||||
assert_account_balances calculated_balances_for(:house), expected_balances
|
||||
end
|
||||
|
||||
test "syncs vehicle account balances" do
|
||||
expected_balances = get_expected_balances_for(:car)
|
||||
assert_account_balances calculated_balances_for(:car), expected_balances
|
||||
end
|
||||
|
||||
private
|
||||
def assert_account_balances(actual_balances, expected_balances)
|
||||
assert_equal expected_balances.count, actual_balances.count
|
||||
|
||||
actual_balances.each do |ab|
|
||||
expected_balance = expected_balances.find { |eb| eb[:date] == ab[:date] }
|
||||
assert_in_delta expected_balance[:balance], ab[:balance], 0.01, "Balance incorrect on date: #{ab[:date]}"
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
test "syncs account with only valuations" do
|
||||
account = accounts(:collectable)
|
||||
|
||||
calculator = Account::Balance::Calculator.new(account)
|
||||
calculator.calculate
|
||||
|
||||
expected = @expected_balances.map { |row| row["collectable"].to_d }
|
||||
actual = calculator.daily_balances.map { |b| b[:balance] }
|
||||
|
||||
assert_equal expected, actual
|
||||
end
|
||||
|
||||
test "syncs account with only transactions" do
|
||||
account = accounts(:checking)
|
||||
|
||||
calculator = Account::Balance::Calculator.new(account)
|
||||
calculator.calculate
|
||||
|
||||
expected = @expected_balances.map { |row| row["checking"].to_d }
|
||||
actual = calculator.daily_balances.map { |b| b[:balance] }
|
||||
|
||||
assert_equal expected, actual
|
||||
end
|
||||
|
||||
test "syncs account with both valuations and transactions" do
|
||||
account = accounts(:savings_with_valuation_overrides)
|
||||
|
||||
calculator = Account::Balance::Calculator.new(account)
|
||||
calculator.calculate
|
||||
|
||||
expected = @expected_balances.map { |row| row["savings_with_valuation_overrides"].to_d }
|
||||
actual = calculator.daily_balances.map { |b| b[:balance] }
|
||||
|
||||
assert_equal expected, actual
|
||||
end
|
||||
|
||||
test "syncs liability account" do
|
||||
account = accounts(:credit_card)
|
||||
|
||||
calculator = Account::Balance::Calculator.new(account)
|
||||
calculator.calculate
|
||||
|
||||
expected = @expected_balances.map { |row| row["credit_card"].to_d }
|
||||
actual = calculator.daily_balances.map { |b| b[:balance] }
|
||||
|
||||
assert_equal expected, actual
|
||||
end
|
||||
|
||||
test "syncs foreign currency account" do
|
||||
account = accounts(:eur_checking)
|
||||
calculator = Account::Balance::Calculator.new(account)
|
||||
calculator.calculate
|
||||
|
||||
# Calculator should calculate balances in both account and family currency
|
||||
expected_eur_balances = @expected_balances.map { |row| row["eur_checking_eur"].to_d }
|
||||
expected_usd_balances = @expected_balances.map { |row| row["eur_checking_usd"].to_d }
|
||||
|
||||
actual_eur_balances = calculator.daily_balances.select { |b| b[:currency] == "EUR" }.sort_by { |b| b[:date] }.map { |b| b[:balance] }
|
||||
actual_usd_balances = calculator.daily_balances.select { |b| b[:currency] == "USD" }.sort_by { |b| b[:date] }.map { |b| b[:balance] }
|
||||
|
||||
assert_equal expected_eur_balances, actual_eur_balances
|
||||
assert_equal expected_usd_balances, actual_usd_balances
|
||||
end
|
||||
|
||||
test "syncs multi currency account" do
|
||||
account = accounts(:multi_currency)
|
||||
calculator = Account::Balance::Calculator.new(account)
|
||||
calculator.calculate
|
||||
|
||||
expected_balances = @expected_balances.map { |row| row["multi_currency"].to_d }
|
||||
|
||||
actual_balances = calculator.daily_balances.map { |b| b[:balance] }
|
||||
|
||||
assert_equal expected_balances, actual_balances
|
||||
end
|
||||
|
||||
test "syncs with overridden start date" do
|
||||
account = accounts(:multi_currency)
|
||||
account.sync
|
||||
calc_start_date = 10.days.ago.to_date
|
||||
calculator = Account::Balance::Calculator.new(account, { calc_start_date: })
|
||||
calculator.calculate
|
||||
|
||||
expected_balances = @expected_balances.filter { |row| row["date"] >= calc_start_date }.map { |row| row["multi_currency"].to_d }
|
||||
|
||||
actual_balances = calculator.daily_balances.map { |b| b[:balance] }
|
||||
|
||||
assert_equal expected_balances, actual_balances
|
||||
end
|
||||
def calculated_balances_for(account_key)
|
||||
Account::Balance::Calculator.new(accounts(account_key)).calculate.daily_balances
|
||||
end
|
||||
end
|
||||
|
|
|
@ -4,7 +4,7 @@ class Account::SyncableTest < ActiveSupport::TestCase
|
|||
include ActiveJob::TestHelper
|
||||
|
||||
setup do
|
||||
@account = accounts(:savings_with_valuation_overrides)
|
||||
@account = accounts(:savings)
|
||||
end
|
||||
|
||||
test "triggers sync job" do
|
||||
|
@ -14,27 +14,27 @@ class Account::SyncableTest < ActiveSupport::TestCase
|
|||
end
|
||||
|
||||
test "account has no balances until synced" do
|
||||
account = accounts(:savings_with_valuation_overrides)
|
||||
account = accounts(:savings)
|
||||
|
||||
assert_equal 0, account.balances.count
|
||||
end
|
||||
|
||||
test "account has balances after syncing" do
|
||||
account = accounts(:savings_with_valuation_overrides)
|
||||
account = accounts(:savings)
|
||||
account.sync
|
||||
|
||||
assert_equal 31, account.balances.count
|
||||
assert_equal 32, account.balances.count
|
||||
end
|
||||
|
||||
test "partial sync with missing historical balances performs a full sync" do
|
||||
account = accounts(:savings_with_valuation_overrides)
|
||||
account = accounts(:savings)
|
||||
account.sync 10.days.ago.to_date
|
||||
|
||||
assert_equal 31, account.balances.count
|
||||
assert_equal 32, account.balances.count
|
||||
end
|
||||
|
||||
test "balances are updated after syncing" do
|
||||
account = accounts(:savings_with_valuation_overrides)
|
||||
account = accounts(:savings)
|
||||
balance_date = 10.days.ago
|
||||
account.balances.create!(date: balance_date, balance: 1000)
|
||||
account.sync
|
||||
|
@ -43,7 +43,7 @@ class Account::SyncableTest < ActiveSupport::TestCase
|
|||
end
|
||||
|
||||
test "balances before sync start date are not updated after syncing" do
|
||||
account = accounts(:savings_with_valuation_overrides)
|
||||
account = accounts(:savings)
|
||||
balance_date = 10.days.ago
|
||||
account.balances.create!(date: balance_date, balance: 1000)
|
||||
account.sync 5.days.ago.to_date
|
||||
|
@ -52,7 +52,7 @@ class Account::SyncableTest < ActiveSupport::TestCase
|
|||
end
|
||||
|
||||
test "balances after sync start date are updated after syncing" do
|
||||
account = accounts(:savings_with_valuation_overrides)
|
||||
account = accounts(:savings)
|
||||
balance_date = 10.days.ago
|
||||
account.balances.create!(date: balance_date, balance: 1000)
|
||||
account.sync 20.days.ago.to_date
|
||||
|
@ -61,7 +61,7 @@ class Account::SyncableTest < ActiveSupport::TestCase
|
|||
end
|
||||
|
||||
test "balance on the sync date is updated after syncing" do
|
||||
account = accounts(:savings_with_valuation_overrides)
|
||||
account = accounts(:savings)
|
||||
balance_date = 5.days.ago
|
||||
account.balances.create!(date: balance_date, balance: 1000)
|
||||
account.sync balance_date.to_date
|
||||
|
@ -73,13 +73,13 @@ class Account::SyncableTest < ActiveSupport::TestCase
|
|||
account = accounts(:eur_checking)
|
||||
account.sync
|
||||
|
||||
assert_equal 62, account.balances.count
|
||||
assert_equal 31, account.balances.where(currency: "EUR").count
|
||||
assert_equal 31, account.balances.where(currency: "USD").count
|
||||
assert_equal 64, account.balances.count
|
||||
assert_equal 32, account.balances.where(currency: "EUR").count
|
||||
assert_equal 32, account.balances.where(currency: "USD").count
|
||||
end
|
||||
|
||||
test "stale balances are purged after syncing" do
|
||||
account = accounts(:savings_with_valuation_overrides)
|
||||
account = accounts(:savings)
|
||||
|
||||
# Create old, stale balances that should be purged (since they are before account start date)
|
||||
account.balances.create!(date: 1.year.ago, balance: 1000)
|
||||
|
@ -88,14 +88,6 @@ class Account::SyncableTest < ActiveSupport::TestCase
|
|||
|
||||
account.sync
|
||||
|
||||
assert_equal 31, account.balances.count
|
||||
end
|
||||
|
||||
test "account balance is updated after sync" do
|
||||
account = accounts(:savings_with_valuation_overrides)
|
||||
|
||||
assert_changes -> { account.balance }, to: 20500 do
|
||||
account.sync
|
||||
end
|
||||
assert_equal 32, account.balances.count
|
||||
end
|
||||
end
|
||||
|
|
|
@ -5,16 +5,6 @@ class AccountTest < ActiveSupport::TestCase
|
|||
def setup
|
||||
@account = accounts(:checking)
|
||||
@family = families(:dylan_family)
|
||||
@snapshots = CSV.read("test/fixtures/family/expected_snapshots.csv", headers: true).map do |row|
|
||||
{
|
||||
"date" => (Date.current + row["date_offset"].to_i.days).to_date,
|
||||
"assets" => row["assets"],
|
||||
"liabilities" => row["liabilities"],
|
||||
"Account::Depository" => row["depositories"],
|
||||
"Account::Credit" => row["credits"],
|
||||
"Account::OtherAsset" => row["other_assets"]
|
||||
}
|
||||
end
|
||||
end
|
||||
|
||||
test "new account should be valid" do
|
||||
|
@ -47,26 +37,23 @@ class AccountTest < ActiveSupport::TestCase
|
|||
test "syncs regular account" do
|
||||
@account.sync
|
||||
assert_equal "ok", @account.status
|
||||
assert_equal 31, @account.balances.count
|
||||
assert_equal 32, @account.balances.count
|
||||
end
|
||||
|
||||
test "syncs foreign currency account" do
|
||||
account = accounts(:eur_checking)
|
||||
account.sync
|
||||
assert_equal "ok", account.status
|
||||
assert_equal 31, account.balances.where(currency: "USD").count
|
||||
assert_equal 31, account.balances.where(currency: "EUR").count
|
||||
assert_equal 32, account.balances.where(currency: "USD").count
|
||||
assert_equal 32, account.balances.where(currency: "EUR").count
|
||||
end
|
||||
|
||||
test "groups accounts by type" do
|
||||
@family.accounts.each do |account|
|
||||
account.sync
|
||||
end
|
||||
|
||||
result = @family.accounts.by_group(period: Period.all)
|
||||
|
||||
expected_assets = @snapshots.last["assets"].to_d
|
||||
expected_liabilities = @snapshots.last["liabilities"].to_d
|
||||
|
||||
assets = result[:assets]
|
||||
liabilities = result[:liabilities]
|
||||
|
||||
|
@ -84,14 +71,14 @@ class AccountTest < ActiveSupport::TestCase
|
|||
other_liabilities = liabilities.children.find { |group| group.name == "Account::OtherLiability" }
|
||||
|
||||
assert_equal 4, depositories.children.count
|
||||
assert_equal 0, properties.children.count
|
||||
assert_equal 0, vehicles.children.count
|
||||
assert_equal 0, investments.children.count
|
||||
assert_equal 1, properties.children.count
|
||||
assert_equal 1, vehicles.children.count
|
||||
assert_equal 1, investments.children.count
|
||||
assert_equal 1, other_assets.children.count
|
||||
|
||||
assert_equal 1, credits.children.count
|
||||
assert_equal 0, loans.children.count
|
||||
assert_equal 0, other_liabilities.children.count
|
||||
assert_equal 1, loans.children.count
|
||||
assert_equal 1, other_liabilities.children.count
|
||||
end
|
||||
|
||||
test "generates series with last balance equal to current account balance" do
|
||||
|
|
|
@ -2,26 +2,13 @@ require "test_helper"
|
|||
require "csv"
|
||||
|
||||
class FamilyTest < ActiveSupport::TestCase
|
||||
include FamilySnapshotTestHelper
|
||||
|
||||
def setup
|
||||
@family = families(:dylan_family)
|
||||
|
||||
@family.accounts.each do |account|
|
||||
account.sync
|
||||
end
|
||||
|
||||
# See this Google Sheet for calculations and expected results for dylan_family:
|
||||
# https://docs.google.com/spreadsheets/d/18LN5N-VLq4b49Mq1fNwF7_eBiHSQB46qQduRtdAEN98/edit?usp=sharing
|
||||
@expected_snapshots = CSV.read("test/fixtures/family/expected_snapshots.csv", headers: true).map do |row|
|
||||
{
|
||||
"date" => (Date.current + row["date_offset"].to_i.days).to_date,
|
||||
"net_worth" => row["net_worth"],
|
||||
"assets" => row["assets"],
|
||||
"liabilities" => row["liabilities"],
|
||||
"rolling_spend" => row["rolling_spend"],
|
||||
"rolling_income" => row["rolling_income"],
|
||||
"savings_rate" => row["savings_rate"]
|
||||
}
|
||||
end
|
||||
end
|
||||
|
||||
test "should have many users" do
|
||||
|
@ -58,57 +45,62 @@ class FamilyTest < ActiveSupport::TestCase
|
|||
end
|
||||
|
||||
test "should calculate total assets" do
|
||||
expected = @expected_snapshots.last["assets"].to_d
|
||||
assert_equal Money.new(expected), @family.assets
|
||||
expected = get_today_snapshot_value_for :assets
|
||||
assert_in_delta expected, @family.assets.amount, 0.01
|
||||
end
|
||||
|
||||
test "should calculate total liabilities" do
|
||||
expected = @expected_snapshots.last["liabilities"].to_d
|
||||
assert_equal Money.new(expected), @family.liabilities
|
||||
expected = get_today_snapshot_value_for :liabilities
|
||||
assert_in_delta expected, @family.liabilities.amount, 0.01
|
||||
end
|
||||
|
||||
test "should calculate net worth" do
|
||||
expected = @expected_snapshots.last["net_worth"].to_d
|
||||
assert_equal Money.new(expected), @family.net_worth
|
||||
expected = get_today_snapshot_value_for :net_worth
|
||||
assert_in_delta expected, @family.net_worth.amount, 0.01
|
||||
end
|
||||
|
||||
test "should calculate snapshot correctly" do
|
||||
asset_series = @family.snapshot[:asset_series]
|
||||
liability_series = @family.snapshot[:liability_series]
|
||||
net_worth_series = @family.snapshot[:net_worth_series]
|
||||
test "calculates asset time series" do
|
||||
series = @family.snapshot[:asset_series]
|
||||
expected_series = get_expected_balances_for :assets
|
||||
|
||||
assert_equal @expected_snapshots.count, asset_series.values.count
|
||||
assert_equal @expected_snapshots.count, liability_series.values.count
|
||||
assert_equal @expected_snapshots.count, net_worth_series.values.count
|
||||
|
||||
@expected_snapshots.each_with_index do |row, index|
|
||||
expected_assets = TimeSeries::Value.new(date: row["date"], value: Money.new(row["assets"].to_d))
|
||||
expected_liabilities = TimeSeries::Value.new(date: row["date"], value: Money.new(row["liabilities"].to_d))
|
||||
expected_net_worth = TimeSeries::Value.new(date: row["date"], value: Money.new(row["net_worth"].to_d))
|
||||
|
||||
assert_in_delta expected_assets.value.amount, Money.new(asset_series.values[index].value).amount, 0.01
|
||||
assert_in_delta expected_liabilities.value.amount, Money.new(liability_series.values[index].value).amount, 0.01
|
||||
assert_in_delta expected_net_worth.value.amount, Money.new(net_worth_series.values[index].value).amount, 0.01
|
||||
end
|
||||
assert_time_series_balances series, expected_series
|
||||
end
|
||||
|
||||
test "should calculate transaction snapshot correctly" do
|
||||
spending_series = @family.snapshot_transactions[:spending_series]
|
||||
income_series = @family.snapshot_transactions[:income_series]
|
||||
savings_rate_series = @family.snapshot_transactions[:savings_rate_series]
|
||||
test "calculates liability time series" do
|
||||
series = @family.snapshot[:liability_series]
|
||||
expected_series = get_expected_balances_for :liabilities
|
||||
|
||||
assert_equal @expected_snapshots.count, spending_series.values.count
|
||||
assert_equal @expected_snapshots.count, income_series.values.count
|
||||
assert_equal @expected_snapshots.count, savings_rate_series.values.count
|
||||
assert_time_series_balances series, expected_series
|
||||
end
|
||||
|
||||
@expected_snapshots.each_with_index do |row, index|
|
||||
expected_spending = TimeSeries::Value.new(date: row["date"], value: Money.new(row["rolling_spend"].to_d))
|
||||
expected_income = TimeSeries::Value.new(date: row["date"], value: Money.new(row["rolling_income"].to_d))
|
||||
expected_savings_rate = TimeSeries::Value.new(date: row["date"], value: Money.new(row["savings_rate"].to_d))
|
||||
test "calculates net worth time series" do
|
||||
series = @family.snapshot[:net_worth_series]
|
||||
expected_series = get_expected_balances_for :net_worth
|
||||
|
||||
assert_in_delta expected_spending.value.amount, Money.new(spending_series.values[index].value).amount, 0.01
|
||||
assert_in_delta expected_income.value.amount, Money.new(income_series.values[index].value).amount, 0.01
|
||||
assert_in_delta expected_savings_rate.value.amount, savings_rate_series.values[index].value, 0.01
|
||||
assert_time_series_balances series, expected_series
|
||||
end
|
||||
|
||||
test "calculates rolling expenses" do
|
||||
series = @family.snapshot_transactions[:spending_series]
|
||||
expected_series = get_expected_balances_for :rolling_spend
|
||||
|
||||
assert_time_series_balances series, expected_series, ignore_count: true
|
||||
end
|
||||
|
||||
test "calculates rolling income" do
|
||||
series = @family.snapshot_transactions[:income_series]
|
||||
expected_series = get_expected_balances_for :rolling_income
|
||||
|
||||
assert_time_series_balances series, expected_series, ignore_count: true
|
||||
end
|
||||
|
||||
test "calculates savings rate series" do
|
||||
series = @family.snapshot_transactions[:savings_rate_series]
|
||||
expected_series = get_expected_balances_for :savings_rate
|
||||
|
||||
series.values.each do |tsb|
|
||||
expected_balance = expected_series.find { |eb| eb[:date] == tsb.date }
|
||||
assert_in_delta expected_balance[:balance], tsb.value, 0.0001, "Balance incorrect on date: #{tsb.date}"
|
||||
end
|
||||
end
|
||||
|
||||
|
@ -127,4 +119,15 @@ class FamilyTest < ActiveSupport::TestCase
|
|||
assert_equal liabilities_before - disabled_cc.balance, @family.liabilities
|
||||
assert_equal net_worth_before - disabled_checking.balance + disabled_cc.balance, @family.net_worth
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
def assert_time_series_balances(time_series_balances, expected_balances, ignore_count: false)
|
||||
assert_equal time_series_balances.values.count, expected_balances.count unless ignore_count
|
||||
|
||||
time_series_balances.values.each do |tsb|
|
||||
expected_balance = expected_balances.find { |eb| eb[:date] == tsb.date }
|
||||
assert_in_delta expected_balance[:balance], tsb.value.amount, 0.01, "Balance incorrect on date: #{tsb.date}"
|
||||
end
|
||||
end
|
||||
end
|
||||
|
|
|
@ -3,6 +3,7 @@ require "test_helper"
|
|||
class TransactionTest < ActiveSupport::TestCase
|
||||
setup do
|
||||
@transaction = transactions(:checking_one)
|
||||
@family = families(:dylan_family)
|
||||
end
|
||||
|
||||
# See: https://github.com/maybe-finance/maybe/wiki/vision#signage-of-money
|
||||
|
@ -41,4 +42,14 @@ class TransactionTest < ActiveSupport::TestCase
|
|||
current_transaction.account.expects(:sync_later).with(prior_transaction.date)
|
||||
current_transaction.sync_account_later
|
||||
end
|
||||
|
||||
test "can calculate total spending for a group of transactions" do
|
||||
assert_equal Money.new(2135), @family.transactions.expense_total("USD")
|
||||
assert_equal Money.new(1010.85, "EUR"), @family.transactions.expense_total("EUR")
|
||||
end
|
||||
|
||||
test "can calculate total income for a group of transactions" do
|
||||
assert_equal -Money.new(2075), @family.transactions.income_total("USD")
|
||||
assert_equal -Money.new(250, "EUR"), @family.transactions.income_total("EUR")
|
||||
end
|
||||
end
|
||||
|
|
49
test/models/transfer_test.rb
Normal file
49
test/models/transfer_test.rb
Normal file
|
@ -0,0 +1,49 @@
|
|||
require "test_helper"
|
||||
|
||||
class TransferTest < ActiveSupport::TestCase
|
||||
setup do
|
||||
# Transfers can be posted on different dates
|
||||
@outflow = accounts(:checking).transactions.create! date: 1.day.ago.to_date, name: "Transfer to Savings", amount: 100, marked_as_transfer: true
|
||||
@inflow = accounts(:savings).transactions.create! date: Date.current, name: "Transfer from Savings", amount: -100, marked_as_transfer: true
|
||||
end
|
||||
|
||||
test "transfer valid if it has inflow and outflow from different accounts for the same amount" do
|
||||
transfer = Transfer.create! transactions: [ @inflow, @outflow ]
|
||||
|
||||
assert transfer.valid?
|
||||
end
|
||||
|
||||
test "transfer must have 2 transactions" do
|
||||
invalid_transfer_1 = Transfer.new transactions: [ @outflow ]
|
||||
invalid_transfer_2 = Transfer.new transactions: [ @inflow, @outflow, transactions(:savings_four) ]
|
||||
|
||||
assert invalid_transfer_1.invalid?
|
||||
assert invalid_transfer_2.invalid?
|
||||
end
|
||||
|
||||
test "transfer cannot have 2 transactions from the same account" do
|
||||
account = accounts(:checking)
|
||||
inflow = account.transactions.create! date: Date.current, name: "Inflow", amount: -100
|
||||
outflow = account.transactions.create! date: Date.current, name: "Outflow", amount: 100
|
||||
|
||||
assert_raise ActiveRecord::RecordInvalid do
|
||||
Transfer.create! transactions: [ inflow, outflow ]
|
||||
end
|
||||
end
|
||||
|
||||
test "all transfer transactions must be marked as transfers" do
|
||||
@inflow.update! marked_as_transfer: false
|
||||
|
||||
assert_raise ActiveRecord::RecordInvalid do
|
||||
Transfer.create! transactions: [ @inflow, @outflow ]
|
||||
end
|
||||
end
|
||||
|
||||
test "transfer transactions must net to zero" do
|
||||
@outflow.update! amount: 105
|
||||
|
||||
assert_raises ActiveRecord::RecordInvalid do
|
||||
Transfer.create! transactions: [ @inflow, @outflow ]
|
||||
end
|
||||
end
|
||||
end
|
|
@ -3,7 +3,7 @@ require "ostruct"
|
|||
class ValueGroupTest < ActiveSupport::TestCase
|
||||
setup do
|
||||
checking = accounts(:checking)
|
||||
savings = accounts(:savings_with_valuation_overrides)
|
||||
savings = accounts(:savings)
|
||||
collectable = accounts(:collectable)
|
||||
|
||||
# Level 1
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue