require 'securerandom' require 'digest' module ActiveMerchant #:nodoc: module Billing #:nodoc: # ==== USA ePay Advanced SOAP Interface # # This class encapuslates USA ePay's Advanced SOAP Interface. The Advanced Soap Interface allows # standard transactions, storing customer information, and recurring billing. Storing sensitive # information on USA ePay's servers can help with PCI DSS compliance, since customer and card data # do not need to be stored locally. # # Make sure you have enabled this functionality for your account with USA ePay. # # Information about the Advanced SOAP interface is available on the {USA ePay wiki}[http://wiki.usaepay.com/developer/soap]. # # ==== Login, Password, and Software ID # # Please follow all of USA ePay's directions for acquiring all accounts and settings. # # The value used for :login is the Key value found in the Merchant Console under Settings > Source # Key. You will have to add this key in the USA ePay Merchant Console. # # The value used for :password is the pin value also found and assigned in the Merchant Console under # Settings > Source Key. The pin is required to use all but basic transactions in the SOAP interface. # You will have to add the pin to your source key, as it defaults to none. # # The value used for the :software_id is found in the Developer's Login under the Developers Center # in your WSDL. It is the 8 character value in tag. A masked example: # # It is also found in the link to your WSDL. This is required as every account has a different path # SOAP requests are submitted to. Optionally, you can provide the entire urls via :live_url and :test_url, if your prefer. # # ==== Responses # * #success? -- +true+ if transmitted and returned correctly # * #message -- response or fault message # * #authorization -- reference_number or nil # * #params -- hash of entire soap response contents # # ==== Address Options # * :billing_address/:shipping_address -- contains some extra options # * :name -- virtual attribute; will split to first and last name # * :first_name # * :last_name # * :address1 # * :address2 # * :city # * :state # * :zip # * :country # * :phone # * :email # * :fax # * :company # # ==== Support: # * Questions: post to {active_merchant google group}[http://groups.google.com/group/activemerchant] # * Feedback/fixes: matt (at) nearapogee (dot) com # # ==== Links: # * {USA ePay Merchant Console}[https://sandbox.usaepay.com/login] # * {USA ePay Developer Login}[https://www.usaepay.com/developer/login] # class UsaEpayAdvancedGateway < Gateway API_VERSION = "1.4" class_attribute :test_url, :live_url TEST_URL_BASE = 'https://sandbox.usaepay.com/soap/gate/' #:nodoc: LIVE_URL_BASE = 'https://www.usaepay.com/soap/gate/' #:nodoc: FAILURE_MESSAGE = "Default Failure" #:nodoc: self.supported_countries = ['US'] self.supported_cardtypes = [:visa, :master, :american_express, :discover, :diners_club, :jcb] self.homepage_url = 'http://www.usaepay.com/' self.display_name = 'USA ePay Advanced SOAP Interface' CUSTOMER_OPTIONS = { :id => [:string, 'CustomerID'], # merchant assigned number :notes => [:string, 'Notes'], :data => [:string, 'CustomData'], :url => [:string, 'URL'], # Recurring Billing :enabled => [:boolean, 'Enabled'], :schedule => [:string, 'Schedule'], :number_left => [:integer, 'NumLeft'], :currency => [:string, 'Currency'], :description => [:string, 'Description'], :order_id => [:string, 'OrderID'], :user => [:string, 'User'], :source => [:string, 'Source'], :send_receipt => [:boolean, 'SendReceipt'], :receipt_note => [:string, 'ReceiptNote'], # Point of Sale :price_tier => [:string, 'PriceTier'], :tax_class => [:string, 'TaxClass'], :lookup_code => [:string, 'LookupCode'] } #:nodoc: ADDRESS_OPTIONS = { :first_name => [:string, 'FirstName'], :last_name => [:string, 'LastName'], :address1 => [:string, 'Street'], :address2 => [:string, 'Street2'], :city => [:string, 'City'], :state => [:string, 'State'], :zip => [:string, 'Zip'], :country => [:string, 'Country'], :phone => [:string, 'Phone'], :email => [:string, 'Email'], :fax => [:string, 'Fax'], :company => [:string, 'Company'] } #:nodoc: CUSTOMER_TRANSACTION_REQUEST_OPTIONS = { :command => [:string, 'Command'], :ignore_duplicate => [:boolean, 'IgnoreDuplicate'], :client_ip => [:string, 'ClientIP'], :customer_receipt => [:boolean, 'CustReceipt'], :customer_email => [:boolean, 'CustReceiptEmail'], :customer_template => [:boolean, 'CustReceiptName'], :merchant_receipt => [:boolean, 'MerchReceipt'], :merchant_email => [:boolean, 'MerchReceiptEmail'], :merchant_template => [:boolean, 'MerchReceiptName'], :verification_value => [:boolean, 'isRecurring'], :software => [:string, 'Software'] } #:nodoc: TRANSACTION_REQUEST_OBJECT_OPTIONS = { :command => [:string, 'Command'], :ignore_duplicate => [:boolean, 'IgnoreDuplicate'], :authorization_code => [:string, 'AuthCode'], :reference_number => [:string, 'RefNum'], :account_holder => [:string, 'AccountHolder'], :client_ip => [:string, 'ClientIP'], :customer_id => [:string, 'CustomerID'], :customer_receipt => [:boolean, 'CustReceipt'], :customer_template => [:boolean, 'CustReceiptName'], :software => [:string, 'Software'] } #:nodoc: TRANSACTION_DETAIL_OPTIONS = { :invoice => [:string, 'Invoice'], :po_number => [:string, 'PONum'], :order_id => [:string, 'OrderID'], :clerk => [:string, 'Clerk'], :terminal => [:string, 'Terminal'], :table => [:string, 'Table'], :description => [:string, 'Description'], :comments => [:string, 'Comments'], :allow_partial_auth => [:boolean, 'AllowPartialAuth'], :currency => [:string, 'Currency'], :non_tax => [:boolean, 'NonTax'], } #:nodoc: TRANSACTION_DETAIL_MONEY_OPTIONS = { :amount => [:double, 'Amount'], :tax => [:double, 'Tax'], :tip => [:double, 'Tip'], :non_tax => [:boolean, 'NonTax'], :shipping => [:double, 'Shipping'], :discount => [:double, 'Discount'], :subtotal => [:double, 'Subtotal'] } #:nodoc: CREDIT_CARD_DATA_OPTIONS = { :magnetic_stripe => [:string, 'MagStripe'], :dukpt => [:string, 'DUKPT'], :signature => [:string, 'Signature'], :terminal_type => [:string, 'TermType'], :magnetic_support => [:string, 'MagSupport'], :xid => [:string, 'XID'], :cavv => [:string, 'CAVV'], :eci => [:integer, 'ECI'], :internal_card_authorization => [:boolean, 'InternalCardAduth'], :pares => [:string, 'Pares'] } #:nodoc: CHECK_DATA_OPTIONS = { :check_number => [:integer, 'CheckNumber'], :drivers_license => [:string, 'DriversLicense'], :drivers_license_state => [:string, 'DriversLicenseState'], :record_type => [:string, 'RecordType'], :aux_on_us => [:string, 'AuxOnUS'], :epc_code => [:string, 'EpcCode'], :front_image => [:string, 'FrontImage'], :back_image => [:string, 'BackImage'] } #:nodoc: RECURRING_BILLING_OPTIONS = { :schedule => [:string, 'Schedule'], :number_left => [:integer, 'NumLeft'], :enabled => [:boolean, 'Enabled'] } #:nodoc: AVS_RESULTS = { 'Y' => %w( YYY Y YYA YYD ), 'Z' => %w( NYZ Z ), 'A' => %w( YNA A YNY ), 'N' => %w( NNN N NN ), 'X' => %w( YYX X ), 'W' => %w( NYW W ), 'XXW' => %w( XXW ), 'XXU' => %w( XXU ), 'R' => %w( XXR R U E ), 'S' => %w( XXS S ), 'XXE' => %w( XXE ), 'G' => %w( XXG G C I ), 'B' => %w( YYG B M ), 'D' => %w( GGG D ), 'P' => %w( YGG P ) }.inject({}) do |map, (type, codes)| codes.each { |code| map[code] = type } map end #:nodoc: AVS_CUSTOM_MESSAGES = { 'XXW' => 'Card number not on file.', 'XXU' => 'Address information not verified for domestic transaction.', 'XXE' => 'Address verification not allowed for card type.' } #:nodoc: # Create a new gateway. # # ==== Required # * At least the live_url OR the software_id must be present. # * :software_id -- 8 character software id # OR # * :test_url -- full url for testing # * :live_url -- full url for live/production # # ==== Optional # * :soap_response -- set to +true+ to add :soap_response to the params hash containing the entire soap xml message # def initialize(options = {}) requires! options, :login, :password @options = options if @options[:software_id] self.live_url = "#{LIVE_URL_BASE}#{@options[:software_id].to_s}" self.test_url = "#{TEST_URL_BASE}#{@options[:software_id].to_s}" else self.live_url = @options[:live_url].to_s self.test_url = @options[:test_url].to_s if @options[:test_url] end super end # Standard Gateway Methods ====================================== # Make a purchase with a credit card. (Authorize and # capture for settlement.) # # Note: See run_transaction for additional options. # def purchase(money, creditcard, options={}) run_sale(options.merge!(:amount => money, :payment_method => creditcard)) end # Authorize an amount on a credit card or account. # # Note: See run_transaction for additional options. # def authorize(money, creditcard, options={}) run_auth_only(options.merge!(:amount => money, :payment_method => creditcard)) end # Capture an authorized transaction. # # Note: See run_transaction for additional options. # def capture(money, identification, options={}) capture_transaction(options.merge!(:amount => money, :reference_number => identification)) end # Void a previous transaction that has not been settled. # # Note: See run_transaction for additional options. # def void(identification, options={}) void_transaction(options.merge!(:reference_number => identification)) end # Credit a previous transaction. # # Note: See run_transaction for additional options. # def credit(money, identification, options={}) refund_transaction(options.merge!(:amount => money, :reference_number => identification)) end # Customer ====================================================== # Add a customer. # # ==== Options # * :id -- merchant assigned id # * :notes -- notes about customer # * :data -- base64 data about customer # * :url -- customer website # * :billing_address -- usual options # * :payment_methods -- array of payment method hashes. # * :method -- credit_card or check # * :name -- optional name/label for the method # * :sort -- optional integer value specifying the backup sort order, 0 is default # # ==== Recurring Options # * :enabled -- +true+ enables recurring # * :schedule -- daily, weekly, bi-weekly (every two weeks), monthly, bi-monthly (every two months), quarterly, bi-annually (every six months), annually, first of month, last day of month # * :number_left -- number of payments left; -1 for unlimited # * :next -- date of next payment (Date/Time) # * :amount -- amount of recurring payment # * :tax -- tax portion of amount # * :currency -- numeric currency code # * :description -- description of transaction # * :order_id -- transaction order id # * :user -- merchant username assigned to transaction # * :source -- name of source key assigned to billing # * :send_receipt -- +true+ to send client a receipt # * :receipt_note -- leave a note on the receipt # # ==== Point of Sale Options # * :price_tier -- name of customer price tier # * :tax_class -- tax class # * :lookup_code -- lookup code from customer/member id card; barcode or magnetic stripe; can be assigned by merchant; defaults to system assigned if blank # # ==== Response # * #message -- customer number assigned by gateway # def add_customer(options={}) request = build_request(__method__, options) commit(__method__, request) end # Update a customer by replacing all of the customer details.. # # Use quickUpdateCustomer to just update a few attributes. # # ==== Options # * Same as add_customer # def update_customer(options={}) requires! options, :customer_number request = build_request(__method__, options) commit(__method__, request) end # Enable a customer for recurring billing. # # Note: Customer does not need to have all recurring paramerters to succeed. # # ==== Required # * :customer_number # def enable_customer(options={}) requires! options, :customer_number request = build_request(__method__, options) commit(__method__, request) end # Disable a customer for recurring billing. # # ==== Required # * :customer_number # def disable_customer(options={}) requires! options, :customer_number request = build_request(__method__, options) commit(__method__, request) end # Add a payment method to a customer. # # ==== Required # * :customer_number -- number returned by add_customer response.message # * :payment_method # * :method -- credit_card or check # * :name -- optional name/label for the method # * :sort -- an integer value specifying the backup sort order, 0 is default # # ==== Optional # * :make_default -- set +true+ to make default # * :verify -- set +true+ to run auth_only verification; throws fault if cannot verify # # ==== Response # * #message -- method_id of new customer payment method # def add_customer_payment_method(options={}) requires! options, :customer_number request = build_request(__method__, options) commit(__method__, request) end # Retrive all of the payment methods belonging to a customer # # ==== Required # * :customer_number # # ==== Response # * #message -- either a single hash or an array of hashes of payment methods # def get_customer_payment_methods(options={}) requires! options, :customer_number request = build_request(__method__, options) commit(__method__, request) end # Retrive one of the payment methods belonging to a customer # # ==== Required # * :customer_number # * :method_id # # ==== Response # * #message -- hash of payment method # def get_customer_payment_method(options={}) requires! options, :customer_number, :method_id request = build_request(__method__, options) commit(__method__, request) end # Update a customer payment method. # # ==== Required # * :method_id -- method_id to update # # ==== Options # * :method -- credit_card or check # * :name -- optional name/label for the method # * :sort -- an integer value specifying the backup sort order, 0 is default # * :verify -- set +true+ to run auth_only verification; throws fault if cannot verify # # ==== Response # * #message -- hash of payment method # def update_customer_payment_method(options={}) requires! options, :method_id request = build_request(__method__, options) commit(__method__, request) end # Delete one the payment methods beloning to a customer # # ==== Required # * :customer_number # * :method_id # def delete_customer_payment_method(options={}) requires! options, :customer_number, :method_id request = build_request(__method__, options) commit(__method__, request) end # Delete a customer. # # ==== Required # * :customer_number # def delete_customer(options={}) requires! options, :customer_number request = build_request(__method__, options) commit(__method__, request) end # Run a transaction for an existing customer in the database. # # ==== Required Options # * :customer_number -- gateway assigned identifier # * :command -- Sale, AuthOnly, Credit, Check, CheckCredit # * :amount -- total amount # # ==== Options # * :method_id -- which payment method to use, 0/nil/omitted for default method # * :ignore_duplicate -- +true+ overrides duplicate transaction # * :client_ip -- client ip address # * :customer_receipt -- +true+, sends receipt to customer. active_merchant defaults to +false+ # * :customer_email -- specify if different than customer record # * :customer_template -- name of template # * :merchant_receipt -- +true+, sends receipt to merchant. active_merchant defaults to +false+ # * :merchant_email -- required if :merchant_receipt set to +true+ # * :merchant_template -- name of template # * :recurring -- defaults to +false+ *see documentation* # * :verification_value -- pci forbids storage of this value, only required for CVV2 validation # * :software -- active_merchant sets to required gateway option value # * :line_items -- XXX not implemented yet # * :custom_fields -- XXX not implemented yet # # ==== Transaction Options # * :invoice -- transaction invoice number; truncated to 10 characters; defaults to reference_number # * :po_number -- commercial purchase order number; upto 25 characters # * :order_id -- should be used to assign a unique id; upto 64 characters # * :clerk -- sales clerk # * :terminal -- terminal name # * :table -- table name/number # * :description -- description # * :comments -- comments # * :allow_partial_auth -- allow partial authorization if full amount is not available; defaults +false+ # * :currency -- numeric currency code # * :tax -- tax portion of amount # * :tip -- tip portion of amount # * :non_tax -- +true+ if transaction is non-taxable # * :shipping -- shipping portion of amount # * :discount -- amount of discount # * :subtotal -- amount of transaction before tax, tip, shipping, and discount are applied # # ==== Response # * #message -- transaction response hash # def run_customer_transaction(options={}) requires! options, :customer_number, :command, :amount request = build_request(__method__, options) commit(__method__, request) end # Transactions ================================================== # Run a transaction. # # Note: run_sale, run_auth_only, run_credit, run_check_sale, run_check_credit # methods are also available. Each takes the same options as # run_transaction, but the :command option is not required. # # Recurring Note: If recurring options are included USA ePay will create a # new customer record with the supplied information. The customer number # will be returned in the response. # # ==== Options # * :method -- credit_card or check # * :command -- sale, credit, void, creditvoid, authonly, capture, postauth, check, checkcredit; defaults to sale; only required for run_transaction when other than sale # * :reference_number -- for the original transaction; obtained by sale or authonly # * :authorization_code -- required for postauth; obtained offline # * :ignore_duplicate -- set +true+ if you want to override the duplicate tranaction handling # * :account_holder -- name of account holder # * :customer_id -- merchant assigned id # * :customer_receipt -- set +true+ to email receipt to billing email address # * :customer_template -- name of template # * :software -- stamp merchant software version for tracking # * :billing_address -- see UsaEpayCimGateway documentation for all address fields # * :shipping_address -- see UsaEpayCimGateway documentation for all address fields # * :recurring -- used for recurring billing transactions # * :schedule -- disabled, daily, weekly, bi-weekly (every two weeks), monthly, bi-monthly (every two months), quarterly, bi-annually (every six months), annually # * :next -- date customer billed next (Date/Time) # * :expire -- date the recurring transactions end (Date/Time) # * :number_left -- transactions remaining in billing cycle # * :amount -- amount to be billed each recurring transaction # * :enabled -- states if currently active # * :line_items -- XXX not implemented yet # * :custom_fields -- XXX not implemented yet # # ==== Transaction Options # * :amount -- total amount # * :invoice -- transaction invoice number; truncated to 10 characters; defaults to reference_number # * :po_number -- commercial purchase order number; upto 25 characters # * :order_id -- should be used to assign a unique id; upto 64 characters # * :clerk -- sales clerk # * :terminal -- terminal name # * :table -- table name/number # * :description -- description # * :comments -- comments # * :allow_partial_auth -- allow partial authorization if full amount is not available; defaults +false+ # * :currency -- numeric currency code # * :tax -- tax portion of amount # * :tip -- tip portion of amount # * :non_tax -- +true+ if transaction is non-taxable # * :shipping -- shipping portion of amount # * :discount -- amount of discount # * :subtotal -- amount of transaction before tax, tip, shipping, and discount are applied # # ==== Response # * #message -- transaction response hash # def run_transaction(options={}) request = build_request(__method__, options) commit(__method__, request) end TRANSACTION_METHODS = [ :run_sale, :run_auth_only, :run_credit, :run_check_sale, :run_check_credit ] #:nodoc: TRANSACTION_METHODS.each do |method| define_method method do |options| request = build_request(method, options) commit(method, request) end end # Post an authorization code obtained offline. # # ==== Required # * :authorization_code -- obtained offline # # ==== Options # * Same as run_transaction # # ==== Response # * #message -- transaction response hash # def post_auth(options={}) requires! options, :authorization_code request = build_request(__method__, options) commit(__method__, request) end # Capture an authorized transaction and move it into the current batch # for settlement. # # Note: Check with merchant bank for details/restrictions on differing # amounts than the original authorization. # # ==== Required # * :reference_number # # ==== Options # * :amount -- may be different than original amount; 0 will void authorization # # ==== Response # * #message -- transaction response hash # def capture_transaction(options={}) requires! options, :reference_number request = build_request(__method__, options) commit(__method__, request) end # Void a transaction. # # Note: Can only be voided before being settled. # # ==== Required # * :reference_number # # ==== Response # * #message -- transaction response hash # def void_transaction(options={}) requires! options, :reference_number request = build_request(__method__, options) commit(__method__, request) end # Refund transaction. # # Note: Required after a transaction has been settled. Refunds # both credit card and check transactions. # # ==== Required # * :reference_number # * :amount -- amount to refund; 0 will refund original amount # # ==== Response # * #message -- transaction response hash # def refund_transaction(options={}) requires! options, :reference_number, :amount request = build_request(__method__, options) commit(__method__, request) end # Override transaction flagged for mananager approval. # # Note: Checks only! # # ==== Required # * :reference_number # # ==== Options # * :reason # # ==== Response # * #message -- transaction response hash # def override_transaction(options={}) requires! options, :reference_number request = build_request(__method__, options) commit(__method__, request) end # Quick Transactions ============================================ # Run a sale transaction based off of a past transaction. # # Transfers referenced transaction's payment method to this # transaction. As of 6/2011, USA ePay blocks credit card numbers # at 3 years. # # ==== Required # * :reference_number -- transaction to reference payment from # * :amount -- total amount # # ==== Options # * :authorize_only -- set +true+ if you just want to authorize # # ==== Transaction Options # * :invoice -- transaction invoice number; truncated to 10 characters; defaults to reference_number # * :po_number -- commercial purchase order number; upto 25 characters # * :order_id -- should be used to assign a unique id; upto 64 characters # * :clerk -- sales clerk # * :terminal -- terminal name # * :table -- table name/number # * :description -- description # * :comments -- comments # * :allow_partial_auth -- allow partial authorization if full amount is not available; defaults +false+ # * :currency -- numeric currency code # * :tax -- tax portion of amount # * :tip -- tip portion of amount # * :non_tax -- +true+ if transaction is non-taxable # * :shipping -- shipping portion of amount # * :discount -- amount of discount # * :subtotal -- amount of transaction before tax, tip, shipping, and discount are applied # # ==== Response # * #message -- transaction response hash # def run_quick_sale(options={}) requires! options, :reference_number, :amount request = build_request(__method__, options) commit(__method__, request) end # Run a credit based off of a past transaction. # # Transfers referenced transaction's payment method to this # transaction. As of 6/2011, USA ePay blocks credit card numbers # at 3 years. # # ==== Required # * :reference_number -- transaction to reference payment from # # ==== Transaction Options # * :amount -- total amount # * :invoice -- transaction invoice number; truncated to 10 characters; defaults to reference_number # * :po_number -- commercial purchase order number; upto 25 characters # * :order_id -- should be used to assign a unique id; upto 64 characters # * :clerk -- sales clerk # * :terminal -- terminal name # * :table -- table name/number # * :description -- description # * :comments -- comments # * :allow_partial_auth -- allow partial authorization if full amount is not available; defaults +false+ # * :currency -- numeric currency code # * :tax -- tax portion of amount # * :tip -- tip portion of amount # * :non_tax -- +true+ if transaction is non-taxable # * :shipping -- shipping portion of amount # * :discount -- amount of discount # * :subtotal -- amount of transaction before tax, tip, shipping, and discount are applied # # ==== Response # * #message -- transaction response hash # def run_quick_credit(options={}) requires! options, :reference_number request = build_request(__method__, options) commit(__method__, request) end # Transaction Status ============================================ # Retrieve details of a specified transaction. # # ==== Required # * :reference_number # # ==== Response # * #message -- transaction hash # def get_transaction(options={}) requires! options, :reference_number request = build_request(__method__, options) commit(__method__, request) end # Check status of a transaction. # # ==== Required # * :reference_number # # ==== Response # * response.success -- success of the referenced transaction # * response.message -- message of the referenced transaction # * response.authorization -- same as :reference_number in options # def get_transaction_status(options={}) requires! options, :reference_number request = build_request(__method__, options) commit(__method__, request) end # Check status of a transaction (custom). # # ==== Required # * :reference_number # * :fields -- string array of fields to retrieve # * Response.AuthCode # * Response.AvsResult # * Response.AvsResultCode # * Response.BatchNum # * Response.CardCodeResult # * Response.CardCodeResultCode # * Response.ConversionRate # * Response.ConvertedAmount # * Response.ConvertedAmountCurrency # * Response.Error # * Response.ErrorCode # * Response.RefNum # * Response.Result # * Response.ResultCode # * Response.Status # * Response.StatusCode # * CheckTrace.TrackingNum # * CheckTrace.Effective # * CheckTrace.Processed # * CheckTrace.Settled # * CheckTrace.Returned # * CheckTrace.BankNote # * DateTime # * AccountHolder # * Details.Invoice # * Details.PoNum # * Details.OrderID # * Details.Clerk # * Details.Terminal # * Details.Table # * Details.Description # * Details.Amount # * Details.Currency # * Details.Tax # * Details.Tip # * Details.NonTax # * Details.Shipping # * Details.Discount # * Details.Subtotal # * CreditCardData.CardType # * CreditCardData.CardNumber # * CreditCardData.CardExpiration # * CreditCardData.CardCode # * CreditCardData.AvsStreet # * CreditCardData.AvsZip # * CreditCardData.CardPresent # * CheckData.CheckNumber # * CheckData.Routing # * CheckData.Account # * CheckData.SSN # * CheckData.DriversLicense # * CheckData.DriversLicenseState # * CheckData.RecordType # * User # * Source # * ServerIP # * ClientIP # * CustomerID # * BillingAddress.FirstName # * BillingAddress.LastName # * BillingAddress.Company # * BillingAddress.Street # * BillingAddress.Street2 # * BillingAddress.City # * BillingAddress.State # * BillingAddress.Zip # * BillingAddress.Country # * BillingAddress.Phone # * BillingAddress.Fax # * BillingAddress.Email # * ShippingAddress.FirstName # * ShippingAddress.LastName # * ShippingAddress.Company # * ShippingAddress.Street # * ShippingAddress.Street2 # * ShippingAddress.City # * ShippingAddress.State # * ShippingAddress.Zip # * ShippingAddress.Country # * ShippingAddress.Phone # * ShippingAddress.Fax # * ShippingAddress.Email # # ==== Response # * #message -- hash; keys are the field values # def get_transaction_custom(options={}) requires! options, :reference_number, :fields request = build_request(__method__, options) commit(__method__, request) end # Check status of a check transaction. # # ==== Required # * :reference_number # # ==== Response # * #message -- check trace hash # def get_check_trace(options={}) requires! options, :reference_number request = build_request(__method__, options) commit(__method__, request) end # Account ======================================================= # Retrieve merchant account details # # ==== Response # * #message -- account hash # def get_account_details request = build_request(__method__) commit(__method__, request) end # Builders ====================================================== private # Build soap header, etc. def build_request(action, options = {}) soap = Builder::XmlMarkup.new soap.instruct!(:xml, :version => '1.0', :encoding => 'utf-8') soap.tag! "SOAP-ENV:Envelope", 'xmlns:SOAP-ENV' => 'http://schemas.xmlsoap.org/soap/envelope/', 'xmlns:ns1' => 'urn:usaepay', 'xmlns:xsd' => 'http://www.w3.org/2001/XMLSchema', 'xmlns:xsi' => 'http://www.w3.org/2001/XMLSchema-instance', 'xmlns:SOAP-ENC' => 'http://schemas.xmlsoap.org/soap/encoding/', 'SOAP-ENV:encodingStyle' => 'http://schemas.xmlsoap.org/soap/encoding/' do |soap| soap.tag! "SOAP-ENV:Body" do |soap| send("build_#{action}", soap, options) end end soap.target! end # Build generic tag. def build_tag(soap, type, tag, value) soap.tag!(tag, value, 'xsi:type' => "xsd:#{type}") if value != nil end # Build token. def build_token(soap, options) seed = SecureRandom.base64(32) hash = Digest::SHA1.hexdigest("#{@options[:login]}#{seed}#{@options[:password].to_s.strip}") soap.Token 'xsi:type' => 'ns1:ueSecurityToken' do |soap| build_tag soap, :string, 'ClientIP', options[:client_ip] soap.PinHash 'xsi:type' => 'ns1:ueHash' do |soap| build_tag soap, :string, "HashValue", hash build_tag soap, :string, "Seed", seed build_tag soap, :string, "Type", 'sha1' end build_tag soap, :string, 'SourceKey', @options[:login] end end # Customer ====================================================== def build_add_customer(soap, options) soap.tag! "ns1:addCustomer" do |soap| build_token soap, options build_customer_data soap, options build_tag soap, :double, 'Amount', amount(options[:amount]) build_tag soap, :double, 'Tax', amount(options[:tax]) build_tag soap, :string, 'Next', options[:next].strftime("%Y-%m-%d") if options[:next] end end def build_customer(soap, options, type, add_customer_data=false) soap.tag! "ns1:#{type}" do |soap| build_token soap, options build_tag soap, :integer, 'CustNum', options[:customer_number] build_customer_data soap, options if add_customer_data end end def build_update_customer(soap, options) build_customer(soap, options, 'updateCustomer', true) end def build_enable_customer(soap, options) build_customer(soap, options, 'enableCustomer') end def build_disable_customer(soap, options) build_customer(soap, options, 'disableCustomer') end def build_delete_customer(soap, options) build_customer(soap, options, 'deleteCustomer') end def build_add_customer_payment_method(soap, options) soap.tag! "ns1:addCustomerPaymentMethod" do |soap| build_token soap, options build_tag soap, :integer, 'CustNum', options[:customer_number] build_customer_payment_methods soap, options build_tag soap, :boolean, 'MakeDefault', options[:make_default] build_tag soap, :boolean, 'Verify', options[:verify] end end def build_get_customer_payment_method(soap, options) soap.tag! 'ns1:getCustomerPaymentMethod' do |soap| build_token soap, options build_tag soap, :integer, 'CustNum', options[:customer_number] build_tag soap, :integer, 'MethodID', options[:method_id] end end def build_get_customer_payment_methods(soap, options) build_customer(soap, options, 'getCustomerPaymentMethods') end def build_update_customer_payment_method(soap, options) soap.tag! 'ns1:updateCustomerPaymentMethod' do |soap| build_token soap, options build_customer_payment_methods soap, options build_tag soap, :boolean, 'Verify', options[:verify] end end def build_delete_customer_payment_method(soap, options) soap.tag! "ns1:deleteCustomerPaymentMethod" do |soap| build_token soap, options build_tag soap, :integer, 'Custnum', options[:customer_number] build_tag soap, :integer, 'PaymentMethodID', options[:method_id] end end def build_run_customer_transaction(soap, options) soap.tag! "ns1:runCustomerTransaction" do |soap| build_token soap, options build_tag soap, :integer, 'CustNum', options[:customer_number] build_tag soap, :integer, 'PaymentMethodID', options[:method_id] || 0 build_customer_transaction soap, options end end # Transactions ================================================== def build_run_transaction(soap, options) soap.tag! 'ns1:runTransaction' do |soap| build_token soap, options build_transaction_request_object soap, options, 'Parameters' end end def build_run_sale(soap, options) soap.tag! 'ns1:runSale' do |soap| build_token soap, options build_transaction_request_object soap, options end end def build_run_auth_only(soap, options) soap.tag! 'ns1:runAuthOnly' do |soap| build_token soap, options build_transaction_request_object soap, options end end def build_run_credit(soap, options) soap.tag! 'ns1:runCredit' do |soap| build_token soap, options build_transaction_request_object soap, options end end def build_run_check_sale(soap, options) soap.tag! 'ns1:runCheckSale' do |soap| build_token soap, options build_transaction_request_object soap, options end end def build_run_check_credit(soap, options) soap.tag! 'ns1:runCheckCredit' do |soap| build_token soap, options build_transaction_request_object soap, options end end def build_post_auth(soap, options) soap.tag! 'ns1:postAuth' do |soap| build_token soap, options build_transaction_request_object soap, options end end def build_run_quick_sale(soap, options) soap.tag! 'ns1:runQuickSale' do |soap| build_token soap, options build_tag soap, :integer, 'RefNum', options[:reference_number] build_transaction_detail soap, options build_tag soap, :boolean, 'AuthOnly', options[:authorize_only] || false end end def build_run_quick_credit(soap, options) soap.tag! 'ns1:runQuickCredit' do |soap| build_token soap, options build_tag soap, :integer, 'RefNum', options[:reference_number] build_transaction_detail soap, options end end def build_get_transaction(soap, options) soap.tag! "ns1:getTransaction" do |soap| build_token soap, options build_tag soap, :integer, 'RefNum', options[:reference_number] end end def build_get_transaction_status(soap, options) soap.tag! "ns1:getTransactionStatus" do |soap| build_token soap, options build_tag soap, :integer, 'RefNum', options[:reference_number] end end def build_get_transaction_custom(soap, options) soap.tag! "ns1:getTransactionCustom" do |soap| build_token soap, options build_tag soap, :integer, 'RefNum', options[:reference_number] build_transaction_field_array soap, options end end def build_get_check_trace(soap, options) soap.tag! "ns1:getCheckTrace" do |soap| build_token soap, options build_tag soap, :integer, 'RefNum', options[:reference_number] end end def build_capture_transaction(soap, options) soap.tag! "ns1:captureTransaction" do |soap| build_token soap, options build_tag soap, :integer, 'RefNum', options[:reference_number] build_tag soap, :double, 'RefNum', amount(options[:amount]) end end def build_void_transaction(soap, options) soap.tag! "ns1:voidTransaction" do |soap| build_token soap, options build_tag soap, :integer, 'RefNum', options[:reference_number] end end def build_refund_transaction(soap, options) soap.tag! "ns1:refundTransaction" do |soap| build_token soap, options build_tag soap, :integer, 'RefNum', options[:reference_number] build_tag soap, :integer, 'Amount', amount(options[:amount]) end end def build_override_transaction(soap, options) soap.tag! "ns1:overrideTransaction" do |soap| build_token soap, options build_tag soap, :integer, 'RefNum', options[:reference_number] build_tag soap, :string, 'Reason', options[:reason] end end # Account ======================================================= def build_get_account_details(soap, options) soap.tag! "ns1:getAccountDetails" do |soap| build_token soap, options end end # Customer Helpers ============================================== def build_customer_data(soap, options) soap.CustomerData 'xsi:type' => 'ns1:CustomerObject' do CUSTOMER_OPTIONS.each do |k,v| build_tag soap, v[0], v[1], options[k] end build_billing_address soap, options build_customer_payments soap, options build_custom_fields soap, options end end def build_customer_payments(soap, options) if options[:payment_methods] length = options[:payment_methods].length soap.PaymentMethods 'SOAP-ENC:arrayType' => "ns1:PaymentMethod[#{length}]", 'xsi:type' =>"ns1:PaymentMethodArray" do |soap| build_customer_payment_methods soap, options end end end def extract_methods_and_tag(options) case when options[:payment_method] && !options[:payment_methods] payment_methods = [options[:payment_method]] tag_name = 'PaymentMethod' when options[:payment_methods] && !options[:payment_method] payment_methods = options[:payment_methods] tag_name = 'item' else payment_methods = [options] tag_name = 'PaymentMethod' end [payment_methods, tag_name] end def build_credit_card_or_check(soap, payment_method) case when payment_method[:method].kind_of?(ActiveMerchant::Billing::CreditCard) build_tag soap, :string, 'CardNumber', payment_method[:method].number build_tag soap, :string, 'CardExpiration', "#{payment_method[:method].year}-#{"%02d" % payment_method[:method].month}" if options[:billing_address] build_tag soap, :string, 'AvsStreet', options[:billing_address][:address1] build_tag soap, :string, 'AvsZip', options[:billing_address][:zip] end build_tag soap, :string, 'CardCode', payment_method[:method].verification_value when payment_method[:method].kind_of?(ActiveMerchant::Billing::Check) build_tag soap, :string, 'Account', payment_method[:method].number build_tag soap, :string, 'Routing', payment_method[:method].routing_number build_tag soap, :string, 'AccountType', payment_method[:method].account_type.capitalize build_tag soap, :string, 'DriversLicense', options[:drivers_license] build_tag soap, :string, 'DriversLicenseState', options[:drivers_license_state] build_tag soap, :string, 'RecordType', options[:record_type] end end def build_customer_payment_methods(soap, options) payment_methods, tag_name = extract_methods_and_tag(options) payment_methods.each do |payment_method| soap.tag! tag_name, 'xsi:type' => "ns1:PaymentMethod" do |soap| build_tag soap, :integer, 'MethodID', payment_method[:method_id] build_tag soap, :string, 'MethodType', payment_method[:type] build_tag soap, :string, 'MethodName', payment_method[:name] build_tag soap, :integer, 'SecondarySort', payment_method[:sort] build_credit_card_or_check(soap, payment_method) end end end def build_customer_transaction(soap, options) soap.Parameters 'xsi:type' => "ns1:CustomerTransactionRequest" do |soap| build_transaction_detail soap, options CUSTOMER_TRANSACTION_REQUEST_OPTIONS.each do |k,v| build_tag soap, v[0], v[1], options[k] end build_custom_fields soap, options build_line_items soap, options end end # Transaction Helpers =========================================== def build_transaction_request_object(soap, options, name='Params') soap.tag! name, 'xsi:type' => "ns1:TransactionRequestObject" do |soap| TRANSACTION_REQUEST_OBJECT_OPTIONS.each do |k,v| build_tag soap, v[0], v[1], options[k] end case when options[:payment_method] == nil when options[:payment_method].kind_of?(ActiveMerchant::Billing::CreditCard) build_credit_card_data soap, options when options[:payment_method].kind_of?(ActiveMerchant::Billing::Check) build_check_data soap, options else raise ArgumentError, 'options[:payment_method] must be a CreditCard or Check' end build_transaction_detail soap, options build_billing_address soap, options build_shipping_address soap, options build_recurring_billing soap, options build_line_items soap, options build_custom_fields soap, options end end def build_transaction_detail(soap, options) soap.Details 'xsi:type' => "ns1:TransactionDetail" do |soap| TRANSACTION_DETAIL_OPTIONS.each do |k,v| build_tag soap, v[0], v[1], options[k] end TRANSACTION_DETAIL_MONEY_OPTIONS.each do |k,v| build_tag soap, v[0], v[1], amount(options[k]) end end end def build_credit_card_data(soap, options) soap.CreditCardData 'xsi:type' => "ns1:CreditCardData" do |soap| build_tag soap, :string, 'CardNumber', options[:payment_method].number build_tag soap, :string, 'CardExpiration', "#{options[:payment_method].year}-#{"%02d" % options[:payment_method].month}" if options[:billing_address] build_tag soap, :string, 'AvsStreet', options[:billing_address][:address1] build_tag soap, :string, 'AvsZip', options[:billing_address][:zip] end build_tag soap, :string, 'CardCode', options[:payment_method].verification_value build_tag soap, :boolean, 'CardPresent', options[:card_present] || false CREDIT_CARD_DATA_OPTIONS.each do |k,v| build_tag soap, v[0], v[1], options[k] end end end def build_check_data(soap, options) soap.CheckData 'xsi:type' => "ns1:CheckData" do |soap| build_tag soap, :string, 'Account', options[:payment_method].number build_tag soap, :string, 'Routing', options[:payment_method].routing_number build_tag soap, :string, 'AccountType', options[:payment_method].account_type.capitalize CHECK_DATA_OPTIONS.each do |k,v| build_tag soap, v[0], v[1], options[k] end end end def build_recurring_billing(soap, options) if options[:recurring] soap.RecurringBilling 'xsi:type' => "ns1:RecurringBilling" do |soap| build_tag soap, :double, 'Amount', amount(options[:recurring][:amount]) build_tag soap, :string, 'Next', options[:recurring][:next].strftime("%Y-%m-%d") if options[:recurring][:next] build_tag soap, :string, 'Expire', options[:recurring][:expire].strftime("%Y-%m-%d") if options[:recurring][:expire] RECURRING_BILLING_OPTIONS.each do |k,v| build_tag soap, v[0], v[1], options[:recurring][k] end end end end def build_transaction_field_array(soap, options) soap.Fields 'SOAP-ENC:arryType' => "xsd:string[#{options[:fields].length}]", 'xsi:type' => 'ns1:stringArray' do |soap| options[:fields].each do |field| build_tag soap, :string, 'item', field end end end # General Helpers =============================================== def build_billing_address(soap, options) if options[:billing_address] if options[:billing_address][:name] name = options[:billing_address][:name].split(nil,2) # divide name options[:billing_address][:first_name], options[:billing_address][:last_name] = name[0], name[1] end soap.BillingAddress 'xsi:type' => "ns1:Address" do ADDRESS_OPTIONS.each do |k,v| build_tag soap, v[0], v[1], options[:billing_address][k] end end end end def build_shipping_address(soap, options) if options[:shipping_address] if options[:shipping_address][:name] name = options[:shipping_address][:name].split(nil,2) # divide name options[:shipping_address][:first_name], options[:shipping_address][:last_name] = name[0], name[1] end soap.ShippingAddress 'xsi:type' => "ns1:Address" do ADDRESS_OPTIONS.each do |k,v| build_tag soap, v[0], v[1], options[:shipping_address][k] end end end end def build_line_items(soap, options) # TODO end def build_custom_fields(soap, options) # TODO end # Request ======================================================= def commit(action, request) url = test? ? test_url : live_url begin soap = ssl_post(url, request, "Content-Type" => "text/xml") rescue ActiveMerchant::ResponseError => error soap = error.response.body end response = build_response(action, soap) end def build_response(action, soap) response_params, success, message, authorization, avs, cvv = parse(action, soap) response_params.merge!('soap_response' => soap) if @options[:soap_response] response = Response.new( success, message, response_params, :test => test?, :authorization => authorization, :avs_result => avs_from(avs), :cvv_result => cvv ) end def avs_from(avs) avs_params = { :code => avs } avs_params.merge!(:message => AVS_CUSTOM_MESSAGES[avs]) if AVS_CUSTOM_MESSAGES.key?(avs) avs_params end def parse(action, soap) xml = REXML::Document.new(soap) root = REXML::XPath.first(xml, "//SOAP-ENV:Body") response = root ? parse_element(root[0]) : { :response => soap } success, message, authorization, avs, cvv = false, FAILURE_MESSAGE, nil, nil, nil fault = (!response) || (response.length < 1) || response.has_key?('faultcode') return [response, success, response['faultstring'], authorization, avs, cvv] if fault if response.respond_to?(:[]) && p = response["#{action}_return"] if p.respond_to?(:key?) && p.key?('result_code') success = p['result_code'] == 'A' ? true : false authorization = p['ref_num'] avs = AVS_RESULTS[p['avs_result_code']] cvv = p['card_code_result_code'] else success = true end message = case action when :get_customer_payment_methods p['item'] when :get_transaction_custom p['item'].inject({}) { |map, field| map[field['field']] = field['value']; map } else p end elsif response.respond_to?(:[]) && p = response[:response] message = p # when response is html end [response, success, message, authorization, avs, cvv] end def parse_element(node) if node.has_elements? response = {} node.elements.each do |e| key = e.name.underscore value = parse_element(e) if response.has_key?(key) if response[key].is_a?(Array) response[key].push(value) else response[key] = [response[key], value] end else response[key] = parse_element(e) end end else response = node.text end response end end end end