aktivemerchant 2.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/CHANGELOG +1596 -0
- data/CONTRIBUTORS +511 -0
- data/MIT-LICENSE +20 -0
- data/README.md +18 -0
- data/lib/active_merchant.rb +108 -0
- data/lib/active_merchant/billing.rb +13 -0
- data/lib/active_merchant/billing/apple_pay_payment_token.rb +22 -0
- data/lib/active_merchant/billing/avs_result.rb +98 -0
- data/lib/active_merchant/billing/base.rb +72 -0
- data/lib/active_merchant/billing/check.rb +76 -0
- data/lib/active_merchant/billing/compatibility.rb +120 -0
- data/lib/active_merchant/billing/credit_card.rb +352 -0
- data/lib/active_merchant/billing/credit_card_formatting.rb +24 -0
- data/lib/active_merchant/billing/credit_card_methods.rb +160 -0
- data/lib/active_merchant/billing/cvv_result.rb +38 -0
- data/lib/active_merchant/billing/gateway.rb +268 -0
- data/lib/active_merchant/billing/gateways.rb +17 -0
- data/lib/active_merchant/billing/gateways/adyen.rb +209 -0
- data/lib/active_merchant/billing/gateways/alfabank.rb +117 -0
- data/lib/active_merchant/billing/gateways/app55.rb +176 -0
- data/lib/active_merchant/billing/gateways/authorize_net.rb +419 -0
- data/lib/active_merchant/billing/gateways/authorize_net_arb.rb +417 -0
- data/lib/active_merchant/billing/gateways/authorize_net_cim.rb +976 -0
- data/lib/active_merchant/billing/gateways/balanced.rb +256 -0
- data/lib/active_merchant/billing/gateways/bank_frick.rb +225 -0
- data/lib/active_merchant/billing/gateways/banwire.rb +105 -0
- data/lib/active_merchant/billing/gateways/barclays_epdq.rb +314 -0
- data/lib/active_merchant/billing/gateways/barclays_epdq_extra_plus.rb +15 -0
- data/lib/active_merchant/billing/gateways/be2bill.rb +131 -0
- data/lib/active_merchant/billing/gateways/beanstream.rb +188 -0
- data/lib/active_merchant/billing/gateways/beanstream/beanstream_core.rb +393 -0
- data/lib/active_merchant/billing/gateways/beanstream_interac.rb +54 -0
- data/lib/active_merchant/billing/gateways/blue_pay.rb +506 -0
- data/lib/active_merchant/billing/gateways/bogus.rb +140 -0
- data/lib/active_merchant/billing/gateways/borgun.rb +210 -0
- data/lib/active_merchant/billing/gateways/braintree.rb +19 -0
- data/lib/active_merchant/billing/gateways/braintree/braintree_common.rb +9 -0
- data/lib/active_merchant/billing/gateways/braintree_blue.rb +515 -0
- data/lib/active_merchant/billing/gateways/braintree_orange.rb +20 -0
- data/lib/active_merchant/billing/gateways/bridge_pay.rb +189 -0
- data/lib/active_merchant/billing/gateways/card_save.rb +23 -0
- data/lib/active_merchant/billing/gateways/card_stream.rb +220 -0
- data/lib/active_merchant/billing/gateways/cashnet.rb +191 -0
- data/lib/active_merchant/billing/gateways/cc5.rb +201 -0
- data/lib/active_merchant/billing/gateways/cecabank.rb +229 -0
- data/lib/active_merchant/billing/gateways/certo_direct.rb +278 -0
- data/lib/active_merchant/billing/gateways/checkout.rb +213 -0
- data/lib/active_merchant/billing/gateways/commercegate.rb +143 -0
- data/lib/active_merchant/billing/gateways/conekta.rb +209 -0
- data/lib/active_merchant/billing/gateways/cyber_source.rb +709 -0
- data/lib/active_merchant/billing/gateways/data_cash.rb +600 -0
- data/lib/active_merchant/billing/gateways/efsnet.rb +219 -0
- data/lib/active_merchant/billing/gateways/elavon.rb +348 -0
- data/lib/active_merchant/billing/gateways/epay.rb +275 -0
- data/lib/active_merchant/billing/gateways/evo_ca.rb +308 -0
- data/lib/active_merchant/billing/gateways/eway.rb +214 -0
- data/lib/active_merchant/billing/gateways/eway_managed.rb +291 -0
- data/lib/active_merchant/billing/gateways/eway_rapid.rb +524 -0
- data/lib/active_merchant/billing/gateways/exact.rb +218 -0
- data/lib/active_merchant/billing/gateways/fat_zebra.rb +173 -0
- data/lib/active_merchant/billing/gateways/federated_canada.rb +160 -0
- data/lib/active_merchant/billing/gateways/finansbank.rb +23 -0
- data/lib/active_merchant/billing/gateways/first_giving.rb +143 -0
- data/lib/active_merchant/billing/gateways/first_pay.rb +160 -0
- data/lib/active_merchant/billing/gateways/firstdata_e4.rb +355 -0
- data/lib/active_merchant/billing/gateways/garanti.rb +257 -0
- data/lib/active_merchant/billing/gateways/global_transport.rb +183 -0
- data/lib/active_merchant/billing/gateways/hdfc.rb +207 -0
- data/lib/active_merchant/billing/gateways/hps.rb +288 -0
- data/lib/active_merchant/billing/gateways/iats_payments.rb +251 -0
- data/lib/active_merchant/billing/gateways/ideal/ideal_base.rb +246 -0
- data/lib/active_merchant/billing/gateways/ideal/ideal_rabobank.pem +13 -0
- data/lib/active_merchant/billing/gateways/ideal/ideal_response.rb +29 -0
- data/lib/active_merchant/billing/gateways/ideal_rabobank.rb +66 -0
- data/lib/active_merchant/billing/gateways/inspire.rb +213 -0
- data/lib/active_merchant/billing/gateways/instapay.rb +163 -0
- data/lib/active_merchant/billing/gateways/iridium.rb +457 -0
- data/lib/active_merchant/billing/gateways/itransact.rb +448 -0
- data/lib/active_merchant/billing/gateways/jetpay.rb +275 -0
- data/lib/active_merchant/billing/gateways/linkpoint.rb +438 -0
- data/lib/active_merchant/billing/gateways/litle.rb +346 -0
- data/lib/active_merchant/billing/gateways/maxipago.rb +197 -0
- data/lib/active_merchant/billing/gateways/merchant_e_solutions.rb +170 -0
- data/lib/active_merchant/billing/gateways/merchant_one.rb +114 -0
- data/lib/active_merchant/billing/gateways/merchant_ware.rb +319 -0
- data/lib/active_merchant/billing/gateways/merchant_ware_version_four.rb +268 -0
- data/lib/active_merchant/billing/gateways/merchant_warrior.rb +195 -0
- data/lib/active_merchant/billing/gateways/mercury.rb +333 -0
- data/lib/active_merchant/billing/gateways/metrics_global.rb +303 -0
- data/lib/active_merchant/billing/gateways/migs.rb +265 -0
- data/lib/active_merchant/billing/gateways/migs/migs_codes.rb +100 -0
- data/lib/active_merchant/billing/gateways/modern_payments.rb +37 -0
- data/lib/active_merchant/billing/gateways/modern_payments_cim.rb +219 -0
- data/lib/active_merchant/billing/gateways/moneris.rb +309 -0
- data/lib/active_merchant/billing/gateways/moneris_us.rb +291 -0
- data/lib/active_merchant/billing/gateways/money_movers.rb +152 -0
- data/lib/active_merchant/billing/gateways/nab_transact.rb +280 -0
- data/lib/active_merchant/billing/gateways/net_registry.rb +198 -0
- data/lib/active_merchant/billing/gateways/netaxept.rb +181 -0
- data/lib/active_merchant/billing/gateways/netbilling.rb +190 -0
- data/lib/active_merchant/billing/gateways/netpay.rb +223 -0
- data/lib/active_merchant/billing/gateways/network_merchants.rb +242 -0
- data/lib/active_merchant/billing/gateways/nmi.rb +256 -0
- data/lib/active_merchant/billing/gateways/ogone.rb +435 -0
- data/lib/active_merchant/billing/gateways/openpay.rb +194 -0
- data/lib/active_merchant/billing/gateways/optimal_payment.rb +313 -0
- data/lib/active_merchant/billing/gateways/orbital.rb +803 -0
- data/lib/active_merchant/billing/gateways/orbital/orbital_soft_descriptors.rb +47 -0
- data/lib/active_merchant/billing/gateways/pac_net_raven.rb +207 -0
- data/lib/active_merchant/billing/gateways/pago_facil.rb +122 -0
- data/lib/active_merchant/billing/gateways/pay_gate_xml.rb +261 -0
- data/lib/active_merchant/billing/gateways/pay_junction.rb +390 -0
- data/lib/active_merchant/billing/gateways/pay_secure.rb +112 -0
- data/lib/active_merchant/billing/gateways/pay_u_latam.rb +462 -0
- data/lib/active_merchant/billing/gateways/paybox_direct.rb +188 -0
- data/lib/active_merchant/billing/gateways/payex.rb +412 -0
- data/lib/active_merchant/billing/gateways/payflow.rb +304 -0
- data/lib/active_merchant/billing/gateways/payflow/payflow_common_api.rb +209 -0
- data/lib/active_merchant/billing/gateways/payflow/payflow_express_response.rb +39 -0
- data/lib/active_merchant/billing/gateways/payflow/payflow_response.rb +13 -0
- data/lib/active_merchant/billing/gateways/payflow_express.rb +224 -0
- data/lib/active_merchant/billing/gateways/payflow_express_uk.rb +15 -0
- data/lib/active_merchant/billing/gateways/payflow_uk.rb +21 -0
- data/lib/active_merchant/billing/gateways/payment_express.rb +353 -0
- data/lib/active_merchant/billing/gateways/paymill.rb +281 -0
- data/lib/active_merchant/billing/gateways/paypal.rb +117 -0
- data/lib/active_merchant/billing/gateways/paypal/paypal_common_api.rb +670 -0
- data/lib/active_merchant/billing/gateways/paypal/paypal_express_response.rb +65 -0
- data/lib/active_merchant/billing/gateways/paypal/paypal_recurring_api.rb +262 -0
- data/lib/active_merchant/billing/gateways/paypal_ca.rb +13 -0
- data/lib/active_merchant/billing/gateways/paypal_digital_goods.rb +44 -0
- data/lib/active_merchant/billing/gateways/paypal_express.rb +264 -0
- data/lib/active_merchant/billing/gateways/paypal_express_common.rb +30 -0
- data/lib/active_merchant/billing/gateways/payscout.rb +162 -0
- data/lib/active_merchant/billing/gateways/paystation.rb +199 -0
- data/lib/active_merchant/billing/gateways/payway.rb +207 -0
- data/lib/active_merchant/billing/gateways/pin.rb +197 -0
- data/lib/active_merchant/billing/gateways/plugnpay.rb +283 -0
- data/lib/active_merchant/billing/gateways/psigate.rb +216 -0
- data/lib/active_merchant/billing/gateways/psl_card.rb +303 -0
- data/lib/active_merchant/billing/gateways/qbms.rb +292 -0
- data/lib/active_merchant/billing/gateways/quantum.rb +276 -0
- data/lib/active_merchant/billing/gateways/quickpay.rb +367 -0
- data/lib/active_merchant/billing/gateways/realex.rb +298 -0
- data/lib/active_merchant/billing/gateways/redsys.rb +391 -0
- data/lib/active_merchant/billing/gateways/sage.rb +175 -0
- data/lib/active_merchant/billing/gateways/sage/sage_bankcard.rb +87 -0
- data/lib/active_merchant/billing/gateways/sage/sage_core.rb +114 -0
- data/lib/active_merchant/billing/gateways/sage/sage_vault.rb +149 -0
- data/lib/active_merchant/billing/gateways/sage/sage_virtual_check.rb +102 -0
- data/lib/active_merchant/billing/gateways/sage_pay.rb +398 -0
- data/lib/active_merchant/billing/gateways/sallie_mae.rb +143 -0
- data/lib/active_merchant/billing/gateways/secure_net.rb +252 -0
- data/lib/active_merchant/billing/gateways/secure_pay.rb +201 -0
- data/lib/active_merchant/billing/gateways/secure_pay_au.rb +281 -0
- data/lib/active_merchant/billing/gateways/secure_pay_tech.rb +105 -0
- data/lib/active_merchant/billing/gateways/skip_jack.rb +452 -0
- data/lib/active_merchant/billing/gateways/smart_ps.rb +283 -0
- data/lib/active_merchant/billing/gateways/so_easy_pay.rb +194 -0
- data/lib/active_merchant/billing/gateways/spreedly_core.rb +247 -0
- data/lib/active_merchant/billing/gateways/stripe.rb +411 -0
- data/lib/active_merchant/billing/gateways/swipe_checkout.rb +157 -0
- data/lib/active_merchant/billing/gateways/tns.rb +227 -0
- data/lib/active_merchant/billing/gateways/trans_first.rb +126 -0
- data/lib/active_merchant/billing/gateways/transax.rb +23 -0
- data/lib/active_merchant/billing/gateways/transnational.rb +10 -0
- data/lib/active_merchant/billing/gateways/trust_commerce.rb +416 -0
- data/lib/active_merchant/billing/gateways/usa_epay.rb +25 -0
- data/lib/active_merchant/billing/gateways/usa_epay_advanced.rb +1516 -0
- data/lib/active_merchant/billing/gateways/usa_epay_transaction.rb +254 -0
- data/lib/active_merchant/billing/gateways/verifi.rb +225 -0
- data/lib/active_merchant/billing/gateways/viaklix.rb +183 -0
- data/lib/active_merchant/billing/gateways/vindicia.rb +385 -0
- data/lib/active_merchant/billing/gateways/webpay.rb +97 -0
- data/lib/active_merchant/billing/gateways/wepay.rb +189 -0
- data/lib/active_merchant/billing/gateways/wirecard.rb +421 -0
- data/lib/active_merchant/billing/gateways/worldpay.rb +331 -0
- data/lib/active_merchant/billing/gateways/worldpay_us.rb +181 -0
- data/lib/active_merchant/billing/model.rb +30 -0
- data/lib/active_merchant/billing/payment_token.rb +21 -0
- data/lib/active_merchant/billing/rails.rb +3 -0
- data/lib/active_merchant/billing/response.rb +91 -0
- data/lib/active_merchant/country.rb +332 -0
- data/lib/active_merchant/empty.rb +20 -0
- data/lib/active_merchant/errors.rb +29 -0
- data/lib/active_merchant/offsite_payments_shim.rb +19 -0
- data/lib/active_merchant/version.rb +3 -0
- data/lib/activemerchant.rb +1 -0
- data/lib/support/gateway_support.rb +71 -0
- data/lib/support/outbound_hosts.rb +25 -0
- data/lib/support/ssl_verify.rb +93 -0
- metadata +400 -0
@@ -0,0 +1,143 @@
|
|
1
|
+
module ActiveMerchant #:nodoc:
|
2
|
+
module Billing #:nodoc:
|
3
|
+
class CommercegateGateway < Gateway
|
4
|
+
self.test_url = self.live_url = 'https://secure.commercegate.com/gateway/nvp'
|
5
|
+
|
6
|
+
self.supported_countries = %w(
|
7
|
+
AD AT AX BE BG CH CY CZ DE DK ES FI FR GB GG
|
8
|
+
GI GR HR HU IE IM IS IT JE LI LT LU LV MC MT
|
9
|
+
NL NO PL PT RO SE SI SK VA
|
10
|
+
)
|
11
|
+
|
12
|
+
self.money_format = :dollars
|
13
|
+
self.default_currency = 'EUR'
|
14
|
+
self.supported_cardtypes = [:visa, :master, :american_express, :discover]
|
15
|
+
self.homepage_url = 'http://www.commercegate.com/'
|
16
|
+
self.display_name = 'CommerceGate'
|
17
|
+
|
18
|
+
def initialize(options = {})
|
19
|
+
requires!(options, :login, :password, :site_id, :offer_id)
|
20
|
+
super
|
21
|
+
end
|
22
|
+
|
23
|
+
def authorize(money, creditcard, options = {})
|
24
|
+
post = {}
|
25
|
+
add_creditcard(post, creditcard)
|
26
|
+
add_auth_purchase_options(post, money, options)
|
27
|
+
commit('AUTH', post)
|
28
|
+
end
|
29
|
+
|
30
|
+
def capture(money, authorization, options = {})
|
31
|
+
post = {}
|
32
|
+
post[:currencyCode] = (options[:currency] || currency(money))
|
33
|
+
post[:amount] = amount(money)
|
34
|
+
post[:transID] = authorization
|
35
|
+
commit('CAPTURE', post)
|
36
|
+
end
|
37
|
+
|
38
|
+
def purchase(money, creditcard, options = {})
|
39
|
+
post = {}
|
40
|
+
add_creditcard(post, creditcard)
|
41
|
+
add_auth_purchase_options(post, money, options)
|
42
|
+
commit('SALE', post)
|
43
|
+
end
|
44
|
+
|
45
|
+
def refund(money, identification, options = {})
|
46
|
+
post = {}
|
47
|
+
post[:currencyCode] = options[:currency] || currency(money)
|
48
|
+
post[:amount] = amount(money)
|
49
|
+
post[:transID] = identification
|
50
|
+
commit('REFUND', post)
|
51
|
+
end
|
52
|
+
|
53
|
+
def void(identification, options = {})
|
54
|
+
post = {}
|
55
|
+
post[:transID] = identification
|
56
|
+
commit('VOID_AUTH', post)
|
57
|
+
end
|
58
|
+
|
59
|
+
private
|
60
|
+
|
61
|
+
def add_address(post, address)
|
62
|
+
if address
|
63
|
+
post[:address] = address[:address1]
|
64
|
+
post[:city] = address[:city]
|
65
|
+
post[:state] = address[:state]
|
66
|
+
post[:postalCode] = address[:zip]
|
67
|
+
end
|
68
|
+
post[:countryCode] = ((address && address[:country]) || "US")
|
69
|
+
end
|
70
|
+
|
71
|
+
def add_auth_purchase_options(post, money, options)
|
72
|
+
add_address(post, options[:address])
|
73
|
+
|
74
|
+
post[:customerIP] = options[:ip] || "127.0.0.1"
|
75
|
+
post[:amount] = amount(money)
|
76
|
+
post[:email] = options[:email] || "unknown@example.com"
|
77
|
+
post[:currencyCode]= options[:currency] || currency(money)
|
78
|
+
post[:merchAcct] = options[:merchant]
|
79
|
+
|
80
|
+
end
|
81
|
+
|
82
|
+
def add_creditcard(params, creditcard)
|
83
|
+
params[:firstName] = creditcard.first_name
|
84
|
+
params[:lastName] = creditcard.last_name
|
85
|
+
params[:cardNumber] = creditcard.number
|
86
|
+
params[:expiryMonth] = creditcard.month
|
87
|
+
params[:expiryYear] = creditcard.year
|
88
|
+
params[:cvv] = creditcard.verification_value if creditcard.verification_value?
|
89
|
+
end
|
90
|
+
|
91
|
+
def commit(action, parameters)
|
92
|
+
parameters[:apiUsername] = @options[:login]
|
93
|
+
parameters[:apiPassword] = @options[:password]
|
94
|
+
parameters[:siteID] = @options[:site_id]
|
95
|
+
parameters[:offerID] = @options[:offer_id]
|
96
|
+
parameters[:action] = action
|
97
|
+
|
98
|
+
response = parse(ssl_post(self.live_url, post_data(parameters)))
|
99
|
+
|
100
|
+
Response.new(
|
101
|
+
successful?(response),
|
102
|
+
message_from(response),
|
103
|
+
response,
|
104
|
+
authorization: response['transID'],
|
105
|
+
test: test?,
|
106
|
+
avs_result: {code: response['avsCode']},
|
107
|
+
cvv_result: response['cvvCode']
|
108
|
+
)
|
109
|
+
end
|
110
|
+
|
111
|
+
def parse(body)
|
112
|
+
results = {}
|
113
|
+
|
114
|
+
body.split(/\&/).each do |pair|
|
115
|
+
key,val = pair.split(%r{=})
|
116
|
+
results[key] = CGI.unescape(val)
|
117
|
+
end
|
118
|
+
|
119
|
+
results
|
120
|
+
end
|
121
|
+
|
122
|
+
def successful?(response)
|
123
|
+
response['returnCode'] == '0'
|
124
|
+
end
|
125
|
+
|
126
|
+
def message_from(response)
|
127
|
+
if response['returnText'].present?
|
128
|
+
response['returnText']
|
129
|
+
else
|
130
|
+
"Invalid response received from the CommerceGate API. " +
|
131
|
+
"Please contact CommerceGate support if you continue to receive this message. " +
|
132
|
+
"(The raw response returned by the API was #{response.inspect})"
|
133
|
+
end
|
134
|
+
end
|
135
|
+
|
136
|
+
def post_data(parameters)
|
137
|
+
parameters.collect do |key, value|
|
138
|
+
"#{key}=#{CGI.escape(value.to_s)}"
|
139
|
+
end.join("&")
|
140
|
+
end
|
141
|
+
end
|
142
|
+
end
|
143
|
+
end
|
@@ -0,0 +1,209 @@
|
|
1
|
+
module ActiveMerchant #:nodoc:
|
2
|
+
module Billing #:nodoc:
|
3
|
+
class ConektaGateway < Gateway
|
4
|
+
self.live_url = 'https://api.conekta.io/'
|
5
|
+
|
6
|
+
self.supported_countries = ['MX']
|
7
|
+
self.supported_cardtypes = [:visa, :master, :american_express]
|
8
|
+
self.homepage_url = 'https://conekta.io/'
|
9
|
+
self.display_name = 'Conekta Gateway'
|
10
|
+
self.money_format = :cents
|
11
|
+
self.default_currency = 'MXN'
|
12
|
+
|
13
|
+
def initialize(options = {})
|
14
|
+
requires!(options, :key)
|
15
|
+
options[:version] ||= '0.3.0'
|
16
|
+
super
|
17
|
+
end
|
18
|
+
|
19
|
+
def purchase(money, payment_source, options = {})
|
20
|
+
post = {}
|
21
|
+
|
22
|
+
add_order(post, money, options)
|
23
|
+
add_payment_source(post, payment_source, options)
|
24
|
+
add_details_data(post, options)
|
25
|
+
|
26
|
+
commit(:post, 'charges', post)
|
27
|
+
end
|
28
|
+
|
29
|
+
def authorize(money, payment_source, options = {})
|
30
|
+
post = {}
|
31
|
+
|
32
|
+
add_order(post, money, options)
|
33
|
+
add_payment_source(post, payment_source, options)
|
34
|
+
add_details_data(post, options)
|
35
|
+
|
36
|
+
post[:capture] = false
|
37
|
+
commit(:post, "charges", post)
|
38
|
+
end
|
39
|
+
|
40
|
+
def capture(money, identifier, options = {})
|
41
|
+
post = {}
|
42
|
+
|
43
|
+
post[:order_id] = identifier
|
44
|
+
add_order(post, money, options)
|
45
|
+
|
46
|
+
commit(:post, "charges/#{identifier}/capture", post)
|
47
|
+
end
|
48
|
+
|
49
|
+
def refund(money, identifier, options)
|
50
|
+
post = {}
|
51
|
+
|
52
|
+
post[:order_id] = identifier
|
53
|
+
add_order(post, money, options)
|
54
|
+
|
55
|
+
commit(:post, "charges/#{identifier}/refund", post)
|
56
|
+
end
|
57
|
+
|
58
|
+
private
|
59
|
+
|
60
|
+
def add_order(post, money, options)
|
61
|
+
post[:description] = options[:description] || "Active Merchant Purchase"
|
62
|
+
post[:reference_id] = options[:order_id] if options[:order_id]
|
63
|
+
post[:currency] = (options[:currency] || currency(money)).downcase
|
64
|
+
post[:amount] = amount(money)
|
65
|
+
end
|
66
|
+
|
67
|
+
def add_details_data(post, options)
|
68
|
+
details = {}
|
69
|
+
details[:name] = options[:customer] if options[:customer]
|
70
|
+
details[:email] = options[:email] if options[:email]
|
71
|
+
details[:phone] = options[:phone] if options[:phone]
|
72
|
+
details[:device_fingerprint] = options[:device_fingerprint] if options[:device_fingerprint]
|
73
|
+
details[:ip] = options[:ip] if options[:ip]
|
74
|
+
add_billing_address(details, options)
|
75
|
+
add_line_items(details, options)
|
76
|
+
add_shipment(details, options)
|
77
|
+
|
78
|
+
post[:details] = details
|
79
|
+
end
|
80
|
+
|
81
|
+
def add_shipment(post, options)
|
82
|
+
shipment = {}
|
83
|
+
shipment[:carrier] = options[:carrier] if options[:carrier]
|
84
|
+
shipment[:service] = options[:service] if options[:service]
|
85
|
+
shipment[:tracking_number] = options[:tracking_number] if options[:tracking_number]
|
86
|
+
shipment[:price] = options[:price] if options[:price]
|
87
|
+
add_shipment_address(shipment, options)
|
88
|
+
post[:shipment] = shipment
|
89
|
+
end
|
90
|
+
|
91
|
+
def add_shipment_address(post, options)
|
92
|
+
if(address = options[:shipping_address])
|
93
|
+
post[:address] = {}
|
94
|
+
post[:address][:street1] = address[:address1] if address[:address1]
|
95
|
+
post[:address][:street2] = address[:address2] if address[:address2]
|
96
|
+
post[:address][:street3] = address[:address3] if address[:address3]
|
97
|
+
post[:address][:city] = address[:city] if address[:city]
|
98
|
+
post[:address][:state] = address[:state] if address[:state]
|
99
|
+
post[:address][:country] = address[:country] if address[:country]
|
100
|
+
post[:address][:zip] = address[:zip] if address[:zip]
|
101
|
+
end
|
102
|
+
end
|
103
|
+
|
104
|
+
def add_line_items(post, options)
|
105
|
+
post[:line_items] = (options[:line_items] || []).collect do |line_item|
|
106
|
+
line_item
|
107
|
+
end
|
108
|
+
end
|
109
|
+
|
110
|
+
def add_billing_address(post, options)
|
111
|
+
if(address = (options[:billing_address] || options[:address]))
|
112
|
+
post[:billing_address] = {}
|
113
|
+
post[:billing_address][:street1] = address[:address1] if address[:address1]
|
114
|
+
post[:billing_address][:street2] = address[:address2] if address[:address2]
|
115
|
+
post[:billing_address][:street3] = address[:address3] if address[:address3]
|
116
|
+
post[:billing_address][:city] = address[:city] if address[:city]
|
117
|
+
post[:billing_address][:state] = address[:state] if address[:state]
|
118
|
+
post[:billing_address][:country] = address[:country] if address[:country]
|
119
|
+
post[:billing_address][:zip] = address[:zip] if address[:zip]
|
120
|
+
post[:billing_address][:company_name] = address[:company_name] if address[:company_name]
|
121
|
+
post[:billing_address][:tax_id] = address[:tax_id] if address[:tax_id]
|
122
|
+
post[:billing_address][:name] = address[:name] if address[:name]
|
123
|
+
post[:billing_address][:phone] = address[:phone] if address[:phone]
|
124
|
+
post[:billing_address][:email] = address[:email] if address[:email]
|
125
|
+
end
|
126
|
+
end
|
127
|
+
|
128
|
+
def add_address(post, options)
|
129
|
+
if(address = (options[:billing_address] || options[:address]))
|
130
|
+
post[:address] = {}
|
131
|
+
post[:address][:street1] = address[:address1] if address[:address1]
|
132
|
+
post[:address][:street2] = address[:address2] if address[:address2]
|
133
|
+
post[:address][:street3] = address[:address3] if address[:address3]
|
134
|
+
post[:address][:city] = address[:city] if address[:city]
|
135
|
+
post[:address][:state] = address[:state] if address[:state]
|
136
|
+
post[:address][:country] = address[:country] if address[:country]
|
137
|
+
post[:address][:zip] = address[:zip] if address[:zip]
|
138
|
+
end
|
139
|
+
end
|
140
|
+
|
141
|
+
def add_payment_source(post, payment_source, options)
|
142
|
+
if payment_source.kind_of?(String)
|
143
|
+
post[:card] = payment_source
|
144
|
+
elsif payment_source.respond_to?(:number)
|
145
|
+
post[:card] = {}
|
146
|
+
post[:card][:name] = payment_source.name
|
147
|
+
post[:card][:cvc] = payment_source.verification_value
|
148
|
+
post[:card][:number] = payment_source.number
|
149
|
+
post[:card][:exp_month] = "#{sprintf("%02d", payment_source.month)}"
|
150
|
+
post[:card][:exp_year] = "#{"#{payment_source.year}"[-2, 2]}"
|
151
|
+
add_address(post[:card], options)
|
152
|
+
end
|
153
|
+
end
|
154
|
+
|
155
|
+
def parse(body)
|
156
|
+
return {} unless body
|
157
|
+
JSON.parse(body)
|
158
|
+
end
|
159
|
+
|
160
|
+
def headers(meta)
|
161
|
+
{
|
162
|
+
"Accept" => "application/vnd.conekta-v#{options[:version]}+json",
|
163
|
+
"Authorization" => "Basic " + Base64.encode64("#{options[:key]}:"),
|
164
|
+
"RaiseHtmlError" => "false",
|
165
|
+
"Conekta-Client-User-Agent" => {"agent"=>"Conekta ActiveMerchantBindings/#{ActiveMerchant::VERSION}"}.to_json,
|
166
|
+
"X-Conekta-Client-User-Agent" => user_agent,
|
167
|
+
"X-Conekta-Client-User-Metadata" => meta.to_json
|
168
|
+
}
|
169
|
+
end
|
170
|
+
|
171
|
+
def commit(method, url, parameters, options = {})
|
172
|
+
success = false
|
173
|
+
begin
|
174
|
+
raw_response = parse(ssl_request(method, live_url + url, (parameters ? parameters.to_query : nil), headers(options[:meta])))
|
175
|
+
success = (raw_response.key?("object") && (raw_response["object"] != "error"))
|
176
|
+
rescue ResponseError => e
|
177
|
+
raw_response = response_error(e.response.body)
|
178
|
+
rescue JSON::ParserError
|
179
|
+
raw_response = json_error(raw_response)
|
180
|
+
end
|
181
|
+
|
182
|
+
Response.new(
|
183
|
+
success,
|
184
|
+
raw_response["message"],
|
185
|
+
raw_response,
|
186
|
+
test: test?,
|
187
|
+
authorization: raw_response["id"]
|
188
|
+
)
|
189
|
+
end
|
190
|
+
|
191
|
+
def response_error(raw_response)
|
192
|
+
begin
|
193
|
+
parse(raw_response)
|
194
|
+
rescue JSON::ParserError
|
195
|
+
json_error(raw_response)
|
196
|
+
end
|
197
|
+
end
|
198
|
+
|
199
|
+
def json_error(raw_response)
|
200
|
+
msg = 'Invalid response received from the Conekta API.'
|
201
|
+
msg += " (The raw response returned by the API was #{raw_response.inspect})"
|
202
|
+
{
|
203
|
+
"message" => msg
|
204
|
+
}
|
205
|
+
end
|
206
|
+
end
|
207
|
+
end
|
208
|
+
end
|
209
|
+
|
@@ -0,0 +1,709 @@
|
|
1
|
+
module ActiveMerchant #:nodoc:
|
2
|
+
module Billing #:nodoc:
|
3
|
+
# See the remote and mocked unit test files for example usage. Pay special
|
4
|
+
# attention to the contents of the options hash.
|
5
|
+
#
|
6
|
+
# Initial setup instructions can be found in
|
7
|
+
# http://cybersource.com/support_center/implementation/downloads/soap_api/SOAP_toolkits.pdf
|
8
|
+
#
|
9
|
+
# Debugging
|
10
|
+
# If you experience an issue with this gateway be sure to examine the
|
11
|
+
# transaction information from a general transaction search inside the
|
12
|
+
# CyberSource Business Center for the full error messages including field
|
13
|
+
# names.
|
14
|
+
#
|
15
|
+
# Important Notes
|
16
|
+
# * For checks you can purchase and store.
|
17
|
+
# * AVS and CVV only work against the production server. You will always
|
18
|
+
# get back X for AVS and no response for CVV against the test server.
|
19
|
+
# * Nexus is the list of states or provinces where you have a physical
|
20
|
+
# presence. Nexus is used to calculate tax. Leave blank to tax everyone.
|
21
|
+
# * If you want to calculate VAT for overseas customers you must supply a
|
22
|
+
# registration number in the options hash as vat_reg_number.
|
23
|
+
# * productCode is a value in the line_items hash that is used to tell
|
24
|
+
# CyberSource what kind of item you are selling. It is used when
|
25
|
+
# calculating tax/VAT.
|
26
|
+
# * All transactions use dollar values.
|
27
|
+
# * To process pinless debit cards through the pinless debit card
|
28
|
+
# network, your Cybersource merchant account must accept pinless
|
29
|
+
# debit card payments.
|
30
|
+
class CyberSourceGateway < Gateway
|
31
|
+
self.test_url = 'https://ics2wstest.ic3.com/commerce/1.x/transactionProcessor'
|
32
|
+
self.live_url = 'https://ics2ws.ic3.com/commerce/1.x/transactionProcessor'
|
33
|
+
|
34
|
+
XSD_VERSION = "1.104"
|
35
|
+
|
36
|
+
# visa, master, american_express, discover
|
37
|
+
self.supported_cardtypes = [:visa, :master, :american_express, :discover]
|
38
|
+
self.supported_countries = %w(US BR CA CN DK FI FR DE JP MX NO SE GB SG)
|
39
|
+
self.default_currency = 'USD'
|
40
|
+
self.homepage_url = 'http://www.cybersource.com'
|
41
|
+
self.display_name = 'CyberSource'
|
42
|
+
|
43
|
+
# map credit card to the CyberSource expected representation
|
44
|
+
@@credit_card_codes = {
|
45
|
+
:visa => '001',
|
46
|
+
:master => '002',
|
47
|
+
:american_express => '003',
|
48
|
+
:discover => '004'
|
49
|
+
}
|
50
|
+
|
51
|
+
# map response codes to something humans can read
|
52
|
+
@@response_codes = {
|
53
|
+
:r100 => "Successful transaction",
|
54
|
+
:r101 => "Request is missing one or more required fields" ,
|
55
|
+
:r102 => "One or more fields contains invalid data",
|
56
|
+
:r150 => "General failure",
|
57
|
+
:r151 => "The request was received but a server time-out occurred",
|
58
|
+
:r152 => "The request was received, but a service timed out",
|
59
|
+
:r200 => "The authorization request was approved by the issuing bank but declined by CyberSource because it did not pass the AVS check",
|
60
|
+
:r201 => "The issuing bank has questions about the request",
|
61
|
+
:r202 => "Expired card",
|
62
|
+
:r203 => "General decline of the card",
|
63
|
+
:r204 => "Insufficient funds in the account",
|
64
|
+
:r205 => "Stolen or lost card",
|
65
|
+
:r207 => "Issuing bank unavailable",
|
66
|
+
:r208 => "Inactive card or card not authorized for card-not-present transactions",
|
67
|
+
:r209 => "American Express Card Identifiction Digits (CID) did not match",
|
68
|
+
:r210 => "The card has reached the credit limit",
|
69
|
+
:r211 => "Invalid card verification number",
|
70
|
+
:r221 => "The customer matched an entry on the processor's negative file",
|
71
|
+
:r230 => "The authorization request was approved by the issuing bank but declined by CyberSource because it did not pass the card verification check",
|
72
|
+
:r231 => "Invalid account number",
|
73
|
+
:r232 => "The card type is not accepted by the payment processor",
|
74
|
+
:r233 => "General decline by the processor",
|
75
|
+
:r234 => "A problem exists with your CyberSource merchant configuration",
|
76
|
+
:r235 => "The requested amount exceeds the originally authorized amount",
|
77
|
+
:r236 => "Processor failure",
|
78
|
+
:r237 => "The authorization has already been reversed",
|
79
|
+
:r238 => "The authorization has already been captured",
|
80
|
+
:r239 => "The requested transaction amount must match the previous transaction amount",
|
81
|
+
:r240 => "The card type sent is invalid or does not correlate with the credit card number",
|
82
|
+
:r241 => "The request ID is invalid",
|
83
|
+
:r242 => "You requested a capture, but there is no corresponding, unused authorization record.",
|
84
|
+
:r243 => "The transaction has already been settled or reversed",
|
85
|
+
:r244 => "The bank account number failed the validation check",
|
86
|
+
:r246 => "The capture or credit is not voidable because the capture or credit information has already been submitted to your processor",
|
87
|
+
:r247 => "You requested a credit for a capture that was previously voided",
|
88
|
+
:r250 => "The request was received, but a time-out occurred with the payment processor",
|
89
|
+
:r254 => "Your CyberSource account is prohibited from processing stand-alone refunds",
|
90
|
+
:r255 => "Your CyberSource account is not configured to process the service in the country you specified"
|
91
|
+
}
|
92
|
+
|
93
|
+
# These are the options that can be used when creating a new CyberSource
|
94
|
+
# Gateway object.
|
95
|
+
#
|
96
|
+
# :login => your username
|
97
|
+
#
|
98
|
+
# :password => the transaction key you generated in the Business Center
|
99
|
+
#
|
100
|
+
# :test => true sets the gateway to test mode
|
101
|
+
#
|
102
|
+
# :vat_reg_number => your VAT registration number
|
103
|
+
#
|
104
|
+
# :nexus => "WI CA QC" sets the states/provinces where you have a physical
|
105
|
+
# presence for tax purposes
|
106
|
+
#
|
107
|
+
# :ignore_avs => true don't want to use AVS so continue processing even
|
108
|
+
# if AVS would have failed
|
109
|
+
#
|
110
|
+
# :ignore_cvv => true don't want to use CVV so continue processing even
|
111
|
+
# if CVV would have failed
|
112
|
+
def initialize(options = {})
|
113
|
+
requires!(options, :login, :password)
|
114
|
+
super
|
115
|
+
end
|
116
|
+
|
117
|
+
# Request an authorization for an amount from CyberSource
|
118
|
+
#
|
119
|
+
# You must supply an :order_id in the options hash
|
120
|
+
def authorize(money, creditcard_or_reference, options = {})
|
121
|
+
requires!(options, :order_id)
|
122
|
+
setup_address_hash(options)
|
123
|
+
commit(build_auth_request(money, creditcard_or_reference, options), options )
|
124
|
+
end
|
125
|
+
|
126
|
+
def auth_reversal(money, identification, options = {})
|
127
|
+
commit(build_auth_reversal_request(money, identification, options), options)
|
128
|
+
end
|
129
|
+
|
130
|
+
# Capture an authorization that has previously been requested
|
131
|
+
def capture(money, authorization, options = {})
|
132
|
+
setup_address_hash(options)
|
133
|
+
commit(build_capture_request(money, authorization, options), options)
|
134
|
+
end
|
135
|
+
|
136
|
+
# Purchase is an auth followed by a capture
|
137
|
+
# You must supply an order_id in the options hash
|
138
|
+
# options[:pinless_debit_card] => true # attempts to process as pinless debit card
|
139
|
+
def purchase(money, payment_method_or_reference, options = {})
|
140
|
+
requires!(options, :order_id)
|
141
|
+
setup_address_hash(options)
|
142
|
+
commit(build_purchase_request(money, payment_method_or_reference, options), options)
|
143
|
+
end
|
144
|
+
|
145
|
+
def void(identification, options = {})
|
146
|
+
commit(build_void_request(identification, options), options)
|
147
|
+
end
|
148
|
+
|
149
|
+
def refund(money, identification, options = {})
|
150
|
+
commit(build_refund_request(money, identification, options), options)
|
151
|
+
end
|
152
|
+
|
153
|
+
# Adds credit to a subscription (stand alone credit).
|
154
|
+
def credit(money, reference, options = {})
|
155
|
+
requires!(options, :order_id)
|
156
|
+
commit(build_credit_request(money, reference, options), options)
|
157
|
+
end
|
158
|
+
|
159
|
+
# Stores a customer subscription/profile with type "on-demand".
|
160
|
+
# To charge the card while creating a profile, pass
|
161
|
+
# options[:setup_fee] => money
|
162
|
+
def store(payment_method, options = {})
|
163
|
+
requires!(options, :order_id)
|
164
|
+
setup_address_hash(options)
|
165
|
+
commit(build_create_subscription_request(payment_method, options), options)
|
166
|
+
end
|
167
|
+
|
168
|
+
# Updates a customer subscription/profile
|
169
|
+
def update(reference, creditcard, options = {})
|
170
|
+
requires!(options, :order_id)
|
171
|
+
setup_address_hash(options)
|
172
|
+
commit(build_update_subscription_request(reference, creditcard, options), options)
|
173
|
+
end
|
174
|
+
|
175
|
+
# Removes a customer subscription/profile
|
176
|
+
def unstore(reference, options = {})
|
177
|
+
requires!(options, :order_id)
|
178
|
+
commit(build_delete_subscription_request(reference, options), options)
|
179
|
+
end
|
180
|
+
|
181
|
+
# Retrieves a customer subscription/profile
|
182
|
+
def retrieve(reference, options = {})
|
183
|
+
requires!(options, :order_id)
|
184
|
+
commit(build_retrieve_subscription_request(reference, options), options)
|
185
|
+
end
|
186
|
+
|
187
|
+
# CyberSource requires that you provide line item information for tax
|
188
|
+
# calculations. If you do not have prices for each item or want to
|
189
|
+
# simplify the situation then pass in one fake line item that costs the
|
190
|
+
# subtotal of the order
|
191
|
+
#
|
192
|
+
# The line_item hash goes in the options hash and should look like
|
193
|
+
#
|
194
|
+
# :line_items => [
|
195
|
+
# {
|
196
|
+
# :declared_value => '1',
|
197
|
+
# :quantity => '2',
|
198
|
+
# :code => 'default',
|
199
|
+
# :description => 'Giant Walrus',
|
200
|
+
# :sku => 'WA323232323232323'
|
201
|
+
# },
|
202
|
+
# {
|
203
|
+
# :declared_value => '6',
|
204
|
+
# :quantity => '1',
|
205
|
+
# :code => 'default',
|
206
|
+
# :description => 'Marble Snowcone',
|
207
|
+
# :sku => 'FAKE1232132113123'
|
208
|
+
# }
|
209
|
+
# ]
|
210
|
+
#
|
211
|
+
# This functionality is only supported by this particular gateway may
|
212
|
+
# be changed at any time
|
213
|
+
def calculate_tax(creditcard, options)
|
214
|
+
requires!(options, :line_items)
|
215
|
+
setup_address_hash(options)
|
216
|
+
commit(build_tax_calculation_request(creditcard, options), options)
|
217
|
+
end
|
218
|
+
|
219
|
+
# Determines if a card can be used for Pinless Debit Card transactions
|
220
|
+
def validate_pinless_debit_card(creditcard, options = {})
|
221
|
+
requires!(options, :order_id)
|
222
|
+
commit(build_validate_pinless_debit_request(creditcard,options), options)
|
223
|
+
end
|
224
|
+
|
225
|
+
private
|
226
|
+
|
227
|
+
# Create all address hash key value pairs so that we still function if we
|
228
|
+
# were only provided with one or two of them
|
229
|
+
def setup_address_hash(options)
|
230
|
+
options[:billing_address] = options[:billing_address] || options[:address] || {}
|
231
|
+
options[:shipping_address] = options[:shipping_address] || {}
|
232
|
+
end
|
233
|
+
|
234
|
+
def build_auth_request(money, creditcard_or_reference, options)
|
235
|
+
xml = Builder::XmlMarkup.new :indent => 2
|
236
|
+
add_payment_method_or_subscription(xml, money, creditcard_or_reference, options)
|
237
|
+
add_auth_service(xml, creditcard_or_reference, options)
|
238
|
+
add_business_rules_data(xml)
|
239
|
+
add_network_token(xml, creditcard_or_reference, options) if token_based_card?(creditcard_or_reference, options)
|
240
|
+
xml.target!
|
241
|
+
end
|
242
|
+
|
243
|
+
def build_tax_calculation_request(creditcard, options)
|
244
|
+
xml = Builder::XmlMarkup.new :indent => 2
|
245
|
+
add_address(xml, creditcard, options[:billing_address], options, false)
|
246
|
+
add_address(xml, creditcard, options[:shipping_address], options, true)
|
247
|
+
add_line_item_data(xml, options)
|
248
|
+
add_purchase_data(xml, 0, false, options)
|
249
|
+
add_tax_service(xml)
|
250
|
+
add_business_rules_data(xml)
|
251
|
+
xml.target!
|
252
|
+
end
|
253
|
+
|
254
|
+
def build_capture_request(money, authorization, options)
|
255
|
+
order_id, request_id, request_token = authorization.split(";")
|
256
|
+
options[:order_id] = order_id
|
257
|
+
|
258
|
+
xml = Builder::XmlMarkup.new :indent => 2
|
259
|
+
add_purchase_data(xml, money, true, options)
|
260
|
+
add_capture_service(xml, request_id, request_token)
|
261
|
+
add_business_rules_data(xml)
|
262
|
+
xml.target!
|
263
|
+
end
|
264
|
+
|
265
|
+
def build_purchase_request(money, payment_method_or_reference, options)
|
266
|
+
xml = Builder::XmlMarkup.new :indent => 2
|
267
|
+
add_payment_method_or_subscription(xml, money, payment_method_or_reference, options)
|
268
|
+
if !payment_method_or_reference.is_a?(String) && card_brand(payment_method_or_reference) == 'check'
|
269
|
+
add_check_service(xml)
|
270
|
+
else
|
271
|
+
add_purchase_service(xml, payment_method_or_reference, options)
|
272
|
+
add_business_rules_data(xml) unless options[:pinless_debit_card]
|
273
|
+
add_network_token(xml, payment_method_or_reference, options) if token_based_card?(payment_method_or_reference, options)
|
274
|
+
end
|
275
|
+
xml.target!
|
276
|
+
end
|
277
|
+
|
278
|
+
def build_void_request(identification, options)
|
279
|
+
order_id, request_id, request_token = identification.split(";")
|
280
|
+
options[:order_id] = order_id
|
281
|
+
|
282
|
+
xml = Builder::XmlMarkup.new :indent => 2
|
283
|
+
add_void_service(xml, request_id, request_token)
|
284
|
+
xml.target!
|
285
|
+
end
|
286
|
+
|
287
|
+
def build_auth_reversal_request(money, identification, options)
|
288
|
+
order_id, request_id, request_token = identification.split(";")
|
289
|
+
options[:order_id] = order_id
|
290
|
+
xml = Builder::XmlMarkup.new :indent => 2
|
291
|
+
add_purchase_data(xml, money, true, options)
|
292
|
+
add_auth_reversal_service(xml, request_id, request_token)
|
293
|
+
xml.target!
|
294
|
+
end
|
295
|
+
|
296
|
+
def build_refund_request(money, identification, options)
|
297
|
+
order_id, request_id, request_token = identification.split(";")
|
298
|
+
options[:order_id] = order_id
|
299
|
+
|
300
|
+
xml = Builder::XmlMarkup.new :indent => 2
|
301
|
+
add_purchase_data(xml, money, true, options)
|
302
|
+
add_credit_service(xml, request_id, request_token)
|
303
|
+
|
304
|
+
xml.target!
|
305
|
+
end
|
306
|
+
|
307
|
+
def build_credit_request(money, reference, options)
|
308
|
+
xml = Builder::XmlMarkup.new :indent => 2
|
309
|
+
|
310
|
+
add_purchase_data(xml, money, true, options)
|
311
|
+
add_subscription(xml, options, reference)
|
312
|
+
add_credit_service(xml)
|
313
|
+
|
314
|
+
xml.target!
|
315
|
+
end
|
316
|
+
|
317
|
+
def build_create_subscription_request(payment_method, options)
|
318
|
+
default_subscription_params = {:frequency => "on-demand", :amount => 0, :automatic_renew => false}
|
319
|
+
options[:subscription] = default_subscription_params.update(
|
320
|
+
options[:subscription] || {}
|
321
|
+
)
|
322
|
+
|
323
|
+
xml = Builder::XmlMarkup.new :indent => 2
|
324
|
+
add_address(xml, payment_method, options[:billing_address], options)
|
325
|
+
add_purchase_data(xml, options[:setup_fee] || 0, true, options)
|
326
|
+
if card_brand(payment_method) == 'check'
|
327
|
+
add_check(xml, payment_method)
|
328
|
+
add_check_payment_method(xml)
|
329
|
+
else
|
330
|
+
add_creditcard(xml, payment_method)
|
331
|
+
add_creditcard_payment_method(xml)
|
332
|
+
end
|
333
|
+
add_subscription(xml, options)
|
334
|
+
if options[:setup_fee]
|
335
|
+
if card_brand(payment_method) == 'check'
|
336
|
+
add_check_service(xml, options)
|
337
|
+
else
|
338
|
+
add_purchase_service(xml, payment_method, options)
|
339
|
+
end
|
340
|
+
end
|
341
|
+
add_subscription_create_service(xml, options)
|
342
|
+
add_business_rules_data(xml)
|
343
|
+
add_network_token(xml, payment_method, options) if options[:setup_fee] and token_based_card?(payment_method, options)
|
344
|
+
xml.target!
|
345
|
+
end
|
346
|
+
|
347
|
+
def build_update_subscription_request(reference, creditcard, options)
|
348
|
+
xml = Builder::XmlMarkup.new :indent => 2
|
349
|
+
add_address(xml, creditcard, options[:billing_address], options) unless options[:billing_address].blank?
|
350
|
+
add_purchase_data(xml, options[:setup_fee], true, options) unless options[:setup_fee].blank?
|
351
|
+
add_creditcard(xml, creditcard) if creditcard
|
352
|
+
add_creditcard_payment_method(xml) if creditcard
|
353
|
+
add_subscription(xml, options, reference)
|
354
|
+
add_subscription_update_service(xml, options)
|
355
|
+
add_business_rules_data(xml)
|
356
|
+
xml.target!
|
357
|
+
end
|
358
|
+
|
359
|
+
def build_delete_subscription_request(reference, options)
|
360
|
+
xml = Builder::XmlMarkup.new :indent => 2
|
361
|
+
add_subscription(xml, options, reference)
|
362
|
+
add_subscription_delete_service(xml, options)
|
363
|
+
xml.target!
|
364
|
+
end
|
365
|
+
|
366
|
+
def build_retrieve_subscription_request(reference, options)
|
367
|
+
xml = Builder::XmlMarkup.new :indent => 2
|
368
|
+
add_subscription(xml, options, reference)
|
369
|
+
add_subscription_retrieve_service(xml, options)
|
370
|
+
xml.target!
|
371
|
+
end
|
372
|
+
|
373
|
+
def build_validate_pinless_debit_request(creditcard,options)
|
374
|
+
xml = Builder::XmlMarkup.new :indent => 2
|
375
|
+
add_creditcard(xml, creditcard)
|
376
|
+
add_validate_pinless_debit_service(xml)
|
377
|
+
xml.target!
|
378
|
+
end
|
379
|
+
|
380
|
+
def add_business_rules_data(xml)
|
381
|
+
xml.tag! 'businessRules' do
|
382
|
+
xml.tag!('ignoreAVSResult', 'true') if @options[:ignore_avs]
|
383
|
+
xml.tag!('ignoreCVResult', 'true') if @options[:ignore_cvv]
|
384
|
+
end
|
385
|
+
end
|
386
|
+
|
387
|
+
def add_line_item_data(xml, options)
|
388
|
+
options[:line_items].each_with_index do |value, index|
|
389
|
+
xml.tag! 'item', {'id' => index} do
|
390
|
+
xml.tag! 'unitPrice', amount(value[:declared_value])
|
391
|
+
xml.tag! 'quantity', value[:quantity]
|
392
|
+
xml.tag! 'productCode', value[:code] || 'shipping_only'
|
393
|
+
xml.tag! 'productName', value[:description]
|
394
|
+
xml.tag! 'productSKU', value[:sku]
|
395
|
+
end
|
396
|
+
end
|
397
|
+
end
|
398
|
+
|
399
|
+
def add_merchant_data(xml, options)
|
400
|
+
xml.tag! 'merchantID', @options[:login]
|
401
|
+
xml.tag! 'merchantReferenceCode', options[:order_id]
|
402
|
+
xml.tag! 'clientLibrary' ,'Ruby Active Merchant'
|
403
|
+
xml.tag! 'clientLibraryVersion', VERSION
|
404
|
+
xml.tag! 'clientEnvironment' , RUBY_PLATFORM
|
405
|
+
end
|
406
|
+
|
407
|
+
def add_purchase_data(xml, money = 0, include_grand_total = false, options={})
|
408
|
+
xml.tag! 'purchaseTotals' do
|
409
|
+
xml.tag! 'currency', options[:currency] || currency(money)
|
410
|
+
xml.tag!('grandTotalAmount', amount(money)) if include_grand_total
|
411
|
+
end
|
412
|
+
end
|
413
|
+
|
414
|
+
def add_address(xml, payment_method, address, options, shipTo = false)
|
415
|
+
xml.tag! shipTo ? 'shipTo' : 'billTo' do
|
416
|
+
xml.tag! 'firstName', payment_method.first_name if payment_method
|
417
|
+
xml.tag! 'lastName', payment_method.last_name if payment_method
|
418
|
+
xml.tag! 'street1', address[:address1]
|
419
|
+
xml.tag! 'street2', address[:address2] unless address[:address2].blank?
|
420
|
+
xml.tag! 'city', address[:city]
|
421
|
+
xml.tag! 'state', address[:state]
|
422
|
+
xml.tag! 'postalCode', address[:zip]
|
423
|
+
xml.tag! 'country', address[:country]
|
424
|
+
xml.tag! 'company', address[:company] unless address[:company].blank?
|
425
|
+
xml.tag! 'companyTaxID', address[:companyTaxID] unless address[:company_tax_id].blank?
|
426
|
+
xml.tag! 'phoneNumber', address[:phone] unless address[:phone].blank?
|
427
|
+
xml.tag! 'email', options[:email]
|
428
|
+
xml.tag! 'driversLicenseNumber', options[:drivers_license_number] unless options[:drivers_license_number].blank?
|
429
|
+
xml.tag! 'driversLicenseState', options[:drivers_license_state] unless options[:drivers_license_state].blank?
|
430
|
+
end
|
431
|
+
end
|
432
|
+
|
433
|
+
def add_creditcard(xml, creditcard)
|
434
|
+
xml.tag! 'card' do
|
435
|
+
xml.tag! 'accountNumber', creditcard.number
|
436
|
+
xml.tag! 'expirationMonth', format(creditcard.month, :two_digits)
|
437
|
+
xml.tag! 'expirationYear', format(creditcard.year, :four_digits)
|
438
|
+
xml.tag!('cvNumber', creditcard.verification_value) unless (@options[:ignore_cvv] || creditcard.verification_value.blank? )
|
439
|
+
xml.tag! 'cardType', @@credit_card_codes[card_brand(creditcard).to_sym]
|
440
|
+
end
|
441
|
+
end
|
442
|
+
|
443
|
+
def add_check(xml, check)
|
444
|
+
xml.tag! 'check' do
|
445
|
+
xml.tag! 'accountNumber', check.account_number
|
446
|
+
xml.tag! 'accountType', check.account_type[0]
|
447
|
+
xml.tag! 'bankTransitNumber', check.routing_number
|
448
|
+
end
|
449
|
+
end
|
450
|
+
|
451
|
+
def add_tax_service(xml)
|
452
|
+
xml.tag! 'taxService', {'run' => 'true'} do
|
453
|
+
xml.tag!('nexus', @options[:nexus]) unless @options[:nexus].blank?
|
454
|
+
xml.tag!('sellerRegistration', @options[:vat_reg_number]) unless @options[:vat_reg_number].blank?
|
455
|
+
end
|
456
|
+
end
|
457
|
+
|
458
|
+
def add_auth_service(xml, payment_method_or_reference, options)
|
459
|
+
if token_based_card?(payment_method_or_reference, options)
|
460
|
+
# Sadly, for each type of card there is different rules on how to send this
|
461
|
+
# See http://apps.cybersource.com/library/documentation/dev_guides/tokenization/Tokenization.pdf
|
462
|
+
if card_brand(payment_method_or_reference) == 'visa'
|
463
|
+
xml.tag! 'ccAuthService', {'run' => 'true'} do
|
464
|
+
# Cryptogram for payment network tokenization transactions. The value for this field must be 28-character base64 or 40-character hex binary
|
465
|
+
xml.tag! 'cavv', options[:cryptogram]
|
466
|
+
# Type of payer authentication fields that are being used for the payment network tokenization transaction
|
467
|
+
xml.tag! 'commerceIndicator', 'vbv' # Verified by Visa
|
468
|
+
# Cryptogram for payment network tokenization transactions. The value for this field must be 28-character base64 or 40-character hex binary
|
469
|
+
xml.tag! 'xid', options[:cryptogram]
|
470
|
+
end
|
471
|
+
elsif card_brand(payment_method_or_reference) == 'master'
|
472
|
+
xml.tag! 'ucaf' do
|
473
|
+
# Cryptogram for payment network tokenization transactions with MasterCard
|
474
|
+
xml.tag! 'authenticationData', options[:cryptogram]
|
475
|
+
# Required field for payment network tokenization transactions with MasterCard
|
476
|
+
xml.tag! 'collectionIndicator', '2'
|
477
|
+
end
|
478
|
+
xml.tag! 'ccAuthService', {'run' => 'true'} do
|
479
|
+
# Type of payer authentication fields that are being used for the payment network tokenization transaction
|
480
|
+
xml.tag! 'commerceIndicator', 'spa' # MasterCard SecureCode
|
481
|
+
end
|
482
|
+
elsif card_brand(payment_method_or_reference) == 'american_express'
|
483
|
+
# For the American Express card type, the cryptogram is a 40-byte binary value that
|
484
|
+
# you must split into two 20-byte binary values (block A and block B). Send the first
|
485
|
+
# 20-byte value (block A) in the cardholder authentication verification value (CAVV)
|
486
|
+
# field. Send the second 20-byte value (block B) in the transaction ID (XID) field.
|
487
|
+
# The incoming cryptogram is base64 encoded, so we first decode it, then split it
|
488
|
+
# then re-encode it for transmission (chomping off a trailing newline ruby adds).
|
489
|
+
base64_decoded_cryptogram = Base64.decode64(options[:cryptogram])
|
490
|
+
block_a = base64_decoded_cryptogram[0, 20]
|
491
|
+
block_b = base64_decoded_cryptogram[20, 20]
|
492
|
+
xml.tag! 'ccAuthService', {'run' => 'true'} do
|
493
|
+
xml.tag! 'cavv', Base64.encode64(block_a).chomp
|
494
|
+
# Type of payer authentication fields that are being used for the payment network tokenization transaction
|
495
|
+
xml.tag! 'commerceIndicator', 'aesk' # American Express SafeKey
|
496
|
+
xml.tag! 'xid', Base64.encode64(block_b).chomp
|
497
|
+
end
|
498
|
+
end
|
499
|
+
else
|
500
|
+
xml.tag! 'ccAuthService', {'run' => 'true'}
|
501
|
+
end
|
502
|
+
end
|
503
|
+
|
504
|
+
def add_capture_service(xml, request_id, request_token)
|
505
|
+
xml.tag! 'ccCaptureService', {'run' => 'true'} do
|
506
|
+
xml.tag! 'authRequestID', request_id
|
507
|
+
xml.tag! 'authRequestToken', request_token
|
508
|
+
end
|
509
|
+
end
|
510
|
+
|
511
|
+
def add_purchase_service(xml, payment_method_or_reference, options)
|
512
|
+
if options[:pinless_debit_card]
|
513
|
+
xml.tag! 'pinlessDebitService', {'run' => 'true'}
|
514
|
+
else
|
515
|
+
add_auth_service(xml, payment_method_or_reference, options)
|
516
|
+
xml.tag! 'ccCaptureService', {'run' => 'true'}
|
517
|
+
end
|
518
|
+
end
|
519
|
+
|
520
|
+
def add_void_service(xml, request_id, request_token)
|
521
|
+
xml.tag! 'voidService', {'run' => 'true'} do
|
522
|
+
xml.tag! 'voidRequestID', request_id
|
523
|
+
xml.tag! 'voidRequestToken', request_token
|
524
|
+
end
|
525
|
+
end
|
526
|
+
|
527
|
+
def add_auth_reversal_service(xml, request_id, request_token)
|
528
|
+
xml.tag! 'ccAuthReversalService', {'run' => 'true'} do
|
529
|
+
xml.tag! 'authRequestID', request_id
|
530
|
+
xml.tag! 'authRequestToken', request_token
|
531
|
+
end
|
532
|
+
end
|
533
|
+
|
534
|
+
def add_credit_service(xml, request_id = nil, request_token = nil)
|
535
|
+
xml.tag! 'ccCreditService', {'run' => 'true'} do
|
536
|
+
xml.tag! 'captureRequestID', request_id if request_id
|
537
|
+
xml.tag! 'captureRequestToken', request_token if request_token
|
538
|
+
end
|
539
|
+
end
|
540
|
+
|
541
|
+
def add_check_service(xml)
|
542
|
+
xml.tag! 'ecDebitService', {'run' => 'true'}
|
543
|
+
end
|
544
|
+
|
545
|
+
def add_subscription_create_service(xml, options)
|
546
|
+
xml.tag! 'paySubscriptionCreateService', {'run' => 'true'}
|
547
|
+
end
|
548
|
+
|
549
|
+
def add_subscription_update_service(xml, options)
|
550
|
+
xml.tag! 'paySubscriptionUpdateService', {'run' => 'true'}
|
551
|
+
end
|
552
|
+
|
553
|
+
def add_subscription_delete_service(xml, options)
|
554
|
+
xml.tag! 'paySubscriptionDeleteService', {'run' => 'true'}
|
555
|
+
end
|
556
|
+
|
557
|
+
def add_subscription_retrieve_service(xml, options)
|
558
|
+
xml.tag! 'paySubscriptionRetrieveService', {'run' => 'true'}
|
559
|
+
end
|
560
|
+
|
561
|
+
def add_subscription(xml, options, reference = nil)
|
562
|
+
options[:subscription] ||= {}
|
563
|
+
|
564
|
+
xml.tag! 'recurringSubscriptionInfo' do
|
565
|
+
if reference
|
566
|
+
_, subscription_id, _ = reference.split(";")
|
567
|
+
xml.tag! 'subscriptionID', subscription_id
|
568
|
+
end
|
569
|
+
|
570
|
+
xml.tag! 'status', options[:subscription][:status] if options[:subscription][:status]
|
571
|
+
xml.tag! 'amount', amount(options[:subscription][:amount]) if options[:subscription][:amount]
|
572
|
+
xml.tag! 'numberOfPayments', options[:subscription][:occurrences] if options[:subscription][:occurrences]
|
573
|
+
xml.tag! 'automaticRenew', options[:subscription][:automatic_renew] if options[:subscription][:automatic_renew]
|
574
|
+
xml.tag! 'frequency', options[:subscription][:frequency] if options[:subscription][:frequency]
|
575
|
+
xml.tag! 'startDate', options[:subscription][:start_date].strftime("%Y%m%d") if options[:subscription][:start_date]
|
576
|
+
xml.tag! 'endDate', options[:subscription][:end_date].strftime("%Y%m%d") if options[:subscription][:end_date]
|
577
|
+
xml.tag! 'approvalRequired', options[:subscription][:approval_required] || false
|
578
|
+
xml.tag! 'event', options[:subscription][:event] if options[:subscription][:event]
|
579
|
+
xml.tag! 'billPayment', options[:subscription][:bill_payment] if options[:subscription][:bill_payment]
|
580
|
+
end
|
581
|
+
end
|
582
|
+
|
583
|
+
def add_creditcard_payment_method(xml)
|
584
|
+
xml.tag! 'subscription' do
|
585
|
+
xml.tag! 'paymentMethod', "credit card"
|
586
|
+
end
|
587
|
+
end
|
588
|
+
|
589
|
+
def add_check_payment_method(xml)
|
590
|
+
xml.tag! 'subscription' do
|
591
|
+
xml.tag! 'paymentMethod', "check"
|
592
|
+
end
|
593
|
+
end
|
594
|
+
|
595
|
+
def add_payment_method_or_subscription(xml, money, payment_method_or_reference, options)
|
596
|
+
if payment_method_or_reference.is_a?(String)
|
597
|
+
add_purchase_data(xml, money, true, options)
|
598
|
+
add_subscription(xml, options, payment_method_or_reference)
|
599
|
+
elsif card_brand(payment_method_or_reference) == 'check'
|
600
|
+
add_address(xml, payment_method_or_reference, options[:billing_address], options)
|
601
|
+
add_purchase_data(xml, money, true, options)
|
602
|
+
add_check(xml, payment_method_or_reference)
|
603
|
+
else
|
604
|
+
add_address(xml, payment_method_or_reference, options[:billing_address], options)
|
605
|
+
add_address(xml, payment_method_or_reference, options[:shipping_address], options, true)
|
606
|
+
add_purchase_data(xml, money, true, options)
|
607
|
+
add_creditcard(xml, payment_method_or_reference)
|
608
|
+
end
|
609
|
+
end
|
610
|
+
|
611
|
+
def add_validate_pinless_debit_service(xml)
|
612
|
+
xml.tag!'pinlessDebitValidateService', {'run' => 'true'}
|
613
|
+
end
|
614
|
+
|
615
|
+
def add_network_token(xml, payment_method_or_reference, options)
|
616
|
+
xml.tag! 'paymentNetworkToken' do
|
617
|
+
# Type of transaction that provided the token data. This value does not specify the token
|
618
|
+
# service provider; it specifies the entity that provided you with information about the token.
|
619
|
+
# 1: In-app transaction.
|
620
|
+
# An application on the customer’s mobile device provided the
|
621
|
+
# token data for an e-commerce transaction.
|
622
|
+
# For recurring transactions, use this value if the original transaction was an in-app
|
623
|
+
# e-commerce transaction.
|
624
|
+
# See http://apps.cybersource.com/library/documentation/dev_guides/tokenization/Tokenization.pdf
|
625
|
+
xml.tag! 'transactionType', '1'
|
626
|
+
end
|
627
|
+
end
|
628
|
+
|
629
|
+
def token_based_card?(payment_method_or_reference, options)
|
630
|
+
!payment_method_or_reference.is_a?(String) && card_brand(payment_method_or_reference) && options[:cryptogram]
|
631
|
+
end
|
632
|
+
|
633
|
+
# Where we actually build the full SOAP request using builder
|
634
|
+
def build_request(body, options)
|
635
|
+
xml = Builder::XmlMarkup.new :indent => 2
|
636
|
+
xml.instruct!
|
637
|
+
xml.tag! 's:Envelope', {'xmlns:s' => 'http://schemas.xmlsoap.org/soap/envelope/'} do
|
638
|
+
xml.tag! 's:Header' do
|
639
|
+
xml.tag! 'wsse:Security', {'s:mustUnderstand' => '1', 'xmlns:wsse' => 'http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-secext-1.0.xsd'} do
|
640
|
+
xml.tag! 'wsse:UsernameToken' do
|
641
|
+
xml.tag! 'wsse:Username', @options[:login]
|
642
|
+
xml.tag! 'wsse:Password', @options[:password], 'Type' => 'http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-username-token-profile-1.0#PasswordText'
|
643
|
+
end
|
644
|
+
end
|
645
|
+
end
|
646
|
+
xml.tag! 's:Body', {'xmlns:xsi' => 'http://www.w3.org/2001/XMLSchema-instance', 'xmlns:xsd' => 'http://www.w3.org/2001/XMLSchema'} do
|
647
|
+
xml.tag! 'requestMessage', {'xmlns' => "urn:schemas-cybersource-com:transaction-data-#{XSD_VERSION}"} do
|
648
|
+
add_merchant_data(xml, options)
|
649
|
+
xml << body
|
650
|
+
end
|
651
|
+
end
|
652
|
+
end
|
653
|
+
xml.target!
|
654
|
+
end
|
655
|
+
|
656
|
+
# Contact CyberSource, make the SOAP request, and parse the reply into a
|
657
|
+
# Response object
|
658
|
+
def commit(request, options)
|
659
|
+
response = parse(ssl_post(test? ? self.test_url : self.live_url, build_request(request, options)))
|
660
|
+
|
661
|
+
success = response[:decision] == "ACCEPT"
|
662
|
+
message = @@response_codes[('r' + response[:reasonCode]).to_sym] rescue response[:message]
|
663
|
+
authorization = success ? [ options[:order_id], response[:requestID], response[:requestToken] ].compact.join(";") : nil
|
664
|
+
|
665
|
+
Response.new(success, message, response,
|
666
|
+
:test => test?,
|
667
|
+
:authorization => authorization,
|
668
|
+
:avs_result => { :code => response[:avsCode] },
|
669
|
+
:cvv_result => response[:cvCode]
|
670
|
+
)
|
671
|
+
end
|
672
|
+
|
673
|
+
# Parse the SOAP response
|
674
|
+
# Technique inspired by the Paypal Gateway
|
675
|
+
def parse(xml)
|
676
|
+
reply = {}
|
677
|
+
xml = REXML::Document.new(xml)
|
678
|
+
if root = REXML::XPath.first(xml, "//c:replyMessage")
|
679
|
+
root.elements.to_a.each do |node|
|
680
|
+
case node.name
|
681
|
+
when 'c:reasonCode'
|
682
|
+
reply[:message] = reply(node.text)
|
683
|
+
else
|
684
|
+
parse_element(reply, node)
|
685
|
+
end
|
686
|
+
end
|
687
|
+
elsif root = REXML::XPath.first(xml, "//soap:Fault")
|
688
|
+
parse_element(reply, root)
|
689
|
+
reply[:message] = "#{reply[:faultcode]}: #{reply[:faultstring]}"
|
690
|
+
end
|
691
|
+
return reply
|
692
|
+
end
|
693
|
+
|
694
|
+
def parse_element(reply, node)
|
695
|
+
if node.has_elements?
|
696
|
+
node.elements.each{|e| parse_element(reply, e) }
|
697
|
+
else
|
698
|
+
if node.parent.name =~ /item/
|
699
|
+
parent = node.parent.name + (node.parent.attributes["id"] ? "_" + node.parent.attributes["id"] : '')
|
700
|
+
reply[(parent + '_' + node.name).to_sym] = node.text
|
701
|
+
else
|
702
|
+
reply[node.name.to_sym] = node.text
|
703
|
+
end
|
704
|
+
end
|
705
|
+
return reply
|
706
|
+
end
|
707
|
+
end
|
708
|
+
end
|
709
|
+
end
|