active_accountability_merchant 1.97.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (259) hide show
  1. checksums.yaml +7 -0
  2. data/CHANGELOG +2948 -0
  3. data/CONTRIBUTORS +568 -0
  4. data/MIT-LICENSE +20 -0
  5. data/README.md +246 -0
  6. data/lib/active_merchant.rb +63 -0
  7. data/lib/active_merchant/billing.rb +15 -0
  8. data/lib/active_merchant/billing/apple_pay_payment_token.rb +22 -0
  9. data/lib/active_merchant/billing/avs_result.rb +96 -0
  10. data/lib/active_merchant/billing/base.rb +61 -0
  11. data/lib/active_merchant/billing/check.rb +80 -0
  12. data/lib/active_merchant/billing/compatibility.rb +117 -0
  13. data/lib/active_merchant/billing/credit_card.rb +406 -0
  14. data/lib/active_merchant/billing/credit_card_formatting.rb +23 -0
  15. data/lib/active_merchant/billing/credit_card_methods.rb +343 -0
  16. data/lib/active_merchant/billing/cvv_result.rb +38 -0
  17. data/lib/active_merchant/billing/gateway.rb +330 -0
  18. data/lib/active_merchant/billing/gateways.rb +14 -0
  19. data/lib/active_merchant/billing/gateways/adyen.rb +503 -0
  20. data/lib/active_merchant/billing/gateways/allied_wallet.rb +205 -0
  21. data/lib/active_merchant/billing/gateways/authorize_net.rb +1055 -0
  22. data/lib/active_merchant/billing/gateways/authorize_net_arb.rb +417 -0
  23. data/lib/active_merchant/billing/gateways/authorize_net_cim.rb +980 -0
  24. data/lib/active_merchant/billing/gateways/axcessms.rb +181 -0
  25. data/lib/active_merchant/billing/gateways/balanced.rb +256 -0
  26. data/lib/active_merchant/billing/gateways/bambora_apac.rb +226 -0
  27. data/lib/active_merchant/billing/gateways/bank_frick.rb +225 -0
  28. data/lib/active_merchant/billing/gateways/banwire.rb +116 -0
  29. data/lib/active_merchant/billing/gateways/barclaycard_smartpay.rb +384 -0
  30. data/lib/active_merchant/billing/gateways/barclays_epdq_extra_plus.rb +15 -0
  31. data/lib/active_merchant/billing/gateways/be2bill.rb +131 -0
  32. data/lib/active_merchant/billing/gateways/beanstream.rb +217 -0
  33. data/lib/active_merchant/billing/gateways/beanstream/beanstream_core.rb +473 -0
  34. data/lib/active_merchant/billing/gateways/beanstream_interac.rb +57 -0
  35. data/lib/active_merchant/billing/gateways/blue_pay.rb +522 -0
  36. data/lib/active_merchant/billing/gateways/blue_snap.rb +522 -0
  37. data/lib/active_merchant/billing/gateways/bogus.rb +186 -0
  38. data/lib/active_merchant/billing/gateways/borgun.rb +220 -0
  39. data/lib/active_merchant/billing/gateways/bpoint.rb +277 -0
  40. data/lib/active_merchant/billing/gateways/braintree.rb +19 -0
  41. data/lib/active_merchant/billing/gateways/braintree/braintree_common.rb +22 -0
  42. data/lib/active_merchant/billing/gateways/braintree_blue.rb +740 -0
  43. data/lib/active_merchant/billing/gateways/braintree_orange.rb +19 -0
  44. data/lib/active_merchant/billing/gateways/bridge_pay.rb +244 -0
  45. data/lib/active_merchant/billing/gateways/cams.rb +230 -0
  46. data/lib/active_merchant/billing/gateways/card_connect.rb +316 -0
  47. data/lib/active_merchant/billing/gateways/card_save.rb +22 -0
  48. data/lib/active_merchant/billing/gateways/card_stream.rb +367 -0
  49. data/lib/active_merchant/billing/gateways/cardknox.rb +327 -0
  50. data/lib/active_merchant/billing/gateways/cardprocess.rb +254 -0
  51. data/lib/active_merchant/billing/gateways/cashnet.rb +219 -0
  52. data/lib/active_merchant/billing/gateways/cc5.rb +198 -0
  53. data/lib/active_merchant/billing/gateways/cecabank.rb +249 -0
  54. data/lib/active_merchant/billing/gateways/cenpos.rb +327 -0
  55. data/lib/active_merchant/billing/gateways/checkout.rb +214 -0
  56. data/lib/active_merchant/billing/gateways/checkout_v2.rb +270 -0
  57. data/lib/active_merchant/billing/gateways/citrus_pay.rb +22 -0
  58. data/lib/active_merchant/billing/gateways/clearhaus.rb +220 -0
  59. data/lib/active_merchant/billing/gateways/commercegate.rb +142 -0
  60. data/lib/active_merchant/billing/gateways/conekta.rb +228 -0
  61. data/lib/active_merchant/billing/gateways/creditcall.rb +271 -0
  62. data/lib/active_merchant/billing/gateways/credorax.rb +382 -0
  63. data/lib/active_merchant/billing/gateways/ct_payment.rb +268 -0
  64. data/lib/active_merchant/billing/gateways/culqi.rb +277 -0
  65. data/lib/active_merchant/billing/gateways/cyber_source.rb +828 -0
  66. data/lib/active_merchant/billing/gateways/d_local.rb +226 -0
  67. data/lib/active_merchant/billing/gateways/data_cash.rb +305 -0
  68. data/lib/active_merchant/billing/gateways/decidir.rb +233 -0
  69. data/lib/active_merchant/billing/gateways/dibs.rb +199 -0
  70. data/lib/active_merchant/billing/gateways/digitzs.rb +292 -0
  71. data/lib/active_merchant/billing/gateways/ebanx.rb +296 -0
  72. data/lib/active_merchant/billing/gateways/efsnet.rb +215 -0
  73. data/lib/active_merchant/billing/gateways/elavon.rb +320 -0
  74. data/lib/active_merchant/billing/gateways/element.rb +356 -0
  75. data/lib/active_merchant/billing/gateways/epay.rb +285 -0
  76. data/lib/active_merchant/billing/gateways/evo_ca.rb +308 -0
  77. data/lib/active_merchant/billing/gateways/eway.rb +226 -0
  78. data/lib/active_merchant/billing/gateways/eway_managed.rb +290 -0
  79. data/lib/active_merchant/billing/gateways/eway_rapid.rb +563 -0
  80. data/lib/active_merchant/billing/gateways/exact.rb +224 -0
  81. data/lib/active_merchant/billing/gateways/ezic.rb +195 -0
  82. data/lib/active_merchant/billing/gateways/fat_zebra.rb +218 -0
  83. data/lib/active_merchant/billing/gateways/federated_canada.rb +159 -0
  84. data/lib/active_merchant/billing/gateways/finansbank.rb +22 -0
  85. data/lib/active_merchant/billing/gateways/first_giving.rb +142 -0
  86. data/lib/active_merchant/billing/gateways/first_pay.rb +182 -0
  87. data/lib/active_merchant/billing/gateways/firstdata_e4.rb +451 -0
  88. data/lib/active_merchant/billing/gateways/firstdata_e4_v27.rb +485 -0
  89. data/lib/active_merchant/billing/gateways/flo2cash.rb +215 -0
  90. data/lib/active_merchant/billing/gateways/flo2cash_simple.rb +20 -0
  91. data/lib/active_merchant/billing/gateways/forte.rb +270 -0
  92. data/lib/active_merchant/billing/gateways/garanti.rb +259 -0
  93. data/lib/active_merchant/billing/gateways/global_collect.rb +336 -0
  94. data/lib/active_merchant/billing/gateways/global_transport.rb +194 -0
  95. data/lib/active_merchant/billing/gateways/hdfc.rb +206 -0
  96. data/lib/active_merchant/billing/gateways/hps.rb +350 -0
  97. data/lib/active_merchant/billing/gateways/iats_payments.rb +290 -0
  98. data/lib/active_merchant/billing/gateways/in_context_paypal_express.rb +15 -0
  99. data/lib/active_merchant/billing/gateways/inspire.rb +218 -0
  100. data/lib/active_merchant/billing/gateways/instapay.rb +162 -0
  101. data/lib/active_merchant/billing/gateways/ipp.rb +176 -0
  102. data/lib/active_merchant/billing/gateways/iridium.rb +468 -0
  103. data/lib/active_merchant/billing/gateways/itransact.rb +447 -0
  104. data/lib/active_merchant/billing/gateways/iveri.rb +251 -0
  105. data/lib/active_merchant/billing/gateways/jetpay.rb +402 -0
  106. data/lib/active_merchant/billing/gateways/jetpay_v2.rb +437 -0
  107. data/lib/active_merchant/billing/gateways/komoju.rb +115 -0
  108. data/lib/active_merchant/billing/gateways/kushki.rb +219 -0
  109. data/lib/active_merchant/billing/gateways/latitude19.rb +411 -0
  110. data/lib/active_merchant/billing/gateways/linkpoint.rb +449 -0
  111. data/lib/active_merchant/billing/gateways/litle.rb +507 -0
  112. data/lib/active_merchant/billing/gateways/mastercard.rb +267 -0
  113. data/lib/active_merchant/billing/gateways/maxipago.rb +220 -0
  114. data/lib/active_merchant/billing/gateways/mercado_pago.rb +277 -0
  115. data/lib/active_merchant/billing/gateways/merchant_e_solutions.rb +194 -0
  116. data/lib/active_merchant/billing/gateways/merchant_one.rb +113 -0
  117. data/lib/active_merchant/billing/gateways/merchant_partners.rb +245 -0
  118. data/lib/active_merchant/billing/gateways/merchant_ware.rb +319 -0
  119. data/lib/active_merchant/billing/gateways/merchant_ware_version_four.rb +286 -0
  120. data/lib/active_merchant/billing/gateways/merchant_warrior.rb +210 -0
  121. data/lib/active_merchant/billing/gateways/mercury.rb +356 -0
  122. data/lib/active_merchant/billing/gateways/metrics_global.rb +303 -0
  123. data/lib/active_merchant/billing/gateways/micropayment.rb +183 -0
  124. data/lib/active_merchant/billing/gateways/migs.rb +332 -0
  125. data/lib/active_merchant/billing/gateways/migs/migs_codes.rb +100 -0
  126. data/lib/active_merchant/billing/gateways/modern_payments.rb +37 -0
  127. data/lib/active_merchant/billing/gateways/modern_payments_cim.rb +217 -0
  128. data/lib/active_merchant/billing/gateways/monei.rb +338 -0
  129. data/lib/active_merchant/billing/gateways/moneris.rb +377 -0
  130. data/lib/active_merchant/billing/gateways/moneris_us.rb +352 -0
  131. data/lib/active_merchant/billing/gateways/money_movers.rb +151 -0
  132. data/lib/active_merchant/billing/gateways/mundipagg.rb +293 -0
  133. data/lib/active_merchant/billing/gateways/nab_transact.rb +301 -0
  134. data/lib/active_merchant/billing/gateways/ncr_secure_pay.rb +165 -0
  135. data/lib/active_merchant/billing/gateways/net_registry.rb +199 -0
  136. data/lib/active_merchant/billing/gateways/netaxept.rb +180 -0
  137. data/lib/active_merchant/billing/gateways/netbanx.rb +294 -0
  138. data/lib/active_merchant/billing/gateways/netbilling.rb +232 -0
  139. data/lib/active_merchant/billing/gateways/netpay.rb +222 -0
  140. data/lib/active_merchant/billing/gateways/network_merchants.rb +241 -0
  141. data/lib/active_merchant/billing/gateways/nmi.rb +324 -0
  142. data/lib/active_merchant/billing/gateways/ogone.rb +478 -0
  143. data/lib/active_merchant/billing/gateways/omise.rb +324 -0
  144. data/lib/active_merchant/billing/gateways/openpay.rb +228 -0
  145. data/lib/active_merchant/billing/gateways/opp.rb +388 -0
  146. data/lib/active_merchant/billing/gateways/optimal_payment.rb +332 -0
  147. data/lib/active_merchant/billing/gateways/orbital.rb +1006 -0
  148. data/lib/active_merchant/billing/gateways/orbital/orbital_soft_descriptors.rb +47 -0
  149. data/lib/active_merchant/billing/gateways/pac_net_raven.rb +206 -0
  150. data/lib/active_merchant/billing/gateways/pagarme.rb +246 -0
  151. data/lib/active_merchant/billing/gateways/pago_facil.rb +122 -0
  152. data/lib/active_merchant/billing/gateways/pay_conex.rb +245 -0
  153. data/lib/active_merchant/billing/gateways/pay_gate_xml.rb +280 -0
  154. data/lib/active_merchant/billing/gateways/pay_hub.rb +213 -0
  155. data/lib/active_merchant/billing/gateways/pay_junction.rb +390 -0
  156. data/lib/active_merchant/billing/gateways/pay_junction_v2.rb +188 -0
  157. data/lib/active_merchant/billing/gateways/pay_secure.rb +111 -0
  158. data/lib/active_merchant/billing/gateways/paybox_direct.rb +200 -0
  159. data/lib/active_merchant/billing/gateways/payeezy.rb +410 -0
  160. data/lib/active_merchant/billing/gateways/payex.rb +410 -0
  161. data/lib/active_merchant/billing/gateways/payflow.rb +383 -0
  162. data/lib/active_merchant/billing/gateways/payflow/payflow_common_api.rb +235 -0
  163. data/lib/active_merchant/billing/gateways/payflow/payflow_express_response.rb +43 -0
  164. data/lib/active_merchant/billing/gateways/payflow/payflow_response.rb +13 -0
  165. data/lib/active_merchant/billing/gateways/payflow_express.rb +220 -0
  166. data/lib/active_merchant/billing/gateways/payflow_express_uk.rb +14 -0
  167. data/lib/active_merchant/billing/gateways/payflow_uk.rb +20 -0
  168. data/lib/active_merchant/billing/gateways/payment_express.rb +369 -0
  169. data/lib/active_merchant/billing/gateways/paymentez.rb +300 -0
  170. data/lib/active_merchant/billing/gateways/paymill.rb +371 -0
  171. data/lib/active_merchant/billing/gateways/paypal.rb +128 -0
  172. data/lib/active_merchant/billing/gateways/paypal/paypal_common_api.rb +717 -0
  173. data/lib/active_merchant/billing/gateways/paypal/paypal_express_response.rb +65 -0
  174. data/lib/active_merchant/billing/gateways/paypal/paypal_recurring_api.rb +262 -0
  175. data/lib/active_merchant/billing/gateways/paypal_ca.rb +13 -0
  176. data/lib/active_merchant/billing/gateways/paypal_digital_goods.rb +44 -0
  177. data/lib/active_merchant/billing/gateways/paypal_express.rb +269 -0
  178. data/lib/active_merchant/billing/gateways/paypal_express_common.rb +30 -0
  179. data/lib/active_merchant/billing/gateways/payscout.rb +160 -0
  180. data/lib/active_merchant/billing/gateways/paystation.rb +207 -0
  181. data/lib/active_merchant/billing/gateways/payu_in.rb +248 -0
  182. data/lib/active_merchant/billing/gateways/payu_latam.rb +449 -0
  183. data/lib/active_merchant/billing/gateways/payway.rb +207 -0
  184. data/lib/active_merchant/billing/gateways/pin.rb +234 -0
  185. data/lib/active_merchant/billing/gateways/plugnpay.rb +284 -0
  186. data/lib/active_merchant/billing/gateways/pro_pay.rb +326 -0
  187. data/lib/active_merchant/billing/gateways/psigate.rb +227 -0
  188. data/lib/active_merchant/billing/gateways/psl_card.rb +296 -0
  189. data/lib/active_merchant/billing/gateways/qbms.rb +303 -0
  190. data/lib/active_merchant/billing/gateways/quantum.rb +276 -0
  191. data/lib/active_merchant/billing/gateways/quickbooks.rb +290 -0
  192. data/lib/active_merchant/billing/gateways/quickpay.rb +25 -0
  193. data/lib/active_merchant/billing/gateways/quickpay/quickpay_common.rb +184 -0
  194. data/lib/active_merchant/billing/gateways/quickpay/quickpay_v10.rb +301 -0
  195. data/lib/active_merchant/billing/gateways/quickpay/quickpay_v4to7.rb +226 -0
  196. data/lib/active_merchant/billing/gateways/qvalent.rb +289 -0
  197. data/lib/active_merchant/billing/gateways/realex.rb +374 -0
  198. data/lib/active_merchant/billing/gateways/redsys.rb +534 -0
  199. data/lib/active_merchant/billing/gateways/s5.rb +246 -0
  200. data/lib/active_merchant/billing/gateways/safe_charge.rb +262 -0
  201. data/lib/active_merchant/billing/gateways/sage.rb +448 -0
  202. data/lib/active_merchant/billing/gateways/sage_pay.rb +434 -0
  203. data/lib/active_merchant/billing/gateways/sallie_mae.rb +142 -0
  204. data/lib/active_merchant/billing/gateways/secure_net.rb +267 -0
  205. data/lib/active_merchant/billing/gateways/secure_pay.rb +200 -0
  206. data/lib/active_merchant/billing/gateways/secure_pay_au.rb +292 -0
  207. data/lib/active_merchant/billing/gateways/secure_pay_tech.rb +104 -0
  208. data/lib/active_merchant/billing/gateways/securion_pay.rb +264 -0
  209. data/lib/active_merchant/billing/gateways/skip_jack.rb +450 -0
  210. data/lib/active_merchant/billing/gateways/smart_ps.rb +280 -0
  211. data/lib/active_merchant/billing/gateways/so_easy_pay.rb +194 -0
  212. data/lib/active_merchant/billing/gateways/spreedly_core.rb +306 -0
  213. data/lib/active_merchant/billing/gateways/stripe.rb +790 -0
  214. data/lib/active_merchant/billing/gateways/stripe_payment_intents.rb +267 -0
  215. data/lib/active_merchant/billing/gateways/swipe_checkout.rb +152 -0
  216. data/lib/active_merchant/billing/gateways/telr.rb +274 -0
  217. data/lib/active_merchant/billing/gateways/tns.rb +22 -0
  218. data/lib/active_merchant/billing/gateways/trans_first.rb +239 -0
  219. data/lib/active_merchant/billing/gateways/trans_first_transaction_express.rb +608 -0
  220. data/lib/active_merchant/billing/gateways/transact_pro.rb +224 -0
  221. data/lib/active_merchant/billing/gateways/transax.rb +22 -0
  222. data/lib/active_merchant/billing/gateways/transnational.rb +9 -0
  223. data/lib/active_merchant/billing/gateways/trexle.rb +218 -0
  224. data/lib/active_merchant/billing/gateways/trust_commerce.rb +490 -0
  225. data/lib/active_merchant/billing/gateways/usa_epay.rb +25 -0
  226. data/lib/active_merchant/billing/gateways/usa_epay_advanced.rb +1620 -0
  227. data/lib/active_merchant/billing/gateways/usa_epay_transaction.rb +355 -0
  228. data/lib/active_merchant/billing/gateways/vanco.rb +294 -0
  229. data/lib/active_merchant/billing/gateways/verifi.rb +225 -0
  230. data/lib/active_merchant/billing/gateways/viaklix.rb +176 -0
  231. data/lib/active_merchant/billing/gateways/visanet_peru.rb +245 -0
  232. data/lib/active_merchant/billing/gateways/webpay.rb +97 -0
  233. data/lib/active_merchant/billing/gateways/wepay.rb +237 -0
  234. data/lib/active_merchant/billing/gateways/wirecard.rb +432 -0
  235. data/lib/active_merchant/billing/gateways/world_net.rb +344 -0
  236. data/lib/active_merchant/billing/gateways/worldpay.rb +634 -0
  237. data/lib/active_merchant/billing/gateways/worldpay_online_payments.rb +215 -0
  238. data/lib/active_merchant/billing/gateways/worldpay_us.rb +221 -0
  239. data/lib/active_merchant/billing/model.rb +30 -0
  240. data/lib/active_merchant/billing/network_tokenization_credit_card.rb +39 -0
  241. data/lib/active_merchant/billing/payment_token.rb +21 -0
  242. data/lib/active_merchant/billing/rails.rb +3 -0
  243. data/lib/active_merchant/billing/response.rb +92 -0
  244. data/lib/active_merchant/connection.rb +195 -0
  245. data/lib/active_merchant/country.rb +336 -0
  246. data/lib/active_merchant/empty.rb +20 -0
  247. data/lib/active_merchant/errors.rb +35 -0
  248. data/lib/active_merchant/net_http_ssl_connection.rb +10 -0
  249. data/lib/active_merchant/network_connection_retries.rb +80 -0
  250. data/lib/active_merchant/post_data.rb +25 -0
  251. data/lib/active_merchant/posts_data.rb +92 -0
  252. data/lib/active_merchant/version.rb +3 -0
  253. data/lib/activemerchant.rb +1 -0
  254. data/lib/certs/cacert.pem +3988 -0
  255. data/lib/support/gateway_support.rb +69 -0
  256. data/lib/support/outbound_hosts.rb +28 -0
  257. data/lib/support/ssl_verify.rb +92 -0
  258. data/lib/support/ssl_version.rb +87 -0
  259. metadata +435 -0
@@ -0,0 +1,324 @@
1
+ module ActiveMerchant #:nodoc:
2
+ module Billing #:nodoc:
3
+ class NmiGateway < Gateway
4
+ include Empty
5
+
6
+ DUP_WINDOW_DEPRECATION_MESSAGE = 'The class-level duplicate_window variable is deprecated. Please use the :dup_seconds transaction option instead.'
7
+
8
+ self.test_url = self.live_url = 'https://secure.nmi.com/api/transact.php'
9
+ self.default_currency = 'USD'
10
+ self.money_format = :dollars
11
+ self.supported_countries = ['US']
12
+ self.supported_cardtypes = [:visa, :master, :american_express, :discover]
13
+ self.homepage_url = 'http://nmi.com/'
14
+ self.display_name = 'NMI'
15
+
16
+ def self.duplicate_window=(seconds)
17
+ ActiveMerchant.deprecated(DUP_WINDOW_DEPRECATION_MESSAGE)
18
+ @dup_seconds = seconds
19
+ end
20
+
21
+ def self.duplicate_window
22
+ instance_variable_defined?(:@dup_seconds) ? @dup_seconds : nil
23
+ end
24
+
25
+ def initialize(options = {})
26
+ requires!(options, :login, :password)
27
+ super
28
+ end
29
+
30
+ def purchase(amount, payment_method, options={})
31
+ post = {}
32
+ add_invoice(post, amount, options)
33
+ add_payment_method(post, payment_method, options)
34
+ add_stored_credential(post, options)
35
+ add_customer_data(post, options)
36
+ add_vendor_data(post, options)
37
+ add_merchant_defined_fields(post, options)
38
+ add_level3_fields(post, options)
39
+
40
+ commit('sale', post)
41
+ end
42
+
43
+ def authorize(amount, payment_method, options={})
44
+ post = {}
45
+ add_invoice(post, amount, options)
46
+ add_payment_method(post, payment_method, options)
47
+ add_stored_credential(post, options)
48
+ add_customer_data(post, options)
49
+ add_vendor_data(post, options)
50
+ add_merchant_defined_fields(post, options)
51
+ add_level3_fields(post, options)
52
+
53
+ commit('auth', post)
54
+ end
55
+
56
+ def capture(amount, authorization, options={})
57
+ post = {}
58
+ add_invoice(post, amount, options)
59
+ add_reference(post, authorization)
60
+ add_merchant_defined_fields(post, options)
61
+
62
+ commit('capture', post)
63
+ end
64
+
65
+ def void(authorization, options={})
66
+ post = {}
67
+ add_reference(post, authorization)
68
+ add_payment_type(post, authorization)
69
+
70
+ commit('void', post)
71
+ end
72
+
73
+ def refund(amount, authorization, options={})
74
+ post = {}
75
+ add_invoice(post, amount, options)
76
+ add_reference(post, authorization)
77
+ add_payment_type(post, authorization)
78
+
79
+ commit('refund', post)
80
+ end
81
+
82
+ def credit(amount, payment_method, options={})
83
+ post = {}
84
+ add_invoice(post, amount, options)
85
+ add_payment_method(post, payment_method, options)
86
+ add_customer_data(post, options)
87
+ add_vendor_data(post, options)
88
+ add_level3_fields(post, options)
89
+
90
+ commit('credit', post)
91
+ end
92
+
93
+ def verify(payment_method, options={})
94
+ post = {}
95
+ add_payment_method(post, payment_method, options)
96
+ add_customer_data(post, options)
97
+ add_vendor_data(post, options)
98
+ add_merchant_defined_fields(post, options)
99
+ add_level3_fields(post, options)
100
+
101
+ commit('validate', post)
102
+ end
103
+
104
+ def store(payment_method, options = {})
105
+ post = {}
106
+ add_invoice(post, nil, options)
107
+ add_payment_method(post, payment_method, options)
108
+ add_customer_data(post, options)
109
+ add_vendor_data(post, options)
110
+ add_merchant_defined_fields(post, options)
111
+
112
+ commit('add_customer', post)
113
+ end
114
+
115
+ def verify_credentials
116
+ response = void('0')
117
+ response.message != 'Authentication Failed'
118
+ end
119
+
120
+ def supports_scrubbing?
121
+ true
122
+ end
123
+
124
+ def scrub(transcript)
125
+ transcript.
126
+ gsub(%r((password=)[^&\n]*), '\1[FILTERED]').
127
+ gsub(%r((ccnumber=)\d+), '\1[FILTERED]').
128
+ gsub(%r((cvv=)\d+), '\1[FILTERED]').
129
+ gsub(%r((checkaba=)\d+), '\1[FILTERED]').
130
+ gsub(%r((checkaccount=)\d+), '\1[FILTERED]').
131
+ gsub(%r((cryptogram=)[^&]+(&?)), '\1[FILTERED]\2')
132
+ end
133
+
134
+ def supports_network_tokenization?
135
+ true
136
+ end
137
+
138
+ private
139
+
140
+ def add_level3_fields(post, options)
141
+ add_fields_to_post_if_present(post, options, [:tax, :shipping, :ponumber])
142
+ end
143
+
144
+ def add_invoice(post, money, options)
145
+ post[:amount] = amount(money)
146
+ post[:orderid] = options[:order_id]
147
+ post[:orderdescription] = options[:description]
148
+ post[:currency] = options[:currency] || currency(money)
149
+ post[:billing_method] = 'recurring' if options[:recurring]
150
+ if (dup_seconds = (options[:dup_seconds] || self.class.duplicate_window))
151
+ post[:dup_seconds] = dup_seconds
152
+ end
153
+ end
154
+
155
+ def add_payment_method(post, payment_method, options)
156
+ if(payment_method.is_a?(String))
157
+ customer_vault_id, _ = split_authorization(payment_method)
158
+ post[:customer_vault_id] = customer_vault_id
159
+ elsif payment_method.is_a?(NetworkTokenizationCreditCard)
160
+ post[:ccnumber] = payment_method.number
161
+ post[:ccexp] = exp_date(payment_method)
162
+ post[:token_cryptogram] = payment_method.payment_cryptogram
163
+ elsif(card_brand(payment_method) == 'check')
164
+ post[:payment] = 'check'
165
+ post[:firstname] = payment_method.first_name
166
+ post[:lastname] = payment_method.last_name
167
+ post[:checkname] = payment_method.name
168
+ post[:checkaba] = payment_method.routing_number
169
+ post[:checkaccount] = payment_method.account_number
170
+ post[:account_holder_type] = payment_method.account_holder_type
171
+ post[:account_type] = payment_method.account_type
172
+ post[:sec_code] = options[:sec_code] || 'WEB'
173
+ else
174
+ post[:payment] = 'creditcard'
175
+ post[:firstname] = payment_method.first_name
176
+ post[:lastname] = payment_method.last_name
177
+ post[:ccnumber] = payment_method.number
178
+ post[:cvv] = payment_method.verification_value unless empty?(payment_method.verification_value)
179
+ post[:ccexp] = exp_date(payment_method)
180
+ end
181
+ end
182
+
183
+ def add_stored_credential(post, options)
184
+ return unless (stored_credential = options[:stored_credential])
185
+
186
+ if stored_credential[:initiator] == 'cardholder'
187
+ post[:initiated_by] = 'customer'
188
+ else
189
+ post[:initiated_by] = 'merchant'
190
+ end
191
+
192
+ # :reason_type, when provided, overrides anything previously set in
193
+ # post[:billing_method] (see `add_invoice` and the :recurring) option
194
+ case stored_credential[:reason_type]
195
+ when 'recurring'
196
+ post[:billing_method] = 'recurring'
197
+ when 'installment'
198
+ post[:billing_method] = 'installment'
199
+ when 'unscheduled'
200
+ post.delete(:billing_method)
201
+ end
202
+
203
+ if stored_credential[:initial_transaction]
204
+ post[:stored_credential_indicator] = 'stored'
205
+ else
206
+ post[:stored_credential_indicator] = 'used'
207
+ post[:initial_transaction_id] = stored_credential[:network_transaction_id]
208
+ end
209
+ end
210
+
211
+ def add_customer_data(post, options)
212
+ post[:email] = options[:email]
213
+ post[:ipaddress] = options[:ip]
214
+ post[:customer_id] = options[:customer_id] || options[:customer]
215
+
216
+ if(billing_address = options[:billing_address] || options[:address])
217
+ post[:company] = billing_address[:company]
218
+ post[:address1] = billing_address[:address1]
219
+ post[:address2] = billing_address[:address2]
220
+ post[:city] = billing_address[:city]
221
+ post[:state] = billing_address[:state]
222
+ post[:country] = billing_address[:country]
223
+ post[:zip] = billing_address[:zip]
224
+ post[:phone] = billing_address[:phone]
225
+ end
226
+
227
+ if(shipping_address = options[:shipping_address])
228
+ post[:shipping_company] = shipping_address[:company]
229
+ post[:shipping_address1] = shipping_address[:address1]
230
+ post[:shipping_address2] = shipping_address[:address2]
231
+ post[:shipping_city] = shipping_address[:city]
232
+ post[:shipping_state] = shipping_address[:state]
233
+ post[:shipping_country] = shipping_address[:country]
234
+ post[:shipping_zip] = shipping_address[:zip]
235
+ post[:shipping_phone] = shipping_address[:phone]
236
+ end
237
+ end
238
+
239
+ def add_vendor_data(post, options)
240
+ post[:vendor_id] = options[:vendor_id] if options[:vendor_id]
241
+ post[:processor_id] = options[:processor_id] if options[:processor_id]
242
+ end
243
+
244
+ def add_merchant_defined_fields(post, options)
245
+ (1..20).each do |each|
246
+ key = "merchant_defined_field_#{each}".to_sym
247
+ post[key] = options[key] if options[key]
248
+ end
249
+ end
250
+
251
+ def add_reference(post, authorization)
252
+ transaction_id, _ = split_authorization(authorization)
253
+ post[:transactionid] = transaction_id
254
+ end
255
+
256
+ def add_payment_type(post, authorization)
257
+ _, payment_type = split_authorization(authorization)
258
+ post[:payment] = payment_type if payment_type
259
+ end
260
+
261
+ def exp_date(payment_method)
262
+ "#{format(payment_method.month, :two_digits)}#{format(payment_method.year, :two_digits)}"
263
+ end
264
+
265
+ def commit(action, params)
266
+ params[action == 'add_customer' ? :customer_vault : :type] = action
267
+ params[:username] = @options[:login]
268
+ params[:password] = @options[:password]
269
+
270
+ raw_response = ssl_post(url, post_data(action, params), headers)
271
+ response = parse(raw_response)
272
+ succeeded = success_from(response)
273
+
274
+ Response.new(
275
+ succeeded,
276
+ message_from(succeeded, response),
277
+ response,
278
+ authorization: authorization_from(response, params[:payment], action),
279
+ avs_result: AVSResult.new(code: response[:avsresponse]),
280
+ cvv_result: CVVResult.new(response[:cvvresponse]),
281
+ test: test?
282
+ )
283
+ end
284
+
285
+ def authorization_from(response, payment_type, action)
286
+ authorization = (action == 'add_customer' ? response[:customer_vault_id] : response[:transactionid])
287
+ [ authorization, payment_type ].join('#')
288
+ end
289
+
290
+ def split_authorization(authorization)
291
+ authorization.split('#')
292
+ end
293
+
294
+ def headers
295
+ { 'Content-Type' => 'application/x-www-form-urlencoded;charset=UTF-8' }
296
+ end
297
+
298
+ def post_data(action, params)
299
+ params.map { |k, v| "#{k}=#{CGI.escape(v.to_s)}" }.join('&')
300
+ end
301
+
302
+ def url
303
+ test? ? test_url : live_url
304
+ end
305
+
306
+ def parse(body)
307
+ Hash[CGI::parse(body).map { |k, v| [k.intern, v.first] }]
308
+ end
309
+
310
+ def success_from(response)
311
+ response[:response] == '1'
312
+ end
313
+
314
+ def message_from(succeeded, response)
315
+ if succeeded
316
+ 'Succeeded'
317
+ else
318
+ response[:responsetext]
319
+ end
320
+ end
321
+
322
+ end
323
+ end
324
+ end
@@ -0,0 +1,478 @@
1
+ # coding: utf-8
2
+
3
+ require 'rexml/document'
4
+
5
+ module ActiveMerchant #:nodoc:
6
+ module Billing #:nodoc:
7
+ # = Ogone DirectLink Gateway
8
+ #
9
+ # DirectLink is the API version of the Ogone Payment Platform. It allows server to server
10
+ # communication between Ogone systems and your e-commerce website.
11
+ #
12
+ # This implementation follows the specification provided in the DirectLink integration
13
+ # guide version 4.3.0 (25 April 2012), available here:
14
+ # https://secure.ogone.com/ncol/Ogone_DirectLink_EN.pdf
15
+ #
16
+ # It also features aliases, which allow to store/unstore credit cards, as specified in
17
+ # the Alias Manager Option guide version 3.2.1 (25 April 2012) available here:
18
+ # https://secure.ogone.com/ncol/Ogone_Alias_EN.pdf
19
+ #
20
+ # It also implements the 3-D Secure feature, as specified in the DirectLink with
21
+ # 3-D Secure guide version 3.0 (25 April 2012) available here:
22
+ # https://secure.ogone.com/ncol/Ogone_DirectLink-3-D_EN.pdf
23
+ #
24
+ # It was last tested on Release 4.92 of Ogone DirectLink + AliasManager + Direct Link 3D
25
+ # (25 April 2012).
26
+ #
27
+ # For any questions or comments, please contact one of the following:
28
+ # - Joel Cogen (joel.cogen@belighted.com)
29
+ # - Nicolas Jacobeus (nicolas.jacobeus@belighted.com),
30
+ # - Sébastien Grosjean (public@zencocoon.com),
31
+ # - Rémy Coutable (remy@jilion.com).
32
+ #
33
+ # == Usage
34
+ #
35
+ # gateway = ActiveMerchant::Billing::OgoneGateway.new(
36
+ # :login => "my_ogone_psp_id",
37
+ # :user => "my_ogone_user_id",
38
+ # :password => "my_ogone_pswd",
39
+ # :signature => "my_ogone_sha_signature", # Only if you configured your Ogone environment so.
40
+ # :signature_encryptor => "sha512" # Can be "none" (default), "sha1", "sha256" or "sha512".
41
+ # # Must be the same as the one configured in your Ogone account.
42
+ # )
43
+ #
44
+ # # set up credit card object as in main ActiveMerchant example
45
+ # creditcard = ActiveMerchant::Billing::CreditCard.new(
46
+ # :type => 'visa',
47
+ # :number => '4242424242424242',
48
+ # :month => 8,
49
+ # :year => 2009,
50
+ # :first_name => 'Bob',
51
+ # :last_name => 'Bobsen'
52
+ # )
53
+ #
54
+ # # run request
55
+ # response = gateway.purchase(1000, creditcard, :order_id => "1") # charge 10 EUR
56
+ #
57
+ # If you don't provide an :order_id, the gateway will generate a random one for you.
58
+ #
59
+ # puts response.success? # Check whether the transaction was successful
60
+ # puts response.message # Retrieve the message returned by Ogone
61
+ # puts response.authorization # Retrieve the unique transaction ID returned by Ogone
62
+ # puts response.order_id # Retrieve the order ID
63
+ #
64
+ # == Alias feature
65
+ #
66
+ # To use the alias feature, simply add :billing_id in the options hash:
67
+ #
68
+ # # Associate the alias to that credit card
69
+ # gateway.purchase(1000, creditcard, :order_id => "1", :billing_id => "myawesomecustomer")
70
+ #
71
+ # # You can use the alias instead of the credit card for subsequent orders
72
+ # gateway.purchase(2000, "myawesomecustomer", :order_id => "2")
73
+ #
74
+ # # You can also create an alias without making a purchase using store
75
+ # gateway.store(creditcard, :billing_id => "myawesomecustomer")
76
+ #
77
+ # # When using store, you can also let Ogone generate the alias for you
78
+ # response = gateway.store(creditcard)
79
+ # puts response.billing_id # Retrieve the generated alias
80
+ #
81
+ # # By default, Ogone tries to authorize 0.01 EUR but you can change this
82
+ # # amount using the :store_amount option when creating the gateway object:
83
+ # gateway = ActiveMerchant::Billing::OgoneGateway.new(
84
+ # :login => "my_ogone_psp_id",
85
+ # :user => "my_ogone_user_id",
86
+ # :password => "my_ogone_pswd",
87
+ # :signature => "my_ogone_sha_signature",
88
+ # :signature_encryptor => "sha512",
89
+ # :store_amount => 100 # The store method will try to authorize 1 EUR instead of 0.01 EUR
90
+ # )
91
+ # response = gateway.store(creditcard) # authorize 1 EUR and void the authorization right away
92
+ #
93
+ # == 3-D Secure feature
94
+ #
95
+ # To use the 3-D Secure feature, simply add :d3d => true in the options hash:
96
+ # gateway.purchase(2000, "myawesomecustomer", :order_id => "2", :d3d => true)
97
+ #
98
+ # Specific 3-D Secure request options are (please refer to the documentation for more infos about these options):
99
+ # :win_3ds => :main_window (default), :pop_up or :pop_ix.
100
+ # :http_accept => "*/*" (default), or any other HTTP_ACCEPT header value.
101
+ # :http_user_agent => The cardholder's User-Agent string
102
+ # :accept_url => URL of the web page to show the customer when the payment is authorized.
103
+ # (or waiting to be authorized).
104
+ # :decline_url => URL of the web page to show the customer when the acquirer rejects the authorization
105
+ # more than the maximum permitted number of authorization attempts (10 by default, but can
106
+ # be changed in the "Global transaction parameters" tab, "Payment retry" section of the
107
+ # Technical Information page).
108
+ # :exception_url => URL of the web page to show the customer when the payment result is uncertain.
109
+ # :paramplus => Field to submit the miscellaneous parameters and their values that you wish to be
110
+ # returned in the post sale request or final redirection.
111
+ # :complus => Field to submit a value you wish to be returned in the post sale request or output.
112
+ # :language => Customer's language, for example: "en_EN"
113
+ #
114
+ class OgoneGateway < Gateway
115
+ CVV_MAPPING = { 'OK' => 'M',
116
+ 'KO' => 'N',
117
+ 'NO' => 'P' }
118
+
119
+ AVS_MAPPING = { 'OK' => 'M',
120
+ 'KO' => 'N',
121
+ 'NO' => 'R' }
122
+
123
+ SUCCESS_MESSAGE = 'The transaction was successful'
124
+
125
+ THREE_D_SECURE_DISPLAY_WAYS = { :main_window => 'MAINW', # display the identification page in the main window (default value).
126
+
127
+ :pop_up => 'POPUP', # display the identification page in a pop-up window and return to the main window at the end.
128
+ :pop_ix => 'POPIX' } # display the identification page in a pop-up window and remain in the pop-up window.
129
+
130
+ OGONE_NO_SIGNATURE_DEPRECATION_MESSAGE = 'Signature usage will be the default for a future release of ActiveMerchant. You should either begin using it, or update your configuration to explicitly disable it (signature_encryptor: none)'
131
+ OGONE_STORE_OPTION_DEPRECATION_MESSAGE = "The 'store' option has been renamed to 'billing_id', and its usage is deprecated."
132
+
133
+ self.test_url = 'https://secure.ogone.com/ncol/test/'
134
+ self.live_url = 'https://secure.ogone.com/ncol/prod/'
135
+
136
+ self.supported_countries = ['BE', 'DE', 'FR', 'NL', 'AT', 'CH']
137
+ # also supports Airplus and UATP
138
+ self.supported_cardtypes = [:visa, :master, :american_express, :diners_club, :discover, :jcb, :maestro]
139
+ self.homepage_url = 'http://www.ogone.com/'
140
+ self.display_name = 'Ogone'
141
+ self.default_currency = 'EUR'
142
+ self.money_format = :cents
143
+
144
+ def initialize(options = {})
145
+ requires!(options, :login, :user, :password)
146
+ super
147
+ end
148
+
149
+ # Verify and reserve the specified amount on the account, without actually doing the transaction.
150
+ def authorize(money, payment_source, options = {})
151
+ post = {}
152
+ action = payment_source.brand == 'mastercard' ? 'PAU' : 'RES'
153
+ add_invoice(post, options)
154
+ add_payment_source(post, payment_source, options)
155
+ add_address(post, payment_source, options)
156
+ add_customer_data(post, options)
157
+ add_money(post, money, options)
158
+ commit(action, post)
159
+ end
160
+
161
+ # Verify and transfer the specified amount.
162
+ def purchase(money, payment_source, options = {})
163
+ post = {}
164
+ action = options[:action] || 'SAL'
165
+ add_invoice(post, options)
166
+ add_payment_source(post, payment_source, options)
167
+ add_address(post, payment_source, options)
168
+ add_customer_data(post, options)
169
+ add_money(post, money, options)
170
+ commit(action, post)
171
+ end
172
+
173
+ # Complete a previously authorized transaction.
174
+ def capture(money, authorization, options = {})
175
+ post = {}
176
+ action = options[:action] || 'SAL'
177
+ add_authorization(post, reference_from(authorization))
178
+ add_invoice(post, options)
179
+ add_customer_data(post, options)
180
+ add_money(post, money, options)
181
+ commit(action, post)
182
+ end
183
+
184
+ # Cancels a previously authorized transaction.
185
+ def void(identification, options = {})
186
+ post = {}
187
+ add_authorization(post, reference_from(identification))
188
+ commit('DES', post)
189
+ end
190
+
191
+ # Credit the specified account by a specific amount.
192
+ def credit(money, identification_or_credit_card, options = {})
193
+ if reference_transaction?(identification_or_credit_card)
194
+ ActiveMerchant.deprecated CREDIT_DEPRECATION_MESSAGE
195
+ # Referenced credit: refund of a settled transaction
196
+ refund(money, identification_or_credit_card, options)
197
+ else # must be a credit card or card reference
198
+ perform_non_referenced_credit(money, identification_or_credit_card, options)
199
+ end
200
+ end
201
+
202
+ # Refund of a settled transaction
203
+ def refund(money, reference, options = {})
204
+ perform_reference_credit(money, reference, options)
205
+ end
206
+
207
+ def verify(credit_card, options={})
208
+ MultiResponse.run(:use_first_response) do |r|
209
+ r.process { authorize(100, credit_card, options) }
210
+ r.process(:ignore_result) { void(r.authorization, options) }
211
+ end
212
+ end
213
+
214
+ # Store a credit card by creating an Ogone Alias
215
+ def store(payment_source, options = {})
216
+ options[:alias_operation] = 'BYPSP' unless(options.has_key?(:billing_id) || options.has_key?(:store))
217
+ response = authorize(@options[:store_amount] || 1, payment_source, options)
218
+ void(response.authorization) if response.success?
219
+ response
220
+ end
221
+
222
+ def supports_scrubbing?
223
+ true
224
+ end
225
+
226
+ def scrub(transcript)
227
+ transcript.
228
+ gsub(%r((Authorization: Basic )\w+), '\1[FILTERED]').
229
+ gsub(%r((&?cardno=)[^&]*)i, '\1[FILTERED]').
230
+ gsub(%r((&?cvc=)[^&]*)i, '\1[FILTERED]').
231
+ gsub(%r((&?pswd=)[^&]*)i, '\1[FILTERED]')
232
+ end
233
+
234
+ private
235
+
236
+ def reference_from(authorization)
237
+ authorization.split(';').first
238
+ end
239
+
240
+ def reference_transaction?(identifier)
241
+ return false unless identifier.is_a?(String)
242
+ _, action = identifier.split(';')
243
+ !action.nil?
244
+ end
245
+
246
+ def perform_reference_credit(money, payment_target, options = {})
247
+ post = {}
248
+ add_authorization(post, reference_from(payment_target))
249
+ add_money(post, money, options)
250
+ commit('RFD', post)
251
+ end
252
+
253
+ def perform_non_referenced_credit(money, payment_target, options = {})
254
+ # Non-referenced credit: acts like a reverse purchase
255
+ post = {}
256
+ add_invoice(post, options)
257
+ add_payment_source(post, payment_target, options)
258
+ add_address(post, payment_target, options)
259
+ add_customer_data(post, options)
260
+ add_money(post, money, options)
261
+ commit('RFD', post)
262
+ end
263
+
264
+ def add_payment_source(post, payment_source, options)
265
+ add_d3d(post, options) if options[:d3d]
266
+
267
+ if payment_source.is_a?(String)
268
+ add_alias(post, payment_source, options[:alias_operation])
269
+ add_eci(post, options[:eci] || '9')
270
+ else
271
+ if options.has_key?(:store)
272
+ ActiveMerchant.deprecated OGONE_STORE_OPTION_DEPRECATION_MESSAGE
273
+ options[:billing_id] ||= options[:store]
274
+ end
275
+ add_alias(post, options[:billing_id], options[:alias_operation])
276
+ add_eci(post, options[:eci] || '7')
277
+ add_creditcard(post, payment_source)
278
+ end
279
+ end
280
+
281
+ def add_d3d(post, options)
282
+ add_pair post, 'FLAG3D', 'Y'
283
+ win_3ds = THREE_D_SECURE_DISPLAY_WAYS.key?(options[:win_3ds]) ?
284
+ THREE_D_SECURE_DISPLAY_WAYS[options[:win_3ds]] :
285
+ THREE_D_SECURE_DISPLAY_WAYS[:main_window]
286
+ add_pair post, 'WIN3DS', win_3ds
287
+
288
+ add_pair post, 'HTTP_ACCEPT', options[:http_accept] || '*/*'
289
+ add_pair post, 'HTTP_USER_AGENT', options[:http_user_agent] if options[:http_user_agent]
290
+ add_pair post, 'ACCEPTURL', options[:accept_url] if options[:accept_url]
291
+ add_pair post, 'DECLINEURL', options[:decline_url] if options[:decline_url]
292
+ add_pair post, 'EXCEPTIONURL', options[:exception_url] if options[:exception_url]
293
+ add_pair post, 'CANCELURL', options[:cancel_url] if options[:cancel_url]
294
+ add_pair post, 'PARAMVAR', options[:paramvar] if options[:paramvar]
295
+ add_pair post, 'PARAMPLUS', options[:paramplus] if options[:paramplus]
296
+ add_pair post, 'COMPLUS', options[:complus] if options[:complus]
297
+ add_pair post, 'LANGUAGE', options[:language] if options[:language]
298
+ end
299
+
300
+ def add_eci(post, eci)
301
+ add_pair post, 'ECI', eci.to_s
302
+ end
303
+
304
+ def add_alias(post, alias_name, alias_operation = nil)
305
+ add_pair post, 'ALIAS', alias_name
306
+ add_pair post, 'ALIASOPERATION', alias_operation unless alias_operation.nil?
307
+ end
308
+
309
+ def add_authorization(post, authorization)
310
+ add_pair post, 'PAYID', authorization
311
+ end
312
+
313
+ def add_money(post, money, options)
314
+ add_pair post, 'currency', options[:currency] || @options[:currency] || currency(money)
315
+ add_pair post, 'amount', amount(money)
316
+ end
317
+
318
+ def add_customer_data(post, options)
319
+ add_pair post, 'EMAIL', options[:email]
320
+ add_pair post, 'REMOTE_ADDR', options[:ip]
321
+ end
322
+
323
+ def add_address(post, creditcard, options)
324
+ return unless options[:billing_address]
325
+ add_pair post, 'Owneraddress', options[:billing_address][:address1]
326
+ add_pair post, 'OwnerZip', options[:billing_address][:zip]
327
+ add_pair post, 'ownertown', options[:billing_address][:city]
328
+ add_pair post, 'ownercty', options[:billing_address][:country]
329
+ add_pair post, 'ownertelno', options[:billing_address][:phone]
330
+ end
331
+
332
+ def add_invoice(post, options)
333
+ add_pair post, 'orderID', options[:order_id] || generate_unique_id[0...30]
334
+ add_pair post, 'COM', options[:description]
335
+ add_pair post, 'ORIG', options[:origin] if options[:origin]
336
+ end
337
+
338
+ def add_creditcard(post, creditcard)
339
+ add_pair post, 'CN', creditcard.name
340
+ add_pair post, 'CARDNO', creditcard.number
341
+ add_pair post, 'ED', '%02d%02s' % [creditcard.month, creditcard.year.to_s[-2..-1]]
342
+ add_pair post, 'CVC', creditcard.verification_value
343
+ end
344
+
345
+ def parse(body)
346
+ xml_root = REXML::Document.new(body).root
347
+ response = convert_attributes_to_hash(xml_root.attributes)
348
+
349
+ # Add HTML_ANSWER element (3-D Secure specific to the response's params)
350
+ # Note: HTML_ANSWER is not an attribute so we add it "by hand" to the response
351
+ if html_answer = REXML::XPath.first(xml_root, '//HTML_ANSWER')
352
+ response['HTML_ANSWER'] = html_answer.text
353
+ end
354
+
355
+ response
356
+ end
357
+
358
+ def commit(action, parameters)
359
+ add_pair parameters, 'RTIMEOUT', @options[:timeout] if @options[:timeout]
360
+ add_pair parameters, 'PSPID', @options[:login]
361
+ add_pair parameters, 'USERID', @options[:user]
362
+ add_pair parameters, 'PSWD', @options[:password]
363
+
364
+ response = parse(ssl_post(url(parameters['PAYID']), post_data(action, parameters)))
365
+
366
+ options = {
367
+ :authorization => [response['PAYID'], action].join(';'),
368
+ :test => test?,
369
+ :avs_result => { :code => AVS_MAPPING[response['AAVCheck']] },
370
+ :cvv_result => CVV_MAPPING[response['CVCCheck']]
371
+ }
372
+ OgoneResponse.new(successful?(response), message_from(response), response, options)
373
+ end
374
+
375
+ def url(payid)
376
+ (test? ? test_url : live_url) + (payid ? 'maintenancedirect.asp' : 'orderdirect.asp')
377
+ end
378
+
379
+ def successful?(response)
380
+ response['NCERROR'] == '0'
381
+ end
382
+
383
+ def message_from(response)
384
+ if successful?(response)
385
+ SUCCESS_MESSAGE
386
+ else
387
+ format_error_message(response['NCERRORPLUS'])
388
+ end
389
+ end
390
+
391
+ def format_error_message(message)
392
+ raw_message = message.to_s.strip
393
+ case raw_message
394
+ when /\|/
395
+ raw_message.split('|').join(', ').capitalize
396
+ when /\//
397
+ raw_message.split('/').first.to_s.capitalize
398
+ else
399
+ raw_message.to_s.capitalize
400
+ end
401
+ end
402
+
403
+ def post_data(action, parameters = {})
404
+ add_pair parameters, 'Operation', action
405
+ add_signature(parameters)
406
+ parameters.to_query
407
+ end
408
+
409
+ def add_signature(parameters)
410
+ if @options[:signature].blank?
411
+ ActiveMerchant.deprecated(OGONE_NO_SIGNATURE_DEPRECATION_MESSAGE) unless(@options[:signature_encryptor] == 'none')
412
+ return
413
+ end
414
+
415
+ add_pair parameters, 'SHASign', calculate_signature(parameters, @options[:signature_encryptor], @options[:signature])
416
+ end
417
+
418
+ def calculate_signature(signed_parameters, algorithm, secret)
419
+ return legacy_calculate_signature(signed_parameters, secret) unless algorithm
420
+
421
+ sha_encryptor = case algorithm
422
+ when 'sha256'
423
+ Digest::SHA256
424
+ when 'sha512'
425
+ Digest::SHA512
426
+ when 'sha1'
427
+ Digest::SHA1
428
+ else
429
+ raise "Unknown signature algorithm #{algorithm}"
430
+ end
431
+
432
+ filtered_params = signed_parameters.select { |k, v| !v.blank? }
433
+ sha_encryptor.hexdigest(
434
+ filtered_params.sort_by { |k, v| k.upcase }.map { |k, v| "#{k.upcase}=#{v}#{secret}" }.join('')
435
+ ).upcase
436
+ end
437
+
438
+ def legacy_calculate_signature(parameters, secret)
439
+ Digest::SHA1.hexdigest(
440
+ (
441
+ %w(
442
+ orderID
443
+ amount
444
+ currency
445
+ CARDNO
446
+ PSPID
447
+ Operation
448
+ ALIAS
449
+ ).map { |key| parameters[key] } +
450
+ [secret]
451
+ ).join('')
452
+ ).upcase
453
+ end
454
+
455
+ def add_pair(post, key, value)
456
+ post[key] = value if !value.blank?
457
+ end
458
+
459
+ def convert_attributes_to_hash(rexml_attributes)
460
+ response_hash = {}
461
+ rexml_attributes.each do |key, value|
462
+ response_hash[key] = value
463
+ end
464
+ response_hash
465
+ end
466
+ end
467
+
468
+ class OgoneResponse < Response
469
+ def order_id
470
+ @params['orderID']
471
+ end
472
+
473
+ def billing_id
474
+ @params['ALIAS']
475
+ end
476
+ end
477
+ end
478
+ end