frankfurter/lib/quote.rb
2016-04-11 14:40:24 +01:00

68 lines
1.4 KiB
Ruby

# frozen_string_literal: true
require 'virtus'
require 'currency'
class Quote
include Virtus.value_object
Invalid = Class.new(StandardError)
DEFAULT_BASE = 'EUR'.freeze
values do
attribute :base, String, default: DEFAULT_BASE
attribute :date, Date, default: proc { Currency.current_date }
attribute :rates, Hash, default: :find_rates, lazy: true
end
private
def base=(base)
base.upcase!
super base
end
def date=(date)
current_date = Currency.current_date_before(date)
raise Invalid, 'Date too old' unless current_date
super current_date
rescue Sequel::DatabaseError => ex
if ex.wrapped_exception.is_a?(PG::DatetimeFieldOverflow)
raise Invalid, 'Invalid date'
else
raise
end
end
def find_rates
quoted_against_default_base? ? find_default_rates : find_rebased_rates
end
def quoted_against_default_base?
base == DEFAULT_BASE
end
def find_default_rates
Currency.where(date: date).reduce({}) do |rates, currency|
rates.update(currency.to_h)
end
end
def find_rebased_rates
rates = find_default_rates
denominator = rates.update(DEFAULT_BASE => 1.0).delete(base)
raise Invalid, 'Invalid base' unless denominator
rates.each do |iso_code, rate|
rates[iso_code] = round_rate(rate / denominator)
end
rates
end
# I'm mimicking the apparent convention of the ECB here.
def round_rate(rate)
Float(format('%.5g', rate))
end
end