2025-05-25 20:40:18 -04:00
|
|
|
class Balance::ChartSeriesBuilder
|
|
|
|
def initialize(account_ids:, currency:, period: Period.last_30_days, interval: "1 day", favorable_direction: "up")
|
|
|
|
@account_ids = account_ids
|
|
|
|
@currency = currency
|
|
|
|
@period = period
|
|
|
|
@interval = interval
|
|
|
|
@favorable_direction = favorable_direction
|
|
|
|
end
|
|
|
|
|
|
|
|
def balance_series
|
2025-07-23 18:15:14 -04:00
|
|
|
build_series_for(:end_balance)
|
2025-05-26 20:05:16 -05:00
|
|
|
rescue => e
|
|
|
|
Rails.logger.error "Balance series error: #{e.message} for accounts #{@account_ids}"
|
|
|
|
raise
|
2025-05-25 20:40:18 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def cash_balance_series
|
2025-07-23 18:15:14 -04:00
|
|
|
build_series_for(:end_cash_balance)
|
2025-05-26 20:05:16 -05:00
|
|
|
rescue => e
|
|
|
|
Rails.logger.error "Cash balance series error: #{e.message} for accounts #{@account_ids}"
|
|
|
|
raise
|
2025-05-25 20:40:18 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def holdings_balance_series
|
2025-07-23 18:15:14 -04:00
|
|
|
build_series_for(:end_holdings_balance)
|
2025-05-26 20:05:16 -05:00
|
|
|
rescue => e
|
|
|
|
Rails.logger.error "Holdings balance series error: #{e.message} for accounts #{@account_ids}"
|
|
|
|
raise
|
2025-05-25 20:40:18 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
attr_reader :account_ids, :currency, :period, :favorable_direction
|
|
|
|
|
|
|
|
def interval
|
|
|
|
@interval || period.interval
|
|
|
|
end
|
|
|
|
|
|
|
|
def build_series_for(column)
|
|
|
|
values = query_data.map do |datum|
|
2025-07-23 18:15:14 -04:00
|
|
|
# Map column names to their start equivalents
|
|
|
|
previous_column = case column
|
|
|
|
when :end_balance then :start_balance
|
|
|
|
when :end_cash_balance then :start_cash_balance
|
|
|
|
when :end_holdings_balance then :start_holdings_balance
|
|
|
|
end
|
|
|
|
|
2025-05-25 20:40:18 -04:00
|
|
|
Series::Value.new(
|
|
|
|
date: datum.date,
|
|
|
|
date_formatted: I18n.l(datum.date, format: :long),
|
|
|
|
value: Money.new(datum.send(column), currency),
|
|
|
|
trend: Trend.new(
|
|
|
|
current: Money.new(datum.send(column), currency),
|
2025-07-23 18:15:14 -04:00
|
|
|
previous: Money.new(datum.send(previous_column), currency),
|
2025-05-25 20:40:18 -04:00
|
|
|
favorable_direction: favorable_direction
|
|
|
|
)
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
Series.new(
|
|
|
|
start_date: period.start_date,
|
|
|
|
end_date: period.end_date,
|
|
|
|
interval: interval,
|
|
|
|
values: values,
|
|
|
|
favorable_direction: favorable_direction
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
def query_data
|
|
|
|
@query_data ||= Balance.find_by_sql([
|
|
|
|
query,
|
|
|
|
{
|
|
|
|
account_ids: account_ids,
|
|
|
|
target_currency: currency,
|
|
|
|
start_date: period.start_date,
|
|
|
|
end_date: period.end_date,
|
|
|
|
interval: interval,
|
|
|
|
sign_multiplier: sign_multiplier
|
|
|
|
}
|
|
|
|
])
|
2025-05-26 20:05:16 -05:00
|
|
|
rescue => e
|
|
|
|
Rails.logger.error "Query data error: #{e.message} for accounts #{account_ids}, period #{period.start_date} to #{period.end_date}"
|
|
|
|
raise
|
2025-05-25 20:40:18 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
# Since the query aggregates the *net* of assets - liabilities, this means that if we're looking at
|
|
|
|
# a single liability account, we'll get a negative set of values. This is not what the user expects
|
|
|
|
# to see. When favorable direction is "down" (i.e. liability, decrease is "good"), we need to invert
|
|
|
|
# the values by multiplying by -1.
|
|
|
|
def sign_multiplier
|
|
|
|
favorable_direction == "down" ? -1 : 1
|
|
|
|
end
|
|
|
|
|
|
|
|
def query
|
|
|
|
<<~SQL
|
|
|
|
WITH dates AS (
|
|
|
|
SELECT generate_series(DATE :start_date, DATE :end_date, :interval::interval)::date AS date
|
|
|
|
UNION DISTINCT
|
2025-07-23 18:15:14 -04:00
|
|
|
SELECT :end_date::date -- Ensure end date is included
|
2025-05-25 20:40:18 -04:00
|
|
|
)
|
|
|
|
SELECT
|
2025-07-23 18:15:14 -04:00
|
|
|
d.date,
|
|
|
|
-- Use flows_factor: already handles asset (+1) vs liability (-1)
|
|
|
|
COALESCE(SUM(last_bal.end_balance * last_bal.flows_factor * COALESCE(er.rate, 1) * :sign_multiplier::integer), 0) AS end_balance,
|
|
|
|
COALESCE(SUM(last_bal.end_cash_balance * last_bal.flows_factor * COALESCE(er.rate, 1) * :sign_multiplier::integer), 0) AS end_cash_balance,
|
|
|
|
-- Holdings only for assets (flows_factor = 1)
|
|
|
|
COALESCE(SUM(
|
|
|
|
CASE WHEN last_bal.flows_factor = 1
|
|
|
|
THEN last_bal.end_non_cash_balance
|
|
|
|
ELSE 0
|
|
|
|
END * COALESCE(er.rate, 1) * :sign_multiplier::integer
|
|
|
|
), 0) AS end_holdings_balance,
|
|
|
|
-- Previous balances
|
|
|
|
COALESCE(SUM(last_bal.start_balance * last_bal.flows_factor * COALESCE(er.rate, 1) * :sign_multiplier::integer), 0) AS start_balance,
|
|
|
|
COALESCE(SUM(last_bal.start_cash_balance * last_bal.flows_factor * COALESCE(er.rate, 1) * :sign_multiplier::integer), 0) AS start_cash_balance,
|
|
|
|
COALESCE(SUM(
|
|
|
|
CASE WHEN last_bal.flows_factor = 1
|
|
|
|
THEN last_bal.start_non_cash_balance
|
|
|
|
ELSE 0
|
|
|
|
END * COALESCE(er.rate, 1) * :sign_multiplier::integer
|
|
|
|
), 0) AS start_holdings_balance
|
|
|
|
FROM dates d
|
|
|
|
CROSS JOIN accounts
|
|
|
|
LEFT JOIN LATERAL (
|
|
|
|
SELECT b.end_balance,
|
|
|
|
b.end_cash_balance,
|
|
|
|
b.end_non_cash_balance,
|
|
|
|
b.start_balance,
|
|
|
|
b.start_cash_balance,
|
|
|
|
b.start_non_cash_balance,
|
|
|
|
b.flows_factor
|
|
|
|
FROM balances b
|
|
|
|
WHERE b.account_id = accounts.id
|
|
|
|
AND b.date <= d.date
|
|
|
|
ORDER BY b.date DESC
|
|
|
|
LIMIT 1
|
|
|
|
) last_bal ON TRUE
|
|
|
|
LEFT JOIN LATERAL (
|
|
|
|
SELECT er.rate
|
|
|
|
FROM exchange_rates er
|
|
|
|
WHERE er.from_currency = accounts.currency
|
|
|
|
AND er.to_currency = :target_currency
|
|
|
|
AND er.date <= d.date
|
|
|
|
ORDER BY er.date DESC
|
|
|
|
LIMIT 1
|
|
|
|
) er ON TRUE
|
|
|
|
WHERE accounts.id = ANY(array[:account_ids]::uuid[])
|
|
|
|
GROUP BY d.date
|
|
|
|
ORDER BY d.date
|
2025-05-25 20:40:18 -04:00
|
|
|
SQL
|
|
|
|
end
|
|
|
|
end
|