require 'nokogiri' require 'securerandom' module ActiveMerchant #:nodoc: module Billing #:nodoc: class ElementGateway < Gateway self.test_url = 'https://certtransaction.elementexpress.com/express.asmx' self.live_url = 'https://transaction.elementexpress.com/express.asmx' self.supported_countries = ['US'] self.default_currency = 'USD' self.supported_cardtypes = %i[visa master american_express discover diners_club jcb] self.homepage_url = 'http://www.elementps.com' self.display_name = 'Element' SERVICE_TEST_URL = 'https://certservices.elementexpress.com/express.asmx' SERVICE_LIVE_URL = 'https://services.elementexpress.com/express.asmx' def initialize(options={}) requires!(options, :account_id, :account_token, :application_id, :acceptor_id, :application_name, :application_version) super end def purchase(money, payment, options={}) action = payment.is_a?(Check) ? 'CheckSale' : 'CreditCardSale' request = build_soap_request do |xml| xml.send(action, xmlns: 'https://transaction.elementexpress.com') do add_credentials(xml) add_payment_method(xml, payment) add_transaction(xml, money, options) add_terminal(xml, options) add_address(xml, options) end end commit(action, request, money) end def authorize(money, payment, options={}) request = build_soap_request do |xml| xml.CreditCardAuthorization(xmlns: 'https://transaction.elementexpress.com') do add_credentials(xml) add_payment_method(xml, payment) add_transaction(xml, money, options) add_terminal(xml, options) add_address(xml, options) end end commit('CreditCardAuthorization', request, money) end def capture(money, authorization, options={}) trans_id, = split_authorization(authorization) options[:trans_id] = trans_id request = build_soap_request do |xml| xml.CreditCardAuthorizationCompletion(xmlns: 'https://transaction.elementexpress.com') do add_credentials(xml) add_transaction(xml, money, options) add_terminal(xml, options) end end commit('CreditCardAuthorizationCompletion', request, money) end def refund(money, authorization, options={}) trans_id, = split_authorization(authorization) options[:trans_id] = trans_id request = build_soap_request do |xml| xml.CreditCardReturn(xmlns: 'https://transaction.elementexpress.com') do add_credentials(xml) add_transaction(xml, money, options) add_terminal(xml, options) end end commit('CreditCardReturn', request, money) end def void(authorization, options={}) trans_id, trans_amount = split_authorization(authorization) options.merge!({trans_id: trans_id, trans_amount: trans_amount, reversal_type: 'Full'}) request = build_soap_request do |xml| xml.CreditCardReversal(xmlns: 'https://transaction.elementexpress.com') do add_credentials(xml) add_transaction(xml, trans_amount, options) add_terminal(xml, options) end end commit('CreditCardReversal', request, trans_amount) end def store(payment, options = {}) request = build_soap_request do |xml| xml.PaymentAccountCreate(xmlns: 'https://services.elementexpress.com') do add_credentials(xml) add_payment_method(xml, payment) add_payment_account(xml, payment, options[:payment_account_reference_number] || SecureRandom.hex(20)) add_address(xml, options) end end commit('PaymentAccountCreate', request, nil) end def verify(credit_card, options={}) request = build_soap_request do |xml| xml.CreditCardAVSOnly(xmlns: 'https://transaction.elementexpress.com') do add_credentials(xml) add_payment_method(xml, credit_card) add_transaction(xml, 0, options) add_terminal(xml, options) add_address(xml, options) end end # send request with the transaction amount set to 0 commit('CreditCardAVSOnly', request, 0) 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'). gsub(%r(().+?())i, '\1[FILTERED]\2'). gsub(%r(().+?())i, '\1[FILTERED]\2') end private def add_credentials(xml) xml.credentials do xml.AccountID @options[:account_id] xml.AccountToken @options[:account_token] xml.AcceptorID @options[:acceptor_id] end xml.application do xml.ApplicationID @options[:application_id] xml.ApplicationName @options[:application_name] xml.ApplicationVersion @options[:application_version] end end def add_payment_method(xml, payment) if payment.is_a?(String) add_payment_account_id(xml, payment) elsif payment.is_a?(Check) add_echeck(xml, payment) else add_credit_card(xml, payment) end end def add_payment_account(xml, payment, payment_account_reference_number) xml.paymentAccount do xml.PaymentAccountType payment_account_type(payment) xml.PaymentAccountReferenceNumber payment_account_reference_number end end def add_payment_account_id(xml, payment) xml.extendedParameters do xml.ExtendedParameters do xml.Key 'PaymentAccount' xml.Value('xsi:type' => 'PaymentAccount') do xml.PaymentAccountID payment end end end end def add_transaction(xml, money, options = {}) xml.transaction do xml.ReversalType options[:reversal_type] if options[:reversal_type] xml.TransactionID options[:trans_id] if options[:trans_id] xml.TransactionAmount amount(money.to_i) if money xml.MarketCode 'Default' if money xml.ReferenceNumber options[:order_id] || SecureRandom.hex(20) end end def add_terminal(xml, options) xml.terminal do xml.TerminalID options[:terminal_id] || '01' xml.CardPresentCode options[:card_present_code] || 'UseDefault' xml.CardholderPresentCode 'UseDefault' xml.CardInputCode 'UseDefault' xml.CVVPresenceCode 'UseDefault' xml.TerminalCapabilityCode 'UseDefault' xml.TerminalEnvironmentCode 'UseDefault' xml.MotoECICode 'NonAuthenticatedSecureECommerceTransaction' end end def add_credit_card(xml, payment) xml.card do xml.CardNumber payment.number xml.ExpirationMonth format(payment.month, :two_digits) xml.ExpirationYear format(payment.year, :two_digits) xml.CardholderName payment.first_name + ' ' + payment.last_name xml.CVV payment.verification_value end end def add_echeck(xml, payment) xml.demandDepositAccount do xml.AccountNumber payment.account_number xml.RoutingNumber payment.routing_number xml.DDAAccountType payment.account_type.capitalize end end def add_address(xml, options) if address = options[:billing_address] || options[:address] xml.address do xml.BillingAddress1 address[:address1] if address[:address1] xml.BillingAddress2 address[:address2] if address[:address2] xml.BillingCity address[:city] if address[:city] xml.BillingState address[:state] if address[:state] xml.BillingZipcode address[:zip] if address[:zip] xml.BillingEmail address[:email] if address[:email] xml.BillingPhone address[:phone_number] if address[:phone_number] end end if shipping_address = options[:shipping_address] xml.address do xml.ShippingAddress1 shipping_address[:address1] if shipping_address[:address1] xml.ShippingAddress2 shipping_address[:address2] if shipping_address[:address2] xml.ShippingCity shipping_address[:city] if shipping_address[:city] xml.ShippingState shipping_address[:state] if shipping_address[:state] xml.ShippingZipcode shipping_address[:zip] if shipping_address[:zip] xml.ShippingEmail shipping_address[:email] if shipping_address[:email] xml.ShippingPhone shipping_address[:phone_number] if shipping_address[:phone_number] end end end def parse(xml) response = {} doc = Nokogiri::XML(xml) doc.remove_namespaces! root = doc.root.xpath('//response/*') root = doc.root.xpath('//Response/*') if root.empty? root.each do |node| if node.elements.empty? response[node.name.downcase] = node.text else node_name = node.name.downcase response[node_name] = Hash.new node.elements.each do |childnode| response[node_name][childnode.name.downcase] = childnode.text end end end response end def commit(action, xml, amount) response = parse(ssl_post(url(action), xml, headers(action))) Response.new( success_from(response), message_from(response), response, authorization: authorization_from(action, response, amount), avs_result: success_from(response) ? avs_from(response) : nil, cvv_result: success_from(response) ? cvv_from(response) : nil, test: test? ) end def authorization_from(action, response, amount) if action == 'PaymentAccountCreate' response['paymentaccount']['paymentaccountid'] else "#{response['transaction']['transactionid']}|#{amount}" if response['transaction'] end end def success_from(response) response['expressresponsecode'] == '0' end def message_from(response) response['expressresponsemessage'] end def avs_from(response) AVSResult.new(code: response['card']['avsresponsecode']) if response['card'] end def cvv_from(response) CVVResult.new(response['card']['cvvresponsecode']) if response['card'] end def split_authorization(authorization) authorization.split('|') end def build_soap_request builder = Nokogiri::XML::Builder.new(encoding: 'UTF-8') do |xml| xml['soap'].Envelope('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/') do xml['soap'].Body do yield(xml) end end end builder.to_xml end def payment_account_type(payment) if payment.is_a?(Check) payment_account_type = payment.account_type else payment_account_type = 'CreditCard' end payment_account_type end def url(action) if action == 'PaymentAccountCreate' test? ? SERVICE_TEST_URL : SERVICE_LIVE_URL else test? ? test_url : live_url end end def interface(action) return 'transaction' if action != 'PaymentAccountCreate' return 'services' if action == 'PaymentAccountCreate' end def headers(action) { 'Content-Type' => 'text/xml; charset=utf-8', 'SOAPAction' => "https://#{interface(action)}.elementexpress.com/#{action}" } end end end end