module ActiveMerchant #:nodoc:
module Billing #:nodoc:
class MerchantWareVersionFourGateway < Gateway
self.live_url = 'https://ps1.merchantware.net/Merchantware/ws/RetailTransaction/v4/Credit.asmx'
self.test_url = 'https://ps1.merchantware.net/Merchantware/ws/RetailTransaction/v4/Credit.asmx'
self.supported_countries = ['US']
self.supported_cardtypes = %i[visa master american_express discover]
self.homepage_url = 'http://merchantwarehouse.com/merchantware'
self.display_name = 'MerchantWARE'
ENV_NAMESPACES = { 'xmlns:xsi' => 'http://www.w3.org/2001/XMLSchema-instance',
'xmlns:xsd' => 'http://www.w3.org/2001/XMLSchema',
'xmlns:soap' => 'http://schemas.xmlsoap.org/soap/envelope/' }
TX_NAMESPACE = 'http://schemas.merchantwarehouse.com/merchantware/40/Credit/'
ACTIONS = {
purchase: 'SaleKeyed',
reference_purchase: 'RepeatSale',
authorize: 'PreAuthorizationKeyed',
capture: 'PostAuthorization',
void: 'Void',
refund: 'Refund'
}
# Creates a new MerchantWareVersionFourGateway
#
# The gateway requires that a valid login, password, and name be passed
# in the +options+ hash.
#
# ==== Options
#
# * :login - The MerchantWARE SiteID.
# * :password - The MerchantWARE Key.
# * :name - The MerchantWARE Name.
def initialize(options = {})
requires!(options, :login, :password, :name)
super
end
# Authorize a credit card for a given amount.
#
# ==== Parameters
# * money - The amount to be authorized as an Integer value in cents.
# * credit_card - The CreditCard details for the transaction.
# * options
# * :order_id - A unique reference for this order (required).
# * :billing_address - The billing address for the cardholder.
def authorize(money, credit_card, options = {})
request = build_purchase_request(:authorize, money, credit_card, options)
commit(:authorize, request)
end
# Authorize and immediately capture funds from a credit card.
#
# ==== Parameters
# * money - The amount to be authorized as anInteger value in cents.
# * payment_source - The CreditCard details or 'token' from prior transaction
# * options
# * :order_id - A unique reference for this order (required).
# * :billing_address - The billing address for the cardholder.
def purchase(money, payment_source, options = {})
action = payment_source.is_a?(String) ? :reference_purchase : :purchase
request = build_purchase_request(action, money, payment_source, options)
commit(action, request)
end
# Capture authorized funds from a credit card.
#
# ==== Parameters
# * money - The amount to be captured as anInteger value in cents.
# * authorization - The authorization string returned from the initial authorization.
def capture(money, authorization, options = {})
request = build_capture_request(:capture, money, authorization, options)
commit(:capture, request)
end
# Void a transaction.
#
# ==== Parameters
# * authorization - The authorization string returned from the initial authorization or purchase.
def void(authorization, options = {})
reference, options[:order_id] = split_reference(authorization)
request = soap_request(:void) do |xml|
add_reference_token(xml, reference)
end
commit(:void, request)
end
# Refund an amount back a cardholder
#
# ==== Parameters
#
# * money - The amount to be refunded as an Integer value in cents.
# * identification - The credit card you want to refund or the authorization for the existing transaction you are refunding.
# * options
# * :order_id - A unique reference for this order (required when performing a non-referenced credit)
def refund(money, identification, options = {})
reference, options[:order_id] = split_reference(identification)
request = soap_request(:refund) do |xml|
add_reference_token(xml, reference)
add_invoice(xml, options)
add_amount(xml, money, 'overrideAmount')
end
commit(:refund, request)
end
def verify(credit_card, options = {})
MultiResponse.run(:use_first_response) do |r|
r.process { authorize(100, credit_card, options) }
r.process(:ignore_result) { void(r.authorization, options) }
end
end
def supports_scrubbing?
true
end
def scrub(transcript)
transcript.
gsub(%r(().+?())i, '\1[FILTERED]\2').
gsub(%r(().+?())i, '\1[FILTERED]\2').
gsub(%r(().+?())i, '\1[FILTERED]\2')
end
private
def soap_request(action)
xml = Builder::XmlMarkup.new indent: 2
xml.instruct!
xml.tag! 'soap:Envelope', ENV_NAMESPACES do
xml.tag! 'soap:Body' do
xml.tag! ACTIONS[action], 'xmlns' => TX_NAMESPACE do
xml.tag! 'merchantName', @options[:name]
xml.tag! 'merchantSiteId', @options[:login]
xml.tag! 'merchantKey', @options[:password]
yield xml
end
end
end
xml.target!
end
def build_purchase_request(action, money, payment_source, options)
requires!(options, :order_id)
soap_request(action) do |xml|
add_invoice(xml, options)
add_amount(xml, money)
add_payment_source(xml, payment_source)
add_address(xml, options)
end
end
def build_capture_request(action, money, identification, options)
reference, options[:order_id] = split_reference(identification)
soap_request(action) do |xml|
add_reference_token(xml, reference)
add_invoice(xml, options)
add_amount(xml, money)
end
end
def add_invoice(xml, options)
xml.tag! 'invoiceNumber', truncate(options[:order_id].to_s.gsub(/[^\w]/, ''), 8)
end
def add_amount(xml, money, tag = 'amount')
xml.tag! tag, amount(money)
end
def add_reference_token(xml, reference)
xml.tag! 'token', reference
end
def add_address(xml, options)
address = options[:billing_address] || options[:address] || {}
xml.tag! 'avsStreetAddress', address[:address1]
xml.tag! 'avsStreetZipCode', address[:zip]
end
def add_payment_source(xml, source)
if source.is_a?(String)
add_reference_token(xml, source)
else
add_credit_card(xml, source)
end
end
def add_credit_card(xml, credit_card)
xml.tag! 'cardNumber', credit_card.number
xml.tag! 'expirationDate', expdate(credit_card)
xml.tag! 'cardholder', credit_card.name
xml.tag! 'cardSecurityCode', credit_card.verification_value if credit_card.verification_value?
end
def split_reference(reference)
reference.to_s.split(';')
end
def parse(action, data)
response = {}
xml = REXML::Document.new(data)
root = REXML::XPath.first(xml, "//#{ACTIONS[action]}Response/#{ACTIONS[action]}Result")
root.elements.each do |element|
response[element.name] = element.text
end
if response['ErrorMessage'].present?
response[:message] = response['ErrorMessage']
response[:success] = false
else
status, code, message = response['ApprovalStatus'].split(';')
response[:status] = status
if response[:success] = status == 'APPROVED'
response[:message] = status
else
response[:message] = message
response[:failure_code] = code
end
end
response
end
def parse_error(http_response, action)
response = {}
response[:http_code] = http_response.code
response[:http_message] = http_response.message
response[:success] = false
document = REXML::Document.new(http_response.body)
node = REXML::XPath.first(document, "//#{ACTIONS[action]}Response/#{ACTIONS[action]}Result")
node.elements.each do |element|
response[element.name] = element.text
end
response[:message] = response['ErrorMessage'].to_s.tr("\n", ' ')
response
rescue REXML::ParseException
response[:http_body] = http_response.body
response[:message] = 'Failed to parse the failed response'
response
end
def soap_action(action)
"#{TX_NAMESPACE}#{ACTIONS[action]}"
end
def url
test? ? test_url : live_url
end
def commit(action, request)
begin
data = ssl_post(url, request,
'Content-Type' => 'text/xml; charset=utf-8',
'SOAPAction' => soap_action(action))
response = parse(action, data)
rescue ActiveMerchant::ResponseError => e
response = parse_error(e.response, action)
end
Response.new(response[:success], response[:message], response,
test: test?,
authorization: authorization_from(response),
avs_result: { code: response['AvsResponse'] },
cvv_result: response['CvResponse'])
end
def authorization_from(response)
response['Token']
end
end
end
end