require File.dirname(__FILE__) + '/paypal/paypal_common_api' require File.dirname(__FILE__) + '/paypal_express' module ActiveMerchant #:nodoc: module Billing #:nodoc: class PaypalGateway < Gateway include PaypalCommonAPI self.supported_cardtypes = [:visa, :master, :american_express, :discover] self.supported_countries = ['US'] self.homepage_url = 'https://www.paypal.com/cgi-bin/webscr?cmd=_wp-pro-overview-outside' self.display_name = 'PayPal Website Payments Pro (US)' def authorize(money, credit_card, options = {}) requires!(options, :ip) commit 'DoDirectPayment', build_sale_or_authorization_request('Authorization', money, credit_card, options) end def purchase(money, credit_card, options = {}) requires!(options, :ip) commit 'DoDirectPayment', build_sale_or_authorization_request('Sale', money, credit_card, options) end def express @express ||= PaypalExpressGateway.new(@options) end private def build_sale_or_authorization_request(action, money, credit_card, options) billing_address = options[:billing_address] || options[:address] currency_code = options[:currency] || currency(money) xml = Builder::XmlMarkup.new :indent => 2 xml.tag! 'DoDirectPaymentReq', 'xmlns' => PAYPAL_NAMESPACE do xml.tag! 'DoDirectPaymentRequest', 'xmlns:n2' => EBAY_NAMESPACE do xml.tag! 'n2:Version', API_VERSION xml.tag! 'n2:DoDirectPaymentRequestDetails' do xml.tag! 'n2:PaymentAction', action xml.tag! 'n2:PaymentDetails' do xml.tag! 'n2:OrderTotal', amount(money), 'currencyID' => currency_code # All of the values must be included together and add up to the order total if [:subtotal, :shipping, :handling, :tax].all?{ |o| options.has_key?(o) } xml.tag! 'n2:ItemTotal', amount(options[:subtotal]), 'currencyID' => currency_code xml.tag! 'n2:ShippingTotal', amount(options[:shipping]),'currencyID' => currency_code xml.tag! 'n2:HandlingTotal', amount(options[:handling]),'currencyID' => currency_code xml.tag! 'n2:TaxTotal', amount(options[:tax]), 'currencyID' => currency_code end xml.tag! 'n2:NotifyURL', options[:notify_url] xml.tag! 'n2:OrderDescription', options[:description] xml.tag! 'n2:InvoiceID', options[:order_id] xml.tag! 'n2:ButtonSource', application_id.to_s.slice(0,32) unless application_id.blank? add_address(xml, 'n2:ShipToAddress', options[:shipping_address]) if options[:shipping_address] end add_credit_card(xml, credit_card, billing_address, options) xml.tag! 'n2:IPAddress', options[:ip] end end end xml.target! end def add_credit_card(xml, credit_card, address, options) xml.tag! 'n2:CreditCard' do xml.tag! 'n2:CreditCardType', credit_card_type(card_brand(credit_card)) xml.tag! 'n2:CreditCardNumber', credit_card.number xml.tag! 'n2:ExpMonth', format(credit_card.month, :two_digits) xml.tag! 'n2:ExpYear', format(credit_card.year, :four_digits) xml.tag! 'n2:CVV2', credit_card.verification_value if [ 'switch', 'solo' ].include?(card_brand(credit_card).to_s) xml.tag! 'n2:StartMonth', format(credit_card.start_month, :two_digits) unless credit_card.start_month.blank? xml.tag! 'n2:StartYear', format(credit_card.start_year, :four_digits) unless credit_card.start_year.blank? xml.tag! 'n2:IssueNumber', format(credit_card.issue_number, :two_digits) unless credit_card.issue_number.blank? end xml.tag! 'n2:CardOwner' do xml.tag! 'n2:PayerName' do xml.tag! 'n2:FirstName', credit_card.first_name xml.tag! 'n2:LastName', credit_card.last_name end xml.tag! 'n2:Payer', options[:email] add_address(xml, 'n2:Address', address) end end end def credit_card_type(type) case type when 'visa' then 'Visa' when 'master' then 'MasterCard' when 'discover' then 'Discover' when 'american_express' then 'Amex' when 'switch' then 'Switch' when 'solo' then 'Solo' end end def build_response(success, message, response, options = {}) Response.new(success, message, response, options) end end end end