tlconnor-activemerchant 1.20.4 → 1.23.0
Sign up to get free protection for your applications and to get access to all the features.
- data/CHANGELOG +86 -6
- data/CONTRIBUTORS +33 -0
- data/lib/active_merchant/billing/gateways/authorize_net_cim.rb +2 -0
- data/lib/active_merchant/billing/gateways/barclays_epdq.rb +4 -4
- data/lib/active_merchant/billing/gateways/blue_pay.rb +492 -11
- data/lib/active_merchant/billing/gateways/braintree_blue.rb +46 -19
- data/lib/active_merchant/billing/gateways/certo_direct.rb +1 -1
- data/lib/active_merchant/billing/gateways/elavon.rb +2 -0
- data/lib/active_merchant/billing/gateways/epay.rb +3 -1
- data/lib/active_merchant/billing/gateways/itransact.rb +450 -0
- data/lib/active_merchant/billing/gateways/litle.rb +275 -0
- data/lib/active_merchant/billing/gateways/migs.rb +259 -0
- data/lib/active_merchant/billing/gateways/migs/migs_codes.rb +100 -0
- data/lib/active_merchant/billing/gateways/moneris.rb +4 -30
- data/lib/active_merchant/billing/gateways/moneris_us.rb +211 -0
- data/lib/active_merchant/billing/gateways/nab_transact.rb +1 -1
- data/lib/active_merchant/billing/gateways/ogone.rb +104 -12
- data/lib/active_merchant/billing/gateways/orbital.rb +15 -6
- data/lib/active_merchant/billing/gateways/paybox_direct.rb +1 -4
- data/lib/active_merchant/billing/gateways/payflow.rb +8 -3
- data/lib/active_merchant/billing/gateways/payflow/payflow_common_api.rb +4 -1
- data/lib/active_merchant/billing/gateways/payflow_express.rb +4 -2
- data/lib/active_merchant/billing/gateways/payment_express.rb +60 -13
- data/lib/active_merchant/billing/gateways/paypal.rb +3 -18
- data/lib/active_merchant/billing/gateways/paypal/paypal_common_api.rb +333 -3
- data/lib/active_merchant/billing/gateways/paypal/paypal_recurring_api.rb +245 -0
- data/lib/active_merchant/billing/gateways/paypal_digital_goods.rb +43 -0
- data/lib/active_merchant/billing/gateways/paypal_express.rb +14 -65
- data/lib/active_merchant/billing/gateways/paypal_express_common.rb +8 -3
- data/lib/active_merchant/billing/gateways/realex.rb +5 -7
- data/lib/active_merchant/billing/gateways/secure_pay_au.rb +3 -2
- data/lib/active_merchant/billing/gateways/stripe.rb +1 -9
- data/lib/active_merchant/billing/gateways/usa_epay_advanced.rb +2 -2
- data/lib/active_merchant/billing/gateways/usa_epay_transaction.rb +1 -5
- data/lib/active_merchant/billing/gateways/viaklix.rb +7 -2
- data/lib/active_merchant/billing/gateways/vindicia.rb +359 -0
- data/lib/active_merchant/billing/integrations/dotpay.rb +22 -0
- data/lib/active_merchant/billing/integrations/dotpay/helper.rb +77 -0
- data/lib/active_merchant/billing/integrations/dotpay/notification.rb +86 -0
- data/lib/active_merchant/billing/integrations/dotpay/return.rb +11 -0
- data/lib/active_merchant/billing/integrations/epay.rb +21 -0
- data/lib/active_merchant/billing/integrations/epay/helper.rb +55 -0
- data/lib/active_merchant/billing/integrations/epay/notification.rb +110 -0
- data/lib/active_merchant/billing/integrations/paypal/notification.rb +2 -1
- data/lib/active_merchant/billing/integrations/quickpay/helper.rb +2 -3
- data/lib/active_merchant/billing/integrations/robokassa.rb +49 -0
- data/lib/active_merchant/billing/integrations/robokassa/common.rb +19 -0
- data/lib/active_merchant/billing/integrations/robokassa/helper.rb +50 -0
- data/lib/active_merchant/billing/integrations/robokassa/notification.rb +55 -0
- data/lib/active_merchant/billing/integrations/robokassa/return.rb +17 -0
- data/lib/active_merchant/billing/integrations/two_checkout.rb +25 -3
- data/lib/active_merchant/billing/integrations/two_checkout/helper.rb +58 -26
- data/lib/active_merchant/billing/integrations/two_checkout/notification.rb +71 -46
- data/lib/active_merchant/billing/integrations/verkkomaksut.rb +20 -0
- data/lib/active_merchant/billing/integrations/verkkomaksut/helper.rb +87 -0
- data/lib/active_merchant/billing/integrations/verkkomaksut/notification.rb +59 -0
- data/lib/active_merchant/version.rb +1 -1
- metadata +28 -5
@@ -64,7 +64,7 @@ module ActiveMerchant #:nodoc:
|
|
64
64
|
end
|
65
65
|
|
66
66
|
def purchase(money, credit_card_or_stored_id, options = {})
|
67
|
-
if credit_card_or_stored_id.
|
67
|
+
if credit_card_or_stored_id.respond_to?(:number)
|
68
68
|
requires!(options, :order_id)
|
69
69
|
commit :purchase, build_purchase_request(money, credit_card_or_stored_id, options)
|
70
70
|
else
|
@@ -126,7 +126,8 @@ module ActiveMerchant #:nodoc:
|
|
126
126
|
def build_reference_request(money, reference)
|
127
127
|
xml = Builder::XmlMarkup.new
|
128
128
|
|
129
|
-
transaction_id, order_id, preauth_id, original_amount = reference.split(
|
129
|
+
transaction_id, order_id, preauth_id, original_amount = reference.split('*')
|
130
|
+
|
130
131
|
xml.tag! 'amount', (money ? amount(money) : original_amount)
|
131
132
|
xml.tag! 'currency', options[:currency] || currency(money)
|
132
133
|
xml.tag! 'txnID', transaction_id
|
@@ -58,14 +58,6 @@ module ActiveMerchant #:nodoc:
|
|
58
58
|
commit(:post, 'charges', post, meta)
|
59
59
|
end
|
60
60
|
|
61
|
-
def authorize(money, creditcard, options = {})
|
62
|
-
raise "Stripe does not support separate authorization and capture"
|
63
|
-
end
|
64
|
-
|
65
|
-
def capture(money, identification, options = {})
|
66
|
-
raise "Stripe does not support separate authorization and capture"
|
67
|
-
end
|
68
|
-
|
69
61
|
def void(identification, options = {})
|
70
62
|
commit(:post, "charges/#{CGI.escape(identification)}/refund", {})
|
71
63
|
end
|
@@ -215,7 +207,7 @@ module ActiveMerchant #:nodoc:
|
|
215
207
|
Response.new(success,
|
216
208
|
success ? "Transaction approved" : response["error"]["message"],
|
217
209
|
response,
|
218
|
-
:test => !response["livemode"],
|
210
|
+
:test => response.has_key?("livemode") ? !response["livemode"] : false,
|
219
211
|
:authorization => response["id"],
|
220
212
|
:avs_result => { :code => avs_code },
|
221
213
|
:cvv_result => cvc_code
|
@@ -1239,7 +1239,7 @@ module ActiveMerchant #:nodoc:
|
|
1239
1239
|
when payment_method[:method].kind_of?(ActiveMerchant::Billing::CreditCard)
|
1240
1240
|
build_tag soap, :string, 'CardNumber', payment_method[:method].number
|
1241
1241
|
build_tag soap, :string, 'CardExpiration',
|
1242
|
-
"#{
|
1242
|
+
"#{"%02d" % payment_method[:method].month}#{payment_method[:method].year}"
|
1243
1243
|
if options[:billing_address]
|
1244
1244
|
build_tag soap, :string, 'AvsStreet', options[:billing_address][:address1]
|
1245
1245
|
build_tag soap, :string, 'AvsZip', options[:billing_address][:zip]
|
@@ -1319,7 +1319,7 @@ module ActiveMerchant #:nodoc:
|
|
1319
1319
|
soap.CreditCardData 'xsi:type' => "ns1:CreditCardData" do |soap|
|
1320
1320
|
build_tag soap, :string, 'CardNumber', options[:payment_method].number
|
1321
1321
|
build_tag soap, :string, 'CardExpiration',
|
1322
|
-
"#{
|
1322
|
+
"#{"%02d" % options[:payment_method].month}#{options[:payment_method].year}"
|
1323
1323
|
if options[:billing_address]
|
1324
1324
|
build_tag soap, :string, 'AvsStreet', options[:billing_address][:address1]
|
1325
1325
|
build_tag soap, :string, 'AvsZip', options[:billing_address][:zip]
|
@@ -175,11 +175,7 @@ module ActiveMerchant #:nodoc:
|
|
175
175
|
:test => @options[:test] || test?,
|
176
176
|
:authorization => response[:ref_num],
|
177
177
|
:cvv_result => response[:cvv2_result_code],
|
178
|
-
:avs_result => {
|
179
|
-
:street_match => response[:avs_result_code].to_s[0,1],
|
180
|
-
:postal_match => response[:avs_result_code].to_s[1,1],
|
181
|
-
:code => response[:avs_result_code].to_s[2,1]
|
182
|
-
}
|
178
|
+
:avs_result => { :code => response[:avs_result_code] }
|
183
179
|
)
|
184
180
|
end
|
185
181
|
|
@@ -43,6 +43,7 @@ module ActiveMerchant #:nodoc:
|
|
43
43
|
add_creditcard(form, creditcard)
|
44
44
|
add_address(form, options)
|
45
45
|
add_customer_data(form, options)
|
46
|
+
add_test_mode(form, options)
|
46
47
|
commit(:purchase, money, form)
|
47
48
|
end
|
48
49
|
|
@@ -58,10 +59,15 @@ module ActiveMerchant #:nodoc:
|
|
58
59
|
add_creditcard(form, creditcard)
|
59
60
|
add_address(form, options)
|
60
61
|
add_customer_data(form, options)
|
62
|
+
add_test_mode(form, options)
|
61
63
|
commit(:credit, money, form)
|
62
64
|
end
|
63
65
|
|
64
66
|
private
|
67
|
+
def add_test_mode(form, options)
|
68
|
+
form[:test_mode] = 'TRUE' if options[:test_mode]
|
69
|
+
end
|
70
|
+
|
65
71
|
def add_customer_data(form, options)
|
66
72
|
form[:email] = options[:email].to_s.slice(0, 100) unless options[:email].blank?
|
67
73
|
form[:customer_code] = options[:customer].to_s.slice(0, 10) unless options[:customer].blank?
|
@@ -129,8 +135,7 @@ module ActiveMerchant #:nodoc:
|
|
129
135
|
'merchant_id' => @options[:login],
|
130
136
|
'pin' => @options[:password],
|
131
137
|
'show_form' => 'false',
|
132
|
-
'
|
133
|
-
'result_format' => 'ASCII',
|
138
|
+
'result_format' => 'ASCII'
|
134
139
|
}
|
135
140
|
|
136
141
|
result['user_id'] = @options[:user] unless @options[:user].blank?
|
@@ -0,0 +1,359 @@
|
|
1
|
+
begin
|
2
|
+
require "vindicia-api"
|
3
|
+
rescue LoadError
|
4
|
+
raise "Could not load the vindicia-api gem. Use `gem install vindicia-api` to install it."
|
5
|
+
end
|
6
|
+
|
7
|
+
module ActiveMerchant #:nodoc:
|
8
|
+
module Billing #:nodoc:
|
9
|
+
|
10
|
+
# For more information on the Vindicia Gateway please visit their {website}[http://vindicia.com/]
|
11
|
+
#
|
12
|
+
# The login and password are not the username and password you use to
|
13
|
+
# login to the Vindicia Merchant Portal.
|
14
|
+
#
|
15
|
+
# ==== Recurring Billing
|
16
|
+
#
|
17
|
+
# AutoBills are an feature of Vindicia's API that allows for creating and managing subscriptions.
|
18
|
+
#
|
19
|
+
# For more information about Vindicia's API and various other services visit their {Resource Center}[http://www.vindicia.com/resources/index.html]
|
20
|
+
class VindiciaGateway < Gateway
|
21
|
+
self.supported_countries = %w{US CA GB AU MX BR DE KR CN HK}
|
22
|
+
self.supported_cardtypes = [:visa, :master, :american_express, :discover]
|
23
|
+
self.homepage_url = 'http://www.vindicia.com/'
|
24
|
+
self.display_name = 'Vindicia'
|
25
|
+
|
26
|
+
class_attribute :test_url, :live_url
|
27
|
+
|
28
|
+
self.test_url = "https://soap.prodtest.sj.vindicia.com/soap.pl"
|
29
|
+
self.live_url = "http://soap.vindicia.com/soap.pl"
|
30
|
+
|
31
|
+
# Creates a new VindiciaGateway
|
32
|
+
#
|
33
|
+
# The gateway requires that a valid login and password be passed
|
34
|
+
# in the +options+ hash.
|
35
|
+
#
|
36
|
+
# ==== Options
|
37
|
+
#
|
38
|
+
# * <tt>:login</tt> -- Vindicia SOAP login (REQUIRED)
|
39
|
+
# * <tt>:password</tt> -- Vindicia SOAP password (REQUIRED)
|
40
|
+
# * <tt>:api_version</tt> -- Vindicia API Version - defaults to 3.6 (OPTIONAL)
|
41
|
+
# * <tt>:account_id</tt> -- Account Id which all transactions will be run against. (REQUIRED)
|
42
|
+
# * <tt>:transaction_prefix</tt> -- Prefix to order id for one-time transactions - defaults to 'X' (OPTIONAL
|
43
|
+
# * <tt>:min_chargeback_probability</tt> -- Minimum score for chargebacks - defaults to 65 (OPTIONAL)
|
44
|
+
# * <tt>:cvn_success</tt> -- Array of valid CVN Check return values - defaults to [M, P] (OPTIONAL)
|
45
|
+
# * <tt>:avs_success</tt> -- Array of valid AVS Check return values - defaults to [X, Y, A, W, Z] (OPTIONAL)
|
46
|
+
def initialize(options = {})
|
47
|
+
requires!(options, :login, :password)
|
48
|
+
|
49
|
+
config = lambda do |config|
|
50
|
+
config.login = options[:login]
|
51
|
+
config.password = options[:password]
|
52
|
+
config.api_version = options[:api_version] || "3.6"
|
53
|
+
config.endpoint = test? ? self.test_url : self.live_url
|
54
|
+
config.namespace = "http://soap.vindicia.com"
|
55
|
+
end
|
56
|
+
|
57
|
+
if Vindicia.config.is_configured?
|
58
|
+
config.call(Vindicia.config)
|
59
|
+
else
|
60
|
+
Vindicia.configure(&config)
|
61
|
+
end
|
62
|
+
|
63
|
+
requires!(options, :account_id)
|
64
|
+
@account_id = options[:account_id]
|
65
|
+
|
66
|
+
@transaction_prefix = options[:transaction_prefix] || "X"
|
67
|
+
|
68
|
+
@min_chargeback_probability = options[:min_chargeback_probability] || 65
|
69
|
+
@cvn_success = options[:cvn_success] || %w{M P}
|
70
|
+
@avs_success = options[:avs_success] || %w{X Y A W Z}
|
71
|
+
|
72
|
+
@options = options
|
73
|
+
@allowed_authorization_statuses = %w{Authorized}
|
74
|
+
end
|
75
|
+
|
76
|
+
# Perform a purchase, which is essentially an authorization and capture in a single operation.
|
77
|
+
#
|
78
|
+
# ==== Parameters
|
79
|
+
#
|
80
|
+
# * <tt>money</tt> -- The amount to be purchased as an Integer value in cents.
|
81
|
+
# * <tt>creditcard</tt> -- The CreditCard details for the transaction.
|
82
|
+
# * <tt>options</tt> -- A hash of optional parameters.
|
83
|
+
def purchase(money, creditcard, options = {})
|
84
|
+
response = authorize(money, creditcard, options)
|
85
|
+
return response if !response.success? || response.fraud_review?
|
86
|
+
|
87
|
+
capture(money, response.authorization, options)
|
88
|
+
end
|
89
|
+
|
90
|
+
# Performs an authorization, which reserves the funds on the customer's credit card, but does not
|
91
|
+
# charge the card.
|
92
|
+
#
|
93
|
+
# ==== Parameters
|
94
|
+
#
|
95
|
+
# * <tt>money</tt> -- The amount to be authorized as an Integer value in cents.
|
96
|
+
# * <tt>creditcard</tt> -- The CreditCard details for the transaction.
|
97
|
+
# * <tt>options</tt> -- A hash of optional parameters.
|
98
|
+
def authorize(money, creditcard, options = {})
|
99
|
+
vindicia_transaction = authorize_transaction(money, creditcard, options)
|
100
|
+
response = check_transaction(vindicia_transaction)
|
101
|
+
|
102
|
+
# if this response is under fraud review because of our AVS/CVV checks void the transaction
|
103
|
+
if !response.success? && response.fraud_review? && !response.authorization.blank?
|
104
|
+
void_response = void([vindicia_transaction[:transaction][:merchantTransactionId]], options)
|
105
|
+
if void_response.success?
|
106
|
+
return response
|
107
|
+
else
|
108
|
+
return void_response
|
109
|
+
end
|
110
|
+
end
|
111
|
+
|
112
|
+
response
|
113
|
+
end
|
114
|
+
|
115
|
+
# Captures the funds from an authorized transaction.
|
116
|
+
#
|
117
|
+
# ==== Parameters
|
118
|
+
#
|
119
|
+
# * <tt>money</tt> -- The amount to be captured as an Integer value in cents.
|
120
|
+
# * <tt>identification</tt> -- The authorization returned from the previous authorize request.
|
121
|
+
def capture(money, identification, options = {})
|
122
|
+
response = post(Vindicia::Transaction.capture({
|
123
|
+
:transactions => [{ :merchantTransactionId => identification }]
|
124
|
+
}))
|
125
|
+
|
126
|
+
if response[:return][:returnCode] != '200' || response[:qtyFail].to_i > 0
|
127
|
+
return fail(response)
|
128
|
+
end
|
129
|
+
|
130
|
+
success(response, identification)
|
131
|
+
end
|
132
|
+
|
133
|
+
# Void a previous transaction
|
134
|
+
#
|
135
|
+
# ==== Parameters
|
136
|
+
#
|
137
|
+
# * <tt>identification</tt> - The authorization returned from the previous authorize request.
|
138
|
+
# * <tt>options</tt> - Extra options (currently only :ip used)
|
139
|
+
def void(identification, options = {})
|
140
|
+
response = post(Vindicia::Transaction.cancel({
|
141
|
+
:transactions => [{
|
142
|
+
:account => { :merchantAccountId => @account_id },
|
143
|
+
:merchantTransactionId => identification,
|
144
|
+
:sourceIp => options[:ip]
|
145
|
+
}]
|
146
|
+
}))
|
147
|
+
|
148
|
+
if response[:return][:returnCode] == '200' && response[:qtyFail].to_i == 0
|
149
|
+
success(response, identification)
|
150
|
+
else
|
151
|
+
fail(response)
|
152
|
+
end
|
153
|
+
end
|
154
|
+
|
155
|
+
# Perform a recurring billing, which is essentially a purchase and autobill setup in a single operation.
|
156
|
+
#
|
157
|
+
# ==== Parameters
|
158
|
+
#
|
159
|
+
# * <tt>money</tt> -- The amount to be purchased as an Integer value in cents.
|
160
|
+
# * <tt>creditcard</tt> -- The CreditCard details for the transaction.
|
161
|
+
# * <tt>options</tt> -- A hash of parameters.
|
162
|
+
#
|
163
|
+
# ==== Options
|
164
|
+
#
|
165
|
+
# * <tt>:product_sku</tt> -- The subscription product's sku
|
166
|
+
# * <tt>:autobill_prefix</tt> -- Prefix to order id for subscriptions - defaults to 'A' (OPTIONAL)
|
167
|
+
def recurring(money, creditcard, options={})
|
168
|
+
options[:recurring] = true
|
169
|
+
@autobill_prefix = options[:autobill_prefix] || "A"
|
170
|
+
|
171
|
+
response = authorize(money, creditcard, options)
|
172
|
+
return response if !response.success? || response.fraud_review?
|
173
|
+
|
174
|
+
capture_resp = capture(money, response.authorization, options)
|
175
|
+
return capture_resp if !response.success?
|
176
|
+
|
177
|
+
# Setting up a recurring AutoBill requires an associated product
|
178
|
+
requires!(options, :product_sku)
|
179
|
+
autobill_response = check_subscription(authorize_subscription(options.merge(:product_sku => options[:product_sku])))
|
180
|
+
|
181
|
+
if autobill_response.success?
|
182
|
+
autobill_response
|
183
|
+
else
|
184
|
+
# If the AutoBill fails to set-up, void the transaction and return it as the response
|
185
|
+
void_response = void(capture_resp.authorization, options)
|
186
|
+
if void_response.success?
|
187
|
+
return autobill_response
|
188
|
+
else
|
189
|
+
return void_response
|
190
|
+
end
|
191
|
+
end
|
192
|
+
end
|
193
|
+
|
194
|
+
protected
|
195
|
+
|
196
|
+
def post(body)
|
197
|
+
parse(ssl_post(Vindicia.config.endpoint, body, "Content-Type" => "text/xml"))
|
198
|
+
end
|
199
|
+
|
200
|
+
def parse(response)
|
201
|
+
# Vindicia always returns in the form of request_type_response => { actual_response }
|
202
|
+
Hash.from_xml(response)["Envelope"]["Body"].values.first.with_indifferent_access
|
203
|
+
end
|
204
|
+
|
205
|
+
def check_transaction(vindicia_transaction)
|
206
|
+
if vindicia_transaction[:return][:returnCode] == '200'
|
207
|
+
status_log = vindicia_transaction[:transaction][:statusLog].first
|
208
|
+
if status_log[:creditCardStatus]
|
209
|
+
avs = status_log[:creditCardStatus][:avsCode]
|
210
|
+
cvn = status_log[:creditCardStatus][:cvnCode]
|
211
|
+
end
|
212
|
+
|
213
|
+
if @allowed_authorization_statuses.include?(status_log[:status]) &&
|
214
|
+
check_cvn(cvn) && check_avs(avs)
|
215
|
+
|
216
|
+
success(vindicia_transaction,
|
217
|
+
vindicia_transaction[:transaction][:merchantTransactionId],
|
218
|
+
avs, cvn)
|
219
|
+
else
|
220
|
+
# If the transaction is authorized, but it didn't pass our AVS/CVV checks send the authorization along so
|
221
|
+
# that is gets voided. Otherwise, send no authorization.
|
222
|
+
fail(vindicia_transaction, avs, cvn, false,
|
223
|
+
@allowed_authorization_statuses.include?(status_log[:status]) ? vindicia_transaction[:transaction][:merchantTransactionId] : "")
|
224
|
+
end
|
225
|
+
else
|
226
|
+
# 406 = Chargeback risk score is higher than minChargebackProbability, transaction not authorized.
|
227
|
+
fail(vindicia_transaction, nil, nil, vindicia_transaction[:return][:return_code] == '406')
|
228
|
+
end
|
229
|
+
end
|
230
|
+
|
231
|
+
def authorize_transaction(money, creditcard, options)
|
232
|
+
parameters = {
|
233
|
+
:amount => amount(money),
|
234
|
+
:currency => options[:currency] || currency(money)
|
235
|
+
}
|
236
|
+
|
237
|
+
add_account_data(parameters, options)
|
238
|
+
add_customer_data(parameters, options)
|
239
|
+
add_payment_source(parameters, creditcard, options)
|
240
|
+
|
241
|
+
post(Vindicia::Transaction.auth({
|
242
|
+
:transaction => parameters,
|
243
|
+
:minChargebackProbability => @min_chargeback_probability
|
244
|
+
}))
|
245
|
+
end
|
246
|
+
|
247
|
+
def add_account_data(parameters, options)
|
248
|
+
parameters[:account] = { :merchantAccountId => @account_id }
|
249
|
+
parameters[:sourceIp] = options[:ip] if options[:ip]
|
250
|
+
end
|
251
|
+
|
252
|
+
def add_customer_data(parameters, options)
|
253
|
+
parameters[:merchantTransactionId] = transaction_id(options[:order_id])
|
254
|
+
parameters[:shippingAddress] = convert_am_address_to_vindicia(options[:shipping_address])
|
255
|
+
|
256
|
+
# Transaction items must be provided for tax purposes
|
257
|
+
requires!(options, :line_items)
|
258
|
+
parameters[:transactionItems] = options[:line_items]
|
259
|
+
|
260
|
+
if options[:recurring]
|
261
|
+
parameters[:nameValues] = [{:name => 'merchantAutoBillIdentifier', :value => autobill_id(options[:order_id])}]
|
262
|
+
end
|
263
|
+
end
|
264
|
+
|
265
|
+
def add_payment_source(parameters, creditcard, options)
|
266
|
+
parameters[:sourcePaymentMethod] = {
|
267
|
+
:type => 'CreditCard',
|
268
|
+
:creditCard => { :account => creditcard.number, :expirationDate => "%4d%02d" % [creditcard.year, creditcard.month] },
|
269
|
+
:accountHolderName => creditcard.name,
|
270
|
+
:nameValues => [{ :name => 'CVN', :value => creditcard.verification_value }],
|
271
|
+
:billingAddress => convert_am_address_to_vindicia(options[:billing_address] || options[:address]),
|
272
|
+
:customerSpecifiedType => creditcard.type.capitalize,
|
273
|
+
:active => !!options[:recurring]
|
274
|
+
}
|
275
|
+
end
|
276
|
+
|
277
|
+
def authorize_subscription(options)
|
278
|
+
parameters = {}
|
279
|
+
|
280
|
+
add_account_data(parameters, options)
|
281
|
+
add_subscription_information(parameters, options)
|
282
|
+
|
283
|
+
post(Vindicia::AutoBill.update({
|
284
|
+
:autobill => parameters,
|
285
|
+
:validatePaymentMethod => false,
|
286
|
+
:minChargebackProbability => 100
|
287
|
+
}))
|
288
|
+
end
|
289
|
+
|
290
|
+
def check_subscription(vindicia_transaction)
|
291
|
+
if vindicia_transaction[:return][:returnCode] == '200'
|
292
|
+
if vindicia_transaction[:autobill] && vindicia_transaction[:autobill][:status] == "Active"
|
293
|
+
success(vindicia_transaction,
|
294
|
+
vindicia_transaction[:autobill][:merchantAutoBillId])
|
295
|
+
else
|
296
|
+
fail(vindicia_transaction)
|
297
|
+
end
|
298
|
+
else
|
299
|
+
fail(vindicia_transaction)
|
300
|
+
end
|
301
|
+
end
|
302
|
+
|
303
|
+
def add_subscription_information(parameters, options)
|
304
|
+
requires!(options, :product_sku)
|
305
|
+
|
306
|
+
if options[:shipping_address]
|
307
|
+
parameters[:account][:shipping_address] = options[:shipping_address]
|
308
|
+
end
|
309
|
+
|
310
|
+
parameters[:merchantAutoBillId] = autobill_id(options[:order_id])
|
311
|
+
parameters[:product] = { :merchantProductId => options[:product_sku] }
|
312
|
+
end
|
313
|
+
|
314
|
+
def check_avs(avs)
|
315
|
+
avs.blank? || @avs_success.include?(avs)
|
316
|
+
end
|
317
|
+
|
318
|
+
def check_cvn(cvn)
|
319
|
+
cvn.blank? || @cvn_success.include?(cvn)
|
320
|
+
end
|
321
|
+
|
322
|
+
def success(response, authorization, avs_code = nil, cvn_code = nil)
|
323
|
+
ActiveMerchant::Billing::Response.new(true, response[:return][:returnString], response,
|
324
|
+
{ :fraud_review => false, :authorization => authorization, :test => test?,
|
325
|
+
:avs_result => { :code => avs_code }, :cvv_result => cvn_code })
|
326
|
+
end
|
327
|
+
|
328
|
+
def fail(response, avs_code = nil, cvn_code = nil, fraud_review = false, authorization = "")
|
329
|
+
ActiveMerchant::Billing::Response.new(false, response[:return][:returnString], response,
|
330
|
+
{ :fraud_review => fraud_review || !authorization.blank?,
|
331
|
+
:authorization => authorization, :test => test?,
|
332
|
+
:avs_result => { :code => avs_code }, :cvv_result => cvn_code })
|
333
|
+
|
334
|
+
end
|
335
|
+
|
336
|
+
def autobill_id(order_id)
|
337
|
+
"#{@autobill_prefix}#{order_id}"
|
338
|
+
end
|
339
|
+
|
340
|
+
def transaction_id(order_id)
|
341
|
+
"#{@transaction_prefix}#{order_id}"
|
342
|
+
end
|
343
|
+
|
344
|
+
# Converts valid ActiveMerchant address hash to proper Vindicia format
|
345
|
+
def convert_am_address_to_vindicia(address)
|
346
|
+
return if address.nil?
|
347
|
+
|
348
|
+
convs = { :address1 => :addr1, :address2 => :addr2,
|
349
|
+
:state => :district, :zip => :postalCode }
|
350
|
+
|
351
|
+
vindicia_address = {}
|
352
|
+
address.each do |key, val|
|
353
|
+
vindicia_address[convs[key] || key] = val
|
354
|
+
end
|
355
|
+
vindicia_address
|
356
|
+
end
|
357
|
+
end
|
358
|
+
end
|
359
|
+
end
|