require 'virtus' require 'currency' # Quotes exchange rates on a specific date class Snapshot include Virtus.model attribute :base, String, default: 'EUR' attribute :date, Date def quote self.date = if date last_date = Currency.where("date <= '#{date}'").order(:date).last fail ArgumentError, 'Date too old' unless last_date last_date.date else Currency.last_date end attributes.merge(rates: rebase(rates)) end private def rates Currency.where(date: date).reduce({}) do |rates, currency| rates.update(currency.to_hash) end end def rebase(rates) if base.upcase! != 'EUR' denominator = rates.update('EUR' => 1.0).delete(base) fail ArgumentError, 'Invalid base' unless denominator rates.each do |iso_code, rate| rates[iso_code] = round(rate / denominator) end end rates end def round(rate) if rate > 100 rate.round(2) elsif rate > 10 rate.round(3) else rate.round(4) end end end