tlconnor-activemerchant 1.20.4
Sign up to get free protection for your applications and to get access to all the features.
- data/CHANGELOG +805 -0
- data/CONTRIBUTORS +274 -0
- data/MIT-LICENSE +20 -0
- data/gem-public_cert.pem +20 -0
- data/lib/active_merchant.rb +63 -0
- data/lib/active_merchant/billing.rb +9 -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 +264 -0
- data/lib/active_merchant/billing/credit_card_formatting.rb +21 -0
- data/lib/active_merchant/billing/credit_card_methods.rb +129 -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 +170 -0
- data/lib/active_merchant/billing/gateways.rb +18 -0
- data/lib/active_merchant/billing/gateways/authorize_net.rb +694 -0
- data/lib/active_merchant/billing/gateways/authorize_net_cim.rb +944 -0
- data/lib/active_merchant/billing/gateways/barclays_epdq.rb +308 -0
- data/lib/active_merchant/billing/gateways/beanstream.rb +167 -0
- data/lib/active_merchant/billing/gateways/beanstream/beanstream_core.rb +388 -0
- data/lib/active_merchant/billing/gateways/beanstream_interac.rb +54 -0
- data/lib/active_merchant/billing/gateways/blue_pay.rb +11 -0
- data/lib/active_merchant/billing/gateways/bogus.rb +142 -0
- data/lib/active_merchant/billing/gateways/braintree.rb +17 -0
- data/lib/active_merchant/billing/gateways/braintree/braintree_common.rb +9 -0
- data/lib/active_merchant/billing/gateways/braintree_blue.rb +308 -0
- data/lib/active_merchant/billing/gateways/braintree_orange.rb +21 -0
- data/lib/active_merchant/billing/gateways/card_save.rb +23 -0
- data/lib/active_merchant/billing/gateways/card_stream.rb +230 -0
- data/lib/active_merchant/billing/gateways/certo_direct.rb +279 -0
- data/lib/active_merchant/billing/gateways/cyber_source.rb +430 -0
- data/lib/active_merchant/billing/gateways/data_cash.rb +597 -0
- data/lib/active_merchant/billing/gateways/efsnet.rb +235 -0
- data/lib/active_merchant/billing/gateways/elavon.rb +135 -0
- data/lib/active_merchant/billing/gateways/epay.rb +274 -0
- data/lib/active_merchant/billing/gateways/eway.rb +277 -0
- data/lib/active_merchant/billing/gateways/eway_managed.rb +265 -0
- data/lib/active_merchant/billing/gateways/exact.rb +227 -0
- data/lib/active_merchant/billing/gateways/federated_canada.rb +168 -0
- data/lib/active_merchant/billing/gateways/first_pay.rb +177 -0
- data/lib/active_merchant/billing/gateways/garanti.rb +262 -0
- data/lib/active_merchant/billing/gateways/ideal/ideal_base.rb +250 -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 +55 -0
- data/lib/active_merchant/billing/gateways/inspire.rb +221 -0
- data/lib/active_merchant/billing/gateways/instapay.rb +164 -0
- data/lib/active_merchant/billing/gateways/iridium.rb +258 -0
- data/lib/active_merchant/billing/gateways/jetpay.rb +276 -0
- data/lib/active_merchant/billing/gateways/linkpoint.rb +454 -0
- data/lib/active_merchant/billing/gateways/merchant_e_solutions.rb +157 -0
- data/lib/active_merchant/billing/gateways/merchant_ware.rb +289 -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 +239 -0
- data/lib/active_merchant/billing/gateways/nab_transact.rb +244 -0
- data/lib/active_merchant/billing/gateways/net_registry.rb +189 -0
- data/lib/active_merchant/billing/gateways/netaxept.rb +239 -0
- data/lib/active_merchant/billing/gateways/netbilling.rb +168 -0
- data/lib/active_merchant/billing/gateways/nmi.rb +13 -0
- data/lib/active_merchant/billing/gateways/ogone.rb +330 -0
- data/lib/active_merchant/billing/gateways/optimal_payment.rb +277 -0
- data/lib/active_merchant/billing/gateways/orbital.rb +344 -0
- data/lib/active_merchant/billing/gateways/orbital/orbital_soft_descriptors.rb +46 -0
- data/lib/active_merchant/billing/gateways/pay_junction.rb +397 -0
- data/lib/active_merchant/billing/gateways/pay_secure.rb +120 -0
- data/lib/active_merchant/billing/gateways/paybox_direct.rb +207 -0
- data/lib/active_merchant/billing/gateways/payflow.rb +261 -0
- data/lib/active_merchant/billing/gateways/payflow/payflow_common_api.rb +208 -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 +222 -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 +235 -0
- data/lib/active_merchant/billing/gateways/paypal.rb +121 -0
- data/lib/active_merchant/billing/gateways/paypal/paypal_common_api.rb +354 -0
- data/lib/active_merchant/billing/gateways/paypal/paypal_express_response.rb +49 -0
- data/lib/active_merchant/billing/gateways/paypal_ca.rb +13 -0
- data/lib/active_merchant/billing/gateways/paypal_express.rb +229 -0
- data/lib/active_merchant/billing/gateways/paypal_express_common.rb +25 -0
- data/lib/active_merchant/billing/gateways/paystation.rb +201 -0
- data/lib/active_merchant/billing/gateways/plugnpay.rb +298 -0
- data/lib/active_merchant/billing/gateways/psigate.rb +219 -0
- data/lib/active_merchant/billing/gateways/psl_card.rb +304 -0
- data/lib/active_merchant/billing/gateways/qbms.rb +297 -0
- data/lib/active_merchant/billing/gateways/quantum.rb +282 -0
- data/lib/active_merchant/billing/gateways/quickpay.rb +298 -0
- data/lib/active_merchant/billing/gateways/realex.rb +315 -0
- data/lib/active_merchant/billing/gateways/sage.rb +146 -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_pay.rb +320 -0
- data/lib/active_merchant/billing/gateways/sallie_mae.rb +144 -0
- data/lib/active_merchant/billing/gateways/samurai.rb +121 -0
- data/lib/active_merchant/billing/gateways/secure_net.rb +330 -0
- data/lib/active_merchant/billing/gateways/secure_pay.rb +31 -0
- data/lib/active_merchant/billing/gateways/secure_pay_au.rb +280 -0
- data/lib/active_merchant/billing/gateways/secure_pay_tech.rb +113 -0
- data/lib/active_merchant/billing/gateways/skip_jack.rb +458 -0
- data/lib/active_merchant/billing/gateways/smart_ps.rb +271 -0
- data/lib/active_merchant/billing/gateways/stripe.rb +244 -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 +423 -0
- data/lib/active_merchant/billing/gateways/usa_epay.rb +23 -0
- data/lib/active_merchant/billing/gateways/usa_epay_advanced.rb +1496 -0
- data/lib/active_merchant/billing/gateways/usa_epay_transaction.rb +206 -0
- data/lib/active_merchant/billing/gateways/verifi.rb +233 -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/worldpay.rb +280 -0
- data/lib/active_merchant/billing/integrations.rb +17 -0
- data/lib/active_merchant/billing/integrations/action_view_helper.rb +72 -0
- data/lib/active_merchant/billing/integrations/authorize_net_sim.rb +38 -0
- data/lib/active_merchant/billing/integrations/authorize_net_sim/helper.rb +228 -0
- data/lib/active_merchant/billing/integrations/authorize_net_sim/notification.rb +340 -0
- data/lib/active_merchant/billing/integrations/bogus.rb +23 -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/chronopay.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/direc_pay.rb +41 -0
- data/lib/active_merchant/billing/integrations/direc_pay/helper.rb +200 -0
- data/lib/active_merchant/billing/integrations/direc_pay/notification.rb +76 -0
- data/lib/active_merchant/billing/integrations/direc_pay/return.rb +32 -0
- data/lib/active_merchant/billing/integrations/direc_pay/status.rb +37 -0
- data/lib/active_merchant/billing/integrations/directebanking.rb +47 -0
- data/lib/active_merchant/billing/integrations/directebanking/helper.rb +90 -0
- data/lib/active_merchant/billing/integrations/directebanking/notification.rb +120 -0
- data/lib/active_merchant/billing/integrations/directebanking/return.rb +11 -0
- data/lib/active_merchant/billing/integrations/dwolla.rb +30 -0
- data/lib/active_merchant/billing/integrations/dwolla/helper.rb +31 -0
- data/lib/active_merchant/billing/integrations/dwolla/notification.rb +55 -0
- data/lib/active_merchant/billing/integrations/dwolla/return.rb +38 -0
- data/lib/active_merchant/billing/integrations/e_payment_plans.rb +48 -0
- data/lib/active_merchant/billing/integrations/e_payment_plans/helper.rb +34 -0
- data/lib/active_merchant/billing/integrations/e_payment_plans/notification.rb +84 -0
- data/lib/active_merchant/billing/integrations/gestpay.rb +25 -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/helper.rb +113 -0
- data/lib/active_merchant/billing/integrations/hi_trust.rb +27 -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/moneybookers.rb +26 -0
- data/lib/active_merchant/billing/integrations/moneybookers/helper.rb +59 -0
- data/lib/active_merchant/billing/integrations/moneybookers/notification.rb +129 -0
- data/lib/active_merchant/billing/integrations/nochex.rb +88 -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/notification.rb +62 -0
- data/lib/active_merchant/billing/integrations/payflow_link.rb +21 -0
- data/lib/active_merchant/billing/integrations/payflow_link/helper.rb +100 -0
- data/lib/active_merchant/billing/integrations/payflow_link/notification.rb +78 -0
- data/lib/active_merchant/billing/integrations/paypal.rb +39 -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/quickpay.rb +21 -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/return.rb +42 -0
- data/lib/active_merchant/billing/integrations/sage_pay_form.rb +37 -0
- data/lib/active_merchant/billing/integrations/sage_pay_form/encryption.rb +33 -0
- data/lib/active_merchant/billing/integrations/sage_pay_form/helper.rb +127 -0
- data/lib/active_merchant/billing/integrations/sage_pay_form/notification.rb +210 -0
- data/lib/active_merchant/billing/integrations/sage_pay_form/return.rb +31 -0
- data/lib/active_merchant/billing/integrations/two_checkout.rb +22 -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/valitor.rb +33 -0
- data/lib/active_merchant/billing/integrations/valitor/helper.rb +86 -0
- data/lib/active_merchant/billing/integrations/valitor/notification.rb +13 -0
- data/lib/active_merchant/billing/integrations/valitor/response_fields.rb +97 -0
- data/lib/active_merchant/billing/integrations/valitor/return.rb +13 -0
- data/lib/active_merchant/billing/integrations/world_pay.rb +27 -0
- data/lib/active_merchant/billing/integrations/world_pay/helper.rb +100 -0
- data/lib/active_merchant/billing/integrations/world_pay/notification.rb +160 -0
- data/lib/active_merchant/billing/response.rb +32 -0
- data/lib/active_merchant/version.rb +3 -0
- data/lib/activemerchant.rb +1 -0
- data/lib/support/gateway_support.rb +58 -0
- data/lib/support/outbound_hosts.rb +25 -0
- metadata +411 -0
@@ -0,0 +1,31 @@
|
|
1
|
+
require File.dirname(__FILE__) + '/authorize_net'
|
2
|
+
|
3
|
+
module ActiveMerchant #:nodoc:
|
4
|
+
module Billing #:nodoc:
|
5
|
+
class SecurePayGateway < AuthorizeNetGateway
|
6
|
+
self.live_url = self.test_url = 'https://www.securepay.com/AuthSpayAdapter/process.aspx'
|
7
|
+
|
8
|
+
self.homepage_url = 'http://www.securepay.com/'
|
9
|
+
self.display_name = 'SecurePay'
|
10
|
+
|
11
|
+
# Limit support to purchase() for the time being
|
12
|
+
# JRuby chokes here
|
13
|
+
# undef_method :authorize, :capture, :void, :credit
|
14
|
+
|
15
|
+
undef_method :authorize
|
16
|
+
undef_method :capture
|
17
|
+
undef_method :void
|
18
|
+
undef_method :credit
|
19
|
+
|
20
|
+
def test?
|
21
|
+
Base.gateway_mode == :test
|
22
|
+
end
|
23
|
+
|
24
|
+
private
|
25
|
+
def split(response)
|
26
|
+
response.split(',')
|
27
|
+
end
|
28
|
+
end
|
29
|
+
end
|
30
|
+
end
|
31
|
+
|
@@ -0,0 +1,280 @@
|
|
1
|
+
require 'rexml/document'
|
2
|
+
|
3
|
+
module ActiveMerchant #:nodoc:
|
4
|
+
module Billing #:nodoc:
|
5
|
+
class SecurePayAuGateway < Gateway
|
6
|
+
API_VERSION = 'xml-4.2'
|
7
|
+
PERIODIC_API_VERSION = 'spxml-3.0'
|
8
|
+
|
9
|
+
TEST_URL = 'https://www.securepay.com.au/test/payment'
|
10
|
+
LIVE_URL = 'https://www.securepay.com.au/xmlapi/payment'
|
11
|
+
TEST_PERIODIC_URL = "https://test.securepay.com.au/xmlapi/periodic"
|
12
|
+
LIVE_PERIODIC_URL = "https://api.securepay.com.au/xmlapi/periodic"
|
13
|
+
|
14
|
+
self.supported_countries = ['AU']
|
15
|
+
self.supported_cardtypes = [:visa, :master, :american_express, :diners_club, :jcb]
|
16
|
+
|
17
|
+
# The homepage URL of the gateway
|
18
|
+
self.homepage_url = 'http://securepay.com.au'
|
19
|
+
|
20
|
+
# The name of the gateway
|
21
|
+
self.display_name = 'SecurePay'
|
22
|
+
|
23
|
+
class_attribute :request_timeout
|
24
|
+
self.request_timeout = 60
|
25
|
+
|
26
|
+
self.money_format = :cents
|
27
|
+
self.default_currency = 'AUD'
|
28
|
+
|
29
|
+
# 0 Standard Payment
|
30
|
+
# 4 Refund
|
31
|
+
# 6 Client Reversal (Void)
|
32
|
+
# 10 Preauthorise
|
33
|
+
# 11 Preauth Complete (Advice)
|
34
|
+
TRANSACTIONS = {
|
35
|
+
:purchase => 0,
|
36
|
+
:authorization => 10,
|
37
|
+
:capture => 11,
|
38
|
+
:void => 6,
|
39
|
+
:refund => 4
|
40
|
+
}
|
41
|
+
|
42
|
+
PERIODIC_ACTIONS = {
|
43
|
+
:add_triggered => "add",
|
44
|
+
:remove_triggered => "delete",
|
45
|
+
:trigger => "trigger"
|
46
|
+
}
|
47
|
+
|
48
|
+
PERIODIC_TYPES = {
|
49
|
+
:add_triggered => 4,
|
50
|
+
:remove_triggered => nil,
|
51
|
+
:trigger => nil
|
52
|
+
}
|
53
|
+
|
54
|
+
SUCCESS_CODES = [ '00', '08', '11', '16', '77' ]
|
55
|
+
|
56
|
+
def initialize(options = {})
|
57
|
+
requires!(options, :login, :password)
|
58
|
+
@options = options
|
59
|
+
super
|
60
|
+
end
|
61
|
+
|
62
|
+
def test?
|
63
|
+
@options[:test] || super
|
64
|
+
end
|
65
|
+
|
66
|
+
def purchase(money, credit_card_or_stored_id, options = {})
|
67
|
+
if credit_card_or_stored_id.is_a?(ActiveMerchant::Billing::CreditCard)
|
68
|
+
requires!(options, :order_id)
|
69
|
+
commit :purchase, build_purchase_request(money, credit_card_or_stored_id, options)
|
70
|
+
else
|
71
|
+
options[:billing_id] = credit_card_or_stored_id.to_s
|
72
|
+
commit_periodic(build_periodic_item(:trigger, money, nil, options))
|
73
|
+
end
|
74
|
+
end
|
75
|
+
|
76
|
+
def authorize(money, credit_card, options = {})
|
77
|
+
requires!(options, :order_id)
|
78
|
+
commit :authorization, build_purchase_request(money, credit_card, options)
|
79
|
+
end
|
80
|
+
|
81
|
+
def capture(money, reference, options = {})
|
82
|
+
commit :capture, build_reference_request(money, reference)
|
83
|
+
end
|
84
|
+
|
85
|
+
def refund(money, reference, options = {})
|
86
|
+
commit :refund, build_reference_request(money, reference)
|
87
|
+
end
|
88
|
+
|
89
|
+
def credit(money, reference, options = {})
|
90
|
+
deprecated CREDIT_DEPRECATION_MESSAGE
|
91
|
+
refund(money, reference)
|
92
|
+
end
|
93
|
+
|
94
|
+
def void(reference, options = {})
|
95
|
+
commit :void, build_reference_request(nil, reference)
|
96
|
+
end
|
97
|
+
|
98
|
+
def store(creditcard, options = {})
|
99
|
+
requires!(options, :billing_id, :amount)
|
100
|
+
commit_periodic(build_periodic_item(:add_triggered, options[:amount], creditcard, options))
|
101
|
+
end
|
102
|
+
|
103
|
+
def unstore(identification, options = {})
|
104
|
+
options[:billing_id] = identification
|
105
|
+
commit_periodic(build_periodic_item(:remove_triggered, options[:amount], nil, options))
|
106
|
+
end
|
107
|
+
|
108
|
+
private
|
109
|
+
|
110
|
+
def build_purchase_request(money, credit_card, options)
|
111
|
+
xml = Builder::XmlMarkup.new
|
112
|
+
|
113
|
+
xml.tag! 'amount', amount(money)
|
114
|
+
xml.tag! 'currency', options[:currency] || currency(money)
|
115
|
+
xml.tag! 'purchaseOrderNo', options[:order_id].to_s.gsub(/[ ']/, '')
|
116
|
+
|
117
|
+
xml.tag! 'CreditCardInfo' do
|
118
|
+
xml.tag! 'cardNumber', credit_card.number
|
119
|
+
xml.tag! 'expiryDate', expdate(credit_card)
|
120
|
+
xml.tag! 'cvv', credit_card.verification_value if credit_card.verification_value?
|
121
|
+
end
|
122
|
+
|
123
|
+
xml.target!
|
124
|
+
end
|
125
|
+
|
126
|
+
def build_reference_request(money, reference)
|
127
|
+
xml = Builder::XmlMarkup.new
|
128
|
+
|
129
|
+
transaction_id, order_id, preauth_id, original_amount = reference.split("*")
|
130
|
+
xml.tag! 'amount', (money ? amount(money) : original_amount)
|
131
|
+
xml.tag! 'currency', options[:currency] || currency(money)
|
132
|
+
xml.tag! 'txnID', transaction_id
|
133
|
+
xml.tag! 'purchaseOrderNo', order_id
|
134
|
+
xml.tag! 'preauthID', preauth_id
|
135
|
+
|
136
|
+
xml.target!
|
137
|
+
end
|
138
|
+
|
139
|
+
def build_request(action, body)
|
140
|
+
xml = Builder::XmlMarkup.new
|
141
|
+
xml.instruct!
|
142
|
+
xml.tag! 'SecurePayMessage' do
|
143
|
+
xml.tag! 'MessageInfo' do
|
144
|
+
xml.tag! 'messageID', ActiveMerchant::Utils.generate_unique_id.slice(0, 30)
|
145
|
+
xml.tag! 'messageTimestamp', generate_timestamp
|
146
|
+
xml.tag! 'timeoutValue', request_timeout
|
147
|
+
xml.tag! 'apiVersion', API_VERSION
|
148
|
+
end
|
149
|
+
|
150
|
+
xml.tag! 'MerchantInfo' do
|
151
|
+
xml.tag! 'merchantID', @options[:login]
|
152
|
+
xml.tag! 'password', @options[:password]
|
153
|
+
end
|
154
|
+
|
155
|
+
xml.tag! 'RequestType', 'Payment'
|
156
|
+
xml.tag! 'Payment' do
|
157
|
+
xml.tag! 'TxnList', "count" => 1 do
|
158
|
+
xml.tag! 'Txn', "ID" => 1 do
|
159
|
+
xml.tag! 'txnType', TRANSACTIONS[action]
|
160
|
+
xml.tag! 'txnSource', 23
|
161
|
+
xml << body
|
162
|
+
end
|
163
|
+
end
|
164
|
+
end
|
165
|
+
end
|
166
|
+
|
167
|
+
xml.target!
|
168
|
+
end
|
169
|
+
|
170
|
+
def commit(action, request)
|
171
|
+
response = parse(ssl_post(test? ? TEST_URL : LIVE_URL, build_request(action, request)))
|
172
|
+
|
173
|
+
Response.new(success?(response), message_from(response), response,
|
174
|
+
:test => test?,
|
175
|
+
:authorization => authorization_from(response)
|
176
|
+
)
|
177
|
+
end
|
178
|
+
|
179
|
+
def build_periodic_item(action, money, credit_card, options)
|
180
|
+
xml = Builder::XmlMarkup.new
|
181
|
+
|
182
|
+
xml.tag! 'actionType', PERIODIC_ACTIONS[action]
|
183
|
+
xml.tag! 'clientID', options[:billing_id].to_s
|
184
|
+
|
185
|
+
if credit_card
|
186
|
+
xml.tag! 'CreditCardInfo' do
|
187
|
+
xml.tag! 'cardNumber', credit_card.number
|
188
|
+
xml.tag! 'expiryDate', expdate(credit_card)
|
189
|
+
xml.tag! 'cvv', credit_card.verification_value if credit_card.verification_value?
|
190
|
+
end
|
191
|
+
end
|
192
|
+
xml.tag! 'amount', amount(money)
|
193
|
+
xml.tag! 'periodicType', PERIODIC_TYPES[action] if PERIODIC_TYPES[action]
|
194
|
+
|
195
|
+
xml.target!
|
196
|
+
end
|
197
|
+
|
198
|
+
def build_periodic_request(body)
|
199
|
+
xml = Builder::XmlMarkup.new
|
200
|
+
xml.instruct!
|
201
|
+
xml.tag! 'SecurePayMessage' do
|
202
|
+
xml.tag! 'MessageInfo' do
|
203
|
+
xml.tag! 'messageID', ActiveMerchant::Utils.generate_unique_id.slice(0, 30)
|
204
|
+
xml.tag! 'messageTimestamp', generate_timestamp
|
205
|
+
xml.tag! 'timeoutValue', request_timeout
|
206
|
+
xml.tag! 'apiVersion', PERIODIC_API_VERSION
|
207
|
+
end
|
208
|
+
|
209
|
+
xml.tag! 'MerchantInfo' do
|
210
|
+
xml.tag! 'merchantID', @options[:login]
|
211
|
+
xml.tag! 'password', @options[:password]
|
212
|
+
end
|
213
|
+
|
214
|
+
xml.tag! 'RequestType', 'Periodic'
|
215
|
+
xml.tag! 'Periodic' do
|
216
|
+
xml.tag! 'PeriodicList', "count" => 1 do
|
217
|
+
xml.tag! 'PeriodicItem', "ID" => 1 do
|
218
|
+
xml << body
|
219
|
+
end
|
220
|
+
end
|
221
|
+
end
|
222
|
+
end
|
223
|
+
xml.target!
|
224
|
+
end
|
225
|
+
|
226
|
+
def commit_periodic(request)
|
227
|
+
my_request = build_periodic_request(request)
|
228
|
+
#puts my_request
|
229
|
+
response = parse(ssl_post(test? ? TEST_PERIODIC_URL : LIVE_PERIODIC_URL, my_request))
|
230
|
+
|
231
|
+
Response.new(success?(response), message_from(response), response,
|
232
|
+
:test => test?,
|
233
|
+
:authorization => authorization_from(response)
|
234
|
+
)
|
235
|
+
end
|
236
|
+
|
237
|
+
def success?(response)
|
238
|
+
SUCCESS_CODES.include?(response[:response_code])
|
239
|
+
end
|
240
|
+
|
241
|
+
def authorization_from(response)
|
242
|
+
[response[:txn_id], response[:purchase_order_no], response[:preauth_id], response[:amount]].join('*')
|
243
|
+
end
|
244
|
+
|
245
|
+
def message_from(response)
|
246
|
+
response[:response_text] || response[:status_description]
|
247
|
+
end
|
248
|
+
|
249
|
+
def expdate(credit_card)
|
250
|
+
"#{format(credit_card.month, :two_digits)}/#{format(credit_card.year, :two_digits)}"
|
251
|
+
end
|
252
|
+
|
253
|
+
def parse(body)
|
254
|
+
xml = REXML::Document.new(body)
|
255
|
+
|
256
|
+
response = {}
|
257
|
+
|
258
|
+
xml.root.elements.to_a.each do |node|
|
259
|
+
parse_element(response, node)
|
260
|
+
end
|
261
|
+
|
262
|
+
response
|
263
|
+
end
|
264
|
+
|
265
|
+
def parse_element(response, node)
|
266
|
+
if node.has_elements?
|
267
|
+
node.elements.each{|element| parse_element(response, element) }
|
268
|
+
else
|
269
|
+
response[node.name.underscore.to_sym] = node.text
|
270
|
+
end
|
271
|
+
end
|
272
|
+
|
273
|
+
# YYYYDDMMHHNNSSKKK000sOOO
|
274
|
+
def generate_timestamp
|
275
|
+
time = Time.now.utc
|
276
|
+
time.strftime("%Y%d%m%H%M%S#{time.usec}+000")
|
277
|
+
end
|
278
|
+
end
|
279
|
+
end
|
280
|
+
end
|
@@ -0,0 +1,113 @@
|
|
1
|
+
module ActiveMerchant #:nodoc:
|
2
|
+
module Billing #:nodoc:
|
3
|
+
class SecurePayTechGateway < Gateway
|
4
|
+
class SecurePayTechPostData < PostData
|
5
|
+
self.required_fields = [ :OrderReference, :CardNumber, :CardExpiry, :CardHolderName, :CardType, :MerchantID, :MerchantKey, :Amount, :Currency ]
|
6
|
+
end
|
7
|
+
|
8
|
+
URL = 'https://tx.securepaytech.com/web/HttpPostPurchase'
|
9
|
+
|
10
|
+
PAYMENT_GATEWAY_RESPONSES = {
|
11
|
+
1 => "Transaction OK",
|
12
|
+
2 => "Insufficient funds",
|
13
|
+
3 => "Card expired",
|
14
|
+
4 => "Card declined",
|
15
|
+
5 => "Server error",
|
16
|
+
6 => "Communications error",
|
17
|
+
7 => "Unsupported transaction type",
|
18
|
+
8 => "Bad or malformed request",
|
19
|
+
9 => "Invalid card number"
|
20
|
+
}
|
21
|
+
|
22
|
+
self.default_currency = 'NZD'
|
23
|
+
self.supported_countries = ['NZ']
|
24
|
+
self.supported_cardtypes = [:visa, :master, :american_express, :diners_club]
|
25
|
+
self.homepage_url = 'http://www.securepaytech.com/'
|
26
|
+
self.display_name = 'SecurePayTech'
|
27
|
+
|
28
|
+
def initialize(options = {})
|
29
|
+
requires!(options, :login, :password)
|
30
|
+
@options = options
|
31
|
+
super
|
32
|
+
end
|
33
|
+
|
34
|
+
def purchase(money, creditcard, options = {})
|
35
|
+
post = SecurePayTechPostData.new
|
36
|
+
|
37
|
+
add_invoice(post, money, options)
|
38
|
+
add_creditcard(post, creditcard)
|
39
|
+
|
40
|
+
commit(:purchase, post)
|
41
|
+
end
|
42
|
+
|
43
|
+
private
|
44
|
+
|
45
|
+
def add_invoice(post, money, options)
|
46
|
+
post[:Amount] = amount(money)
|
47
|
+
post[:Currency] = options[:currency] || currency(money)
|
48
|
+
|
49
|
+
post[:OrderReference] = options[:order_id]
|
50
|
+
end
|
51
|
+
|
52
|
+
def add_creditcard(post, creditcard)
|
53
|
+
post[:CardNumber] = creditcard.number
|
54
|
+
post[:CardExpiry] = expdate(creditcard)
|
55
|
+
post[:CardHolderName] = creditcard.name
|
56
|
+
|
57
|
+
if creditcard.verification_value?
|
58
|
+
post[:EnableCSC] = true
|
59
|
+
post[:CSC] = creditcard.verification_value
|
60
|
+
end
|
61
|
+
|
62
|
+
# SPT will autodetect this
|
63
|
+
post[:CardType] = 0
|
64
|
+
end
|
65
|
+
|
66
|
+
def parse(body)
|
67
|
+
response = CGI.unescape(body).split(',')
|
68
|
+
|
69
|
+
result = {}
|
70
|
+
result[:result_code] = response[0].to_i
|
71
|
+
|
72
|
+
if response.length == 2
|
73
|
+
result[:fail_reason] = response[1]
|
74
|
+
else
|
75
|
+
result[:merchant_transaction_reference] = response[1]
|
76
|
+
result[:receipt_number] = response[2]
|
77
|
+
result[:transaction_number] = response[3]
|
78
|
+
result[:authorisation_id] = response[4]
|
79
|
+
result[:batch_number] = response[5]
|
80
|
+
end
|
81
|
+
|
82
|
+
result
|
83
|
+
end
|
84
|
+
|
85
|
+
def commit(action, post)
|
86
|
+
response = parse( ssl_post(URL, post_data(action, post) ) )
|
87
|
+
|
88
|
+
Response.new(response[:result_code] == 1, message_from(response), response,
|
89
|
+
:test => test?,
|
90
|
+
:authorization => response[:merchant_transaction_reference]
|
91
|
+
)
|
92
|
+
end
|
93
|
+
|
94
|
+
def message_from(result)
|
95
|
+
PAYMENT_GATEWAY_RESPONSES[result[:result_code]]
|
96
|
+
end
|
97
|
+
|
98
|
+
def post_data(action, post)
|
99
|
+
post[:MerchantID] = @options[:login]
|
100
|
+
post[:MerchantKey] = @options[:password]
|
101
|
+
post.to_s
|
102
|
+
end
|
103
|
+
|
104
|
+
def expdate(creditcard)
|
105
|
+
year = sprintf("%.4i", creditcard.year)
|
106
|
+
month = sprintf("%.2i", creditcard.month)
|
107
|
+
|
108
|
+
"#{month}#{year[-2..-1]}"
|
109
|
+
end
|
110
|
+
end
|
111
|
+
end
|
112
|
+
end
|
113
|
+
|
@@ -0,0 +1,458 @@
|
|
1
|
+
#!ruby19
|
2
|
+
# encoding: utf-8
|
3
|
+
|
4
|
+
module ActiveMerchant #:nodoc:
|
5
|
+
module Billing #:nodoc:
|
6
|
+
class SkipJackGateway < Gateway
|
7
|
+
API_VERSION = '?.?'
|
8
|
+
|
9
|
+
LIVE_HOST = "https://www.skipjackic.com"
|
10
|
+
TEST_HOST = "https://developer.skipjackic.com"
|
11
|
+
|
12
|
+
BASIC_PATH = "/scripts/evolvcc.dll"
|
13
|
+
ADVANCED_PATH = "/evolvcc/evolvcc.aspx"
|
14
|
+
|
15
|
+
ACTIONS = {
|
16
|
+
:authorization => 'AuthorizeAPI',
|
17
|
+
:change_status => 'SJAPI_TransactionChangeStatusRequest',
|
18
|
+
:get_status => 'SJAPI_TransactionStatusRequest'
|
19
|
+
}
|
20
|
+
|
21
|
+
SUCCESS_MESSAGE = 'The transaction was successful.'
|
22
|
+
|
23
|
+
MONETARY_CHANGE_STATUSES = ['SETTLE', 'AUTHORIZE', 'AUTHORIZE ADDITIONAL', 'CREDIT', 'SPLITSETTLE']
|
24
|
+
|
25
|
+
CARD_CODE_ERRORS = %w( N S "" )
|
26
|
+
|
27
|
+
CARD_CODE_MESSAGES = {
|
28
|
+
"M" => "Card verification number matched",
|
29
|
+
"N" => "Card verification number didn't match",
|
30
|
+
"P" => "Card verification number was not processed",
|
31
|
+
"S" => "Card verification number should be on card but was not indicated",
|
32
|
+
"U" => "Issuer was not certified for card verification",
|
33
|
+
"" => "Transaction failed because incorrect card verification number was entered or no number was entered"
|
34
|
+
}
|
35
|
+
|
36
|
+
AVS_ERRORS = %w( A B C E I N O P R W Z )
|
37
|
+
|
38
|
+
AVS_MESSAGES = {
|
39
|
+
"A" => "Street address matches billing information, zip/postal code does not",
|
40
|
+
"B" => "Street address match for international transaction. Postal code not verified due to incompatible formats",
|
41
|
+
"C" => "Street address and postal code not verified for internation transaction due to incompatible formats",
|
42
|
+
"D" => "Street address and postal code match for international transaction",
|
43
|
+
"E" => "Address verification service error",
|
44
|
+
"I" => "Address information not verified by international issuer",
|
45
|
+
"M" => "Street address and postal code match for international transaction",
|
46
|
+
"N" => "Neither street address nor zip/postal match billing information",
|
47
|
+
"O" => "Non-US issuer does not participate",
|
48
|
+
"P" => "Postal codes match for international transaction but street address not verified due to incompatible formats",
|
49
|
+
"P" => "Address verification not applicable for this transaction",
|
50
|
+
"R" => "Payment gateway was unavailable or timed out",
|
51
|
+
"S" => "Address verification service not supported by issuer",
|
52
|
+
"U" => "Address information is unavailable",
|
53
|
+
"W" => "9-digit zip/postal code matches billing information, street address does not",
|
54
|
+
"X" => "Street address and 9-digit zip/postal code matches billing information",
|
55
|
+
"Y" => "Street address and 5-digit zip/postal code matches billing information",
|
56
|
+
"Z" => "5-digit zip/postal code matches billing information, street address does not",
|
57
|
+
}
|
58
|
+
|
59
|
+
CHANGE_STATUS_ERROR_MESSAGES = {
|
60
|
+
'0' => 'Success',
|
61
|
+
'-1' => 'Invalid Command',
|
62
|
+
'-2' => 'Parameter Missing',
|
63
|
+
'-3' => 'Failed retrieving response',
|
64
|
+
'-4' => 'Invalid Status',
|
65
|
+
'-5' => 'Failed reading security flags',
|
66
|
+
'-6' => 'Developer serial number not found',
|
67
|
+
'-7' => 'Invalid Serial Number'
|
68
|
+
}
|
69
|
+
|
70
|
+
TRANSACTION_CURRENT_STATUS = {
|
71
|
+
'0' => 'Idle',
|
72
|
+
'1' => 'Authorized',
|
73
|
+
'2' => 'Denied',
|
74
|
+
'3' => 'Settled',
|
75
|
+
'4' => 'Credited',
|
76
|
+
'5' => 'Deleted',
|
77
|
+
'6' => 'Archived',
|
78
|
+
'7' => 'Pre-Authorized',
|
79
|
+
'8' => 'Split Settled'
|
80
|
+
}
|
81
|
+
|
82
|
+
TRANSACTION_PENDING_STATUS = {
|
83
|
+
'0' => 'Idle',
|
84
|
+
'1' => 'Pending Credit',
|
85
|
+
'2' => 'Pending Settlement',
|
86
|
+
'3' => 'Pending Authorization',
|
87
|
+
'4' => 'Pending Manual Settlement',
|
88
|
+
'5' => 'Pending Recurring'
|
89
|
+
}
|
90
|
+
|
91
|
+
RETURN_CODE_MESSAGES = {
|
92
|
+
'-1' => 'Data was not by received intact by Skipjack Transaction Network.',
|
93
|
+
'0' => 'Communication Failure. Error in Request and Response at IP level.',
|
94
|
+
'1' => 'Valid Data. Authorization request was valid.',
|
95
|
+
'-35' => 'Invalid credit card number. Retry with correct credit card number.',
|
96
|
+
'-37' => 'Merchant Processor Unavailable. Skipjack is unable to communicate with payment Processor. Retry',
|
97
|
+
'-39' => 'Length or value of HTML Serial. Number Invalid serial number. Check HTML Serial Number length and that it is a correct/valid number. Confirm you are sending to the correct environment (Development or Production)',
|
98
|
+
'-51' => 'The value or length for billing zip code is incorrect.',
|
99
|
+
'-52' => 'The value or length for shipping zip code is incorrect.',
|
100
|
+
'-53' => 'The value or length for credit card expiration month is incorrect.',
|
101
|
+
'-54' => 'The value or length of the month or year of the credit card account number was incorrect.',
|
102
|
+
'-55' => 'The value or length or billing street address is incorrect.',
|
103
|
+
'-56' => 'The value or length of the shipping address is incorrect.',
|
104
|
+
'-57' => 'The length of the transaction amount must be at least 3 digits long (excluding the decimal place).',
|
105
|
+
'-58' => 'Length or value in Merchant Name Merchant Name associated with Skipjack account is misconfigured or invalid',
|
106
|
+
'-59' => 'Length or value in Merchant Address Merchant Address associated with Skipjack account is misconfigured or invalid',
|
107
|
+
'-60' => 'Length or value in Merchant State Merchant State associated with Skipjack account is misconfigured or invalid',
|
108
|
+
'-61' => 'The value or length for shipping state/province is empty.',
|
109
|
+
'-62' => 'The value for length orderstring is empty.',
|
110
|
+
'-64' => 'The value for the phone number is incorrect.',
|
111
|
+
'-65' => 'The value or length for billing name is empty.',
|
112
|
+
'-66' => 'The value or length for billing e-mail is empty.',
|
113
|
+
'-67' => 'The value or length for billing street address is empty.',
|
114
|
+
'-68' => 'The value or length for billing city is empty.',
|
115
|
+
'-69' => 'The value or length for billing state is empty.',
|
116
|
+
'-70' => 'Empty zipcode Zip Code field is empty.',
|
117
|
+
'-71' => 'Empty ordernumber Ordernumber field is empty.',
|
118
|
+
'-72' => 'Empty accountnumber Accountnumber field is empty',
|
119
|
+
'-73' => 'Empty month Month field is empty.',
|
120
|
+
'-74' => 'Empty year Year field is empty.',
|
121
|
+
'-75' => 'Empty serialnumber Serialnumber field is empty.',
|
122
|
+
'-76' => 'Empty transactionamount Transaction amount field is empty.',
|
123
|
+
'-77' => 'Empty orderstring Orderstring field is empty.',
|
124
|
+
'-78' => 'Empty shiptophone Shiptophone field is empty.',
|
125
|
+
'-79' => 'The value or length for billing name is empty.',
|
126
|
+
'-80' => 'Length shipto name Error in the length or value of shiptophone.',
|
127
|
+
'-81' => 'Length or value of Customer location',
|
128
|
+
'-82' => 'The value or length for billing state is empty.',
|
129
|
+
'-83' => 'The value or length for shipping phone is empty.',
|
130
|
+
'-84' => 'There is already an existing pending transaction in the register sharing the posted Order Number.',
|
131
|
+
'-85' => 'Airline leg info invalid Airline leg field value is invalid or empty.',
|
132
|
+
'-86' => 'Airline ticket info invalid Airline ticket info field is invalid or empty',
|
133
|
+
'-87' => 'Point of Sale check routing number must be 9 numeric digits Point of Sale check routing number is invalid or empty.',
|
134
|
+
'-88' => 'Point of Sale check account number missing or invalid Point of Sale check account number is invalid or empty.',
|
135
|
+
'-89' => 'Point of Sale check MICR missing or invalid Point of Sale check MICR invalid or empty.',
|
136
|
+
'-90' => 'Point of Sale check number missing or invalid Point of Sale check number invalid or empty.',
|
137
|
+
'-91' => 'CVV2 Invalid or empty "Make CVV a required field feature" enabled (New feature 01 April 2006) in the Merchant Account Setup interface but no CVV code was sent in the transaction data.',
|
138
|
+
'-92' => 'Approval Code Invalid Approval Code Invalid. Approval Code is a 6 digit code.',
|
139
|
+
'-93' => 'Blind Credits Request Refused "Allow Blind Credits" option must be enabled on the Skipjack Merchant Account.',
|
140
|
+
'-94' => 'Blind Credits Failed',
|
141
|
+
'-95' => 'Voice Authorization Request Refused Voice Authorization option must be enabled on the Skipjack Merchant Account.',
|
142
|
+
'-96' => 'Voice Authorizations Failed',
|
143
|
+
'-97' => 'Fraud Rejection Violates Velocity Settling.',
|
144
|
+
'-98' => 'Invalid Discount Amount',
|
145
|
+
'-99' => 'POS PIN Debit Pin Block Debit-specific',
|
146
|
+
'-100' => 'POS PIN Debit Invalid Key Serial Number Debit-specific',
|
147
|
+
'-101' => 'Invalid Authentication Data Data for Verified by Visa/MC Secure Code is invalid.',
|
148
|
+
'-102' => 'Authentication Data Not Allowed',
|
149
|
+
'-103' => 'POS Check Invalid Birth Date POS check dateofbirth variable contains a birth date in an incorrect format. Use MM/DD/YYYY format for this variable.',
|
150
|
+
'-104' => 'POS Check Invalid Identification Type POS check identificationtype variable contains a identification type value which is invalid. Use the single digit value where Social Security Number=1, Drivers License=2 for this variable.',
|
151
|
+
'-105' => 'Invalid trackdata Track Data is in invalid format.',
|
152
|
+
'-106' => 'POS Check Invalid Account Type',
|
153
|
+
'-107' => 'POS PIN Debit Invalid Sequence Number',
|
154
|
+
'-108' => 'Invalid Transaction ID',
|
155
|
+
'-109' => 'Invalid From Account Type',
|
156
|
+
'-110' => 'Pos Error Invalid To Account Type',
|
157
|
+
'-112' => 'Pos Error Invalid Auth Option',
|
158
|
+
'-113' => 'Pos Error Transaction Failed',
|
159
|
+
'-114' => 'Pos Error Invalid Incoming Eci',
|
160
|
+
'-115' => 'POS Check Invalid Check Type',
|
161
|
+
'-116' => 'POS Check Invalid Lane Number POS Check lane or cash register number is invalid. Use a valid lane or cash register number that has been configured in the Skipjack Merchant Account.',
|
162
|
+
'-117' => 'POS Check Invalid Cashier Number'
|
163
|
+
}
|
164
|
+
|
165
|
+
self.supported_countries = ['US', 'CA']
|
166
|
+
self.supported_cardtypes = [:visa, :master, :american_express, :jcb, :discover, :diners_club]
|
167
|
+
self.homepage_url = 'http://www.skipjack.com/'
|
168
|
+
self.display_name = 'SkipJack'
|
169
|
+
|
170
|
+
# Creates a new SkipJackGateway
|
171
|
+
#
|
172
|
+
# The gateway requires that a valid login and password be passed
|
173
|
+
# in the +options+ hash.
|
174
|
+
#
|
175
|
+
# ==== Options
|
176
|
+
#
|
177
|
+
# * <tt>:login</tt> -- The SkipJack Merchant Serial Number.
|
178
|
+
# * <tt>:password</tt> -- The SkipJack Developer Serial Number.
|
179
|
+
# * <tt>:test => +true+ or +false+</tt> -- Use the test or live SkipJack url.
|
180
|
+
# * <tt>:advanced => +true+ or +false+</tt> -- Set to true if you're using an advanced processor
|
181
|
+
# See the SkipJack Integration Guide for details. (default: +false+)
|
182
|
+
def initialize(options = {})
|
183
|
+
requires!(options, :login, :password)
|
184
|
+
@options = options
|
185
|
+
super
|
186
|
+
end
|
187
|
+
|
188
|
+
def test?
|
189
|
+
@options[:test] || super
|
190
|
+
end
|
191
|
+
|
192
|
+
def authorize(money, creditcard, options = {})
|
193
|
+
requires!(options, :order_id, :email)
|
194
|
+
post = {}
|
195
|
+
add_invoice(post, options)
|
196
|
+
add_creditcard(post, creditcard)
|
197
|
+
add_address(post, options)
|
198
|
+
add_customer_data(post, options)
|
199
|
+
commit(:authorization, money, post)
|
200
|
+
end
|
201
|
+
|
202
|
+
def purchase(money, creditcard, options = {})
|
203
|
+
post = {}
|
204
|
+
authorization = authorize(money, creditcard, options)
|
205
|
+
if authorization.success?
|
206
|
+
capture(money, authorization.authorization)
|
207
|
+
else
|
208
|
+
authorization
|
209
|
+
end
|
210
|
+
end
|
211
|
+
|
212
|
+
# Captures the funds from an authorized transaction.
|
213
|
+
#
|
214
|
+
# ==== Parameters
|
215
|
+
#
|
216
|
+
# * <tt>money</tt> -- The amount to be capture as an Integer in cents.
|
217
|
+
# * <tt>authorization</tt> -- The authorization returned from the previous authorize request.
|
218
|
+
# * <tt>options</tt> -- A hash of optional parameters.
|
219
|
+
#
|
220
|
+
# ==== Options
|
221
|
+
#
|
222
|
+
# * <tt>:force_settlement</tt> -- Force the settlement to occur as soon as possible. This option is not supported by other gateways. See the SkipJack API reference for more details
|
223
|
+
def capture(money, authorization, options = {})
|
224
|
+
post = { }
|
225
|
+
add_status_action(post, 'SETTLE')
|
226
|
+
add_forced_settlement(post, options)
|
227
|
+
add_transaction_id(post, authorization)
|
228
|
+
commit(:change_status, money, post)
|
229
|
+
end
|
230
|
+
|
231
|
+
def void(authorization, options = {})
|
232
|
+
post = {}
|
233
|
+
add_status_action(post, 'DELETE')
|
234
|
+
add_forced_settlement(post, options)
|
235
|
+
add_transaction_id(post, authorization)
|
236
|
+
commit(:change_status, nil, post)
|
237
|
+
end
|
238
|
+
|
239
|
+
def refund(money, identification, options = {})
|
240
|
+
post = {}
|
241
|
+
add_status_action(post, 'CREDIT')
|
242
|
+
add_forced_settlement(post, options)
|
243
|
+
add_transaction_id(post, identification)
|
244
|
+
commit(:change_status, money, post)
|
245
|
+
end
|
246
|
+
|
247
|
+
def credit(money, identification, options = {})
|
248
|
+
deprecated CREDIT_DEPRECATION_MESSAGE
|
249
|
+
refund(money, identification, options)
|
250
|
+
end
|
251
|
+
|
252
|
+
def status(order_id)
|
253
|
+
commit(:get_status, nil, :szOrderNumber => order_id)
|
254
|
+
end
|
255
|
+
|
256
|
+
private
|
257
|
+
|
258
|
+
def advanced?
|
259
|
+
@options[:advanced]
|
260
|
+
end
|
261
|
+
|
262
|
+
def add_forced_settlement(post, options)
|
263
|
+
post[:szForceSettlement] = options[:force_settlment] ? 1 : 0
|
264
|
+
end
|
265
|
+
|
266
|
+
def add_status_action(post, action)
|
267
|
+
post[:szDesiredStatus] = action
|
268
|
+
end
|
269
|
+
|
270
|
+
def commit(action, money, parameters)
|
271
|
+
response = parse( ssl_post( url_for(action), post_data(action, money, parameters) ), action )
|
272
|
+
|
273
|
+
# Pass along the original transaction id in the case an update transaction
|
274
|
+
Response.new(response[:success], message_from(response, action), response,
|
275
|
+
:test => test?,
|
276
|
+
:authorization => response[:szTransactionFileName] || parameters[:szTransactionId],
|
277
|
+
:avs_result => { :code => response[:szAVSResponseCode] },
|
278
|
+
:cvv_result => response[:szCVV2ResponseCode]
|
279
|
+
)
|
280
|
+
end
|
281
|
+
|
282
|
+
def url_for(action)
|
283
|
+
result = test? ? TEST_HOST : LIVE_HOST
|
284
|
+
result += advanced? && action == :authorization ? ADVANCED_PATH : BASIC_PATH
|
285
|
+
result += "?#{ACTIONS[action]}"
|
286
|
+
end
|
287
|
+
|
288
|
+
def add_credentials(params, action)
|
289
|
+
if action == :authorization
|
290
|
+
params[:SerialNumber] = @options[:login]
|
291
|
+
params[:DeveloperSerialNumber] = @options[:password]
|
292
|
+
else
|
293
|
+
params[:szSerialNumber] = @options[:login]
|
294
|
+
params[:szDeveloperSerialNumber] = @options[:password]
|
295
|
+
end
|
296
|
+
end
|
297
|
+
|
298
|
+
def add_amount(params, action, money)
|
299
|
+
if action == :authorization
|
300
|
+
params[:TransactionAmount] = amount(money)
|
301
|
+
else
|
302
|
+
params[:szAmount] = amount(money) if MONETARY_CHANGE_STATUSES.include?(params[:szDesiredStatus])
|
303
|
+
end
|
304
|
+
end
|
305
|
+
|
306
|
+
def parse(body, action)
|
307
|
+
case action
|
308
|
+
when :authorization
|
309
|
+
parse_authorization_response(body)
|
310
|
+
when :get_status
|
311
|
+
parse_status_response(body, [ :SerialNumber, :TransactionAmount, :TransactionStatusCode, :TransactionStatusMessage, :OrderNumber, :TransactionDateTime, :TransactionID, :ApprovalCode, :BatchNumber ])
|
312
|
+
else
|
313
|
+
parse_status_response(body, [ :SerialNumber, :TransactionAmount, :DesiredStatus, :StatusResponse, :StatusResponseMessage, :OrderNumber, :AuditID ])
|
314
|
+
end
|
315
|
+
end
|
316
|
+
|
317
|
+
def split_lines(body)
|
318
|
+
body.split(/[\r\n]+/)
|
319
|
+
end
|
320
|
+
|
321
|
+
def split_line(line)
|
322
|
+
line.split(/","/).collect { |key| key.sub(/"*([^"]*)"*/, '\1').strip; }
|
323
|
+
end
|
324
|
+
|
325
|
+
def authorize_response_map(body)
|
326
|
+
lines = split_lines(body)
|
327
|
+
keys, values = split_line(lines[0]), split_line(lines[1])
|
328
|
+
Hash[*(keys.zip(values).flatten)].symbolize_keys
|
329
|
+
end
|
330
|
+
|
331
|
+
def parse_authorization_response(body)
|
332
|
+
result = authorize_response_map(body)
|
333
|
+
result[:success] = (result[:szIsApproved] == '1')
|
334
|
+
result
|
335
|
+
end
|
336
|
+
|
337
|
+
def parse_status_response(body, response_keys)
|
338
|
+
lines = split_lines(body)
|
339
|
+
|
340
|
+
keys = [ :szSerialNumber, :szErrorCode, :szNumberRecords]
|
341
|
+
values = split_line(lines[0])[0..2]
|
342
|
+
|
343
|
+
result = Hash[*(keys.zip(values).flatten)]
|
344
|
+
|
345
|
+
result[:szErrorMessage] = ''
|
346
|
+
result[:success] = (result[:szErrorCode] == '0')
|
347
|
+
|
348
|
+
if result[:success]
|
349
|
+
lines[1..-1].each do |line|
|
350
|
+
values = split_line(line)
|
351
|
+
response_keys.each_with_index do |key, index|
|
352
|
+
result[key] = values[index]
|
353
|
+
end
|
354
|
+
end
|
355
|
+
else
|
356
|
+
result[:szErrorMessage] = lines[1]
|
357
|
+
end
|
358
|
+
result
|
359
|
+
end
|
360
|
+
|
361
|
+
def post_data(action, money, params = {})
|
362
|
+
add_credentials(params, action)
|
363
|
+
add_amount(params, action, money)
|
364
|
+
sorted_params = params.to_a.sort{|a,b| a.to_s <=> b.to_s}.reverse
|
365
|
+
sorted_params.collect { |key, value| "#{key.to_s}=#{CGI.escape(value.to_s)}" }.join("&")
|
366
|
+
end
|
367
|
+
|
368
|
+
def add_transaction_id(post, transaction_id)
|
369
|
+
post[:szTransactionId] = transaction_id
|
370
|
+
end
|
371
|
+
|
372
|
+
def add_invoice(post, options)
|
373
|
+
post[:OrderNumber] = sanitize_order_id(options[:order_id])
|
374
|
+
post[:CustomerCode] = options[:customer].to_s.slice(0, 17)
|
375
|
+
post[:InvoiceNumber] = options[:invoice]
|
376
|
+
post[:OrderDescription] = options[:description]
|
377
|
+
|
378
|
+
if order_items = options[:items]
|
379
|
+
post[:OrderString] = order_items.collect { |item| "#{item[:sku]}~#{item[:description].tr('~','-')}~#{item[:declared_value]}~#{item[:quantity]}~#{item[:taxable]}~~~~~~~~#{item[:tax_rate]}~||"}.join
|
380
|
+
else
|
381
|
+
post[:OrderString] = '1~None~0.00~0~N~||'
|
382
|
+
end
|
383
|
+
end
|
384
|
+
|
385
|
+
def add_creditcard(post, creditcard)
|
386
|
+
post[:AccountNumber] = creditcard.number
|
387
|
+
post[:Month] = creditcard.month
|
388
|
+
post[:Year] = creditcard.year
|
389
|
+
post[:CVV2] = creditcard.verification_value if creditcard.verification_value?
|
390
|
+
post[:SJName] = creditcard.name
|
391
|
+
end
|
392
|
+
|
393
|
+
def add_customer_data(post, options)
|
394
|
+
post[:Email] = options[:email]
|
395
|
+
end
|
396
|
+
|
397
|
+
def add_address(post, options)
|
398
|
+
if address = options[:billing_address] || options[:address]
|
399
|
+
post[:StreetAddress] = address[:address1]
|
400
|
+
post[:StreetAddress2] = address[:address2]
|
401
|
+
post[:City] = address[:city]
|
402
|
+
post[:State] = address[:state]
|
403
|
+
post[:ZipCode] = address[:zip]
|
404
|
+
post[:Country] = address[:country]
|
405
|
+
post[:Phone] = address[:phone]
|
406
|
+
post[:Fax] = address[:fax]
|
407
|
+
end
|
408
|
+
|
409
|
+
if address = options[:shipping_address]
|
410
|
+
post[:ShipToName] = address[:name]
|
411
|
+
post[:ShipToStreetAddress] = address[:address1]
|
412
|
+
post[:ShipToStreetAddress2] = address[:address2]
|
413
|
+
post[:ShipToCity] = address[:city]
|
414
|
+
post[:ShipToState] = address[:state]
|
415
|
+
post[:ShipToZipCode] = address[:zip]
|
416
|
+
post[:ShipToCountry] = address[:country]
|
417
|
+
post[:ShipToPhone] = address[:phone]
|
418
|
+
post[:ShipToFax] = address[:fax]
|
419
|
+
end
|
420
|
+
|
421
|
+
# The phone number for the shipping address is required
|
422
|
+
# Use the billing address phone number if a shipping address
|
423
|
+
# phone number wasn't provided
|
424
|
+
post[:ShipToPhone] = post[:Phone] if post[:ShipToPhone].blank?
|
425
|
+
end
|
426
|
+
|
427
|
+
def message_from(response, action)
|
428
|
+
case action
|
429
|
+
when :authorization
|
430
|
+
message_from_authorization(response)
|
431
|
+
when :get_status
|
432
|
+
message_from_status(response)
|
433
|
+
else
|
434
|
+
message_from_status(response)
|
435
|
+
end
|
436
|
+
end
|
437
|
+
|
438
|
+
def message_from_authorization(response)
|
439
|
+
if response[:success]
|
440
|
+
return SUCCESS_MESSAGE
|
441
|
+
else
|
442
|
+
return CARD_CODE_MESSAGES[response[:szCVV2ResponseCode]] if CARD_CODE_ERRORS.include?(response[:szCVV2ResponseCode])
|
443
|
+
return AVS_MESSAGES[response[:szAVSResponseMessage]] if AVS_ERRORS.include?(response[:szAVSResponseCode])
|
444
|
+
return RETURN_CODE_MESSAGES[response[:szReturnCode]] if response[:szReturnCode] != '1'
|
445
|
+
return response[:szAuthorizationDeclinedMessage]
|
446
|
+
end
|
447
|
+
end
|
448
|
+
|
449
|
+
def message_from_status(response)
|
450
|
+
response[:success] ? SUCCESS_MESSAGE : response[:szErrorMessage]
|
451
|
+
end
|
452
|
+
|
453
|
+
def sanitize_order_id(value)
|
454
|
+
value.to_s.gsub(/[^\w.]/, '')
|
455
|
+
end
|
456
|
+
end
|
457
|
+
end
|
458
|
+
end
|