fishman-activemerchant 1.18.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (188) hide show
  1. data/CHANGELOG +733 -0
  2. data/CONTRIBUTORS +257 -0
  3. data/MIT-LICENSE +20 -0
  4. data/gem-public_cert.pem +20 -0
  5. data/lib/active_merchant.rb +47 -0
  6. data/lib/active_merchant/billing.rb +9 -0
  7. data/lib/active_merchant/billing/avs_result.rb +98 -0
  8. data/lib/active_merchant/billing/base.rb +57 -0
  9. data/lib/active_merchant/billing/check.rb +68 -0
  10. data/lib/active_merchant/billing/credit_card.rb +260 -0
  11. data/lib/active_merchant/billing/credit_card_formatting.rb +21 -0
  12. data/lib/active_merchant/billing/credit_card_methods.rb +125 -0
  13. data/lib/active_merchant/billing/cvv_result.rb +38 -0
  14. data/lib/active_merchant/billing/expiry_date.rb +34 -0
  15. data/lib/active_merchant/billing/gateway.rb +170 -0
  16. data/lib/active_merchant/billing/gateways.rb +18 -0
  17. data/lib/active_merchant/billing/gateways/authorize_net.rb +693 -0
  18. data/lib/active_merchant/billing/gateways/authorize_net_cim.rb +858 -0
  19. data/lib/active_merchant/billing/gateways/barclays_epdq.rb +308 -0
  20. data/lib/active_merchant/billing/gateways/beanstream.rb +139 -0
  21. data/lib/active_merchant/billing/gateways/beanstream/beanstream_core.rb +282 -0
  22. data/lib/active_merchant/billing/gateways/beanstream_interac.rb +54 -0
  23. data/lib/active_merchant/billing/gateways/blue_pay.rb +11 -0
  24. data/lib/active_merchant/billing/gateways/bogus.rb +142 -0
  25. data/lib/active_merchant/billing/gateways/braintree.rb +17 -0
  26. data/lib/active_merchant/billing/gateways/braintree/braintree_common.rb +9 -0
  27. data/lib/active_merchant/billing/gateways/braintree_blue.rb +303 -0
  28. data/lib/active_merchant/billing/gateways/braintree_orange.rb +17 -0
  29. data/lib/active_merchant/billing/gateways/card_save.rb +23 -0
  30. data/lib/active_merchant/billing/gateways/card_stream.rb +230 -0
  31. data/lib/active_merchant/billing/gateways/cyber_source.rb +430 -0
  32. data/lib/active_merchant/billing/gateways/data_cash.rb +597 -0
  33. data/lib/active_merchant/billing/gateways/efsnet.rb +235 -0
  34. data/lib/active_merchant/billing/gateways/elavon.rb +134 -0
  35. data/lib/active_merchant/billing/gateways/epay.rb +274 -0
  36. data/lib/active_merchant/billing/gateways/eway.rb +277 -0
  37. data/lib/active_merchant/billing/gateways/eway_managed.rb +264 -0
  38. data/lib/active_merchant/billing/gateways/exact.rb +222 -0
  39. data/lib/active_merchant/billing/gateways/federated_canada.rb +168 -0
  40. data/lib/active_merchant/billing/gateways/first_pay.rb +177 -0
  41. data/lib/active_merchant/billing/gateways/garanti.rb +262 -0
  42. data/lib/active_merchant/billing/gateways/ideal/ideal_base.rb +250 -0
  43. data/lib/active_merchant/billing/gateways/ideal/ideal_rabobank.pem +13 -0
  44. data/lib/active_merchant/billing/gateways/ideal/ideal_response.rb +29 -0
  45. data/lib/active_merchant/billing/gateways/ideal_rabobank.rb +55 -0
  46. data/lib/active_merchant/billing/gateways/inspire.rb +221 -0
  47. data/lib/active_merchant/billing/gateways/instapay.rb +164 -0
  48. data/lib/active_merchant/billing/gateways/iridium.rb +258 -0
  49. data/lib/active_merchant/billing/gateways/jetpay.rb +276 -0
  50. data/lib/active_merchant/billing/gateways/linkpoint.rb +454 -0
  51. data/lib/active_merchant/billing/gateways/merchant_e_solutions.rb +156 -0
  52. data/lib/active_merchant/billing/gateways/merchant_ware.rb +289 -0
  53. data/lib/active_merchant/billing/gateways/modern_payments.rb +36 -0
  54. data/lib/active_merchant/billing/gateways/modern_payments_cim.rb +220 -0
  55. data/lib/active_merchant/billing/gateways/moneris.rb +209 -0
  56. data/lib/active_merchant/billing/gateways/net_registry.rb +189 -0
  57. data/lib/active_merchant/billing/gateways/netaxept.rb +239 -0
  58. data/lib/active_merchant/billing/gateways/netbilling.rb +168 -0
  59. data/lib/active_merchant/billing/gateways/nmi.rb +13 -0
  60. data/lib/active_merchant/billing/gateways/ogone.rb +292 -0
  61. data/lib/active_merchant/billing/gateways/optimal_payment.rb +274 -0
  62. data/lib/active_merchant/billing/gateways/orbital.rb +321 -0
  63. data/lib/active_merchant/billing/gateways/orbital/orbital_soft_descriptors.rb +46 -0
  64. data/lib/active_merchant/billing/gateways/pay_junction.rb +392 -0
  65. data/lib/active_merchant/billing/gateways/pay_secure.rb +120 -0
  66. data/lib/active_merchant/billing/gateways/paybox_direct.rb +207 -0
  67. data/lib/active_merchant/billing/gateways/payflow.rb +253 -0
  68. data/lib/active_merchant/billing/gateways/payflow/payflow_common_api.rb +207 -0
  69. data/lib/active_merchant/billing/gateways/payflow/payflow_express_response.rb +39 -0
  70. data/lib/active_merchant/billing/gateways/payflow/payflow_response.rb +13 -0
  71. data/lib/active_merchant/billing/gateways/payflow_express.rb +222 -0
  72. data/lib/active_merchant/billing/gateways/payflow_express_uk.rb +15 -0
  73. data/lib/active_merchant/billing/gateways/payflow_uk.rb +21 -0
  74. data/lib/active_merchant/billing/gateways/payment_express.rb +235 -0
  75. data/lib/active_merchant/billing/gateways/paypal.rb +121 -0
  76. data/lib/active_merchant/billing/gateways/paypal/paypal_common_api.rb +354 -0
  77. data/lib/active_merchant/billing/gateways/paypal/paypal_express_response.rb +49 -0
  78. data/lib/active_merchant/billing/gateways/paypal_ca.rb +13 -0
  79. data/lib/active_merchant/billing/gateways/paypal_express.rb +184 -0
  80. data/lib/active_merchant/billing/gateways/paypal_express_common.rb +25 -0
  81. data/lib/active_merchant/billing/gateways/paypal_express_de.rb +14 -0
  82. data/lib/active_merchant/billing/gateways/paystation.rb +201 -0
  83. data/lib/active_merchant/billing/gateways/plugnpay.rb +298 -0
  84. data/lib/active_merchant/billing/gateways/psigate.rb +219 -0
  85. data/lib/active_merchant/billing/gateways/psl_card.rb +304 -0
  86. data/lib/active_merchant/billing/gateways/qbms.rb +297 -0
  87. data/lib/active_merchant/billing/gateways/quantum.rb +282 -0
  88. data/lib/active_merchant/billing/gateways/quickpay.rb +297 -0
  89. data/lib/active_merchant/billing/gateways/realex.rb +311 -0
  90. data/lib/active_merchant/billing/gateways/sage.rb +146 -0
  91. data/lib/active_merchant/billing/gateways/sage/sage_bankcard.rb +88 -0
  92. data/lib/active_merchant/billing/gateways/sage/sage_core.rb +116 -0
  93. data/lib/active_merchant/billing/gateways/sage/sage_virtual_check.rb +97 -0
  94. data/lib/active_merchant/billing/gateways/sage_pay.rb +320 -0
  95. data/lib/active_merchant/billing/gateways/sallie_mae.rb +144 -0
  96. data/lib/active_merchant/billing/gateways/secure_net.rb +330 -0
  97. data/lib/active_merchant/billing/gateways/secure_pay.rb +31 -0
  98. data/lib/active_merchant/billing/gateways/secure_pay_au.rb +193 -0
  99. data/lib/active_merchant/billing/gateways/secure_pay_tech.rb +113 -0
  100. data/lib/active_merchant/billing/gateways/skip_jack.rb +453 -0
  101. data/lib/active_merchant/billing/gateways/smart_ps.rb +271 -0
  102. data/lib/active_merchant/billing/gateways/stripe.rb +212 -0
  103. data/lib/active_merchant/billing/gateways/trans_first.rb +127 -0
  104. data/lib/active_merchant/billing/gateways/transax.rb +25 -0
  105. data/lib/active_merchant/billing/gateways/trust_commerce.rb +423 -0
  106. data/lib/active_merchant/billing/gateways/usa_epay.rb +194 -0
  107. data/lib/active_merchant/billing/gateways/verifi.rb +233 -0
  108. data/lib/active_merchant/billing/gateways/viaklix.rb +189 -0
  109. data/lib/active_merchant/billing/gateways/wirecard.rb +318 -0
  110. data/lib/active_merchant/billing/gateways/worldpay.rb +280 -0
  111. data/lib/active_merchant/billing/integrations.rb +17 -0
  112. data/lib/active_merchant/billing/integrations/action_view_helper.rb +68 -0
  113. data/lib/active_merchant/billing/integrations/bogus.rb +23 -0
  114. data/lib/active_merchant/billing/integrations/bogus/helper.rb +17 -0
  115. data/lib/active_merchant/billing/integrations/bogus/notification.rb +11 -0
  116. data/lib/active_merchant/billing/integrations/bogus/return.rb +10 -0
  117. data/lib/active_merchant/billing/integrations/chronopay.rb +23 -0
  118. data/lib/active_merchant/billing/integrations/chronopay/helper.rb +120 -0
  119. data/lib/active_merchant/billing/integrations/chronopay/notification.rb +158 -0
  120. data/lib/active_merchant/billing/integrations/chronopay/return.rb +10 -0
  121. data/lib/active_merchant/billing/integrations/direc_pay.rb +41 -0
  122. data/lib/active_merchant/billing/integrations/direc_pay/helper.rb +200 -0
  123. data/lib/active_merchant/billing/integrations/direc_pay/notification.rb +76 -0
  124. data/lib/active_merchant/billing/integrations/direc_pay/return.rb +32 -0
  125. data/lib/active_merchant/billing/integrations/direc_pay/status.rb +37 -0
  126. data/lib/active_merchant/billing/integrations/directebanking.rb +47 -0
  127. data/lib/active_merchant/billing/integrations/directebanking/helper.rb +90 -0
  128. data/lib/active_merchant/billing/integrations/directebanking/notification.rb +120 -0
  129. data/lib/active_merchant/billing/integrations/directebanking/return.rb +11 -0
  130. data/lib/active_merchant/billing/integrations/dwolla.rb +30 -0
  131. data/lib/active_merchant/billing/integrations/dwolla/helper.rb +28 -0
  132. data/lib/active_merchant/billing/integrations/dwolla/notification.rb +50 -0
  133. data/lib/active_merchant/billing/integrations/dwolla/return.rb +38 -0
  134. data/lib/active_merchant/billing/integrations/e_payment_plans.rb +48 -0
  135. data/lib/active_merchant/billing/integrations/e_payment_plans/helper.rb +34 -0
  136. data/lib/active_merchant/billing/integrations/e_payment_plans/notification.rb +84 -0
  137. data/lib/active_merchant/billing/integrations/gestpay.rb +25 -0
  138. data/lib/active_merchant/billing/integrations/gestpay/common.rb +42 -0
  139. data/lib/active_merchant/billing/integrations/gestpay/helper.rb +70 -0
  140. data/lib/active_merchant/billing/integrations/gestpay/notification.rb +85 -0
  141. data/lib/active_merchant/billing/integrations/gestpay/return.rb +10 -0
  142. data/lib/active_merchant/billing/integrations/helper.rb +96 -0
  143. data/lib/active_merchant/billing/integrations/hi_trust.rb +27 -0
  144. data/lib/active_merchant/billing/integrations/hi_trust/helper.rb +58 -0
  145. data/lib/active_merchant/billing/integrations/hi_trust/notification.rb +59 -0
  146. data/lib/active_merchant/billing/integrations/hi_trust/return.rb +67 -0
  147. data/lib/active_merchant/billing/integrations/moneybookers.rb +26 -0
  148. data/lib/active_merchant/billing/integrations/moneybookers/helper.rb +59 -0
  149. data/lib/active_merchant/billing/integrations/moneybookers/notification.rb +129 -0
  150. data/lib/active_merchant/billing/integrations/nochex.rb +88 -0
  151. data/lib/active_merchant/billing/integrations/nochex/helper.rb +68 -0
  152. data/lib/active_merchant/billing/integrations/nochex/notification.rb +94 -0
  153. data/lib/active_merchant/billing/integrations/nochex/return.rb +10 -0
  154. data/lib/active_merchant/billing/integrations/notification.rb +62 -0
  155. data/lib/active_merchant/billing/integrations/payflow_link.rb +21 -0
  156. data/lib/active_merchant/billing/integrations/payflow_link/helper.rb +58 -0
  157. data/lib/active_merchant/billing/integrations/payflow_link/notification.rb +78 -0
  158. data/lib/active_merchant/billing/integrations/paypal.rb +39 -0
  159. data/lib/active_merchant/billing/integrations/paypal/helper.rb +119 -0
  160. data/lib/active_merchant/billing/integrations/paypal/notification.rb +154 -0
  161. data/lib/active_merchant/billing/integrations/paypal/return.rb +10 -0
  162. data/lib/active_merchant/billing/integrations/quickpay.rb +21 -0
  163. data/lib/active_merchant/billing/integrations/quickpay/helper.rb +72 -0
  164. data/lib/active_merchant/billing/integrations/quickpay/notification.rb +74 -0
  165. data/lib/active_merchant/billing/integrations/return.rb +42 -0
  166. data/lib/active_merchant/billing/integrations/sage_pay_form.rb +37 -0
  167. data/lib/active_merchant/billing/integrations/sage_pay_form/encryption.rb +33 -0
  168. data/lib/active_merchant/billing/integrations/sage_pay_form/helper.rb +111 -0
  169. data/lib/active_merchant/billing/integrations/sage_pay_form/notification.rb +210 -0
  170. data/lib/active_merchant/billing/integrations/sage_pay_form/return.rb +31 -0
  171. data/lib/active_merchant/billing/integrations/two_checkout.rb +23 -0
  172. data/lib/active_merchant/billing/integrations/two_checkout/helper.rb +59 -0
  173. data/lib/active_merchant/billing/integrations/two_checkout/notification.rb +114 -0
  174. data/lib/active_merchant/billing/integrations/two_checkout/return.rb +17 -0
  175. data/lib/active_merchant/billing/integrations/valitor.rb +33 -0
  176. data/lib/active_merchant/billing/integrations/valitor/helper.rb +86 -0
  177. data/lib/active_merchant/billing/integrations/valitor/notification.rb +13 -0
  178. data/lib/active_merchant/billing/integrations/valitor/response_fields.rb +97 -0
  179. data/lib/active_merchant/billing/integrations/valitor/return.rb +13 -0
  180. data/lib/active_merchant/billing/integrations/world_pay.rb +27 -0
  181. data/lib/active_merchant/billing/integrations/world_pay/helper.rb +100 -0
  182. data/lib/active_merchant/billing/integrations/world_pay/notification.rb +160 -0
  183. data/lib/active_merchant/billing/response.rb +32 -0
  184. data/lib/active_merchant/version.rb +3 -0
  185. data/lib/activemerchant.rb +1 -0
  186. data/lib/support/gateway_support.rb +58 -0
  187. data/lib/support/outbound_hosts.rb +25 -0
  188. metadata +335 -0
@@ -0,0 +1,207 @@
1
+ module ActiveMerchant #:nodoc:
2
+ module Billing #:nodoc:
3
+ module PayflowCommonAPI
4
+ def self.included(base)
5
+ base.default_currency = 'USD'
6
+
7
+ base.class_attribute :partner
8
+
9
+ # Set the default partner to PayPal
10
+ base.partner = 'PayPal'
11
+
12
+ base.supported_countries = ['US', 'CA', 'SG', 'AU']
13
+
14
+ base.class_attribute :timeout
15
+ base.timeout = 60
16
+
17
+ # Enable safe retry of failed connections
18
+ # Payflow is safe to retry because retried transactions use the same
19
+ # X-VPS-Request-ID header. If a transaction is detected as a duplicate
20
+ # only the original transaction data will be used by Payflow, and the
21
+ # subsequent Responses will have a :duplicate parameter set in the params
22
+ # hash.
23
+ base.retry_safe = true
24
+ end
25
+
26
+ XMLNS = 'http://www.paypal.com/XMLPay'
27
+ TEST_URL = 'https://pilot-payflowpro.paypal.com'
28
+ LIVE_URL = 'https://payflowpro.paypal.com'
29
+
30
+ CARD_MAPPING = {
31
+ :visa => 'Visa',
32
+ :master => 'MasterCard',
33
+ :discover => 'Discover',
34
+ :american_express => 'Amex',
35
+ :jcb => 'JCB',
36
+ :diners_club => 'DinersClub',
37
+ :switch => 'Switch',
38
+ :solo => 'Solo'
39
+ }
40
+
41
+ TRANSACTIONS = {
42
+ :purchase => "Sale",
43
+ :authorization => "Authorization",
44
+ :capture => "Capture",
45
+ :void => "Void",
46
+ :credit => "Credit"
47
+ }
48
+
49
+ CVV_CODE = {
50
+ 'Match' => 'M',
51
+ 'No Match' => 'N',
52
+ 'Service Not Available' => 'U',
53
+ 'Service not Requested' => 'P'
54
+ }
55
+
56
+ def initialize(options = {})
57
+ requires!(options, :login, :password)
58
+
59
+ @options = options
60
+ @options[:partner] = partner if @options[:partner].blank?
61
+ super
62
+ end
63
+
64
+ def test?
65
+ @options[:test] || super
66
+ end
67
+
68
+ def capture(money, authorization, options = {})
69
+ request = build_reference_request(:capture, money, authorization, options)
70
+ commit(request, options)
71
+ end
72
+
73
+ def void(authorization, options = {})
74
+ request = build_reference_request(:void, nil, authorization, options)
75
+ commit(request, options)
76
+ end
77
+
78
+ private
79
+ def build_request(body, options = {})
80
+ xml = Builder::XmlMarkup.new
81
+ xml.instruct!
82
+ xml.tag! 'XMLPayRequest', 'Timeout' => timeout.to_s, 'version' => "2.1", "xmlns" => XMLNS do
83
+ xml.tag! 'RequestData' do
84
+ xml.tag! 'Vendor', @options[:login]
85
+ xml.tag! 'Partner', @options[:partner]
86
+ if options[:request_type] == :recurring
87
+ xml << body
88
+ else
89
+ xml.tag! 'Transactions' do
90
+ xml.tag! 'Transaction', 'CustRef' => options[:customer] do
91
+ xml.tag! 'Verbosity', 'MEDIUM'
92
+ xml << body
93
+ end
94
+ end
95
+ end
96
+ end
97
+ xml.tag! 'RequestAuth' do
98
+ xml.tag! 'UserPass' do
99
+ xml.tag! 'User', !@options[:user].blank? ? @options[:user] : @options[:login]
100
+ xml.tag! 'Password', @options[:password]
101
+ end
102
+ end
103
+ end
104
+ xml.target!
105
+ end
106
+
107
+ def build_reference_request(action, money, authorization, options)
108
+ xml = Builder::XmlMarkup.new
109
+ xml.tag! TRANSACTIONS[action] do
110
+ xml.tag! 'PNRef', authorization
111
+
112
+ unless money.nil?
113
+ xml.tag! 'Invoice' do
114
+ xml.tag! 'TotalAmt', amount(money), 'Currency' => options[:currency] || currency(money)
115
+ end
116
+ end
117
+ end
118
+
119
+ xml.target!
120
+ end
121
+
122
+ def add_address(xml, tag, address, options)
123
+ return if address.nil?
124
+ xml.tag! tag do
125
+ xml.tag! 'Name', address[:name] unless address[:name].blank?
126
+ xml.tag! 'EMail', options[:email] unless options[:email].blank?
127
+ xml.tag! 'Phone', address[:phone] unless address[:phone].blank?
128
+ xml.tag! 'CustCode', options[:customer] if !options[:customer].blank? && tag == 'BillTo'
129
+
130
+ xml.tag! 'Address' do
131
+ xml.tag! 'Street', address[:address1] unless address[:address1].blank?
132
+ xml.tag! 'City', address[:city] unless address[:city].blank?
133
+ xml.tag! 'State', address[:state].blank? ? "N/A" : address[:state]
134
+ xml.tag! 'Country', address[:country] unless address[:country].blank?
135
+ xml.tag! 'Zip', address[:zip] unless address[:zip].blank?
136
+ end
137
+ end
138
+ end
139
+
140
+ def parse(data)
141
+ response = {}
142
+ xml = REXML::Document.new(data)
143
+ root = REXML::XPath.first(xml, "//ResponseData")
144
+
145
+ # REXML::XPath in Ruby 1.8.6 is now unable to match nodes based on their attributes
146
+ tx_result = REXML::XPath.first(root, "//TransactionResult")
147
+
148
+ if tx_result && tx_result.attributes['Duplicate'] == "true"
149
+ response[:duplicate] = true
150
+ end
151
+
152
+ root.elements.to_a.each do |node|
153
+ parse_element(response, node)
154
+ end
155
+
156
+ response
157
+ end
158
+
159
+ def parse_element(response, node)
160
+ node_name = node.name.underscore.to_sym
161
+ case
162
+ when node_name == :rp_payment_result
163
+ # Since we'll have multiple history items, we can't just flatten everything
164
+ # down as we do everywhere else. RPPaymentResult elements are not contained
165
+ # in an RPPaymentResults element so we'll come here multiple times
166
+ response[node_name] ||= []
167
+ response[node_name] << ( payment_result_response = {} )
168
+ node.elements.each{ |e| parse_element(payment_result_response, e) }
169
+ when node.has_elements?
170
+ node.elements.each{|e| parse_element(response, e) }
171
+ when node_name.to_s =~ /amt$/
172
+ # *Amt elements don't put the value in the #text - instead they use a Currency attribute
173
+ response[node_name] = node.attributes['Currency']
174
+ when node_name == :ext_data
175
+ response[node.attributes['Name'].underscore.to_sym] = node.attributes['Value']
176
+ else
177
+ response[node_name] = node.text
178
+ end
179
+ end
180
+
181
+ def build_headers(content_length)
182
+ {
183
+ "Content-Type" => "text/xml",
184
+ "Content-Length" => content_length.to_s,
185
+ "X-VPS-Client-Timeout" => timeout.to_s,
186
+ "X-VPS-VIT-Integration-Product" => "ActiveMerchant",
187
+ "X-VPS-VIT-Runtime-Version" => RUBY_VERSION,
188
+ "X-VPS-Request-ID" => Utils.generate_unique_id
189
+ }
190
+ end
191
+
192
+ def commit(request_body, options = {})
193
+ request = build_request(request_body, options)
194
+ headers = build_headers(request.size)
195
+
196
+ response = parse(ssl_post(test? ? TEST_URL : LIVE_URL, request, headers))
197
+
198
+ build_response(response[:result] == "0", response[:message], response,
199
+ :test => test?,
200
+ :authorization => response[:pn_ref] || response[:rp_ref],
201
+ :cvv_result => CVV_CODE[response[:cv_result]],
202
+ :avs_result => { :code => response[:avs_result] }
203
+ )
204
+ end
205
+ end
206
+ end
207
+ end
@@ -0,0 +1,39 @@
1
+ module ActiveMerchant #:nodoc:
2
+ module Billing #:nodoc:
3
+ class PayflowExpressResponse < Response
4
+ def email
5
+ @params['e_mail']
6
+ end
7
+
8
+ def full_name
9
+ "#{@params['name']} #{@params['lastname']}"
10
+ end
11
+
12
+ def token
13
+ @params['token']
14
+ end
15
+
16
+ def payer_id
17
+ @params['payer_id']
18
+ end
19
+
20
+ # Really the shipping country, but it is all the information provided
21
+ def payer_country
22
+ address['country']
23
+ end
24
+
25
+ def address
26
+ { 'name' => full_name,
27
+ 'company' => nil,
28
+ 'address1' => @params['street'],
29
+ 'address2' => @params['shiptostreet2'],
30
+ 'city' => @params['city'],
31
+ 'state' => @params['state'],
32
+ 'country' => @params['country'],
33
+ 'zip' => @params['zip'],
34
+ 'phone' => nil
35
+ }
36
+ end
37
+ end
38
+ end
39
+ end
@@ -0,0 +1,13 @@
1
+ module ActiveMerchant #:nodoc:
2
+ module Billing #:nodoc:
3
+ class PayflowResponse < Response
4
+ def profile_id
5
+ @params['profile_id']
6
+ end
7
+
8
+ def payment_history
9
+ @payment_history ||= @params['rp_payment_result'].collect{ |result| result.stringify_keys } rescue []
10
+ end
11
+ end
12
+ end
13
+ end
@@ -0,0 +1,222 @@
1
+ require File.dirname(__FILE__) + '/payflow/payflow_common_api'
2
+ require File.dirname(__FILE__) + '/payflow/payflow_express_response'
3
+ require File.dirname(__FILE__) + '/paypal_express_common'
4
+
5
+ module ActiveMerchant #:nodoc:
6
+ module Billing #:nodoc:
7
+ # ==General Parameters
8
+ # The following parameters are supported for #setup_authorization, #setup_purchase, #authorize and #purchase transactions. I've read
9
+ # in the docs that they recommend you pass the exact same parameters to both setup and authorize/purchase.
10
+ #
11
+ # This information was gleaned from a mix of:
12
+ # * PayFlow documentation
13
+ # * for key value pairs: {Express Checkout for Payflow Pro (PDF)}[https://cms.paypal.com/cms_content/US/en_US/files/developer/PFP_ExpressCheckout_PP.pdf]
14
+ # * XMLPay: {Payflow Pro XMLPay Developer's Guide (PDF)}[https://cms.paypal.com/cms_content/US/en_US/files/developer/PP_PayflowPro_XMLPay_Guide.pdf]
15
+ # * previous ActiveMerchant code
16
+ # * trial & error
17
+ #
18
+ # The following parameters are currently supported.
19
+ # [<tt>:ip</tt>] (opt) Customer IP Address
20
+ # [<tt>:order_id</tt>] (opt) An order or invoice number. This will be passed through to the Payflow backend at manager.paypal.com, and show up as "Supplier Reference #"
21
+ # [<tt>:description</tt>] (opt) Order description, shown to buyer (after redirected to PayPal). If Order Line Items are used (see below), then the description is suppressed. This will not be passed through to the Payflow backend.
22
+ # [<tt>:billing_address</tt>] (opt) See ActiveMerchant::Billing::Gateway for details
23
+ # [<tt>:shipping_address</tt>] (opt) See ActiveMerchant::Billing::Gateway for details
24
+ # [<tt>:currency</tt>] (req) Currency of transaction, will be set to USD by default for PayFlow Express if not specified
25
+ # [<tt>:email</tt>] (opt) Email of buyer; used to pre-fill PayPal login screen
26
+ # [<tt>:payer_id</tt>] (opt) Unique PayPal buyer account identification number, as returned by details_for request
27
+ # [<tt>:token</tt>] (req for #authorize & #purchase) Token returned by setup transaction
28
+ # [<tt>:no_shipping</tt>] (opt) Boolean for whether or not to display shipping address to buyer
29
+ # [<tt>:address_override</tt>] (opt) Boolean. If true, display shipping address passed by parameters, rather than shipping address on file with PayPal
30
+ # [<tt>:allow_note</tt>] (opt) Boolean for permitting buyer to add note during checkout. Note contents can be retrieved with details_for transaction
31
+ # [<tt>:return_url</tt>] (req) URL to which the buyer’s browser is returned after choosing to pay.
32
+ # [<tt>:cancel_return_url</tt>] (req) URL to which the buyer is returned if the buyer cancels the order.
33
+ # [<tt>:notify_url</tt>] (opt) Your URL for receiving Instant Payment Notification (IPN) about this transaction.
34
+ # [<tt>:comment</tt>] (opt) Comment field which will be reported to Payflow backend (at manager.paypal.com) as Comment1
35
+ # [<tt>:comment2</tt>] (opt) Comment field which will be reported to Payflow backend (at manager.paypal.com) as Comment2
36
+ #
37
+ # ==Line Items
38
+ # Support for order line items is available, but has to be enabled on the PayFlow backend. This is what I was told by Todd Sieber at Technical Support:
39
+ #
40
+ # <em>You will need to call Payflow Support at 1-888-883-9770, choose option #2. Request that they update your account in "Pandora" under Product Settings >> PayPal Mark and update the Features Bitmap to 1111111111111112. This is 15 ones and a two.</em>
41
+ #
42
+ # See here[https://www.x.com/message/206214#206214] for the forum discussion (requires login to {x.com}[https://x.com]
43
+ #
44
+ # [<tt>:items</tt>] (opt) Array of Order Line Items hashes. These are shown to the buyer after redirect to PayPal.
45
+ #
46
+ #
47
+ #
48
+ # The following keys are supported for line items:
49
+ # [<tt>:name</tt>] Name of line item
50
+ # [<tt>:description</tt>] Description of line item
51
+ # [<tt>:amount</tt>] Line Item Amount in Cents (as Integer)
52
+ # [<tt>:quantity</tt>] Line Item Quantity (default to 1 if left blank)
53
+ #
54
+ # ====Customization of Payment Page
55
+ # [<tt>:page_style</tt>] (opt) Your URL for receiving Instant Payment Notification (IPN) about this transaction.
56
+ # [<tt>:header_image</tt>] (opt) Your URL for receiving Instant Payment Notification (IPN) about this transaction.
57
+ # [<tt>:background_color</tt>] (opt) Your URL for receiving Instant Payment Notification (IPN) about this transaction.
58
+ # ====Additional options for old Checkout Experience, being phased out in 2010 and 2011
59
+ # [<tt>:header_background_color</tt>] (opt) Your URL for receiving Instant Payment Notification (IPN) about this transaction.
60
+ # [<tt>:header_border_color</tt>] (opt) Your URL for receiving Instant Payment Notification (IPN) about this transaction.
61
+
62
+
63
+ class PayflowExpressGateway < Gateway
64
+ include PayflowCommonAPI
65
+ include PaypalExpressCommon
66
+
67
+ self.test_redirect_url = 'https://www.sandbox.paypal.com/cgi-bin/webscr'
68
+ self.homepage_url = 'https://www.paypal.com/cgi-bin/webscr?cmd=xpt/merchant/ExpressCheckoutIntro-outside'
69
+ self.display_name = 'PayPal Express Checkout'
70
+
71
+ def authorize(money, options = {})
72
+ requires!(options, :token, :payer_id)
73
+ request = build_sale_or_authorization_request('Authorization', money, options)
74
+ commit(request, options)
75
+ end
76
+
77
+ def purchase(money, options = {})
78
+ requires!(options, :token, :payer_id)
79
+ request = build_sale_or_authorization_request('Sale', money, options)
80
+ commit(request, options)
81
+ end
82
+
83
+ def refund(money, identification, options = {})
84
+ request = build_reference_request(:credit, money, identification, options)
85
+ commit(request, options)
86
+ end
87
+
88
+ def credit(money, identification, options = {})
89
+ deprecated CREDIT_DEPRECATION_MESSAGE
90
+ refund(money, identification, options)
91
+ end
92
+
93
+ def setup_authorization(money, options = {})
94
+ requires!(options, :return_url, :cancel_return_url)
95
+
96
+ request = build_setup_express_sale_or_authorization_request('Authorization', money, options)
97
+ commit(request, options)
98
+ end
99
+
100
+ def setup_purchase(money, options = {})
101
+ requires!(options, :return_url, :cancel_return_url)
102
+
103
+ request = build_setup_express_sale_or_authorization_request('Sale', money, options)
104
+ commit(request, options)
105
+ end
106
+
107
+ def details_for(token)
108
+ request = build_get_express_details_request(token)
109
+ commit(request, options)
110
+ end
111
+
112
+ private
113
+ def build_get_express_details_request(token)
114
+ xml = Builder::XmlMarkup.new :indent => 2
115
+ xml.tag! 'GetExpressCheckout' do
116
+ xml.tag! 'Authorization' do
117
+ xml.tag! 'PayData' do
118
+ xml.tag! 'Tender' do
119
+ xml.tag! 'PayPal' do
120
+ xml.tag! 'Token', token
121
+ end
122
+ end
123
+ end
124
+ end
125
+ end
126
+ xml.target!
127
+ end
128
+
129
+ def build_setup_express_sale_or_authorization_request(action, money, options = {})
130
+ xml = Builder::XmlMarkup.new :indent => 2
131
+ xml.tag! 'SetExpressCheckout' do
132
+ xml.tag! action do
133
+ add_pay_data xml, money, options
134
+ end
135
+ end
136
+ xml.target!
137
+ end
138
+
139
+ def build_sale_or_authorization_request(action, money, options)
140
+ xml = Builder::XmlMarkup.new :indent => 2
141
+ xml.tag! 'DoExpressCheckout' do
142
+ xml.tag! action do
143
+ add_pay_data xml, money, options
144
+ end
145
+ end
146
+ xml.target!
147
+ end
148
+
149
+ def add_pay_data(xml, money, options)
150
+ xml.tag! 'PayData' do
151
+ xml.tag! 'Invoice' do
152
+ xml.tag! 'CustIP', options[:ip] unless options[:ip].blank?
153
+ xml.tag! 'InvNum', options[:order_id] unless options[:order_id].blank?
154
+ # Description field will be shown to buyer, unless line items are also being supplied (then only line items are shown).
155
+ xml.tag! 'Description', options[:description] unless options[:description].blank?
156
+ # Comment, Comment2 should make it to the backend at manager.paypal.com, as with Payflow credit card transactions
157
+ # but that doesn't seem to work (yet?). See: https://www.x.com/thread/51908?tstart=0
158
+ xml.tag! 'Comment', options[:comment] unless options[:comment].nil?
159
+ xml.tag!('ExtData', 'Name'=> 'COMMENT2', 'Value'=> options[:comment2]) unless options[:comment2].nil?
160
+
161
+ billing_address = options[:billing_address] || options[:address]
162
+ add_address(xml, 'BillTo', billing_address, options) if billing_address
163
+ add_address(xml, 'ShipTo', options[:shipping_address], options) if options[:shipping_address]
164
+
165
+ # Note: To get order line-items to show up with Payflow Express, this feature has to be enabled on the backend.
166
+ # Call Support at 888 883 9770, press 2. Then request that they update your account in "Pandora" under Product Settings >> PayPal
167
+ # Mark and update the Features Bitmap to 1111111111111112. This is 15 ones and a two.
168
+ # See here for the forum discussion: https://www.x.com/message/206214#206214
169
+ items = options[:items] || []
170
+ items.each_with_index do |item, index|
171
+ xml.tag! 'ExtData', 'Name' => "L_DESC#{index}", 'Value' => item[:description]
172
+ xml.tag! 'ExtData', 'Name' => "L_COST#{index}", 'Value' => amount(item[:amount])
173
+ xml.tag! 'ExtData', 'Name' => "L_QTY#{index}", 'Value' => item[:quantity] || '1'
174
+ xml.tag! 'ExtData', 'Name' => "L_NAME#{index}", 'Value' => item[:name]
175
+ # Note: An ItemURL is supported in Paypal Express (different API), but not PayFlow Express, as far as I can tell.
176
+ # L_URLn nor L_ITEMURLn seem to work
177
+ end
178
+ if items.any?
179
+ xml.tag! 'ExtData', 'Name' => 'CURRENCY', 'Value' => options[:currency] || currency(money)
180
+ xml.tag! 'ExtData', 'Name' => "ITEMAMT", 'Value' => amount(money)
181
+ end
182
+
183
+ xml.tag! 'TotalAmt', amount(money), 'Currency' => options[:currency] || currency(money)
184
+ end
185
+
186
+ xml.tag! 'Tender' do
187
+ add_paypal_details(xml, options)
188
+ end
189
+ end
190
+ end
191
+
192
+ def add_paypal_details(xml, options)
193
+ xml.tag! 'PayPal' do
194
+ xml.tag! 'EMail', options[:email] unless options[:email].blank?
195
+ xml.tag! 'ReturnURL', options[:return_url] unless options[:return_url].blank?
196
+ xml.tag! 'CancelURL', options[:cancel_return_url] unless options[:cancel_return_url].blank?
197
+ xml.tag! 'NotifyURL', options[:notify_url] unless options[:notify_url].blank?
198
+ xml.tag! 'PayerID', options[:payer_id] unless options[:payer_id].blank?
199
+ xml.tag! 'Token', options[:token] unless options[:token].blank?
200
+ xml.tag! 'NoShipping', options[:no_shipping] ? '1' : '0'
201
+ xml.tag! 'AddressOverride', options[:address_override] ? '1' : '0'
202
+ xml.tag! 'ButtonSource', application_id.to_s.slice(0,32) unless application_id.blank?
203
+
204
+ # Customization of the payment page
205
+ xml.tag! 'PageStyle', options[:page_style] unless options[:page_style].blank?
206
+ xml.tag! 'HeaderImage', options[:header_image] unless options[:header_image].blank?
207
+ xml.tag! 'PayflowColor', options[:background_color] unless options[:background_color].blank?
208
+ # Note: HeaderImage and PayflowColor apply to both the new (as of 2010) and the old checkout experience
209
+ # HeaderBackColor and HeaderBorderColor apply only to the old checkout experience which is being phased out.
210
+ xml.tag! 'HeaderBackColor', options[:header_background_color] unless options[:header_background_color].blank?
211
+ xml.tag! 'HeaderBorderColor', options[:header_border_color] unless options[:header_border_color].blank?
212
+ xml.tag! 'ExtData', 'Name' => 'ALLOWNOTE', 'Value' => options[:allow_note]
213
+ end
214
+ end
215
+
216
+ def build_response(success, message, response, options = {})
217
+ PayflowExpressResponse.new(success, message, response, options)
218
+ end
219
+ end
220
+ end
221
+ end
222
+