Class: ActiveMerchant::Billing::AuthorizeNetGateway

Inherits:
Gateway
  • Object
show all
Includes:
Empty
Defined in:
lib/active_merchant/billing/gateways/authorize_net.rb

Constant Summary collapse

STANDARD_AVS_CODE_MAPPING =

Authorize.net has slightly different definitions for returned AVS codes that have been mapped to the closest equivalent AM standard AVSResult codes Authorize.net’s descriptions noted below

{
  'A' => 'A', # Street Address: Match -- First 5 Digits of ZIP: No Match
  'B' => 'I', # Address not provided for AVS check or street address match, postal code could not be verified
  'E' => 'E', # AVS Error
  'G' => 'G', # Non U.S. Card Issuing Bank
  'N' => 'N', # Street Address: No Match -- First 5 Digits of ZIP: No Match
  'P' => 'I', # AVS not applicable for this transaction
  'R' => 'R', # Retry, System Is Unavailable
  'S' => 'S', # AVS Not Supported by Card Issuing Bank
  'U' => 'U', # Address Information For This Cardholder Is Unavailable
  'W' => 'W', # Street Address: No Match -- All 9 Digits of ZIP: Match
  'X' => 'X', # Street Address: Match -- All 9 Digits of ZIP: Match
  'Y' => 'Y', # Street Address: Match - First 5 Digits of ZIP: Match
  'Z' => 'Z'  # Street Address: No Match - First 5 Digits of ZIP: Match
}
STANDARD_ERROR_CODE_MAPPING =
{
  '2127' => STANDARD_ERROR_CODE[:incorrect_address],
  '22' => STANDARD_ERROR_CODE[:card_declined],
  '227' => STANDARD_ERROR_CODE[:incorrect_address],
  '23' => STANDARD_ERROR_CODE[:card_declined],
  '2315' => STANDARD_ERROR_CODE[:invalid_number],
  '2316' => STANDARD_ERROR_CODE[:invalid_expiry_date],
  '2317' => STANDARD_ERROR_CODE[:expired_card],
  '235' => STANDARD_ERROR_CODE[:processing_error],
  '237' => STANDARD_ERROR_CODE[:invalid_number],
  '24' => STANDARD_ERROR_CODE[:pickup_card],
  '244' => STANDARD_ERROR_CODE[:incorrect_cvc],
  '300' => STANDARD_ERROR_CODE[:config_error],
  '3153' => STANDARD_ERROR_CODE[:processing_error],
  '3155' => STANDARD_ERROR_CODE[:unsupported_feature],
  '36' => STANDARD_ERROR_CODE[:incorrect_number],
  '37' => STANDARD_ERROR_CODE[:invalid_expiry_date],
  '378' => STANDARD_ERROR_CODE[:invalid_cvc],
  '38' => STANDARD_ERROR_CODE[:expired_card],
  '384' => STANDARD_ERROR_CODE[:config_error]
}
MARKET_TYPE =
{
  moto: '1',
  retail: '2'
}
DEVICE_TYPE =
{
  unknown: '1',
  unattended_terminal: '2',
  self_service_terminal: '3',
  electronic_cash_register: '4',
  personal_computer_terminal: '5',
  airpay: '6',
  wireless_pos: '7',
  website: '8',
  dial_terminal: '9',
  virtual_terminal: '10'
}
TRANSACTION_ALREADY_ACTIONED =
%w(310 311)
CARD_CODE_ERRORS =
%w(N S)
AVS_ERRORS =
%w(A E I N R W Z)
AVS_REASON_CODES =
%w(27 45)
TRACKS =
{
  1 => /^%(?<format_code>.)(?<pan>[\d]{1,19}+)\^(?<name>.{2,26})\^(?<expiration>[\d]{0,4}|\^)(?<service_code>[\d]{0,3}|\^)(?<discretionary_data>.*)\?\Z/,
  2 => /\A;(?<pan>[\d]{1,19}+)=(?<expiration>[\d]{0,4}|=)(?<service_code>[\d]{0,3}|=)(?<discretionary_data>.*)\?\Z/
}.freeze
APPLE_PAY_DATA_DESCRIPTOR =
'COMMON.APPLE.INAPP.PAYMENT'
PAYMENT_METHOD_NOT_SUPPORTED_ERROR =
'155'
INELIGIBLE_FOR_ISSUING_CREDIT_ERROR =
'54'

Constants inherited from Gateway

Gateway::CREDIT_DEPRECATION_MESSAGE, Gateway::RECURRING_DEPRECATION_MESSAGE, Gateway::STANDARD_ERROR_CODE

Instance Attribute Summary

Attributes inherited from Gateway

#options

Instance Method Summary collapse

Methods inherited from Gateway

#add_field_to_post_if_present, #add_fields_to_post_if_present, #card_brand, card_brand, #generate_unique_id, inherited, #supported_countries, supported_countries, supported_countries=, supports?, #test?

Methods included from CreditCardFormatting

#expdate, #format

Methods included from PostsData

included, #raw_ssl_request, #ssl_get, #ssl_post, #ssl_request

Constructor Details

#initialize(options = {}) ⇒ AuthorizeNetGateway

Returns a new instance of AuthorizeNetGateway.



96
97
98
99
# File 'lib/active_merchant/billing/gateways/authorize_net.rb', line 96

def initialize(options = {})
  requires!(options, :login, :password)
  super
end

Instance Method Details

#amount_for_verify(options) ⇒ Object



189
190
191
192
193
194
195
196
197
# File 'lib/active_merchant/billing/gateways/authorize_net.rb', line 189

def amount_for_verify(options)
  return 100 unless options[:verify_amount].present?

  amount = options[:verify_amount]
  raise ArgumentError.new 'verify_amount value must be an integer' unless amount.is_a?(Integer) && !amount.negative? || amount.is_a?(String) && amount.match?(/^\d+$/) && !amount.to_i.negative?
  raise ArgumentError.new 'Billing address including zip code is required for a 0 amount verify' if amount.to_i.zero? && !validate_billing_address_values?(options)

  amount.to_i
end

#authorize(amount, payment, options = {}) ⇒ Object



113
114
115
116
117
118
119
120
121
122
123
# File 'lib/active_merchant/billing/gateways/authorize_net.rb', line 113

def authorize(amount, payment, options = {})
  if payment.is_a?(String)
    commit(:cim_authorize, options) do |xml|
      add_cim_auth_purchase(xml, 'profileTransAuthOnly', amount, payment, options)
    end
  else
    commit(:authorize) do |xml|
      add_auth_purchase(xml, 'authOnlyTransaction', amount, payment, options)
    end
  end
end

#capture(amount, authorization, options = {}) ⇒ Object



125
126
127
128
129
130
131
# File 'lib/active_merchant/billing/gateways/authorize_net.rb', line 125

def capture(amount, authorization, options = {})
  if auth_was_for_cim?(authorization)
    cim_capture(amount, authorization, options)
  else
    normal_capture(amount, authorization, options)
  end
end

#credit(amount, payment, options = {}) ⇒ Object

Raises:

  • (ArgumentError)


159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
# File 'lib/active_merchant/billing/gateways/authorize_net.rb', line 159

def credit(amount, payment, options = {})
  raise ArgumentError, 'Reference credits are not supported. Please supply the original credit card or use the #refund method.' if payment.is_a?(String)

  commit(:credit) do |xml|
    add_order_id(xml, options)
    xml.transactionRequest do
      xml.transactionType('refundTransaction')
      xml.amount(amount(amount))

      add_payment_source(xml, payment, options, :credit)
      xml.refTransId(transaction_id_from(options[:transaction_id])) if options[:transaction_id]
      add_invoice(xml, 'refundTransaction', options)
      add_customer_data(xml, payment, options)
      add_settings(xml, payment, options)
      add_user_fields(xml, amount, options)
    end
  end
end

#purchase(amount, payment, options = {}) ⇒ Object



101
102
103
104
105
106
107
108
109
110
111
# File 'lib/active_merchant/billing/gateways/authorize_net.rb', line 101

def purchase(amount, payment, options = {})
  if payment.is_a?(String)
    commit(:cim_purchase, options) do |xml|
      add_cim_auth_purchase(xml, 'profileTransAuthCapture', amount, payment, options)
    end
  else
    commit(:purchase) do |xml|
      add_auth_purchase(xml, 'authCaptureTransaction', amount, payment, options)
    end
  end
end

#refund(amount, authorization, options = {}) ⇒ Object



133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
# File 'lib/active_merchant/billing/gateways/authorize_net.rb', line 133

def refund(amount, authorization, options = {})
  response =
    if auth_was_for_cim?(authorization)
      cim_refund(amount, authorization, options)
    else
      normal_refund(amount, authorization, options)
    end

  return response if response.success?
  return response unless options[:force_full_refund_if_unsettled]

  if response.params['response_reason_code'] == INELIGIBLE_FOR_ISSUING_CREDIT_ERROR
    void(authorization, options)
  else
    response
  end
end

#scrub(transcript) ⇒ Object



226
227
228
229
230
231
232
233
234
235
236
237
# File 'lib/active_merchant/billing/gateways/authorize_net.rb', line 226

def scrub(transcript)
  transcript.
    gsub(%r((Authorization: Basic )\w+), '\1[FILTERED]').
    gsub(%r((<transactionKey>).+(</transactionKey>)), '\1[FILTERED]\2').
    gsub(%r((<cardNumber>).+(</cardNumber>)), '\1[FILTERED]\2').
    gsub(%r((<cardCode>).+(</cardCode>)), '\1[FILTERED]\2').
    gsub(%r((<track1>).+(</track1>)), '\1[FILTERED]\2').
    gsub(%r((<track2>).+(</track2>)), '\1[FILTERED]\2').
    gsub(/(<routingNumber>).+(<\/routingNumber>)/, '\1[FILTERED]\2').
    gsub(/(<accountNumber>).+(<\/accountNumber>)/, '\1[FILTERED]\2').
    gsub(%r((<cryptogram>).+(</cryptogram>)), '\1[FILTERED]\2')
end

#store(credit_card, options = {}) ⇒ Object



203
204
205
206
207
208
209
# File 'lib/active_merchant/billing/gateways/authorize_net.rb', line 203

def store(credit_card, options = {})
  if options[:customer_profile_id]
    create_customer_payment_profile(credit_card, options)
  else
    create_customer_profile(credit_card, options)
  end
end

#supports_network_tokenization?Boolean

Returns:

  • (Boolean)


239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
# File 'lib/active_merchant/billing/gateways/authorize_net.rb', line 239

def supports_network_tokenization?
  card = Billing::NetworkTokenizationCreditCard.new({
    number: '4111111111111111',
    month: 12,
    year: 20,
    first_name: 'John',
    last_name: 'Smith',
    brand: 'visa',
    payment_cryptogram: 'EHuWW9PiBkWvqE5juRwDzAUFBAk='
  })

  request = post_data(:authorize) do |xml|
    add_auth_purchase(xml, 'authOnlyTransaction', 1, card, {})
  end
  raw_response = ssl_post(url, request, headers)
  response = parse(:authorize, raw_response)
  response[:response_reason_code].to_s != PAYMENT_METHOD_NOT_SUPPORTED_ERROR
end

#supports_scrubbing?Boolean

Returns:

  • (Boolean)


222
223
224
# File 'lib/active_merchant/billing/gateways/authorize_net.rb', line 222

def supports_scrubbing?
  true
end

#unstore(authorization) ⇒ Object



211
212
213
214
215
# File 'lib/active_merchant/billing/gateways/authorize_net.rb', line 211

def unstore(authorization)
  customer_profile_id, = split_authorization(authorization)

  delete_customer_profile(customer_profile_id)
end

#validate_billing_address_values?(options) ⇒ Boolean

Returns:

  • (Boolean)


199
200
201
# File 'lib/active_merchant/billing/gateways/authorize_net.rb', line 199

def validate_billing_address_values?(options)
  options.dig(:billing_address, :zip).present? && options.dig(:billing_address, :address1).present?
end

#verify(payment_method, options = {}) ⇒ Object



178
179
180
181
182
183
184
185
186
187
# File 'lib/active_merchant/billing/gateways/authorize_net.rb', line 178

def verify(payment_method, options = {})
  amount = amount_for_verify(options)

  MultiResponse.run(:use_first_response) do |r|
    r.process { authorize(amount, payment_method, options) }
    r.process(:ignore_result) { void(r.authorization, options) } unless amount == 0
  end
rescue ArgumentError => e
  Response.new(false, e.message)
end

#verify_credentialsObject



217
218
219
220
# File 'lib/active_merchant/billing/gateways/authorize_net.rb', line 217

def verify_credentials
  response = commit(:verify_credentials) {}
  response.success?
end

#void(authorization, options = {}) ⇒ Object



151
152
153
154
155
156
157
# File 'lib/active_merchant/billing/gateways/authorize_net.rb', line 151

def void(authorization, options = {})
  if auth_was_for_cim?(authorization)
    cim_void(authorization, options)
  else
    normal_void(authorization, options)
  end
end