activemerchant 1.45.0 → 1.126.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (288) hide show
  1. checksums.yaml +5 -5
  2. data/CHANGELOG +2167 -2
  3. data/CONTRIBUTORS +57 -0
  4. data/README.md +62 -41
  5. data/lib/active_merchant/billing/avs_result.rb +25 -28
  6. data/lib/active_merchant/billing/base.rb +11 -35
  7. data/lib/active_merchant/billing/check.rb +28 -24
  8. data/lib/active_merchant/billing/compatibility.rb +10 -12
  9. data/lib/active_merchant/billing/credit_card.rb +141 -69
  10. data/lib/active_merchant/billing/credit_card_formatting.rb +4 -4
  11. data/lib/active_merchant/billing/credit_card_methods.rb +348 -52
  12. data/lib/active_merchant/billing/cvv_result.rb +0 -1
  13. data/lib/active_merchant/billing/gateway.rb +125 -61
  14. data/lib/active_merchant/billing/gateways/adyen.rb +633 -127
  15. data/lib/active_merchant/billing/gateways/airwallex.rb +341 -0
  16. data/lib/active_merchant/billing/gateways/allied_wallet.rb +205 -0
  17. data/lib/active_merchant/billing/gateways/authorize_net.rb +883 -195
  18. data/lib/active_merchant/billing/gateways/authorize_net_arb.rb +31 -24
  19. data/lib/active_merchant/billing/gateways/authorize_net_cim.rb +104 -103
  20. data/lib/active_merchant/billing/gateways/axcessms.rb +179 -0
  21. data/lib/active_merchant/billing/gateways/balanced.rb +72 -65
  22. data/lib/active_merchant/billing/gateways/bambora_apac.rb +222 -0
  23. data/lib/active_merchant/billing/gateways/bank_frick.rb +16 -16
  24. data/lib/active_merchant/billing/gateways/banwire.rb +24 -13
  25. data/lib/active_merchant/billing/gateways/barclaycard_smartpay.rb +397 -0
  26. data/lib/active_merchant/billing/gateways/barclays_epdq_extra_plus.rb +7 -7
  27. data/lib/active_merchant/billing/gateways/be2bill.rb +7 -7
  28. data/lib/active_merchant/billing/gateways/beanstream/beanstream_core.rb +143 -62
  29. data/lib/active_merchant/billing/gateways/beanstream.rb +45 -13
  30. data/lib/active_merchant/billing/gateways/beanstream_interac.rb +13 -10
  31. data/lib/active_merchant/billing/gateways/blue_pay.rb +549 -506
  32. data/lib/active_merchant/billing/gateways/blue_snap.rb +623 -0
  33. data/lib/active_merchant/billing/gateways/bogus.rb +82 -36
  34. data/lib/active_merchant/billing/gateways/borgun.rb +61 -40
  35. data/lib/active_merchant/billing/gateways/bpoint.rb +277 -0
  36. data/lib/active_merchant/billing/gateways/braintree/braintree_common.rb +21 -2
  37. data/lib/active_merchant/billing/gateways/braintree/token_nonce.rb +113 -0
  38. data/lib/active_merchant/billing/gateways/braintree.rb +3 -3
  39. data/lib/active_merchant/billing/gateways/braintree_blue.rb +592 -203
  40. data/lib/active_merchant/billing/gateways/braintree_orange.rb +3 -4
  41. data/lib/active_merchant/billing/gateways/bridge_pay.rb +116 -61
  42. data/lib/active_merchant/billing/gateways/cams.rb +230 -0
  43. data/lib/active_merchant/billing/gateways/card_connect.rb +320 -0
  44. data/lib/active_merchant/billing/gateways/card_save.rb +7 -9
  45. data/lib/active_merchant/billing/gateways/card_stream.rb +248 -97
  46. data/lib/active_merchant/billing/gateways/cardknox.rb +327 -0
  47. data/lib/active_merchant/billing/gateways/cardprocess.rb +256 -0
  48. data/lib/active_merchant/billing/gateways/cashnet.rb +70 -26
  49. data/lib/active_merchant/billing/gateways/cc5.rb +14 -17
  50. data/lib/active_merchant/billing/gateways/cecabank.rb +73 -53
  51. data/lib/active_merchant/billing/gateways/cenpos.rb +328 -0
  52. data/lib/active_merchant/billing/gateways/checkout.rb +32 -33
  53. data/lib/active_merchant/billing/gateways/checkout_v2.rb +340 -0
  54. data/lib/active_merchant/billing/gateways/citrus_pay.rb +21 -0
  55. data/lib/active_merchant/billing/gateways/clearhaus.rb +219 -0
  56. data/lib/active_merchant/billing/gateways/commercegate.rb +11 -12
  57. data/lib/active_merchant/billing/gateways/conekta.rb +55 -34
  58. data/lib/active_merchant/billing/gateways/creditcall.rb +272 -0
  59. data/lib/active_merchant/billing/gateways/credorax.rb +506 -0
  60. data/lib/active_merchant/billing/gateways/ct_payment.rb +269 -0
  61. data/lib/active_merchant/billing/gateways/culqi.rb +279 -0
  62. data/lib/active_merchant/billing/gateways/cyber_source.rb +637 -221
  63. data/lib/active_merchant/billing/gateways/d_local.rb +303 -0
  64. data/lib/active_merchant/billing/gateways/data_cash.rb +57 -355
  65. data/lib/active_merchant/billing/gateways/decidir.rb +347 -0
  66. data/lib/active_merchant/billing/gateways/decidir_plus.rb +344 -0
  67. data/lib/active_merchant/billing/gateways/dibs.rb +199 -0
  68. data/lib/active_merchant/billing/gateways/digitzs.rb +295 -0
  69. data/lib/active_merchant/billing/gateways/ebanx.rb +334 -0
  70. data/lib/active_merchant/billing/gateways/efsnet.rb +46 -50
  71. data/lib/active_merchant/billing/gateways/elavon.rb +377 -250
  72. data/lib/active_merchant/billing/gateways/element.rb +386 -0
  73. data/lib/active_merchant/billing/gateways/epay.rb +92 -71
  74. data/lib/active_merchant/billing/gateways/evo_ca.rb +13 -14
  75. data/lib/active_merchant/billing/gateways/eway.rb +85 -73
  76. data/lib/active_merchant/billing/gateways/eway_managed.rb +85 -87
  77. data/lib/active_merchant/billing/gateways/eway_rapid.rb +134 -80
  78. data/lib/active_merchant/billing/gateways/exact.rb +37 -36
  79. data/lib/active_merchant/billing/gateways/ezic.rb +195 -0
  80. data/lib/active_merchant/billing/gateways/fat_zebra.rb +124 -74
  81. data/lib/active_merchant/billing/gateways/federated_canada.rb +11 -13
  82. data/lib/active_merchant/billing/gateways/finansbank.rb +4 -5
  83. data/lib/active_merchant/billing/gateways/first_giving.rb +23 -23
  84. data/lib/active_merchant/billing/gateways/first_pay.rb +45 -23
  85. data/lib/active_merchant/billing/gateways/firstdata_e4.rb +184 -87
  86. data/lib/active_merchant/billing/gateways/firstdata_e4_v27.rb +505 -0
  87. data/lib/active_merchant/billing/gateways/flo2cash.rb +215 -0
  88. data/lib/active_merchant/billing/gateways/flo2cash_simple.rb +20 -0
  89. data/lib/active_merchant/billing/gateways/forte.rb +286 -0
  90. data/lib/active_merchant/billing/gateways/garanti.rb +31 -32
  91. data/lib/active_merchant/billing/gateways/global_collect.rb +507 -0
  92. data/lib/active_merchant/billing/gateways/global_transport.rb +25 -15
  93. data/lib/active_merchant/billing/gateways/hdfc.rb +55 -57
  94. data/lib/active_merchant/billing/gateways/hps.rb +252 -68
  95. data/lib/active_merchant/billing/gateways/iats_payments.rb +99 -38
  96. data/lib/active_merchant/billing/gateways/in_context_paypal_express.rb +15 -0
  97. data/lib/active_merchant/billing/gateways/inspire.rb +48 -48
  98. data/lib/active_merchant/billing/gateways/instapay.rb +10 -14
  99. data/lib/active_merchant/billing/gateways/ipg.rb +415 -0
  100. data/lib/active_merchant/billing/gateways/ipp.rb +176 -0
  101. data/lib/active_merchant/billing/gateways/iridium.rb +254 -244
  102. data/lib/active_merchant/billing/gateways/itransact.rb +14 -14
  103. data/lib/active_merchant/billing/gateways/iveri.rb +254 -0
  104. data/lib/active_merchant/billing/gateways/ixopay.rb +320 -0
  105. data/lib/active_merchant/billing/gateways/jetpay.rb +200 -80
  106. data/lib/active_merchant/billing/gateways/jetpay_v2.rb +432 -0
  107. data/lib/active_merchant/billing/gateways/komoju.rb +115 -0
  108. data/lib/active_merchant/billing/gateways/kushki.rb +277 -0
  109. data/lib/active_merchant/billing/gateways/latitude19.rb +412 -0
  110. data/lib/active_merchant/billing/gateways/linkpoint.rb +95 -85
  111. data/lib/active_merchant/billing/gateways/litle.rb +371 -94
  112. data/lib/active_merchant/billing/gateways/mastercard.rb +293 -0
  113. data/lib/active_merchant/billing/gateways/maxipago.rb +146 -123
  114. data/lib/active_merchant/billing/gateways/mercado_pago.rb +331 -0
  115. data/lib/active_merchant/billing/gateways/merchant_e_solutions.rb +47 -23
  116. data/lib/active_merchant/billing/gateways/merchant_one.rb +16 -20
  117. data/lib/active_merchant/billing/gateways/merchant_partners.rb +245 -0
  118. data/lib/active_merchant/billing/gateways/merchant_ware.rb +54 -60
  119. data/lib/active_merchant/billing/gateways/merchant_ware_version_four.rb +60 -44
  120. data/lib/active_merchant/billing/gateways/merchant_warrior.rb +76 -23
  121. data/lib/active_merchant/billing/gateways/mercury.rb +73 -54
  122. data/lib/active_merchant/billing/gateways/metrics_global.rb +33 -43
  123. data/lib/active_merchant/billing/gateways/micropayment.rb +182 -0
  124. data/lib/active_merchant/billing/gateways/migs/migs_codes.rb +7 -7
  125. data/lib/active_merchant/billing/gateways/migs.rb +101 -37
  126. data/lib/active_merchant/billing/gateways/mit.rb +260 -0
  127. data/lib/active_merchant/billing/gateways/modern_payments.rb +3 -3
  128. data/lib/active_merchant/billing/gateways/modern_payments_cim.rb +30 -34
  129. data/lib/active_merchant/billing/gateways/moka.rb +290 -0
  130. data/lib/active_merchant/billing/gateways/monei.rb +422 -0
  131. data/lib/active_merchant/billing/gateways/moneris.rb +236 -72
  132. data/lib/active_merchant/billing/gateways/money_movers.rb +11 -13
  133. data/lib/active_merchant/billing/gateways/mundipagg.rb +363 -0
  134. data/lib/active_merchant/billing/gateways/nab_transact.rb +87 -68
  135. data/lib/active_merchant/billing/gateways/ncr_secure_pay.rb +163 -0
  136. data/lib/active_merchant/billing/gateways/net_registry.rb +12 -12
  137. data/lib/active_merchant/billing/gateways/netaxept.rb +33 -34
  138. data/lib/active_merchant/billing/gateways/netbanx.rb +376 -0
  139. data/lib/active_merchant/billing/gateways/netbilling.rb +82 -43
  140. data/lib/active_merchant/billing/gateways/netpay.rb +10 -10
  141. data/lib/active_merchant/billing/gateways/network_merchants.rb +7 -11
  142. data/lib/active_merchant/billing/gateways/nmi.rb +279 -174
  143. data/lib/active_merchant/billing/gateways/ogone.rb +96 -50
  144. data/lib/active_merchant/billing/gateways/omise.rb +323 -0
  145. data/lib/active_merchant/billing/gateways/openpay.rb +58 -23
  146. data/lib/active_merchant/billing/gateways/opp.rb +394 -0
  147. data/lib/active_merchant/billing/gateways/optimal_payment.rb +87 -69
  148. data/lib/active_merchant/billing/gateways/orbital/orbital_soft_descriptors.rb +5 -7
  149. data/lib/active_merchant/billing/gateways/orbital.rb +771 -328
  150. data/lib/active_merchant/billing/gateways/pac_net_raven.rb +31 -32
  151. data/lib/active_merchant/billing/gateways/pagarme.rb +239 -0
  152. data/lib/active_merchant/billing/gateways/pago_facil.rb +12 -14
  153. data/lib/active_merchant/billing/gateways/pay_arc.rb +392 -0
  154. data/lib/active_merchant/billing/gateways/pay_conex.rb +245 -0
  155. data/lib/active_merchant/billing/gateways/pay_gate_xml.rb +84 -68
  156. data/lib/active_merchant/billing/gateways/pay_hub.rb +213 -0
  157. data/lib/active_merchant/billing/gateways/pay_junction.rb +37 -37
  158. data/lib/active_merchant/billing/gateways/pay_junction_v2.rb +206 -0
  159. data/lib/active_merchant/billing/gateways/pay_secure.rb +15 -17
  160. data/lib/active_merchant/billing/gateways/pay_trace.rb +404 -0
  161. data/lib/active_merchant/billing/gateways/paybox_direct.rb +85 -49
  162. data/lib/active_merchant/billing/gateways/payeezy.rb +458 -0
  163. data/lib/active_merchant/billing/gateways/payex.rb +39 -42
  164. data/lib/active_merchant/billing/gateways/payflow/payflow_common_api.rb +67 -41
  165. data/lib/active_merchant/billing/gateways/payflow/payflow_express_response.rb +11 -8
  166. data/lib/active_merchant/billing/gateways/payflow/payflow_response.rb +3 -3
  167. data/lib/active_merchant/billing/gateways/payflow.rb +215 -46
  168. data/lib/active_merchant/billing/gateways/payflow_express.rb +64 -68
  169. data/lib/active_merchant/billing/gateways/payflow_express_uk.rb +2 -3
  170. data/lib/active_merchant/billing/gateways/payflow_uk.rb +5 -6
  171. data/lib/active_merchant/billing/gateways/payment_express.rb +78 -58
  172. data/lib/active_merchant/billing/gateways/paymentez.rb +353 -0
  173. data/lib/active_merchant/billing/gateways/paymill.rb +163 -75
  174. data/lib/active_merchant/billing/gateways/paypal/paypal_common_api.rb +55 -7
  175. data/lib/active_merchant/billing/gateways/paypal/paypal_recurring_api.rb +1 -1
  176. data/lib/active_merchant/billing/gateways/paypal.rb +37 -18
  177. data/lib/active_merchant/billing/gateways/paypal_ca.rb +2 -2
  178. data/lib/active_merchant/billing/gateways/paypal_digital_goods.rb +5 -5
  179. data/lib/active_merchant/billing/gateways/paypal_express.rb +12 -6
  180. data/lib/active_merchant/billing/gateways/paypal_express_common.rb +3 -3
  181. data/lib/active_merchant/billing/gateways/paysafe.rb +412 -0
  182. data/lib/active_merchant/billing/gateways/payscout.rb +10 -13
  183. data/lib/active_merchant/billing/gateways/paystation.rb +96 -91
  184. data/lib/active_merchant/billing/gateways/payu_in.rb +249 -0
  185. data/lib/active_merchant/billing/gateways/payu_latam.rb +479 -0
  186. data/lib/active_merchant/billing/gateways/payway.rb +27 -27
  187. data/lib/active_merchant/billing/gateways/payway_dot_com.rb +253 -0
  188. data/lib/active_merchant/billing/gateways/pin.rb +113 -37
  189. data/lib/active_merchant/billing/gateways/plugnpay.rb +82 -82
  190. data/lib/active_merchant/billing/gateways/priority.rb +369 -0
  191. data/lib/active_merchant/billing/gateways/pro_pay.rb +325 -0
  192. data/lib/active_merchant/billing/gateways/psigate.rb +55 -44
  193. data/lib/active_merchant/billing/gateways/psl_card.rb +32 -40
  194. data/lib/active_merchant/billing/gateways/qbms.rb +86 -76
  195. data/lib/active_merchant/billing/gateways/quantum.rb +21 -23
  196. data/lib/active_merchant/billing/gateways/quickbooks.rb +377 -0
  197. data/lib/active_merchant/billing/gateways/quickpay/quickpay_common.rb +184 -0
  198. data/lib/active_merchant/billing/gateways/quickpay/quickpay_v10.rb +297 -0
  199. data/lib/active_merchant/billing/gateways/quickpay/quickpay_v4to7.rb +226 -0
  200. data/lib/active_merchant/billing/gateways/quickpay.rb +10 -353
  201. data/lib/active_merchant/billing/gateways/qvalent.rb +305 -0
  202. data/lib/active_merchant/billing/gateways/rapyd.rb +258 -0
  203. data/lib/active_merchant/billing/gateways/realex.rb +154 -52
  204. data/lib/active_merchant/billing/gateways/redsys.rb +488 -161
  205. data/lib/active_merchant/billing/gateways/s5.rb +247 -0
  206. data/lib/active_merchant/billing/gateways/safe_charge.rb +298 -0
  207. data/lib/active_merchant/billing/gateways/sage.rb +404 -133
  208. data/lib/active_merchant/billing/gateways/sage_pay.rb +145 -109
  209. data/lib/active_merchant/billing/gateways/sallie_mae.rb +13 -15
  210. data/lib/active_merchant/billing/gateways/secure_net.rb +62 -54
  211. data/lib/active_merchant/billing/gateways/secure_pay.rb +32 -42
  212. data/lib/active_merchant/billing/gateways/secure_pay_au.rb +36 -27
  213. data/lib/active_merchant/billing/gateways/secure_pay_tech.rb +14 -16
  214. data/lib/active_merchant/billing/gateways/securion_pay.rb +265 -0
  215. data/lib/active_merchant/billing/gateways/simetrik.rb +362 -0
  216. data/lib/active_merchant/billing/gateways/skip_jack.rb +52 -54
  217. data/lib/active_merchant/billing/gateways/smart_ps.rb +51 -60
  218. data/lib/active_merchant/billing/gateways/so_easy_pay.rb +29 -29
  219. data/lib/active_merchant/billing/gateways/spreedly_core.rb +108 -43
  220. data/lib/active_merchant/billing/gateways/stripe.rb +536 -120
  221. data/lib/active_merchant/billing/gateways/stripe_payment_intents.rb +516 -0
  222. data/lib/active_merchant/billing/gateways/swipe_checkout.rb +19 -25
  223. data/lib/active_merchant/billing/gateways/telr.rb +273 -0
  224. data/lib/active_merchant/billing/gateways/tns.rb +16 -217
  225. data/lib/active_merchant/billing/gateways/trans_first.rb +148 -34
  226. data/lib/active_merchant/billing/gateways/trans_first_transaction_express.rb +612 -0
  227. data/lib/active_merchant/billing/gateways/transact_pro.rb +222 -0
  228. data/lib/active_merchant/billing/gateways/transax.rb +6 -8
  229. data/lib/active_merchant/billing/gateways/transnational.rb +1 -2
  230. data/lib/active_merchant/billing/gateways/trexle.rb +221 -0
  231. data/lib/active_merchant/billing/gateways/trust_commerce.rb +162 -89
  232. data/lib/active_merchant/billing/gateways/usa_epay.rb +5 -6
  233. data/lib/active_merchant/billing/gateways/usa_epay_advanced.rb +332 -236
  234. data/lib/active_merchant/billing/gateways/usa_epay_transaction.rb +185 -72
  235. data/lib/active_merchant/billing/gateways/vanco.rb +294 -0
  236. data/lib/active_merchant/billing/gateways/verifi.rb +52 -53
  237. data/lib/active_merchant/billing/gateways/viaklix.rb +19 -31
  238. data/lib/active_merchant/billing/gateways/visanet_peru.rb +250 -0
  239. data/lib/active_merchant/billing/gateways/vpos.rb +220 -0
  240. data/lib/active_merchant/billing/gateways/webpay.rb +9 -9
  241. data/lib/active_merchant/billing/gateways/wepay.rb +108 -62
  242. data/lib/active_merchant/billing/gateways/wirecard.rb +52 -43
  243. data/lib/active_merchant/billing/gateways/wompi.rb +193 -0
  244. data/lib/active_merchant/billing/gateways/world_net.rb +345 -0
  245. data/lib/active_merchant/billing/gateways/worldpay.rb +725 -121
  246. data/lib/active_merchant/billing/gateways/worldpay_online_payments.rb +208 -0
  247. data/lib/active_merchant/billing/gateways/worldpay_us.rb +79 -39
  248. data/lib/active_merchant/billing/gateways.rb +8 -11
  249. data/lib/active_merchant/billing/model.rb +2 -2
  250. data/lib/active_merchant/billing/network_tokenization_credit_card.rb +39 -0
  251. data/lib/active_merchant/billing/payment_token.rb +1 -1
  252. data/lib/active_merchant/billing/rails.rb +1 -1
  253. data/lib/active_merchant/billing/response.rb +19 -13
  254. data/lib/active_merchant/billing/three_d_secure_eci_mapper.rb +27 -0
  255. data/lib/active_merchant/billing.rb +3 -0
  256. data/lib/active_merchant/connection.rb +196 -0
  257. data/lib/active_merchant/country.rb +13 -8
  258. data/lib/active_merchant/errors.rb +6 -0
  259. data/lib/active_merchant/net_http_ssl_connection.rb +11 -0
  260. data/lib/active_merchant/network_connection_retries.rb +78 -0
  261. data/lib/active_merchant/post_data.rb +26 -0
  262. data/lib/active_merchant/posts_data.rb +92 -0
  263. data/lib/active_merchant/version.rb +1 -1
  264. data/lib/active_merchant.rb +14 -59
  265. data/lib/activemerchant.rb +1 -1
  266. data/lib/certs/cacert.pem +3214 -0
  267. data/lib/support/gateway_support.rb +8 -10
  268. data/lib/support/outbound_hosts.rb +13 -10
  269. data/lib/support/ssl_verify.rb +9 -14
  270. data/lib/support/ssl_version.rb +86 -0
  271. metadata +136 -94
  272. checksums.yaml.gz.sig +0 -1
  273. data/lib/active_merchant/billing/gateways/app55.rb +0 -176
  274. data/lib/active_merchant/billing/gateways/barclays_epdq.rb +0 -314
  275. data/lib/active_merchant/billing/gateways/certo_direct.rb +0 -278
  276. data/lib/active_merchant/billing/gateways/ideal/ideal_base.rb +0 -246
  277. data/lib/active_merchant/billing/gateways/ideal/ideal_rabobank.pem +0 -13
  278. data/lib/active_merchant/billing/gateways/ideal/ideal_response.rb +0 -29
  279. data/lib/active_merchant/billing/gateways/ideal_rabobank.rb +0 -66
  280. data/lib/active_merchant/billing/gateways/moneris_us.rb +0 -291
  281. data/lib/active_merchant/billing/gateways/sage/sage_bankcard.rb +0 -87
  282. data/lib/active_merchant/billing/gateways/sage/sage_core.rb +0 -114
  283. data/lib/active_merchant/billing/gateways/sage/sage_vault.rb +0 -149
  284. data/lib/active_merchant/billing/gateways/sage/sage_virtual_check.rb +0 -102
  285. data/lib/active_merchant/billing/gateways/vindicia.rb +0 -385
  286. data/lib/active_merchant/offsite_payments_shim.rb +0 -19
  287. data.tar.gz.sig +0 -0
  288. metadata.gz.sig +0 -0
@@ -1,506 +1,549 @@
1
- require 'digest/md5'
2
-
3
- module ActiveMerchant #:nodoc:
4
- module Billing #:nodoc:
5
- class BluePayGateway < Gateway
6
- class_attribute :rebilling_url, :ignore_http_status
7
-
8
- self.live_url = 'https://secure.bluepay.com/interfaces/bp20post'
9
- self.rebilling_url = 'https://secure.bluepay.com/interfaces/bp20rebadmin'
10
-
11
- self.ignore_http_status = true
12
-
13
- CARD_CODE_ERRORS = %w( N S )
14
- AVS_ERRORS = %w( A E N R W Z )
15
- AVS_REASON_CODES = %w(27 45)
16
-
17
- FIELD_MAP = {
18
- 'TRANS_ID' => :transaction_id,
19
- 'STATUS' => :response_code,
20
- 'AVS' => :avs_result_code,
21
- 'CVV2'=> :card_code,
22
- 'AUTH_CODE' => :authorization,
23
- 'MESSAGE' => :message,
24
- 'REBID' => :rebid,
25
- 'TRANS_TYPE' => :trans_type,
26
- 'PAYMENT_ACCOUNT_MASK' => :acct_mask,
27
- 'CARD_TYPE' => :card_type,
28
- }
29
-
30
- REBILL_FIELD_MAP = {
31
- 'REBILL_ID' => :rebill_id,
32
- 'ACCOUNT_ID'=> :account_id,
33
- 'USER_ID' => :user_id,
34
- 'TEMPLATE_ID' => :template_id,
35
- 'STATUS' => :status,
36
- 'CREATION_DATE' => :creation_date,
37
- 'NEXT_DATE' => :next_date,
38
- 'LAST_DATE' => :last_date,
39
- 'SCHED_EXPR' => :schedule,
40
- 'CYCLES_REMAIN' => :cycles_remain,
41
- 'REB_AMOUNT' => :rebill_amount,
42
- 'NEXT_AMOUNT' => :next_amount,
43
- 'USUAL_DATE' => :undoc_usual_date, # Not found in the bp20rebadmin API doc.
44
- }
45
-
46
- self.supported_countries = ['US']
47
- self.supported_cardtypes = [:visa, :master, :american_express, :discover, :diners_club, :jcb]
48
- self.homepage_url = 'http://www.bluepay.com/'
49
- self.display_name = 'BluePay'
50
- self.money_format = :dollars
51
-
52
- # Creates a new BluepayGateway
53
- #
54
- # The gateway requires that a valid Account ID and Secret Key be passed
55
- # in the +options+ hash.
56
- #
57
- # ==== Options
58
- #
59
- # * <tt>:account_id</tt> -- The BluePay gateway Account ID (REQUIRED)
60
- # * <tt>:secret_key</tt> -- The BluePay gateway Secret Key (REQUIRED)
61
- # * <tt>:test</tt> -- set to true for TEST mode or false for LIVE mode
62
- def initialize(options = {})
63
- requires!(options, :login, :password)
64
- super
65
- end
66
-
67
- # Performs an authorization, which reserves the funds on the customer's credit card. This does not actually take funds from the customer
68
- # This is referred to an AUTH transaction in BluePay
69
- #
70
- # ==== Parameters
71
- #
72
- # * <tt>money</tt> -- The amount to be authorized as an Integer value in cents.
73
- # * <tt>payment_object</tt> -- This can either be one of three things:
74
- # A CreditCard object,
75
- # A Check object,
76
- # or a token. The token is called the Master ID. This is a unique transaction ID returned from a previous transaction. This token associates all the stored information for a previous transaction.
77
- # * <tt>options</tt> -- A hash of optional parameters.
78
- def authorize(money, payment_object, options = {})
79
- post = {}
80
- add_payment_method(post, payment_object)
81
- add_invoice(post, options)
82
- add_address(post, options)
83
- add_customer_data(post, options)
84
- add_rebill(post, options) if options[:rebill]
85
- add_duplicate_override(post, options)
86
- post[:TRANS_TYPE] = 'AUTH'
87
- commit('AUTH_ONLY', money, post)
88
- end
89
-
90
- # Perform a purchase, which is essentially an authorization and capture in a single operation.
91
- # This is referred to a SALE transaction in BluePay
92
- #
93
- # ==== Parameters
94
- #
95
- # * <tt>money</tt> -- The amount to be purchased as an Integer value in cents.
96
- # * <tt>payment_object</tt> -- This can either be one of three things:
97
- # A CreditCard object,
98
- # A Check object,
99
- # or a token. The token is called the Master ID. This is a unique transaction ID returned from a previous transaction. This token associates all the stored information for a previous transaction.
100
- # * <tt>options</tt> -- A hash of optional parameters.,
101
- def purchase(money, payment_object, options = {})
102
- post = {}
103
- add_payment_method(post, payment_object)
104
- add_invoice(post, options)
105
- add_address(post, options)
106
- add_customer_data(post, options)
107
- add_rebill(post, options) if options[:rebill]
108
- add_duplicate_override(post, options)
109
- post[:TRANS_TYPE] = 'SALE'
110
- commit('AUTH_CAPTURE', money, post)
111
- end
112
-
113
- # Captures the funds from an authorize transaction.
114
- # This is referred to a CAPTURE transaction in BluePay
115
- #
116
- # ==== Parameters
117
- #
118
- # * <tt>money</tt> -- The amount to be captured as an Integer value in cents.
119
- # * <tt>identification</tt> -- The Master ID, or token, returned from the previous authorize transaction.
120
- def capture(money, identification, options = {})
121
- post = {}
122
- add_address(post, options)
123
- add_customer_data(post, options)
124
- post[:MASTER_ID] = identification
125
- post[:TRANS_TYPE] = 'CAPTURE'
126
- commit('PRIOR_AUTH_CAPTURE', money, post)
127
- end
128
-
129
- # Void a previous transaction
130
- # This is referred to a VOID transaction in BluePay
131
- #
132
- # ==== Parameters
133
- #
134
- # * <tt>identification</tt> - The Master ID, or token, returned from a previous authorize transaction.
135
- def void(identification, options = {})
136
- post = {}
137
- post[:MASTER_ID] = identification
138
- post[:TRANS_TYPE] = 'VOID'
139
- commit('VOID', nil, post)
140
- end
141
-
142
- # Performs a credit.
143
- #
144
- # This transaction indicates that money should flow from the merchant to the customer.
145
- #
146
- # ==== Parameters
147
- #
148
- # * <tt>money</tt> -- The amount to be credited to the customer as an Integer value in cents.
149
- # * <tt>payment_object</tt> -- This can either be one of three things:
150
- # A CreditCard object,
151
- # A Check object,
152
- # or a token. The token is called the Master ID. This is a unique transaction ID returned from a previous transaction. This token associates all the stored information for a previous transaction.
153
- # If the payment_object is a token, then the transaction type will reverse a previous capture or purchase transaction, returning the funds to the customer. If the amount is nil, a full credit will be processed. This is referred to a REFUND transaction in BluePay.
154
- # If the payment_object is either a CreditCard or Check object, then the transaction type will be an unmatched credit placing funds in the specified account. This is referred to a CREDIT transaction in BluePay.
155
- # * <tt>options</tt> -- A hash of parameters.
156
- def refund(money, identification, options = {})
157
- if(identification && !identification.kind_of?(String))
158
- ActiveMerchant.deprecated "refund should only be used to refund a referenced transaction"
159
- return credit(money, identification, options)
160
- end
161
-
162
- post = {}
163
- post[:PAYMENT_ACCOUNT] = ''
164
- post[:MASTER_ID] = identification
165
- post[:TRANS_TYPE] = 'REFUND'
166
- post[:NAME1] = (options[:first_name] ? options[:first_name] : "")
167
- post[:NAME2] = options[:last_name] if options[:last_name]
168
- post[:ZIP] = options[:zip] if options[:zip]
169
- add_invoice(post, options)
170
- add_address(post, options)
171
- add_customer_data(post, options)
172
- commit('CREDIT', money, post)
173
- end
174
-
175
- def credit(money, payment_object, options = {})
176
- if(payment_object && payment_object.kind_of?(String))
177
- ActiveMerchant.deprecated "credit should only be used to credit a payment method"
178
- return refund(money, payment_object, options)
179
- end
180
-
181
- post = {}
182
- post[:PAYMENT_ACCOUNT] = ''
183
- add_payment_method(post, payment_object)
184
- post[:TRANS_TYPE] = 'CREDIT'
185
-
186
- post[:NAME1] = (options[:first_name] ? options[:first_name] : "")
187
- post[:NAME2] = options[:last_name] if options[:last_name]
188
- post[:ZIP] = options[:zip] if options[:zip]
189
- add_invoice(post, options)
190
- add_address(post, options)
191
- add_customer_data(post, options)
192
- commit('CREDIT', money, post)
193
- end
194
-
195
- # Create a new recurring payment.
196
- #
197
- # ==== Parameters
198
- #
199
- # * <tt>money</tt> -- The amount to charge the customer at the time of the recurring payment setup, in cents. Set to zero if you do not want the customer to be charged at this time.
200
- # * <tt>payment_object</tt> -- This can either be one of three things:
201
- # A CreditCard object,
202
- # A Check object,
203
- # or a token. The token is called the Master ID. This is a unique transaction ID returned from a previous transaction. This token associates all the stored information for a previous transaction.
204
- # * <tt>options</tt> -- A hash of optional parameters.,
205
-
206
- # ==== Options
207
- #
208
- # * <tt>:rebill_start_date</tt> is a string that tells the gateway when to start the rebill. (REQUIRED)
209
- # Has two valid formats:
210
- # "YYYY-MM-DD HH:MM:SS" Hours, minutes, and seconds are optional.
211
- # "XX UNITS" Relative date as explained below. Marked from the time of the
212
- # transaction (i.e.: 10 DAYS, 1 MONTH, 1 YEAR)
213
- # * <tt>:rebill_expression</tt> is the period of time in-between rebillings. (REQUIRED)
214
- # It uses the same "XX UNITS" format as rebill_start_date, explained above.
215
- # Optional parameters include:
216
- # * <tt>rebill_cycles</tt>: Number of times to rebill. Don't send or set to nil for infinite rebillings (or
217
- # until canceled).
218
- # * <tt>rebill_amount</tt>: Amount to rebill. Defaults to amount of transaction for rebillings.
219
- #
220
- # For example, to charge the customer $19.95 now and then charge $39.95 in 60 days every 3 months for 5 times, the options hash would be as follows:
221
- # :rebill_start_date => '60 DAYS',
222
- # :rebill_expression => '3 MONTHS',
223
- # :rebill_cycles => '5',
224
- # :rebill_amount => '39.95'
225
- # A money object of 1995 cents would be passed into the 'money' parameter.
226
- def recurring(money, payment_object, options = {})
227
- ActiveMerchant.deprecated RECURRING_DEPRECATION_MESSAGE
228
-
229
- requires!(options, :rebill_start_date, :rebill_expression)
230
- options[:rebill] = true
231
- if money
232
- purchase(money, payment_object, options)
233
- else
234
- authorize(money, payment_object, options)
235
- end
236
- end
237
-
238
- # View a recurring payment
239
- #
240
- # This will pull data associated with a current recurring billing
241
- #
242
- # ==== Parameters
243
- #
244
- # * <tt>rebill_id</tt> -- A string containing the rebill_id of the recurring billing that is already active (REQUIRED)
245
- def status_recurring(rebill_id)
246
- ActiveMerchant.deprecated RECURRING_DEPRECATION_MESSAGE
247
-
248
- post = {}
249
- requires!(rebill_id)
250
- post[:REBILL_ID] = rebill_id
251
- post[:TRANS_TYPE] = 'GET'
252
- commit('rebill', 'nil', post)
253
- end
254
-
255
- # Update a recurring payment's details.
256
- #
257
- # This transaction updates an existing recurring billing
258
- #
259
- # ==== Options
260
- #
261
- # * <tt>:rebill_id</tt> -- The 12 digit rebill ID used to update a particular rebilling cycle. (REQUIRED)
262
- # * <tt>:rebill_amount</tt> -- A string containing the new rebilling amount.
263
- # * <tt>:rebill_next_date</tt> -- A string containing the new rebilling next date.
264
- # * <tt>:rebill_expression</tt> -- A string containing the new rebilling expression.
265
- # * <tt>:rebill_cycles</tt> -- A string containing the new rebilling cycles.
266
- # * <tt>:rebill_next_amount</tt> -- A string containing the next rebilling amount to charge the customer. This ONLY affects the next scheduled charge; all other rebillings will continue at the regular (rebill_amount) amount.
267
- # Take a look above at the recurring_payment method for similar examples on how to use.
268
- def update_recurring(options = {})
269
- ActiveMerchant.deprecated RECURRING_DEPRECATION_MESSAGE
270
-
271
- post = {}
272
- requires!(options, :rebill_id)
273
- post[:REBILL_ID] = options[:rebill_id]
274
- post[:TRANS_TYPE] = 'SET'
275
- post[:REB_AMOUNT] = amount(options[:rebill_amount]) if options[:rebill_amount]
276
- post[:NEXT_DATE] = options[:rebill_next_date]
277
- post[:REB_EXPR] = options[:rebill_expression]
278
- post[:REB_CYCLES] = options[:rebill_cycles]
279
- post[:NEXT_AMOUNT] = options[:rebill_next_amount]
280
- commit('rebill', 'nil', post)
281
- end
282
-
283
- # Cancel a recurring payment.
284
- #
285
- # This transaction cancels an existing recurring billing.
286
- #
287
- # ==== Parameters
288
- #
289
- # * <tt>rebill_id</tt> -- A string containing the rebill_id of the recurring billing that you wish to cancel/stop (REQUIRED)
290
- def cancel_recurring(rebill_id)
291
- ActiveMerchant.deprecated RECURRING_DEPRECATION_MESSAGE
292
-
293
- post = {}
294
- requires!(rebill_id)
295
- post[:REBILL_ID] = rebill_id
296
- post[:TRANS_TYPE] = 'SET'
297
- post[:STATUS] = 'stopped'
298
- commit('rebill', 'nil', post)
299
- end
300
-
301
- private
302
-
303
- def commit(action, money, fields)
304
- fields[:AMOUNT] = amount(money) unless(fields[:TRANS_TYPE] == 'VOID' || action == 'rebill')
305
- fields[:MODE] = (test? ? 'TEST' : 'LIVE')
306
- fields[:ACCOUNT_ID] = @options[:login]
307
-
308
- if action == 'rebill'
309
- url = rebilling_url
310
- fields[:TAMPER_PROOF_SEAL] = calc_rebill_tps(fields)
311
- else
312
- url = live_url
313
- fields[:TAMPER_PROOF_SEAL] = calc_tps(amount(money), fields)
314
- end
315
- parse(ssl_post(url, post_data(action, fields)))
316
- end
317
-
318
- def parse_recurring(response_fields, opts={}) # expected status?
319
- parsed = {}
320
- response_fields.each do |k,v|
321
- mapped_key = REBILL_FIELD_MAP.include?(k) ? REBILL_FIELD_MAP[k] : k
322
- parsed[mapped_key] = v
323
- end
324
-
325
- success = parsed[:status] != 'error'
326
- message = parsed[:status]
327
-
328
- Response.new(success, message, parsed,
329
- :test => test?,
330
- :authorization => parsed[:rebill_id])
331
- end
332
-
333
- def parse(body)
334
- # The bp20api has max one value per form field.
335
- response_fields = Hash[CGI::parse(body).map{|k,v| [k.upcase,v.first]}]
336
-
337
- if response_fields.include? "REBILL_ID"
338
- return parse_recurring(response_fields)
339
- end
340
-
341
- parsed = {}
342
- response_fields.each do |k,v|
343
- mapped_key = FIELD_MAP.include?(k) ? FIELD_MAP[k] : k
344
- parsed[mapped_key] = v
345
- end
346
-
347
- # normalize message
348
- message = message_from(parsed)
349
- success = parsed[:response_code] == '1'
350
- Response.new(success, message, parsed,
351
- :test => test?,
352
- :authorization => (parsed[:rebid] && parsed[:rebid] != '' ? parsed[:rebid] : parsed[:transaction_id]),
353
- :avs_result => { :code => parsed[:avs_result_code] },
354
- :cvv_result => parsed[:card_code]
355
- )
356
- end
357
-
358
- def message_from(parsed)
359
- message = parsed[:message]
360
- if(parsed[:response_code].to_i == 2)
361
- if CARD_CODE_ERRORS.include?(parsed[:card_code])
362
- message = CVVResult.messages[parsed[:card_code]]
363
- elsif AVS_ERRORS.include?(parsed[:avs_result_code])
364
- message = AVSResult.messages[ parsed[:avs_result_code] ]
365
- else
366
- message = message.chomp('.')
367
- end
368
- elsif message == "Missing ACCOUNT_ID"
369
- message = "The merchant login ID or password is invalid"
370
- elsif message =~ /Approved/
371
- message = "This transaction has been approved"
372
- elsif message =~ /Expired/
373
- message = "The credit card has expired"
374
- end
375
- message
376
- end
377
-
378
- def add_invoice(post, options)
379
- post[:ORDER_ID] = options[:order_id]
380
- post[:INVOICE_ID] = options[:invoice]
381
- post[:invoice_num] = options[:order_id]
382
- post[:MEMO] = options[:description]
383
- post[:description] = options[:description]
384
- end
385
-
386
- def add_payment_method(post, payment_object)
387
- post[:MASTER_ID] = ''
388
- case payment_object
389
- when String
390
- post[:MASTER_ID] = payment_object
391
- when Check
392
- add_check(post, payment_object)
393
- else
394
- add_creditcard(post, payment_object)
395
- end
396
- end
397
-
398
- def add_creditcard(post, creditcard)
399
- post[:PAYMENT_TYPE] = 'CREDIT'
400
- post[:PAYMENT_ACCOUNT] = creditcard.number
401
- post[:CARD_CVV2] = creditcard.verification_value
402
- post[:CARD_EXPIRE] = expdate(creditcard)
403
- post[:NAME1] = creditcard.first_name
404
- post[:NAME2] = creditcard.last_name
405
- end
406
-
407
- CHECK_ACCOUNT_TYPES = {
408
- "checking" => "C",
409
- "savings" => "S"
410
- }
411
-
412
- def add_check(post, check)
413
- post[:PAYMENT_TYPE] = 'ACH'
414
- post[:PAYMENT_ACCOUNT] = [CHECK_ACCOUNT_TYPES[check.account_type], check.routing_number, check.account_number].join(":")
415
- post[:NAME1] = check.first_name
416
- post[:NAME2] = check.last_name
417
- end
418
-
419
- def add_customer_data(post, options)
420
- post[:EMAIL] = options[:email]
421
- post[:CUSTOM_ID] = options[:customer]
422
- end
423
-
424
- def add_duplicate_override(post, options)
425
- post[:DUPLICATE_OVERRIDE] = options[:duplicate_override]
426
- end
427
-
428
- def add_address(post, options)
429
- if address = (options[:shipping_address] || options[:billing_address] || options[:address])
430
- post[:ADDR1] = address[:address1]
431
- post[:ADDR2] = address[:address2]
432
- post[:COMPANY_NAME] = address[:company]
433
- post[:PHONE] = address[:phone]
434
- post[:CITY] = address[:city]
435
- post[:STATE] = (address[:state].blank? ? 'n/a' : address[:state])
436
- post[:ZIP] = address[:zip]
437
- post[:COUNTRY] = address[:country]
438
- end
439
- end
440
-
441
- def add_rebill(post, options)
442
- post[:DO_REBILL] = '1'
443
- post[:REB_AMOUNT] = amount(options[:rebill_amount])
444
- post[:REB_FIRST_DATE] = options[:rebill_start_date]
445
- post[:REB_EXPR] = options[:rebill_expression]
446
- post[:REB_CYCLES] = options[:rebill_cycles]
447
- end
448
-
449
- def post_data(action, parameters = {})
450
- post = {}
451
- post[:version] = '1'
452
- post[:login] = ''
453
- post[:tran_key] = ''
454
- post[:relay_response] = "FALSE"
455
- post[:type] = action
456
- post[:delim_data] = "TRUE"
457
- post[:delim_char] = ","
458
- post[:encap_char] = "$"
459
- post[:card_num] = '4111111111111111'
460
- post[:exp_date] = '1212'
461
- post[:solution_ID] = application_id if(application_id && application_id != "ActiveMerchant")
462
- post.merge(parameters).collect { |key, value| "#{key}=#{CGI.escape(value.to_s)}" }.join("&")
463
- end
464
-
465
- def expdate(creditcard)
466
- year = format(creditcard.year, :two_digits)
467
- month = format(creditcard.month, :two_digits)
468
-
469
- "#{month}#{year}"
470
- end
471
-
472
- def calc_tps(amount, post)
473
- post[:NAME1] ||= ''
474
- Digest::MD5.hexdigest(
475
- [
476
- @options[:password],
477
- @options[:login],
478
- post[:TRANS_TYPE],
479
- amount,
480
- post[:MASTER_ID],
481
- post[:NAME1],
482
- post[:PAYMENT_ACCOUNT]
483
- ].join("")
484
- )
485
- end
486
-
487
- def calc_rebill_tps(post)
488
- Digest::MD5.hexdigest(
489
- [
490
- @options[:password],
491
- @options[:login],
492
- post[:TRANS_TYPE],
493
- post[:REBILL_ID]
494
- ].join("")
495
- )
496
- end
497
-
498
- def handle_response(response)
499
- if ignore_http_status || (200...300).include?(response.code.to_i)
500
- return response.body
501
- end
502
- raise ResponseError.new(response)
503
- end
504
- end
505
- end
506
- end
1
+ require 'digest/md5'
2
+
3
+ module ActiveMerchant #:nodoc:
4
+ module Billing #:nodoc:
5
+ class BluePayGateway < Gateway
6
+ class_attribute :rebilling_url, :ignore_http_status
7
+
8
+ self.live_url = 'https://secure.bluepay.com/interfaces/bp20post'
9
+ self.rebilling_url = 'https://secure.bluepay.com/interfaces/bp20rebadmin'
10
+
11
+ self.ignore_http_status = true
12
+
13
+ CARD_CODE_ERRORS = %w(N S)
14
+ AVS_ERRORS = %w(A E N R W Z)
15
+ AVS_REASON_CODES = %w(27 45)
16
+
17
+ FIELD_MAP = {
18
+ 'TRANS_ID' => :transaction_id,
19
+ 'STATUS' => :response_code,
20
+ 'AVS' => :avs_result_code,
21
+ 'CVV2' => :card_code,
22
+ 'AUTH_CODE' => :authorization,
23
+ 'MESSAGE' => :message,
24
+ 'REBID' => :rebid,
25
+ 'TRANS_TYPE' => :trans_type,
26
+ 'PAYMENT_ACCOUNT_MASK' => :acct_mask,
27
+ 'CARD_TYPE' => :card_type
28
+ }
29
+
30
+ REBILL_FIELD_MAP = {
31
+ 'REBILL_ID' => :rebill_id,
32
+ 'ACCOUNT_ID' => :account_id,
33
+ 'USER_ID' => :user_id,
34
+ 'TEMPLATE_ID' => :template_id,
35
+ 'STATUS' => :status,
36
+ 'CREATION_DATE' => :creation_date,
37
+ 'NEXT_DATE' => :next_date,
38
+ 'LAST_DATE' => :last_date,
39
+ 'SCHED_EXPR' => :schedule,
40
+ 'CYCLES_REMAIN' => :cycles_remain,
41
+ 'REB_AMOUNT' => :rebill_amount,
42
+ 'NEXT_AMOUNT' => :next_amount,
43
+ 'USUAL_DATE' => :undoc_usual_date, # Not found in the bp20rebadmin API doc.
44
+ 'CUST_TOKEN' => :cust_token
45
+ }
46
+
47
+ self.supported_countries = %w[US CA]
48
+ self.supported_cardtypes = %i[visa master american_express discover diners_club jcb]
49
+ self.homepage_url = 'http://www.bluepay.com/'
50
+ self.display_name = 'BluePay'
51
+ self.money_format = :dollars
52
+
53
+ # Creates a new BluepayGateway
54
+ #
55
+ # The gateway requires that a valid Account ID and Secret Key be passed
56
+ # in the +options+ hash.
57
+ #
58
+ # ==== Options
59
+ #
60
+ # * <tt>:account_id</tt> -- The BluePay gateway Account ID (REQUIRED)
61
+ # * <tt>:secret_key</tt> -- The BluePay gateway Secret Key (REQUIRED)
62
+ # * <tt>:test</tt> -- set to true for TEST mode or false for LIVE mode
63
+ def initialize(options = {})
64
+ requires!(options, :login, :password)
65
+ super
66
+ end
67
+
68
+ # Performs an authorization, which reserves the funds on the customer's credit card. This does not actually take funds from the customer
69
+ # This is referred to an AUTH transaction in BluePay
70
+ #
71
+ # ==== Parameters
72
+ #
73
+ # * <tt>money</tt> -- The amount to be authorized as an Integer value in cents.
74
+ # * <tt>payment_object</tt> -- This can either be one of three things:
75
+ # A CreditCard object,
76
+ # A Check object,
77
+ # or a token. The token is called the Master ID. This is a unique transaction ID returned from a previous transaction. This token associates all the stored information for a previous transaction.
78
+ # * <tt>options</tt> -- A hash of optional parameters.
79
+ def authorize(money, payment_object, options = {})
80
+ post = {}
81
+ add_payment_method(post, payment_object)
82
+ add_invoice(post, options)
83
+ add_address(post, options)
84
+ add_customer_data(post, options)
85
+ add_rebill(post, options) if options[:rebill]
86
+ add_duplicate_override(post, options)
87
+ add_stored_credential(post, options)
88
+ post[:TRANS_TYPE] = 'AUTH'
89
+ commit('AUTH_ONLY', money, post, options)
90
+ end
91
+
92
+ # Perform a purchase, which is essentially an authorization and capture in a single operation.
93
+ # This is referred to a SALE transaction in BluePay
94
+ #
95
+ # ==== Parameters
96
+ #
97
+ # * <tt>money</tt> -- The amount to be purchased as an Integer value in cents.
98
+ # * <tt>payment_object</tt> -- This can either be one of three things:
99
+ # A CreditCard object,
100
+ # A Check object,
101
+ # or a token. The token is called the Master ID. This is a unique transaction ID returned from a previous transaction. This token associates all the stored information for a previous transaction.
102
+ # * <tt>options</tt> -- A hash of optional parameters.,
103
+ def purchase(money, payment_object, options = {})
104
+ post = {}
105
+ add_payment_method(post, payment_object)
106
+ add_invoice(post, options)
107
+ add_address(post, options)
108
+ add_customer_data(post, options)
109
+ add_rebill(post, options) if options[:rebill]
110
+ add_duplicate_override(post, options)
111
+ add_stored_credential(post, options)
112
+ post[:TRANS_TYPE] = 'SALE'
113
+ commit('AUTH_CAPTURE', money, post, options)
114
+ end
115
+
116
+ # Captures the funds from an authorize transaction.
117
+ # This is referred to a CAPTURE transaction in BluePay
118
+ #
119
+ # ==== Parameters
120
+ #
121
+ # * <tt>money</tt> -- The amount to be captured as an Integer value in cents.
122
+ # * <tt>identification</tt> -- The Master ID, or token, returned from the previous authorize transaction.
123
+ def capture(money, identification, options = {})
124
+ post = {}
125
+ add_address(post, options)
126
+ add_customer_data(post, options)
127
+ post[:MASTER_ID] = identification
128
+ post[:TRANS_TYPE] = 'CAPTURE'
129
+ commit('PRIOR_AUTH_CAPTURE', money, post, options)
130
+ end
131
+
132
+ # Void a previous transaction
133
+ # This is referred to a VOID transaction in BluePay
134
+ #
135
+ # ==== Parameters
136
+ #
137
+ # * <tt>identification</tt> - The Master ID, or token, returned from a previous authorize transaction.
138
+ def void(identification, options = {})
139
+ post = {}
140
+ post[:MASTER_ID] = identification
141
+ post[:TRANS_TYPE] = 'VOID'
142
+ commit('VOID', nil, post, options)
143
+ end
144
+
145
+ # Performs a credit.
146
+ #
147
+ # This transaction indicates that money should flow from the merchant to the customer.
148
+ #
149
+ # ==== Parameters
150
+ #
151
+ # * <tt>money</tt> -- The amount to be credited to the customer as an Integer value in cents.
152
+ # * <tt>payment_object</tt> -- This can either be one of three things:
153
+ # A CreditCard object,
154
+ # A Check object,
155
+ # or a token. The token is called the Master ID. This is a unique transaction ID returned from a previous transaction. This token associates all the stored information for a previous transaction.
156
+ # If the payment_object is a token, then the transaction type will reverse a previous capture or purchase transaction, returning the funds to the customer. If the amount is nil, a full credit will be processed. This is referred to a REFUND transaction in BluePay.
157
+ # If the payment_object is either a CreditCard or Check object, then the transaction type will be an unmatched credit placing funds in the specified account. This is referred to a CREDIT transaction in BluePay.
158
+ # * <tt>options</tt> -- A hash of parameters.
159
+ def refund(money, identification, options = {})
160
+ if identification && !identification.kind_of?(String)
161
+ ActiveMerchant.deprecated 'refund should only be used to refund a referenced transaction'
162
+ return credit(money, identification, options)
163
+ end
164
+
165
+ post = {}
166
+ post[:PAYMENT_ACCOUNT] = ''
167
+ post[:MASTER_ID] = identification
168
+ post[:TRANS_TYPE] = 'REFUND'
169
+ post[:DOC_TYPE] = options[:doc_type] if options[:doc_type]
170
+ post[:NAME1] = options[:first_name] || ''
171
+ post[:NAME2] = options[:last_name] if options[:last_name]
172
+ post[:ZIP] = options[:zip] if options[:zip]
173
+ add_invoice(post, options)
174
+ add_address(post, options)
175
+ add_customer_data(post, options)
176
+ commit('CREDIT', money, post, options)
177
+ end
178
+
179
+ def credit(money, payment_object, options = {})
180
+ if payment_object&.kind_of?(String)
181
+ ActiveMerchant.deprecated 'credit should only be used to credit a payment method'
182
+ return refund(money, payment_object, options)
183
+ end
184
+
185
+ post = {}
186
+ post[:PAYMENT_ACCOUNT] = ''
187
+ add_payment_method(post, payment_object)
188
+ post[:TRANS_TYPE] = 'CREDIT'
189
+ post[:DOC_TYPE] = options[:doc_type] if options[:doc_type]
190
+
191
+ post[:NAME1] = options[:first_name] || ''
192
+ post[:NAME2] = options[:last_name] if options[:last_name]
193
+ post[:ZIP] = options[:zip] if options[:zip]
194
+ add_invoice(post, options)
195
+ add_address(post, options)
196
+ add_customer_data(post, options)
197
+ commit('CREDIT', money, post, options)
198
+ end
199
+
200
+ # Create a new recurring payment.
201
+ #
202
+ # ==== Parameters
203
+ #
204
+ # * <tt>money</tt> -- The amount to charge the customer at the time of the recurring payment setup, in cents. Set to zero if you do not want the customer to be charged at this time.
205
+ # * <tt>payment_object</tt> -- This can either be one of three things:
206
+ # A CreditCard object,
207
+ # A Check object,
208
+ # or a token. The token is called the Master ID. This is a unique transaction ID returned from a previous transaction. This token associates all the stored information for a previous transaction.
209
+ # * <tt>options</tt> -- A hash of optional parameters.,
210
+
211
+ # ==== Options
212
+ #
213
+ # * <tt>:rebill_start_date</tt> is a string that tells the gateway when to start the rebill. (REQUIRED)
214
+ # Has two valid formats:
215
+ # "YYYY-MM-DD HH:MM:SS" Hours, minutes, and seconds are optional.
216
+ # "XX UNITS" Relative date as explained below. Marked from the time of the
217
+ # transaction (i.e.: 10 DAYS, 1 MONTH, 1 YEAR)
218
+ # * <tt>:rebill_expression</tt> is the period of time in-between rebillings. (REQUIRED)
219
+ # It uses the same "XX UNITS" format as rebill_start_date, explained above.
220
+ # Optional parameters include:
221
+ # * <tt>rebill_cycles</tt>: Number of times to rebill. Don't send or set to nil for infinite rebillings (or
222
+ # until canceled).
223
+ # * <tt>rebill_amount</tt>: Amount to rebill. Defaults to amount of transaction for rebillings.
224
+ #
225
+ # For example, to charge the customer $19.95 now and then charge $39.95 in 60 days every 3 months for 5 times, the options hash would be as follows:
226
+ # :rebill_start_date => '60 DAYS',
227
+ # :rebill_expression => '3 MONTHS',
228
+ # :rebill_cycles => '5',
229
+ # :rebill_amount => '39.95'
230
+ # A money object of 1995 cents would be passed into the 'money' parameter.
231
+ def recurring(money, payment_object, options = {})
232
+ ActiveMerchant.deprecated RECURRING_DEPRECATION_MESSAGE
233
+
234
+ requires!(options, :rebill_start_date, :rebill_expression)
235
+ options[:rebill] = true
236
+ if money
237
+ purchase(money, payment_object, options)
238
+ else
239
+ authorize(money, payment_object, options)
240
+ end
241
+ end
242
+
243
+ # View a recurring payment
244
+ #
245
+ # This will pull data associated with a current recurring billing
246
+ #
247
+ # ==== Parameters
248
+ #
249
+ # * <tt>rebill_id</tt> -- A string containing the rebill_id of the recurring billing that is already active (REQUIRED)
250
+ def status_recurring(rebill_id)
251
+ ActiveMerchant.deprecated RECURRING_DEPRECATION_MESSAGE
252
+
253
+ post = {}
254
+ requires!(rebill_id)
255
+ post[:REBILL_ID] = rebill_id
256
+ post[:TRANS_TYPE] = 'GET'
257
+ commit('rebill', 'nil', post)
258
+ end
259
+
260
+ # Update a recurring payment's details.
261
+ #
262
+ # This transaction updates an existing recurring billing
263
+ #
264
+ # ==== Options
265
+ #
266
+ # * <tt>:rebill_id</tt> -- The 12 digit rebill ID used to update a particular rebilling cycle. (REQUIRED)
267
+ # * <tt>:rebill_amount</tt> -- A string containing the new rebilling amount.
268
+ # * <tt>:rebill_next_date</tt> -- A string containing the new rebilling next date.
269
+ # * <tt>:rebill_expression</tt> -- A string containing the new rebilling expression.
270
+ # * <tt>:rebill_cycles</tt> -- A string containing the new rebilling cycles.
271
+ # * <tt>:rebill_next_amount</tt> -- A string containing the next rebilling amount to charge the customer. This ONLY affects the next scheduled charge; all other rebillings will continue at the regular (rebill_amount) amount.
272
+ # Take a look above at the recurring_payment method for similar examples on how to use.
273
+ def update_recurring(options = {})
274
+ ActiveMerchant.deprecated RECURRING_DEPRECATION_MESSAGE
275
+
276
+ post = {}
277
+ requires!(options, :rebill_id)
278
+ post[:REBILL_ID] = options[:rebill_id]
279
+ post[:TRANS_TYPE] = 'SET'
280
+ post[:REB_AMOUNT] = amount(options[:rebill_amount]) if options[:rebill_amount]
281
+ post[:NEXT_DATE] = options[:rebill_next_date]
282
+ post[:REB_EXPR] = options[:rebill_expression]
283
+ post[:REB_CYCLES] = options[:rebill_cycles]
284
+ post[:NEXT_AMOUNT] = options[:rebill_next_amount]
285
+ commit('rebill', 'nil', post)
286
+ end
287
+
288
+ # Cancel a recurring payment.
289
+ #
290
+ # This transaction cancels an existing recurring billing.
291
+ #
292
+ # ==== Parameters
293
+ #
294
+ # * <tt>rebill_id</tt> -- A string containing the rebill_id of the recurring billing that you wish to cancel/stop (REQUIRED)
295
+ def cancel_recurring(rebill_id)
296
+ ActiveMerchant.deprecated RECURRING_DEPRECATION_MESSAGE
297
+
298
+ post = {}
299
+ requires!(rebill_id)
300
+ post[:REBILL_ID] = rebill_id
301
+ post[:TRANS_TYPE] = 'SET'
302
+ post[:STATUS] = 'stopped'
303
+ commit('rebill', 'nil', post)
304
+ end
305
+
306
+ def supports_scrubbing
307
+ true
308
+ end
309
+
310
+ def scrub(transcript)
311
+ transcript.
312
+ gsub(%r((Authorization: Basic )\w+), '\1[FILTERED]').
313
+ gsub(%r((&?card_num=)[^&]*)i, '\1[FILTERED]').
314
+ gsub(%r((&?CARD_CVV2=)[^&]*)i, '\1[FILTERED]').
315
+ gsub(%r((&?PAYMENT_ACCOUNT=)[^&]*)i, '\1[FILTERED]').
316
+ gsub(%r((&?TAMPER_PROOF_SEAL=)[^&"]*)i, '\1[FILTERED]')
317
+ end
318
+
319
+ private
320
+
321
+ def commit(action, money, fields, options = {})
322
+ fields[:AMOUNT] = amount(money) unless fields[:TRANS_TYPE] == 'VOID' || action == 'rebill'
323
+ fields[:MODE] = (test? ? 'TEST' : 'LIVE')
324
+ fields[:ACCOUNT_ID] = @options[:login]
325
+ fields[:CUSTOMER_IP] = options[:ip] if options[:ip]
326
+
327
+ if action == 'rebill'
328
+ url = rebilling_url
329
+ fields[:TAMPER_PROOF_SEAL] = calc_rebill_tps(fields)
330
+ else
331
+ url = live_url
332
+ fields[:TAMPER_PROOF_SEAL] = calc_tps(amount(money), fields)
333
+ end
334
+ parse(ssl_post(url, post_data(action, fields)))
335
+ end
336
+
337
+ def parse_recurring(response_fields, opts = {}) # expected status?
338
+ parsed = {}
339
+ response_fields.each do |k, v|
340
+ mapped_key = REBILL_FIELD_MAP.include?(k) ? REBILL_FIELD_MAP[k] : k
341
+ parsed[mapped_key] = v
342
+ end
343
+
344
+ success = parsed[:status] != 'error'
345
+ message = parsed[:status]
346
+
347
+ Response.new(success, message, parsed,
348
+ test: test?,
349
+ authorization: parsed[:rebill_id])
350
+ end
351
+
352
+ def parse(body)
353
+ # The bp20api has max one value per form field.
354
+ response_fields = Hash[CGI::parse(body).map { |k, v| [k.upcase, v.first] }]
355
+
356
+ return parse_recurring(response_fields) if response_fields.include? 'REBILL_ID'
357
+
358
+ parsed = {}
359
+ response_fields.each do |k, v|
360
+ mapped_key = FIELD_MAP.include?(k) ? FIELD_MAP[k] : k
361
+ parsed[mapped_key] = v
362
+ end
363
+
364
+ # normalize message
365
+ message = message_from(parsed)
366
+ success = parsed[:response_code] == '1'
367
+ Response.new(success, message, parsed,
368
+ test: test?,
369
+ authorization: (parsed[:rebid] && parsed[:rebid] != '' ? parsed[:rebid] : parsed[:transaction_id]),
370
+ avs_result: { code: parsed[:avs_result_code] },
371
+ cvv_result: parsed[:card_code])
372
+ end
373
+
374
+ def message_from(parsed)
375
+ message = parsed[:message]
376
+ if parsed[:response_code].to_i == 2
377
+ if CARD_CODE_ERRORS.include?(parsed[:card_code])
378
+ message = CVVResult.messages[parsed[:card_code]]
379
+ elsif AVS_ERRORS.include?(parsed[:avs_result_code])
380
+ message = AVSResult.messages[parsed[:avs_result_code]]
381
+ else
382
+ message = message.chomp('.')
383
+ end
384
+ elsif message == 'Missing ACCOUNT_ID'
385
+ message = 'The merchant login ID or password is invalid'
386
+ elsif /Approved/.match?(message)
387
+ message = 'This transaction has been approved'
388
+ elsif /Expired/.match?(message)
389
+ message = 'The credit card has expired'
390
+ end
391
+ message
392
+ end
393
+
394
+ def add_invoice(post, options)
395
+ post[:ORDER_ID] = options[:order_id]
396
+ post[:INVOICE_ID] = options[:invoice]
397
+ post[:invoice_num] = options[:order_id]
398
+ post[:MEMO] = options[:description]
399
+ post[:description] = options[:description]
400
+ end
401
+
402
+ def add_payment_method(post, payment_object)
403
+ post[:MASTER_ID] = ''
404
+ case payment_object
405
+ when String
406
+ post[:MASTER_ID] = payment_object
407
+ when Check
408
+ add_check(post, payment_object)
409
+ else
410
+ add_creditcard(post, payment_object)
411
+ end
412
+ end
413
+
414
+ def add_creditcard(post, creditcard)
415
+ post[:PAYMENT_TYPE] = 'CREDIT'
416
+ post[:PAYMENT_ACCOUNT] = creditcard.number
417
+ post[:CARD_CVV2] = creditcard.verification_value
418
+ post[:CARD_EXPIRE] = expdate(creditcard)
419
+ post[:NAME1] = creditcard.first_name
420
+ post[:NAME2] = creditcard.last_name
421
+ end
422
+
423
+ CHECK_ACCOUNT_TYPES = {
424
+ 'checking' => 'C',
425
+ 'savings' => 'S'
426
+ }
427
+
428
+ def add_check(post, check)
429
+ post[:PAYMENT_TYPE] = 'ACH'
430
+ post[:PAYMENT_ACCOUNT] = [CHECK_ACCOUNT_TYPES[check.account_type], check.routing_number, check.account_number].join(':')
431
+ post[:NAME1] = check.first_name
432
+ post[:NAME2] = check.last_name
433
+ end
434
+
435
+ def add_customer_data(post, options)
436
+ post[:EMAIL] = options[:email]
437
+ post[:CUSTOM_ID] = options[:customer]
438
+ post[:CUSTOM_ID2] = options[:custom_id2]
439
+ end
440
+
441
+ def add_duplicate_override(post, options)
442
+ post[:DUPLICATE_OVERRIDE] = options[:duplicate_override]
443
+ end
444
+
445
+ def add_address(post, options)
446
+ if address = (options[:shipping_address] || options[:billing_address] || options[:address])
447
+ post[:ADDR1] = address[:address1]
448
+ post[:ADDR2] = address[:address2]
449
+ post[:COMPANY_NAME] = address[:company]
450
+ post[:PHONE] = address[:phone]
451
+ post[:CITY] = address[:city]
452
+ post[:STATE] = (address[:state].blank? ? 'n/a' : address[:state])
453
+ post[:ZIP] = address[:zip]
454
+ post[:COUNTRY] = address[:country]
455
+ end
456
+ end
457
+
458
+ def add_rebill(post, options)
459
+ post[:DO_REBILL] = '1'
460
+ post[:REB_AMOUNT] = amount(options[:rebill_amount])
461
+ post[:REB_FIRST_DATE] = options[:rebill_start_date]
462
+ post[:REB_EXPR] = options[:rebill_expression]
463
+ post[:REB_CYCLES] = options[:rebill_cycles]
464
+ end
465
+
466
+ def add_stored_credential(post, options)
467
+ post[:cof] = initiator(options)
468
+ post[:cofscheduled] = scheduled(options)
469
+ end
470
+
471
+ def initiator(options)
472
+ return unless initiator = options.dig(:stored_credential, :initiator)
473
+
474
+ case initiator
475
+ when 'merchant'
476
+ 'M'
477
+ when 'cardholder'
478
+ 'C'
479
+ end
480
+ end
481
+
482
+ def scheduled(options)
483
+ return unless reason_type = options.dig(:stored_credential, :reason_type)
484
+
485
+ case reason_type
486
+ when 'recurring', 'installment'
487
+ 'Y'
488
+ when 'unscheduled'
489
+ 'N'
490
+ end
491
+ end
492
+
493
+ def post_data(action, parameters = {})
494
+ post = {}
495
+ post[:version] = '1'
496
+ post[:login] = ''
497
+ post[:tran_key] = ''
498
+ post[:relay_response] = 'FALSE'
499
+ post[:type] = action
500
+ post[:delim_data] = 'TRUE'
501
+ post[:delim_char] = ','
502
+ post[:encap_char] = '$'
503
+ post[:card_num] = '4111111111111111'
504
+ post[:exp_date] = '1212'
505
+ post[:solution_ID] = application_id if application_id
506
+ post.merge(parameters).collect { |key, value| "#{key}=#{CGI.escape(value.to_s)}" }.join('&')
507
+ end
508
+
509
+ def expdate(creditcard)
510
+ year = format(creditcard.year, :two_digits)
511
+ month = format(creditcard.month, :two_digits)
512
+
513
+ "#{month}#{year}"
514
+ end
515
+
516
+ def calc_tps(amount, post)
517
+ post[:NAME1] ||= ''
518
+ Digest::MD5.hexdigest(
519
+ [
520
+ @options[:password],
521
+ @options[:login],
522
+ post[:TRANS_TYPE],
523
+ amount,
524
+ post[:MASTER_ID],
525
+ post[:NAME1],
526
+ post[:PAYMENT_ACCOUNT]
527
+ ].join('')
528
+ )
529
+ end
530
+
531
+ def calc_rebill_tps(post)
532
+ Digest::MD5.hexdigest(
533
+ [
534
+ @options[:password],
535
+ @options[:login],
536
+ post[:TRANS_TYPE],
537
+ post[:REBILL_ID]
538
+ ].join('')
539
+ )
540
+ end
541
+
542
+ def handle_response(response)
543
+ return response.body if ignore_http_status || (200...300).cover?(response.code.to_i)
544
+
545
+ raise ResponseError.new(response)
546
+ end
547
+ end
548
+ end
549
+ end