module Killbill #:nodoc: module Orbital #:nodoc: class OrbitalResponse < ::Killbill::Plugin::ActiveMerchant::ActiveRecord::Response self.table_name = 'orbital_responses' has_one :orbital_transaction scope :succeeded, -> { where(:success => true) } def self.from_response(api_call, kb_account_id, kb_payment_id, kb_payment_transaction_id, transaction_type, payment_processor_account_id, kb_tenant_id, response, extra_params = {}, model = ::Killbill::Orbital::OrbitalResponse) super(api_call, kb_account_id, kb_payment_id, kb_payment_transaction_id, transaction_type, payment_processor_account_id, kb_tenant_id, response, orbital_response_params(response).merge!(extra_params), model) end def self.orbital_response_params(response) { :params_account_num => extract(response, 'account_num'), :params_address_1 => extract(response, 'address_1'), :params_address_2 => extract(response, 'address_2'), :params_approval_status => extract(response, 'approval_status'), :params_auth_code => extract(response, 'auth_code'), :params_avs_resp_code => extract(response, 'avs_resp_code'), :params_card_brand => extract(response, 'card_brand'), :params_cavv_resp_code => extract(response, 'cavv_resp_code'), :params_cc_account_num => extract(response, 'cc_account_num'), :params_cc_expire_date => extract(response, 'cc_expire_date'), :params_country_fraud_filter_status => extract(response, 'country_fraud_filter_status'), :params_customer_bin => extract(response, 'customer_bin'), :params_customer_city => extract(response, 'customer_city'), :params_customer_country_code => extract(response, 'customer_country_code'), :params_customer_email => extract(response, 'customer_email'), :params_customer_merchant_id => extract(response, 'customer_merchant_id'), :params_customer_name => extract(response, 'customer_name'), :params_customer_phone => extract(response, 'customer_phone'), :params_customer_profile_action => extract(response, 'customer_profile_action'), :params_customer_profile_message => extract(response, 'customer_profile_message'), :params_customer_profile_order_override_ind => extract(response, 'customer_profile_order_override_ind'), :params_customer_ref_num => extract(response, 'customer_ref_num'), :params_customer_state => extract(response, 'customer_state'), :params_customer_zip => extract(response, 'customer_zip'), :params_cvv2_resp_code => extract(response, 'cvv2_resp_code'), :params_ecp_account_dda => extract(response, 'ecp_account_dda'), :params_ecp_account_rt => extract(response, 'ecp_account_rt'), :params_ecp_account_type => extract(response, 'ecp_account_type'), :params_ecp_bank_pmt_dlv => extract(response, 'ecp_bank_pmt_dlv'), :params_host_avs_resp_code => extract(response, 'host_avs_resp_code'), :params_host_cvv2_resp_code => extract(response, 'host_cvv2_resp_code'), :params_host_resp_code => extract(response, 'host_resp_code'), :params_industry_type => extract(response, 'industry_type'), :params_iso_country_code => extract(response, 'iso_country_code'), :params_merchant_id => extract(response, 'merchant_id'), :params_message_type => extract(response, 'message_type'), :params_order_default_amount => extract(response, 'order_default_amount'), :params_order_default_description => extract(response, 'order_default_description'), :params_order_id => extract(response, 'order_id'), :params_partial_auth_occurred => extract(response, 'partial_auth_occurred'), :params_proc_status => extract(response, 'proc_status'), :params_profile_proc_status => extract(response, 'profile_proc_status'), :params_recurring_advice_cd => extract(response, 'recurring_advice_cd'), :params_redeemed_amount => extract(response, 'redeemed_amount'), :params_remaining_balance => extract(response, 'remaining_balance'), :params_requested_amount => extract(response, 'requested_amount'), :params_resp_code => extract(response, 'resp_code'), :params_resp_msg => extract(response, 'resp_msg'), :params_resp_time => extract(response, 'resp_time'), :params_status => extract(response, 'status'), :params_status_msg => extract(response, 'status_msg'), :params_switch_solo_issue_num => extract(response, 'switch_solo_issue_num'), :params_switch_solo_start_date => extract(response, 'switch_solo_start_date'), :params_terminal_id => extract(response, 'terminal_id'), :params_tx_ref_idx => extract(response, 'tx_ref_idx'), :params_tx_ref_num => extract(response, 'tx_ref_num'), :params_mit_received_transaction_id => extract(response, 'mit_received_transaction_id') } end def first_reference_id params_tx_ref_num end def second_reference_id params_order_id end def cancel begin error_details = JSON.parse(message) original_message = nil rescue error_details = {} original_message = message end error_details['original_message'] = original_message unless original_message.blank? error_details['payment_plugin_status'] = 'CANCELED' updated_attributes = { :message => error_details.to_json, :success => false, :updated_at => Time.now.utc } # Update the response row update!(updated_attributes) end def update_and_create_transaction(gw_response, amount = nil, currency = nil) updated_attributes = { :message => (gw_response.success? && gw_response.message.nil?) ? "" : gw_response.message, :authorization => gw_response.authorization, :fraud_review => gw_response.fraud_review?, :test => gw_response.test?, :avs_result_code => gw_response.avs_result.kind_of?(::ActiveMerchant::Billing::AVSResult) ? gw_response.avs_result.code : gw_response.avs_result['code'], :avs_result_message => gw_response.avs_result.kind_of?(::ActiveMerchant::Billing::AVSResult) ? gw_response.avs_result.message : gw_response.avs_result['message'], :avs_result_street_match => gw_response.avs_result.kind_of?(::ActiveMerchant::Billing::AVSResult) ? gw_response.avs_result.street_match : gw_response.avs_result['street_match'], :avs_result_postal_match => gw_response.avs_result.kind_of?(::ActiveMerchant::Billing::AVSResult) ? gw_response.avs_result.postal_match : gw_response.avs_result['postal_match'], :cvv_result_code => gw_response.cvv_result.kind_of?(::ActiveMerchant::Billing::CVVResult) ? gw_response.cvv_result.code : gw_response.cvv_result['code'], :cvv_result_message => gw_response.cvv_result.kind_of?(::ActiveMerchant::Billing::CVVResult) ? gw_response.cvv_result.message : gw_response.cvv_result['message'], :success => gw_response.success?, :updated_at => Time.now.utc }.merge(OrbitalResponse.orbital_response_params(gw_response)) # Keep original values as much as possible updated_attributes.delete_if { |k, v| v.blank? && k != :message} # Update the response row update!(updated_attributes) # Create the transaction row if needed (cannot have been created before or the state wouldn't have been UNDEFINED) if gw_response.success? amount = amount.nil? ? gw_response.params['amount'] : amount currency = currency.nil? ? gw_response.params['currency'] : currency amount_in_cents = amount.nil? ? nil : ::Monetize.from_numeric(amount.to_f, currency).cents.to_i build_orbital_transaction(:kb_account_id => kb_account_id, :kb_tenant_id => kb_tenant_id, :amount_in_cents => amount_in_cents, :currency => currency, :api_call => api_call, :kb_payment_id => kb_payment_id, :kb_payment_transaction_id => kb_payment_transaction_id, :transaction_type => transaction_type, :payment_processor_account_id => payment_processor_account_id, :txn_id => txn_id, :created_at => updated_at, :updated_at => updated_at).save! end end def self.auth_responses_from_kb_payment_id(kb_payment_id, kb_tenant_id) where(:kb_payment_id => kb_payment_id, :kb_tenant_id => kb_tenant_id, :api_call => 'authorize').order(:created_at) end def self.find_mit_transaction_ref_id(kb_transaction_id, kb_tenant_id) last_response = where(:kb_payment_transaction_id => kb_transaction_id, :kb_tenant_id => kb_tenant_id).order(:created_at).last return nil if last_response.nil? return last_response.params_mit_received_transaction_id end def gateway_error_code params_resp_code end def to_transaction_info_plugin(transaction=nil) t_info_plugin = super(transaction) t_info_plugin.properties << create_plugin_property('processorResponse', params_host_resp_code) t_info_plugin.properties << create_plugin_property('orbital_response_id', id) t_info_plugin.properties << create_plugin_property('trace_number', params_trace_number) t_info_plugin.properties << create_plugin_property('mit_received_transaction_id', params_mit_received_transaction_id) t_info_plugin end def self.search_where_clause(t, search_key) where_clause = super(t, search_key) search_fields = Killbill::Plugin::ActiveMerchant.glob_config[:search_fields] if search_fields && search_fields.is_a?(Array) where_clauses = search_fields.map { |search_field| t[search_field.to_sym].eq(search_key) } where_clause = where_clauses.reduce(:or) end return where_clause end SIMPLE_PAGINATION_THRESHOLD = 20000 def self.max_nb_records if self.succeeded.limit(1).offset(SIMPLE_PAGINATION_THRESHOLD).nil? self.succeeded.count else SIMPLE_PAGINATION_THRESHOLD + 1 end end end end end