johnreitano-activemerchant 1.5.2
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.
- data/CHANGELOG +508 -0
- data/CONTRIBUTORS +134 -0
- data/MIT-LICENSE +20 -0
- data/README.rdoc +136 -0
- data/gem-public_cert.pem +20 -0
- data/lib/active_merchant/billing/avs_result.rb +98 -0
- data/lib/active_merchant/billing/base.rb +57 -0
- data/lib/active_merchant/billing/check.rb +68 -0
- data/lib/active_merchant/billing/credit_card.rb +159 -0
- data/lib/active_merchant/billing/credit_card_formatting.rb +21 -0
- data/lib/active_merchant/billing/credit_card_methods.rb +125 -0
- data/lib/active_merchant/billing/cvv_result.rb +38 -0
- data/lib/active_merchant/billing/expiry_date.rb +34 -0
- data/lib/active_merchant/billing/gateway.rb +163 -0
- data/lib/active_merchant/billing/gateways/authorize_net.rb +654 -0
- data/lib/active_merchant/billing/gateways/authorize_net_cim.rb +736 -0
- data/lib/active_merchant/billing/gateways/beanstream/beanstream_core.rb +244 -0
- data/lib/active_merchant/billing/gateways/beanstream.rb +102 -0
- data/lib/active_merchant/billing/gateways/beanstream_interac.rb +54 -0
- data/lib/active_merchant/billing/gateways/bogus.rb +98 -0
- data/lib/active_merchant/billing/gateways/braintree.rb +17 -0
- data/lib/active_merchant/billing/gateways/card_stream.rb +230 -0
- data/lib/active_merchant/billing/gateways/cyber_source.rb +594 -0
- data/lib/active_merchant/billing/gateways/data_cash.rb +593 -0
- data/lib/active_merchant/billing/gateways/efsnet.rb +229 -0
- data/lib/active_merchant/billing/gateways/elavon.rb +134 -0
- data/lib/active_merchant/billing/gateways/eway.rb +277 -0
- data/lib/active_merchant/billing/gateways/exact.rb +222 -0
- data/lib/active_merchant/billing/gateways/first_pay.rb +172 -0
- data/lib/active_merchant/billing/gateways/instapay.rb +164 -0
- data/lib/active_merchant/billing/gateways/jetpay.rb +270 -0
- data/lib/active_merchant/billing/gateways/linkpoint.rb +449 -0
- data/lib/active_merchant/billing/gateways/merchant_e_solutions.rb +154 -0
- data/lib/active_merchant/billing/gateways/merchant_ware.rb +283 -0
- data/lib/active_merchant/billing/gateways/modern_payments.rb +36 -0
- data/lib/active_merchant/billing/gateways/modern_payments_cim.rb +220 -0
- data/lib/active_merchant/billing/gateways/moneris.rb +205 -0
- data/lib/active_merchant/billing/gateways/net_registry.rb +189 -0
- data/lib/active_merchant/billing/gateways/netbilling.rb +168 -0
- data/lib/active_merchant/billing/gateways/ogone.rb +279 -0
- data/lib/active_merchant/billing/gateways/pay_junction.rb +392 -0
- data/lib/active_merchant/billing/gateways/pay_secure.rb +120 -0
- data/lib/active_merchant/billing/gateways/payflow/payflow_common_api.rb +207 -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.rb +236 -0
- data/lib/active_merchant/billing/gateways/payflow_express.rb +138 -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 +230 -0
- data/lib/active_merchant/billing/gateways/paypal/paypal_common_api.rb +326 -0
- data/lib/active_merchant/billing/gateways/paypal/paypal_express_response.rb +38 -0
- data/lib/active_merchant/billing/gateways/paypal.rb +121 -0
- data/lib/active_merchant/billing/gateways/paypal_ca.rb +13 -0
- data/lib/active_merchant/billing/gateways/paypal_express.rb +130 -0
- data/lib/active_merchant/billing/gateways/paypal_express_common.rb +20 -0
- data/lib/active_merchant/billing/gateways/plugnpay.rb +292 -0
- data/lib/active_merchant/billing/gateways/psigate.rb +214 -0
- data/lib/active_merchant/billing/gateways/psl_card.rb +304 -0
- data/lib/active_merchant/billing/gateways/quickpay.rb +213 -0
- data/lib/active_merchant/billing/gateways/realex.rb +200 -0
- data/lib/active_merchant/billing/gateways/sage/sage_bankcard.rb +88 -0
- data/lib/active_merchant/billing/gateways/sage/sage_core.rb +116 -0
- data/lib/active_merchant/billing/gateways/sage/sage_virtual_check.rb +97 -0
- data/lib/active_merchant/billing/gateways/sage.rb +146 -0
- data/lib/active_merchant/billing/gateways/sage_pay.rb +309 -0
- data/lib/active_merchant/billing/gateways/sallie_mae.rb +144 -0
- data/lib/active_merchant/billing/gateways/secure_pay.rb +31 -0
- data/lib/active_merchant/billing/gateways/secure_pay_au.rb +157 -0
- data/lib/active_merchant/billing/gateways/secure_pay_tech.rb +113 -0
- data/lib/active_merchant/billing/gateways/skip_jack.rb +453 -0
- data/lib/active_merchant/billing/gateways/smart_ps.rb +265 -0
- data/lib/active_merchant/billing/gateways/trans_first.rb +127 -0
- data/lib/active_merchant/billing/gateways/transax.rb +25 -0
- data/lib/active_merchant/billing/gateways/trust_commerce.rb +418 -0
- data/lib/active_merchant/billing/gateways/usa_epay.rb +194 -0
- data/lib/active_merchant/billing/gateways/verifi.rb +228 -0
- data/lib/active_merchant/billing/gateways/viaklix.rb +189 -0
- data/lib/active_merchant/billing/gateways/wirecard.rb +318 -0
- data/lib/active_merchant/billing/gateways.rb +18 -0
- data/lib/active_merchant/billing/integrations/action_view_helper.rb +79 -0
- data/lib/active_merchant/billing/integrations/bogus/helper.rb +17 -0
- data/lib/active_merchant/billing/integrations/bogus/notification.rb +11 -0
- data/lib/active_merchant/billing/integrations/bogus/return.rb +10 -0
- data/lib/active_merchant/billing/integrations/bogus.rb +23 -0
- data/lib/active_merchant/billing/integrations/chronopay/helper.rb +120 -0
- data/lib/active_merchant/billing/integrations/chronopay/notification.rb +158 -0
- data/lib/active_merchant/billing/integrations/chronopay/return.rb +10 -0
- data/lib/active_merchant/billing/integrations/chronopay.rb +23 -0
- data/lib/active_merchant/billing/integrations/gestpay/common.rb +42 -0
- data/lib/active_merchant/billing/integrations/gestpay/helper.rb +70 -0
- data/lib/active_merchant/billing/integrations/gestpay/notification.rb +85 -0
- data/lib/active_merchant/billing/integrations/gestpay/return.rb +10 -0
- data/lib/active_merchant/billing/integrations/gestpay.rb +25 -0
- data/lib/active_merchant/billing/integrations/helper.rb +93 -0
- data/lib/active_merchant/billing/integrations/hi_trust/helper.rb +58 -0
- data/lib/active_merchant/billing/integrations/hi_trust/notification.rb +59 -0
- data/lib/active_merchant/billing/integrations/hi_trust/return.rb +67 -0
- data/lib/active_merchant/billing/integrations/hi_trust.rb +27 -0
- data/lib/active_merchant/billing/integrations/nochex/helper.rb +68 -0
- data/lib/active_merchant/billing/integrations/nochex/notification.rb +94 -0
- data/lib/active_merchant/billing/integrations/nochex/return.rb +10 -0
- data/lib/active_merchant/billing/integrations/nochex.rb +88 -0
- data/lib/active_merchant/billing/integrations/notification.rb +62 -0
- data/lib/active_merchant/billing/integrations/paypal/helper.rb +119 -0
- data/lib/active_merchant/billing/integrations/paypal/notification.rb +154 -0
- data/lib/active_merchant/billing/integrations/paypal/return.rb +10 -0
- data/lib/active_merchant/billing/integrations/paypal.rb +39 -0
- data/lib/active_merchant/billing/integrations/quickpay/helper.rb +72 -0
- data/lib/active_merchant/billing/integrations/quickpay/notification.rb +74 -0
- data/lib/active_merchant/billing/integrations/quickpay.rb +17 -0
- data/lib/active_merchant/billing/integrations/return.rb +35 -0
- data/lib/active_merchant/billing/integrations/two_checkout/helper.rb +59 -0
- data/lib/active_merchant/billing/integrations/two_checkout/notification.rb +114 -0
- data/lib/active_merchant/billing/integrations/two_checkout/return.rb +17 -0
- data/lib/active_merchant/billing/integrations/two_checkout.rb +23 -0
- data/lib/active_merchant/billing/integrations.rb +29 -0
- data/lib/active_merchant/billing/response.rb +32 -0
- data/lib/active_merchant/billing.rb +9 -0
- data/lib/active_merchant/lib/connection.rb +170 -0
- data/lib/active_merchant/lib/country.rb +319 -0
- data/lib/active_merchant/lib/error.rb +4 -0
- data/lib/active_merchant/lib/post_data.rb +22 -0
- data/lib/active_merchant/lib/posts_data.rb +47 -0
- data/lib/active_merchant/lib/requires_parameters.rb +16 -0
- data/lib/active_merchant/lib/utils.rb +18 -0
- data/lib/active_merchant/lib/validateable.rb +76 -0
- data/lib/active_merchant.rb +46 -0
- data/lib/certs/cacert.pem +7815 -0
- data/lib/support/gateway_support.rb +58 -0
- metadata +218 -0
@@ -0,0 +1,304 @@
|
|
1
|
+
module ActiveMerchant
|
2
|
+
module Billing
|
3
|
+
#
|
4
|
+
# ActiveMerchant PSL Card Gateway
|
5
|
+
#
|
6
|
+
# Notes:
|
7
|
+
# -To be able to use the capture function, the IP address of the machine must be
|
8
|
+
# registered with PSL
|
9
|
+
# -ESALE_KEYED should only be used in situations where the cardholder perceives the
|
10
|
+
# transaction to be Internet-based, such as purchasing from a web site/on-line store.
|
11
|
+
# If the Internet is used purely for the transport of information from the merchant
|
12
|
+
# directly to the gateway then the appropriate cardholder present or not present message
|
13
|
+
# type should be used rather than the ‘E’ equivalent.
|
14
|
+
# -The CV2 / AVS policies are set up with the account settings when signing up for an account
|
15
|
+
class PslCardGateway < Gateway
|
16
|
+
self.money_format = :cents
|
17
|
+
self.default_currency = 'GBP'
|
18
|
+
|
19
|
+
self.supported_countries = ['GB']
|
20
|
+
# Visa Credit, Visa Debit, Mastercard, Maestro, Solo, Electron,
|
21
|
+
# American Express, Diners Club, JCB, International Maestro,
|
22
|
+
# Style, Clydesdale Financial Services, Other
|
23
|
+
|
24
|
+
self.supported_cardtypes = [ :visa, :master, :american_express, :diners_club, :jcb, :switch, :solo, :maestro ]
|
25
|
+
self.homepage_url = 'http://www.paymentsolutionsltd.com/'
|
26
|
+
self.display_name = 'PSL Payment Solutions'
|
27
|
+
|
28
|
+
# Default ISO 3166 country code (GB)
|
29
|
+
cattr_accessor :location
|
30
|
+
self.location = 826
|
31
|
+
|
32
|
+
# PslCard server URL - The url is the same whether testing or live - use
|
33
|
+
# the test account when testing...
|
34
|
+
URL = 'https://pslcard3.paymentsolutionsltd.com/secure/transact.asp?'
|
35
|
+
|
36
|
+
# eCommerce sale transaction, details keyed by merchant or cardholder
|
37
|
+
MESSAGE_TYPE = 'ESALE_KEYED'
|
38
|
+
|
39
|
+
# The type of response that we want to get from PSL, options are HTML, XML or REDIRECT
|
40
|
+
RESPONSE_ACTION = 'HTML'
|
41
|
+
|
42
|
+
# Currency Codes
|
43
|
+
CURRENCY_CODES = {
|
44
|
+
'AUD' => 036,
|
45
|
+
'GBP' => 826,
|
46
|
+
'USD' => 840
|
47
|
+
}
|
48
|
+
|
49
|
+
#The terminal used - only for swipe transactions, so hard coded to 32 for online
|
50
|
+
EMV_TERMINAL_TYPE = 32
|
51
|
+
|
52
|
+
#Different Dispatch types
|
53
|
+
DISPATCH_LATER = 'LATER'
|
54
|
+
DISPATCH_NOW = 'NOW'
|
55
|
+
|
56
|
+
# Return codes
|
57
|
+
APPROVED = '00'
|
58
|
+
|
59
|
+
#Nominal amount to authorize for a 'dispatch later' type
|
60
|
+
#The nominal amount is held straight away, when the goods are ready
|
61
|
+
#to be dispatched, PSL is informed and the full amount is the
|
62
|
+
#taken.
|
63
|
+
NOMINAL_AMOUNT = 101
|
64
|
+
|
65
|
+
AVS_CODE = {
|
66
|
+
"ALL MATCH" => 'Y',
|
67
|
+
"SECURITY CODE MATCH ONLY" => 'N',
|
68
|
+
"ADDRESS MATCH ONLY" => 'Y',
|
69
|
+
"NO DATA MATCHES" => 'N',
|
70
|
+
"DATA NOT CHECKED" => 'R',
|
71
|
+
"SECURITY CHECKS NOT SUPPORTED" => 'X'
|
72
|
+
}
|
73
|
+
|
74
|
+
CVV_CODE = {
|
75
|
+
"ALL MATCH" => 'M',
|
76
|
+
"SECURITY CODE MATCH ONLY" => 'M',
|
77
|
+
"ADDRESS MATCH ONLY" => 'N',
|
78
|
+
"NO DATA MATCHES" => 'N',
|
79
|
+
"DATA NOT CHECKED" => 'P',
|
80
|
+
"SECURITY CHECKS NOT SUPPORTED" => 'X'
|
81
|
+
}
|
82
|
+
|
83
|
+
# Create a new PslCardGateway
|
84
|
+
#
|
85
|
+
# The gateway requires that a valid :login be passed in the options hash
|
86
|
+
#
|
87
|
+
# Paramaters:
|
88
|
+
# -options:
|
89
|
+
# :login - the PslCard account login (required)
|
90
|
+
def initialize(options = {})
|
91
|
+
requires!(options, :login)
|
92
|
+
|
93
|
+
@options = options
|
94
|
+
super
|
95
|
+
end
|
96
|
+
|
97
|
+
# Purchase the item straight away
|
98
|
+
#
|
99
|
+
# Parameters:
|
100
|
+
# -money: Amount to be charged as an Integer value in cents
|
101
|
+
# -authorization: the PSL cross reference from the previous authorization
|
102
|
+
# -options:
|
103
|
+
#
|
104
|
+
# Returns:
|
105
|
+
# -ActiveRecord::Billing::Response object
|
106
|
+
#
|
107
|
+
def purchase(money, credit_card, options = {})
|
108
|
+
post = {}
|
109
|
+
|
110
|
+
add_amount(post, money, DISPATCH_NOW, options)
|
111
|
+
add_credit_card(post, credit_card)
|
112
|
+
add_address(post, options)
|
113
|
+
add_invoice(post, options)
|
114
|
+
add_purchase_details(post)
|
115
|
+
|
116
|
+
commit(post)
|
117
|
+
end
|
118
|
+
|
119
|
+
# Authorize the transaction
|
120
|
+
#
|
121
|
+
# Reserves the funds on the customer's credit card, but does not
|
122
|
+
# charge the card.
|
123
|
+
#
|
124
|
+
# This implementation does not authorize the full amount, rather it checks that the full amount
|
125
|
+
# is available and only 'reserves' the nominal amount (currently a pound and a penny)
|
126
|
+
#
|
127
|
+
# Parameters:
|
128
|
+
# -money: Amount to be charged as an Integer value in cents
|
129
|
+
# -authorization: the PSL cross reference from the previous authorization
|
130
|
+
# -options:
|
131
|
+
#
|
132
|
+
# Returns:
|
133
|
+
# -ActiveRecord::Billing::Response object
|
134
|
+
#
|
135
|
+
def authorize(money, credit_card, options = {})
|
136
|
+
post = {}
|
137
|
+
|
138
|
+
add_amount(post, money, DISPATCH_LATER, options)
|
139
|
+
add_credit_card(post, credit_card)
|
140
|
+
add_address(post, options)
|
141
|
+
add_invoice(post, options)
|
142
|
+
add_purchase_details(post)
|
143
|
+
|
144
|
+
commit(post)
|
145
|
+
end
|
146
|
+
|
147
|
+
# Post an authorization.
|
148
|
+
#
|
149
|
+
# Captures the funds from an authorized transaction.
|
150
|
+
#
|
151
|
+
# Parameters:
|
152
|
+
# -money: Amount to be charged as an Integer value in cents
|
153
|
+
# -authorization: The PSL Cross Reference
|
154
|
+
# -options:
|
155
|
+
#
|
156
|
+
# Returns:
|
157
|
+
# -ActiveRecord::Billing::Response object
|
158
|
+
#
|
159
|
+
def capture(money, authorization, options = {})
|
160
|
+
post = {}
|
161
|
+
|
162
|
+
add_amount(post, money, DISPATCH_NOW, options)
|
163
|
+
add_reference(post, authorization)
|
164
|
+
add_purchase_details(post)
|
165
|
+
|
166
|
+
commit(post)
|
167
|
+
end
|
168
|
+
|
169
|
+
private
|
170
|
+
|
171
|
+
def add_credit_card(post, credit_card)
|
172
|
+
post[:QAName] = credit_card.name
|
173
|
+
post[:CardNumber] = credit_card.number
|
174
|
+
post[:EMVTerminalType] = EMV_TERMINAL_TYPE
|
175
|
+
post[:ExpMonth] = credit_card.month
|
176
|
+
post[:ExpYear] = credit_card.year
|
177
|
+
|
178
|
+
if requires_start_date_or_issue_number?(credit_card)
|
179
|
+
post[:IssueNumber] = credit_card.issue_number unless credit_card.issue_number.blank?
|
180
|
+
post[:StartMonth] = credit_card.start_month unless credit_card.start_month.blank?
|
181
|
+
post[:StartYear] = credit_card.start_year unless credit_card.start_year.blank?
|
182
|
+
end
|
183
|
+
|
184
|
+
# CV2 check
|
185
|
+
post[:AVSCV2Check] = credit_card.verification_value? ? 'YES' : 'NO'
|
186
|
+
post[:CV2] = credit_card.verification_value if credit_card.verification_value?
|
187
|
+
end
|
188
|
+
|
189
|
+
def add_address(post, options)
|
190
|
+
address = options[:billing_address] || options[:address]
|
191
|
+
return if address.nil?
|
192
|
+
|
193
|
+
post[:QAAddress] = [:address1, :address2, :city, :state].collect{|a| address[a]}.reject{|a| a.blank?}.join(' ')
|
194
|
+
post[:QAPostcode] = address[:zip]
|
195
|
+
end
|
196
|
+
|
197
|
+
def add_invoice(post, options)
|
198
|
+
post[:MerchantName] = options[:merchant] || 'Merchant Name' # May use this as the order_id field
|
199
|
+
post[:OrderID] = options[:order_id] unless options[:order_id].blank?
|
200
|
+
end
|
201
|
+
|
202
|
+
def add_reference(post, authorization)
|
203
|
+
post[:CrossReference] = authorization
|
204
|
+
end
|
205
|
+
|
206
|
+
def add_amount(post, money, dispatch_type, options)
|
207
|
+
post[:CurrencyCode] = currency_code(options[:currency] || currency(money))
|
208
|
+
|
209
|
+
if dispatch_type == DISPATCH_LATER
|
210
|
+
post[:amount] = amount(NOMINAL_AMOUNT)
|
211
|
+
post[:DispatchLaterAmount] = amount(money)
|
212
|
+
else
|
213
|
+
post[:amount] = amount(money)
|
214
|
+
end
|
215
|
+
|
216
|
+
post[:Dispatch] = dispatch_type
|
217
|
+
end
|
218
|
+
|
219
|
+
def add_purchase_details(post)
|
220
|
+
post[:EchoAmount] = 'YES'
|
221
|
+
post[:SCBI] = 'YES' # Return information about the transaction
|
222
|
+
post[:MessageType] = MESSAGE_TYPE
|
223
|
+
end
|
224
|
+
|
225
|
+
# Get the currency code for the passed money object
|
226
|
+
#
|
227
|
+
# The money class stores the currency as an ISO 4217:2001 Alphanumeric,
|
228
|
+
# however PSL requires the ISO 4217:2001 Numeric code.
|
229
|
+
#
|
230
|
+
# Parameters:
|
231
|
+
# -money: Integer value in cents
|
232
|
+
#
|
233
|
+
# Returns:
|
234
|
+
# -the ISO 4217:2001 Numberic currency code
|
235
|
+
#
|
236
|
+
def currency_code(currency)
|
237
|
+
CURRENCY_CODES[currency]
|
238
|
+
end
|
239
|
+
|
240
|
+
# Parse the PSL response and create a Response object
|
241
|
+
#
|
242
|
+
# Parameters:
|
243
|
+
# -body: The response string returned from PSL, Formatted:
|
244
|
+
# Key=value&key=value...
|
245
|
+
#
|
246
|
+
# Returns:
|
247
|
+
# -a hash with all of the values returned in the PSL response
|
248
|
+
#
|
249
|
+
def parse(body)
|
250
|
+
|
251
|
+
fields = {}
|
252
|
+
for line in body.split('&')
|
253
|
+
key, value = *line.scan( %r{^(\w+)\=(.*)$} ).flatten
|
254
|
+
fields[key] = CGI.unescape(value)
|
255
|
+
end
|
256
|
+
fields.symbolize_keys
|
257
|
+
end
|
258
|
+
|
259
|
+
# Send the passed data to PSL for processing
|
260
|
+
#
|
261
|
+
# Parameters:
|
262
|
+
# -request: The data that is to be sent to PSL
|
263
|
+
#
|
264
|
+
# Returns:
|
265
|
+
# - ActiveMerchant::Billing::Response object
|
266
|
+
#
|
267
|
+
def commit(request)
|
268
|
+
response = parse( ssl_post(URL, post_data(request)) )
|
269
|
+
|
270
|
+
Response.new(response[:ResponseCode] == APPROVED, response[:Message], response,
|
271
|
+
:test => test?,
|
272
|
+
:authorization => response[:CrossReference],
|
273
|
+
:cvv_result => CVV_CODE[response[:AVSCV2Check]],
|
274
|
+
:avs_result => { :code => AVS_CODE[response[:AVSCV2Check]] }
|
275
|
+
)
|
276
|
+
end
|
277
|
+
|
278
|
+
# Put the passed data into a format that can be submitted to PSL
|
279
|
+
# Key=Value&Key=Value...
|
280
|
+
#
|
281
|
+
# Any ampersands and equal signs are removed from the data being posted
|
282
|
+
# as PSL puts them back into the response string which then cannot be parsed.
|
283
|
+
# This is after escaping before sending the request to PSL - this is a work
|
284
|
+
# around for the time being
|
285
|
+
#
|
286
|
+
# Parameters:
|
287
|
+
# -post: Hash of all the data to be sent
|
288
|
+
#
|
289
|
+
# Returns:
|
290
|
+
# -String: the data to be sent
|
291
|
+
#
|
292
|
+
def post_data(post)
|
293
|
+
post[:CountryCode] = self.location
|
294
|
+
post[:MerchantID] = @options[:login]
|
295
|
+
post[:ValidityID] = @options[:password]
|
296
|
+
post[:ResponseAction] = RESPONSE_ACTION
|
297
|
+
|
298
|
+
post.collect { |key, value|
|
299
|
+
"#{key}=#{CGI.escape(value.to_s.tr('&=', ' '))}"
|
300
|
+
}.join("&")
|
301
|
+
end
|
302
|
+
end
|
303
|
+
end
|
304
|
+
end
|
@@ -0,0 +1,213 @@
|
|
1
|
+
require 'rexml/document'
|
2
|
+
require 'digest/md5'
|
3
|
+
|
4
|
+
module ActiveMerchant #:nodoc:
|
5
|
+
module Billing #:nodoc:
|
6
|
+
class QuickpayGateway < Gateway
|
7
|
+
URL = 'https://secure.quickpay.dk/api'
|
8
|
+
|
9
|
+
self.default_currency = 'DKK'
|
10
|
+
self.money_format = :cents
|
11
|
+
self.supported_cardtypes = [ :dankort, :forbrugsforeningen, :visa, :master, :american_express, :diners_club, :jcb, :maestro ]
|
12
|
+
self.supported_countries = ['DK']
|
13
|
+
self.homepage_url = 'http://quickpay.dk/'
|
14
|
+
self.display_name = 'Quickpay'
|
15
|
+
|
16
|
+
PROTOCOL = 3
|
17
|
+
|
18
|
+
MD5_CHECK_FIELDS = {
|
19
|
+
:authorize => [:protocol, :msgtype, :merchant, :ordernumber, :amount, :currency, :autocapture, :cardnumber, :expirationdate, :cvd, :cardtypelock],
|
20
|
+
:capture => [:protocol, :msgtype, :merchant, :amount, :transaction],
|
21
|
+
:cancel => [:protocol, :msgtype, :merchant, :transaction],
|
22
|
+
:refund => [:protocol, :msgtype, :merchant, :amount, :transaction],
|
23
|
+
:subscribe => [:protocol, :msgtype, :merchant, :ordernumber, :cardnumber, :expirationdate, :cvd, :cardtypelock, :description],
|
24
|
+
:recurring => [:protocol, :msgtype, :merchant, :ordernumber, :amount, :currency, :autocapture, :transaction],
|
25
|
+
:status => [:protocol, :msgtype, :merchant, :transaction],
|
26
|
+
:chstatus => [:protocol, :msgtype, :merchant],
|
27
|
+
}
|
28
|
+
|
29
|
+
APPROVED = '000'
|
30
|
+
|
31
|
+
# The login is the QuickpayId
|
32
|
+
# The password is the md5checkword from the Quickpay admin interface
|
33
|
+
def initialize(options = {})
|
34
|
+
requires!(options, :login, :password)
|
35
|
+
@options = options
|
36
|
+
super
|
37
|
+
end
|
38
|
+
|
39
|
+
def authorize(money, credit_card_or_reference, options = {})
|
40
|
+
post = {}
|
41
|
+
|
42
|
+
add_amount(post, money, options)
|
43
|
+
add_invoice(post, options)
|
44
|
+
add_creditcard_or_reference(post, credit_card_or_reference, options)
|
45
|
+
add_autocapture(post, false)
|
46
|
+
|
47
|
+
commit(recurring_or_authorize(credit_card_or_reference), post)
|
48
|
+
end
|
49
|
+
|
50
|
+
def purchase(money, credit_card_or_reference, options = {})
|
51
|
+
post = {}
|
52
|
+
|
53
|
+
add_amount(post, money, options)
|
54
|
+
add_creditcard_or_reference(post, credit_card_or_reference, options)
|
55
|
+
add_invoice(post, options)
|
56
|
+
add_autocapture(post, true)
|
57
|
+
|
58
|
+
commit(recurring_or_authorize(credit_card_or_reference), post)
|
59
|
+
end
|
60
|
+
|
61
|
+
def capture(money, authorization, options = {})
|
62
|
+
post = {}
|
63
|
+
|
64
|
+
add_reference(post, authorization)
|
65
|
+
add_amount_without_currency(post, money)
|
66
|
+
|
67
|
+
commit(:capture, post)
|
68
|
+
end
|
69
|
+
|
70
|
+
def void(identification, options = {})
|
71
|
+
post = {}
|
72
|
+
|
73
|
+
add_reference(post, identification)
|
74
|
+
|
75
|
+
commit(:cancel, post)
|
76
|
+
end
|
77
|
+
|
78
|
+
def credit(money, identification, options = {})
|
79
|
+
post = {}
|
80
|
+
|
81
|
+
add_amount_without_currency(post, money)
|
82
|
+
add_reference(post, identification)
|
83
|
+
|
84
|
+
commit(:refund, post)
|
85
|
+
end
|
86
|
+
|
87
|
+
def store(creditcard, options = {})
|
88
|
+
post = {}
|
89
|
+
|
90
|
+
add_creditcard(post, creditcard, options)
|
91
|
+
add_invoice(post, options)
|
92
|
+
add_description(post, options)
|
93
|
+
|
94
|
+
commit(:subscribe, post)
|
95
|
+
end
|
96
|
+
|
97
|
+
private
|
98
|
+
|
99
|
+
def add_amount(post, money, options = {})
|
100
|
+
post[:amount] = amount(money)
|
101
|
+
post[:currency] = options[:currency] || currency(money)
|
102
|
+
end
|
103
|
+
|
104
|
+
def add_amount_without_currency(post, money, options = {})
|
105
|
+
post[:amount] = amount(money)
|
106
|
+
end
|
107
|
+
|
108
|
+
def add_invoice(post, options)
|
109
|
+
post[:ordernumber] = format_order_number(options[:order_id])
|
110
|
+
end
|
111
|
+
|
112
|
+
def add_creditcard(post, credit_card, options)
|
113
|
+
post[:cardnumber] = credit_card.number
|
114
|
+
post[:cvd] = credit_card.verification_value
|
115
|
+
post[:expirationdate] = expdate(credit_card)
|
116
|
+
post[:cardtypelock] = options[:cardtypelock] unless options[:cardtypelock].blank?
|
117
|
+
end
|
118
|
+
|
119
|
+
def add_reference(post, identification)
|
120
|
+
post[:transaction] = identification
|
121
|
+
end
|
122
|
+
|
123
|
+
def add_creditcard_or_reference(post, credit_card_or_reference, options)
|
124
|
+
if credit_card_or_reference.is_a?(String)
|
125
|
+
add_reference(post, credit_card_or_reference)
|
126
|
+
else
|
127
|
+
add_creditcard(post, credit_card_or_reference, options)
|
128
|
+
end
|
129
|
+
end
|
130
|
+
|
131
|
+
def add_autocapture(post, autocapture)
|
132
|
+
post[:autocapture] = autocapture ? 1 : 0
|
133
|
+
end
|
134
|
+
|
135
|
+
def recurring_or_authorize(credit_card_or_reference)
|
136
|
+
credit_card_or_reference.is_a?(String) ? :recurring : :authorize
|
137
|
+
end
|
138
|
+
|
139
|
+
def add_description(post, options)
|
140
|
+
post[:description] = options[:description]
|
141
|
+
end
|
142
|
+
|
143
|
+
def commit(action, params)
|
144
|
+
response = parse(ssl_post(URL, post_data(action, params)))
|
145
|
+
|
146
|
+
Response.new(successful?(response), message_from(response), response,
|
147
|
+
:test => test?,
|
148
|
+
:authorization => response[:transaction]
|
149
|
+
)
|
150
|
+
end
|
151
|
+
|
152
|
+
def successful?(response)
|
153
|
+
response[:qpstat] == APPROVED
|
154
|
+
end
|
155
|
+
|
156
|
+
def parse(data)
|
157
|
+
response = {}
|
158
|
+
|
159
|
+
doc = REXML::Document.new(data)
|
160
|
+
|
161
|
+
doc.root.elements.each do |element|
|
162
|
+
response[element.name.to_sym] = element.text
|
163
|
+
end
|
164
|
+
|
165
|
+
response
|
166
|
+
end
|
167
|
+
|
168
|
+
def message_from(response)
|
169
|
+
case response[:qpstat]
|
170
|
+
when '008' # Error in request data
|
171
|
+
response[:qpstatmsg].to_s
|
172
|
+
#.scan(/[A-Z][a-z0-9 \/]+/).to_sentence
|
173
|
+
else
|
174
|
+
response[:qpstatmsg].to_s
|
175
|
+
end
|
176
|
+
end
|
177
|
+
|
178
|
+
def post_data(action, params = {})
|
179
|
+
params[:protocol] = PROTOCOL
|
180
|
+
params[:msgtype] = action.to_s
|
181
|
+
params[:merchant] = @options[:login]
|
182
|
+
#params[:testmode] = '1' if test?
|
183
|
+
params[:md5check] = generate_check_hash(action, params)
|
184
|
+
|
185
|
+
params.collect { |key, value| "#{key}=#{CGI.escape(value.to_s)}" }.join("&")
|
186
|
+
end
|
187
|
+
|
188
|
+
def generate_check_hash(action, params)
|
189
|
+
string = MD5_CHECK_FIELDS[action].collect do |key|
|
190
|
+
params[key]
|
191
|
+
end.join('')
|
192
|
+
|
193
|
+
# Add the md5checkword
|
194
|
+
string << @options[:password].to_s
|
195
|
+
|
196
|
+
Digest::MD5.hexdigest(string)
|
197
|
+
end
|
198
|
+
|
199
|
+
def expdate(credit_card)
|
200
|
+
year = format(credit_card.year, :two_digits)
|
201
|
+
month = format(credit_card.month, :two_digits)
|
202
|
+
|
203
|
+
"#{year}#{month}"
|
204
|
+
end
|
205
|
+
|
206
|
+
# Limited to 20 digits max
|
207
|
+
def format_order_number(number)
|
208
|
+
number.to_s.gsub(/[^\w_]/, '').rjust(4, "0")[0...20]
|
209
|
+
end
|
210
|
+
end
|
211
|
+
end
|
212
|
+
end
|
213
|
+
|