mirror of
https://github.com/maybe-finance/maybe.git
synced 2025-08-02 20:15:22 +02:00
Refactor TimeSeries
artifacts (#651)
* Reindent TimeSeries classes * Fix spacing in time series tests * Remove trend tests where current is nil I think if we've gotten this far with a nil value for current, there's a data integrity problem. If we allow this, we'll have to be very defensive in our code. Best to raise and fix early. * Reindent Money class * Refactor TimeSeries artifacts * Use as_json in TimeSeries * Bring back tests for trends where current is nil * Bring back trend test * Correctly enumerate trend test * Use favorable_direction for trend_styles helper * Make trend public in TimeSeries::Value * Allow nil current values in trends I think I might've gotten it wrong before, nils might appear in trends if values are unavailable for snapshots * Clean up TimeSeries::Trend * Skip trend values same class validations if any values are nil * Refactor Money * Remove object parsing in TimeSeries::Value We're only every passing hashes
This commit is contained in:
parent
fe2a2ac3f9
commit
fc3ade392a
9 changed files with 258 additions and 206 deletions
131
lib/money.rb
131
lib/money.rb
|
@ -1,73 +1,94 @@
|
|||
class Money
|
||||
include Comparable
|
||||
include Arithmetic
|
||||
include Comparable, Arithmetic
|
||||
include ActiveModel::Validations
|
||||
|
||||
attr_reader :amount, :currency
|
||||
attr_reader :amount, :currency
|
||||
|
||||
class << self
|
||||
def default_currency
|
||||
@default ||= Money::Currency.new(:usd)
|
||||
end
|
||||
validate :source_must_be_of_known_type
|
||||
|
||||
def default_currency=(object)
|
||||
@default = Money::Currency.new(object)
|
||||
end
|
||||
class << self
|
||||
def default_currency
|
||||
@default ||= Money::Currency.new(:usd)
|
||||
end
|
||||
|
||||
def initialize(obj, currency = Money.default_currency)
|
||||
unless obj.is_a?(Money) || obj.is_a?(Numeric) || obj.is_a?(BigDecimal)
|
||||
raise ArgumentError, "obj must be an instance of Money, Numeric, or BigDecimal"
|
||||
end
|
||||
|
||||
@amount = obj.is_a?(Money) ? obj.amount : BigDecimal(obj.to_s)
|
||||
@currency = obj.is_a?(Money) ? obj.currency : Money::Currency.new(currency)
|
||||
def default_currency=(object)
|
||||
@default = Money::Currency.new(object)
|
||||
end
|
||||
end
|
||||
|
||||
# TODO: Replace with injected rate store
|
||||
def exchange_to(other_currency, date = Date.current)
|
||||
return self if @currency == Money::Currency.new(other_currency)
|
||||
rate = ExchangeRate.find_rate(from: @currency, to: other_currency, date: date)
|
||||
return nil if rate.nil?
|
||||
Money.new(@amount * rate.rate, other_currency)
|
||||
def initialize(obj, currency = Money.default_currency)
|
||||
@source = obj
|
||||
@amount = obj.is_a?(Money) ? obj.amount : BigDecimal(obj.to_s)
|
||||
@currency = obj.is_a?(Money) ? obj.currency : Money::Currency.new(currency)
|
||||
|
||||
validate!
|
||||
end
|
||||
|
||||
# TODO: Replace with injected rate store
|
||||
def exchange_to(other_currency, date = Date.current)
|
||||
if currency == Money::Currency.new(other_currency)
|
||||
self
|
||||
elsif rate = ExchangeRate.find_rate(from: currency, to: other_currency, date: date)
|
||||
Money.new(amount * rate.rate, other_currency)
|
||||
end
|
||||
end
|
||||
|
||||
def cents_str(precision = @currency.default_precision)
|
||||
format_str = "%.#{precision}f"
|
||||
amount_str = format_str % @amount
|
||||
parts = amount_str.split(@currency.separator)
|
||||
def cents_str(precision = currency.default_precision)
|
||||
format_str = "%.#{precision}f"
|
||||
amount_str = format_str % amount
|
||||
parts = amount_str.split(currency.separator)
|
||||
|
||||
return "" if parts.length < 2
|
||||
|
||||
parts.last.ljust(precision, "0")
|
||||
if parts.length < 2
|
||||
""
|
||||
else
|
||||
parts.last.ljust(precision, "0")
|
||||
end
|
||||
end
|
||||
|
||||
# Basic formatting only. Use the Rails number_to_currency helper for more advanced formatting.
|
||||
alias to_s format
|
||||
def format
|
||||
whole_part, fractional_part = sprintf("%.#{@currency.default_precision}f", @amount).split(".")
|
||||
whole_with_delimiters = whole_part.chars.to_a.reverse.each_slice(3).map(&:join).join(@currency.delimiter).reverse
|
||||
formatted_amount = "#{whole_with_delimiters}#{@currency.separator}#{fractional_part}"
|
||||
@currency.default_format.gsub("%n", formatted_amount).gsub("%u", @currency.symbol)
|
||||
# Use `format` for basic formatting only.
|
||||
# Use the Rails number_to_currency helper for more advanced formatting.
|
||||
def format
|
||||
whole_part, fractional_part = sprintf("%.#{currency.default_precision}f", amount).split(".")
|
||||
whole_with_delimiters = whole_part.chars.to_a.reverse.each_slice(3).map(&:join).join(currency.delimiter).reverse
|
||||
formatted_amount = "#{whole_with_delimiters}#{currency.separator}#{fractional_part}"
|
||||
|
||||
currency.default_format.gsub("%n", formatted_amount).gsub("%u", currency.symbol)
|
||||
end
|
||||
alias_method :to_s, :format
|
||||
|
||||
def as_json
|
||||
{ amount: amount, currency: currency.iso_code }.as_json
|
||||
end
|
||||
|
||||
def <=>(other)
|
||||
raise TypeError, "Money can only be compared with other Money objects except for 0" unless other.is_a?(Money) || other.eql?(0)
|
||||
|
||||
if other.is_a?(Numeric)
|
||||
amount <=> other
|
||||
else
|
||||
amount_comparison = amount <=> other.amount
|
||||
|
||||
if amount_comparison == 0
|
||||
currency <=> other.currency
|
||||
else
|
||||
amount_comparison
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
def to_json(*_args)
|
||||
{ amount: @amount, currency: @currency.iso_code }.to_json
|
||||
end
|
||||
def default_format_options
|
||||
{
|
||||
unit: currency.symbol,
|
||||
precision: currency.default_precision,
|
||||
delimiter: currency.delimiter,
|
||||
separator: currency.separator
|
||||
}
|
||||
end
|
||||
|
||||
def <=>(other)
|
||||
raise TypeError, "Money can only be compared with other Money objects except for 0" unless other.is_a?(Money) || other.eql?(0)
|
||||
return @amount <=> other if other.is_a?(Numeric)
|
||||
amount_comparison = @amount <=> other.amount
|
||||
return amount_comparison unless amount_comparison == 0
|
||||
@currency <=> other.currency
|
||||
end
|
||||
|
||||
def default_format_options
|
||||
{
|
||||
unit: @currency.symbol,
|
||||
precision: @currency.default_precision,
|
||||
delimiter: @currency.delimiter,
|
||||
separator: @currency.separator
|
||||
}
|
||||
private
|
||||
def source_must_be_of_known_type
|
||||
unless @source.is_a?(Money) || @source.is_a?(Numeric) || @source.is_a?(BigDecimal)
|
||||
errors.add :source, "must be a Money, Numeric, or BigDecimal"
|
||||
end
|
||||
end
|
||||
end
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue