1
0
Fork 0
mirror of https://github.com/maybe-finance/maybe.git synced 2025-07-18 20:59:39 +02:00

transfer: Support transfers of different currencies between accounts. (#2243)

Fixes part of #1852.

Co-authored-by: Zach Gollwitzer <zach@maybe.co>
This commit is contained in:
Joseph Ho 2025-06-25 16:34:18 -04:00 committed by GitHub
parent 72a0f87a9c
commit 637d630388
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
2 changed files with 106 additions and 2 deletions

View file

@ -18,6 +18,51 @@ class Family::AutoTransferMatchableTest < ActiveSupport::TestCase
end
end
test "auto-matches multi-currency transfers" do
load_exchange_prices
create_transaction(date: 1.day.ago.to_date, account: @depository, amount: 500)
create_transaction(date: Date.current, account: @credit_card, amount: -700, currency: "CAD")
assert_difference -> { Transfer.count } => 1 do
@family.auto_match_transfers!
end
# test match within lower 5% bound
create_transaction(date: 1.day.ago.to_date, account: @depository, amount: 1000)
create_transaction(date: Date.current, account: @credit_card, amount: -1330, currency: "CAD")
assert_difference -> { Transfer.count } => 1 do
@family.auto_match_transfers!
end
# test match within upper 5% bound
create_transaction(date: 1.day.ago.to_date, account: @depository, amount: 1500)
create_transaction(date: Date.current, account: @credit_card, amount: -2189, currency: "CAD")
assert_difference -> { Transfer.count } => 1 do
@family.auto_match_transfers!
end
# test no match outside of slippage tolerance
create_transaction(date: 1.day.ago.to_date, account: @depository, amount: 1000)
create_transaction(date: Date.current, account: @credit_card, amount: -1320, currency: "CAD")
assert_difference -> { Transfer.count } => 0 do
@family.auto_match_transfers!
end
end
test "only matches inflow with correct currency when duplicate amounts exist" do
load_exchange_prices
create_transaction(date: 1.day.ago.to_date, account: @depository, amount: 500)
create_transaction(date: Date.current, account: @credit_card, amount: -500, currency: "CAD")
create_transaction(date: Date.current, account: @credit_card, amount: -500)
assert_difference -> { Transfer.count } => 1 do
@family.auto_match_transfers!
end
end
# In this scenario, our matching logic should find 4 potential matches. These matches should be ranked based on
# days apart, then de-duplicated so that we aren't auto-matching the same transaction across multiple transfers.
test "when 2 options exist, only auto-match one at a time, ranked by days apart" do
@ -53,4 +98,51 @@ class Family::AutoTransferMatchableTest < ActiveSupport::TestCase
@family.auto_match_transfers!
end
end
test "does not match transactions outside the 4-day window" do
create_transaction(date: 10.days.ago.to_date, account: @depository, amount: 500)
create_transaction(date: Date.current, account: @credit_card, amount: -500)
assert_no_difference -> { Transfer.count } do
@family.auto_match_transfers!
end
end
test "does not match multi-currency transfer with missing exchange rate" do
create_transaction(date: Date.current, account: @depository, amount: 500)
create_transaction(date: Date.current, account: @credit_card, amount: -700, currency: "GBP")
assert_no_difference -> { Transfer.count } do
@family.auto_match_transfers!
end
end
private
def load_exchange_prices
rates = {
4.days.ago.to_date => 1.36,
3.days.ago.to_date => 1.37,
2.days.ago.to_date => 1.38,
1.day.ago.to_date => 1.39,
Date.current => 1.40
}
rates.each do |date, rate|
# USD to CAD
ExchangeRate.create!(
from_currency: "USD",
to_currency: "CAD",
date: date,
rate: rate
)
# CAD to USD (inverse)
ExchangeRate.create!(
from_currency: "CAD",
to_currency: "USD",
date: date,
rate: (1.0 / rate).round(6)
)
end
end
end