public
Description: Library for dealing with money and currency conversion
Homepage: http://money.rubyforge.org/
Clone URL: git://github.com/FooBarWidget/money.git
Click here to lend your support to: money and make a donation at www.pledgie.com !
money /
name age message
file .gitignore Fri Oct 31 11:17:38 -0700 2008 Update documentation and URLs. [Hongli Lai (Phusion)]
file MIT-LICENSE Fri Oct 31 08:03:12 -0700 2008 Add more tests. Add copyright notice for Phusion. [Hongli Lai (Phusion)]
file README.rdoc Fri Oct 31 11:17:38 -0700 2008 Update documentation and URLs. [Hongli Lai (Phusion)]
file Rakefile Sun Feb 15 02:57:04 -0800 2009 Make unit tests work in Ruby 1.9. [Hongli Lai (Phusion)]
directory lib/ Sun Nov 08 02:09:11 -0800 2009 Money#== and Money#<=> will now work on any obj... [Hongli Lai (Phusion)]
file money.gemspec Thu Nov 05 13:21:48 -0800 2009 Bump version to 2.1.5. [Hongli Lai (Phusion)]
directory test/ Sun Nov 08 02:09:11 -0800 2009 Money#== and Money#<=> will now work on any obj... [Hongli Lai (Phusion)]
README.rdoc

Introduction

This library aids one in handling money and different currencies. Features:

  • Provides a Money class which encapsulates all information about an certain amount of money, such as its value and its currency.
  • Represents monetary values as integers, in cents. This avoids floating point rounding errors.
  • Provides APIs for exchanging money from one currency to another.
  • Has the ability to parse a money string into a Money object.

Resources:

Download

Install stable releases with the following command:

  gem install money

The development version (hosted on Github) can be installed with:

  gem sources -a http://gems.github.com
  gem install FooBarWidget-money

Usage

Synopsis

  require 'money'

  # 10.00 USD
  money = Money.new(1000, "USD")
  money.cents     # => 1000
  money.currency  # => "USD"

  Money.new(1000, "USD") == Money.new(1000, "USD")   # => true
  Money.new(1000, "USD") == Money.new(100, "USD")    # => false
  Money.new(1000, "USD") == Money.new(1000, "EUR")   # => false

Currency Exchange

Exchanging money is performed through an exchange bank object. The default exchange bank object requires one to manually specify the exchange rate. Here’s an example of how it works:

  Money.add_rate("USD", "CAD", 1.24515)
  Money.add_rate("CAD", "USD", 0.803115)

  Money.us_dollar(100).exchange_to("CAD")  # => Money.new(124, "CAD")
  Money.ca_dollar(100).exchange_to("USD")  # => Money.new(80, "USD")

Comparison and arithmetic operations work as expected:

  Money.new(1000, "USD") <=> Money.new(900, "USD")   # => 1; 9.00 USD is smaller
  Money.new(1000, "EUR") + Money.new(10, "EUR") == Money.new(1010, "EUR")

  Money.add_rate("USD", "EUR", 0.5)
  Money.new(1000, "EUR") + Money.new(1000, "USD") == Money.new(1500, "EUR")

There is nothing stopping you from creating bank objects which scrapes www.xe.com for the current rates or just returns rand(2):

  Money.default_bank = ExchangeBankWhichScrapesXeDotCom.new

Ruby on Rails

Use the compose_of helper to let Active Record deal with embedding the money object in your models. The following example requires a cents and a currency field.

  class ProductUnit < ActiveRecord::Base
    belongs_to :product
    composed_of :price, :class_name => "Money", :mapping => [%w(cents cents), %w(currency currency)]

    private
      validate :cents_not_zero

      def cents_not_zero
        errors.add("cents", "cannot be zero or less") unless cents > 0
      end

      validates_presence_of :sku, :currency
      validates_uniqueness_of :sku
  end

Default Currency

By default Money defaults to USD as its currency. This can be overwritten using

  Money.default_currency = "CAD"

If you use Rails, then environment.rb is a very good place to put this.