mirror of
https://github.com/maybe-finance/maybe.git
synced 2025-07-19 05:09:38 +02:00
* Setup health check * Security health checker cron * Use resolver throughout codebase * Use resolver for trade builder * Add security health checks to schedule * Handle no provider * Lint fixes
133 lines
4.2 KiB
Ruby
133 lines
4.2 KiB
Ruby
class MarketDataImporter
|
||
# By default, our graphs show 1M as the view, so by fetching 31 days,
|
||
# we ensure we can always show an accurate default graph
|
||
SNAPSHOT_DAYS = 31
|
||
|
||
InvalidModeError = Class.new(StandardError)
|
||
|
||
def initialize(mode: :full, clear_cache: false)
|
||
@mode = set_mode!(mode)
|
||
@clear_cache = clear_cache
|
||
end
|
||
|
||
def import_all
|
||
import_security_prices
|
||
import_exchange_rates
|
||
end
|
||
|
||
# Syncs historical security prices (and details)
|
||
def import_security_prices
|
||
unless Security.provider
|
||
Rails.logger.warn("No provider configured for MarketDataImporter.import_security_prices, skipping sync")
|
||
return
|
||
end
|
||
|
||
# Import all securities that aren't marked as "offline" (i.e. they're available from the provider)
|
||
Security.online.find_each do |security|
|
||
security.import_provider_prices(
|
||
start_date: get_first_required_price_date(security),
|
||
end_date: end_date,
|
||
clear_cache: clear_cache
|
||
)
|
||
|
||
security.import_provider_details(clear_cache: clear_cache)
|
||
end
|
||
end
|
||
|
||
def import_exchange_rates
|
||
unless ExchangeRate.provider
|
||
Rails.logger.warn("No provider configured for MarketDataImporter.import_exchange_rates, skipping sync")
|
||
return
|
||
end
|
||
|
||
required_exchange_rate_pairs.each do |pair|
|
||
# pair is a Hash with keys :source, :target, and :start_date
|
||
start_date = snapshot? ? default_start_date : pair[:start_date]
|
||
|
||
ExchangeRate.import_provider_rates(
|
||
from: pair[:source],
|
||
to: pair[:target],
|
||
start_date: start_date,
|
||
end_date: end_date,
|
||
clear_cache: clear_cache
|
||
)
|
||
end
|
||
end
|
||
|
||
private
|
||
attr_reader :mode, :clear_cache
|
||
|
||
def snapshot?
|
||
mode.to_sym == :snapshot
|
||
end
|
||
|
||
# Builds a unique list of currency pairs with the earliest date we need
|
||
# exchange rates for.
|
||
#
|
||
# Returns: Array of Hashes – [{ source:, target:, start_date: }, ...]
|
||
def required_exchange_rate_pairs
|
||
pair_dates = {} # { [source, target] => earliest_date }
|
||
|
||
# 1. ENTRY-BASED PAIRS – we need rates from the first entry date
|
||
Entry.joins(:account)
|
||
.where.not("entries.currency = accounts.currency")
|
||
.group("entries.currency", "accounts.currency")
|
||
.minimum("entries.date")
|
||
.each do |(source, target), date|
|
||
key = [ source, target ]
|
||
pair_dates[key] = [ pair_dates[key], date ].compact.min
|
||
end
|
||
|
||
# 2. ACCOUNT-BASED PAIRS – use the account's oldest entry date
|
||
account_first_entry_dates = Entry.group(:account_id).minimum(:date)
|
||
|
||
Account.joins(:family)
|
||
.where.not("families.currency = accounts.currency")
|
||
.select("accounts.id, accounts.currency AS source, families.currency AS target")
|
||
.find_each do |account|
|
||
earliest_entry_date = account_first_entry_dates[account.id]
|
||
|
||
chosen_date = [ earliest_entry_date, default_start_date ].compact.min
|
||
|
||
key = [ account.source, account.target ]
|
||
pair_dates[key] = [ pair_dates[key], chosen_date ].compact.min
|
||
end
|
||
|
||
# Convert to array of hashes for ease of use
|
||
pair_dates.map do |(source, target), date|
|
||
{ source: source, target: target, start_date: date }
|
||
end
|
||
end
|
||
|
||
def get_first_required_price_date(security)
|
||
return default_start_date if snapshot?
|
||
|
||
Trade.with_entry.where(security: security).minimum(:date) || default_start_date
|
||
end
|
||
|
||
# An approximation that grabs more than we likely need, but simplifies the logic
|
||
def get_first_required_exchange_rate_date(from_currency:)
|
||
return default_start_date if snapshot?
|
||
|
||
Entry.where(currency: from_currency).minimum(:date) || default_start_date
|
||
end
|
||
|
||
def default_start_date
|
||
SNAPSHOT_DAYS.days.ago.to_date
|
||
end
|
||
|
||
# Since we're querying market data from a US-based API, end date should always be today (EST)
|
||
def end_date
|
||
Date.current.in_time_zone("America/New_York").to_date
|
||
end
|
||
|
||
def set_mode!(mode)
|
||
valid_modes = [ :full, :snapshot ]
|
||
|
||
unless valid_modes.include?(mode.to_sym)
|
||
raise InvalidModeError, "Invalid mode for MarketDataImporter, can only be :full or :snapshot, but was #{mode}"
|
||
end
|
||
|
||
mode.to_sym
|
||
end
|
||
end
|