require 'money' require 'open-uri' require 'multi_json' class Money module Bank class GoogleCurrencyRailsCache < Money::Bank::VariableExchange SERVICE_HOST = "www.google.com" SERVICE_PATH = "/ig/calculator" ## # Clears all rates stored in @rates # # @example # @bank = GoogleCurrencyRailsCache.new #=> # @bank.get_rate(:USD, :EUR) #=> 0.776337241 # @bank.flush_rates #=> nil def flush_rates rates = Rails.cache.read("exchange_rate/all_rates") unless rates.nil? rates.each do |rate_key| Rails.cache.delete "exchange_rate/#{rate_key}" end end Rails.cache.delete("exchange_rate/all_rates") end ## # Clears the specified rate stored in @rates. # # @param [String, Symbol, Currency] from Currency to convert from (used # for key into @rates). # @param [String, Symbol, Currency] to Currency to convert to (used for # key into @rates). # # @return [Boolean] Indicating successful removal. # # @example # @bank = GoogleCurrencyRailsCache.new #=> # @bank.get_rate(:USD, :EUR) #=> 0.776337241 # @bank.flush_rate(:USD, :EUR) #=> 0.776337241 def flush_rate(from, to) rate_key = rate_key_for(from, to) Rails.cache.delete("exchange_rate/#{rate_key}") remove_from_stored_rates(rate_key) end ## # Returns the requested rate. # # @param [String, Symbol, Currency] from Currency to convert from # @param [String, Symbol, Currency] to Currency to convert to # # @return [Float] The requested rate. # # @example # @bank = GoogleCurrencyRailsCache.new #=> # @bank.get_rate(:USD, :EUR) #=> 0.776337241 def get_rate(from, to) Rails.cache.fetch("exchange_rate/#{rate_key_for(from, to)}", :expires_in => 12.hours ) { fetch_rate(from, to) } end ## # Lists all rates currency cached. # # @return [Array] An array of rates. # def cached_rates Rails.cache.read("exchange_rate/all_rates") end private ## # Queries for the requested rate and returns it. # # @param [String, Symbol, Currency] from Currency to convert from # @param [String, Symbol, Currency] to Currency to convert to # # @return [BigDecimal] The requested rate. def fetch_rate(from, to) from, to = Currency.wrap(from), Currency.wrap(to) data = build_uri(from, to).read data = fix_response_json_data(data) error = data['error'] raise UnknownRate unless error == '' || error == '0' add_to_stored_rates(rate_key_for(from, to)) decode_rate data['rhs'] end ## # Build a URI for the given arguments. # # @param [Currency] from The currency to convert from. # @param [Currency] to The currency to convert to. # # @return [URI::HTTP] def build_uri(from, to) uri = URI::HTTP.build( :host => SERVICE_HOST, :path => SERVICE_PATH, :query => "hl=en&q=1#{from.iso_code}%3D%3F#{to.iso_code}" ) end ## # Takes the invalid JSON returned by Google and fixes it. # # @param [String] data The JSON string to fix. # # @return [Hash] def fix_response_json_data(data) data.gsub!(/lhs:/, '"lhs":') data.gsub!(/rhs:/, '"rhs":') data.gsub!(/error:/, '"error":') data.gsub!(/icc:/, '"icc":') data.gsub!(Regexp.new("(\\\\x..|\\\\240)"), '') MultiJson.decode(data) end ## # Takes the 'rhs' response from Google and decodes it. # # @param [String] rhs The google rate string to decode. # # @return [BigDecimal] def decode_rate(rhs) if complex_rate?(rhs) decode_complex_rate(rhs) else decode_basic_rate(rhs) end end ## # Takes the 'rhs' response from Google and decides if it's a complex rate # # @param [String] rhs The google rate string to check. # # @return [Boolean] def complex_rate?(rhs) rhs.match(/10x3csupx3e(-?\d+)x3c\/supx3e/) end ## # Takes a complex 'rhs' response from Google and converts it to a numeric # rate. # # @param [String] rhs The complex google rate string to convert. # # @return [BigDecimal] def decode_complex_rate(rhs) rate = BigDecimal(rhs.match(/\d[\d\s]*\.?\d*/)[0]) power = rhs.match(/10x3csupx3e(-?\d+)x3c\/supx3e/) rate * 10**power[1].to_i end ## # Takes a basic 'rhs' response from Google and converts it to a numeric # rate. # # @param [String] rhs The basic google rate string to convert. # # @return [BigDecimal] def decode_basic_rate(rhs) BigDecimal(rhs.gsub(/[^\d\.]/, '')) end ## # Removes the specified rate from the global list of stored rates # # @param [String] rate The rate to remove # # @return [Boolean] def remove_from_stored_rates(rate) rates = Rails.cache.read("exchange_rate/all_rates") unless rates.nil? rates.delete(rate) Rails.cache.write("exchange_rate/all_rates", rates) end end ## # Add the specified rate to the global list of stored rates # # @param [String] rate The rate to add # # @return [Boolean] def add_to_stored_rates(rate) rates = Rails.cache.read("exchange_rate/all_rates") if rates.nil? rates = [] end rates << rate Rails.cache.write("exchange_rate/all_rates", rates) end end end end