require 'net/http'
require 'net/https'
require 'active_merchant/billing/response'
module ActiveMerchant #:nodoc:
module Billing #:nodoc:
#
# == Description
# The Gateway class is the base class for all ActiveMerchant gateway implementations.
#
# The standard list of gateway functions that most concrete gateway subclasses implement is:
#
# * purchase(money, creditcard, options = {})
# * authorize(money, creditcard, options = {})
# * capture(money, authorization, options = {})
# * void(identification, options = {})
# * credit(money, identification, options = {})
#
# Some gateways include features for recurring billing
#
# * recurring(money, creditcard, options = {})
#
# Some gateways also support features for storing credit cards:
#
# * store(creditcard, options = {})
# * unstore(identification, options = {})
#
# === Gateway Options
# The options hash consists of the following options:
#
# * :order_id - The order number
# * :ip - The IP address of the customer making the purchase
# * :customer - The name, customer number, or other information that identifies the customer
# * :invoice - The invoice number
# * :merchant - The name or description of the merchant offering the product
# * :description - A description of the transaction
# * :email - The email address of the customer
# * :currency - The currency of the transaction. Only important when you are using a currency that is not the default with a gateway that supports multiple currencies.
# * :billing_address - A hash containing the billing address of the customer.
# * :shipping_address - A hash containing the shipping address of the customer.
#
# The :billing_address, and :shipping_address hashes can have the following keys:
#
# * :name - The full name of the customer.
# * :company - The company name of the customer.
# * :address1 - The primary street address of the customer.
# * :address2 - Additional line of address information.
# * :city - The city of the customer.
# * :state - The state of the customer. The 2 digit code for US and Canadian addresses. The full name of the state or province for foreign addresses.
# * :country - The [ISO 3166-1-alpha-2 code](http://www.iso.org/iso/country_codes/iso_3166_code_lists/english_country_names_and_code_elements.htm) for the customer.
# * :zip - The zip or postal code of the customer.
# * :phone - The phone number of the customer.
#
# == Implmenting new gateways
#
# See the {ActiveMerchant Guide to Contributing}[http://code.google.com/p/activemerchant/wiki/Contributing]
#
class Gateway
include PostsData
include RequiresParameters
include CreditCardFormatting
include Utils
DEBIT_CARDS = [ :switch, :solo ]
cattr_reader :implementations
@@implementations = []
def self.inherited(subclass)
super
@@implementations << subclass
end
# The format of the amounts used by the gateway
# :dollars => '12.50'
# :cents => '1250'
class_inheritable_accessor :money_format
self.money_format = :dollars
# The default currency for the transactions if no currency is provided
class_inheritable_accessor :default_currency
# The countries of merchants the gateway supports
class_inheritable_accessor :supported_countries
self.supported_countries = []
# The supported card types for the gateway
class_inheritable_accessor :supported_cardtypes
self.supported_cardtypes = []
# Indicates if the gateway supports 3D Secure authentication or not
class_inheritable_accessor :supports_3d_secure
self.supports_3d_secure = false
class_inheritable_accessor :homepage_url
class_inheritable_accessor :display_name
# The application making the calls to the gateway
# Useful for things like the PayPal build notation (BN) id fields
superclass_delegating_accessor :application_id
self.application_id = 'ActiveMerchant'
attr_reader :options
# Use this method to check if your gateway of interest supports a credit card of some type
def self.supports?(card_type)
supported_cardtypes.include?(card_type.to_sym)
end
def self.card_brand(source)
result = source.respond_to?(:brand) ? source.brand : source.type
result.to_s.downcase
end
def card_brand(source)
self.class.card_brand(source)
end
# Initialize a new gateway.
#
# See the documentation for the gateway you will be using to make sure there are no other
# required options.
def initialize(options = {})
end
# Are we running in test mode?
def test?
Base.gateway_mode == :test
end
private # :nodoc: all
def name
self.class.name.scan(/\:\:(\w+)Gateway/).flatten.first
end
def amount(money)
return nil if money.nil?
cents = money.respond_to?(:cents) ? money.cents : money
if money.is_a?(String) or cents.to_i < 0
raise ArgumentError, 'money amount must be either a Money object or a positive integer in cents.'
end
if self.money_format == :cents
cents.to_s
else
sprintf("%.2f", cents.to_f / 100)
end
end
def currency(money)
money.respond_to?(:currency) ? money.currency : self.default_currency
end
def requires_start_date_or_issue_number?(credit_card)
return false if card_brand(credit_card).blank?
DEBIT_CARDS.include?(card_brand(credit_card).to_sym)
end
end
end
end