yurijmi/better_offsite_payments

View on GitHub
lib/offsite_payments/integrations/valitor.rb

Summary

Maintainability
A
25 mins
Test Coverage
module OffsitePayments #:nodoc:
  module Integrations #:nodoc:
    module Valitor
      mattr_accessor :test_url
      self.test_url = 'https://testvefverslun.valitor.is/1_1/'

      mattr_accessor :production_url
      self.production_url = 'https://vefverslun.valitor.is/1_1/'

      def self.test?
        (OffsitePayments.mode == :test)
      end

      def self.service_url
        (test? ? test_url : production_url)
      end

      def self.notification(params, options={})
        Notification.new(params, options.merge(:test => test?))
      end

      def self.return(query_string, options={})
        Return.new(query_string, options.merge(:test => test?))
      end

      class Helper < OffsitePayments::Helper
        include ActiveUtils::RequiresParameters

        DEFAULT_SUCCESS_TEXT = "The transaction has been completed."

        def initialize(order, account, options={})
          options[:currency] ||= 'ISK'
          super
          add_field 'Adeinsheimild', '0'
          add_field 'KaupandaUpplysingar', '0'
          add_field 'SlokkvaHaus', '0'
          @security_number = options[:credential2]
          @amount          = options[:amount]
          @order           = order
        end

        mapping :account, 'VefverslunID'
        mapping :currency, 'Gjaldmidill'

        mapping :order, 'Tilvisunarnumer'

        mapping :notify_url, 'SlodTokstAdGjaldfaeraServerSide'
        mapping :return_url, 'SlodTokstAdGjaldfaera'
        mapping :cancel_return_url, 'SlodNotandiHaettirVid'

        mapping :success_text, 'SlodTokstAdGjaldfaeraTexti'

        mapping :language, 'Lang'

        def authorize_only
          add_field 'Adeinsheimild', '1'
        end

        def collect_customer_info
          add_field 'KaupandaUpplysingar', '1'
        end

        def hide_header
          add_field 'SlokkvaHaus', '1'
        end

        def product(id, options={})
          raise ArgumentError, "Product id #{id} is not an integer between 1 and 500" unless id.to_i > 0 && id.to_i <= 500
          requires!(options, :amount, :description)
          options.assert_valid_keys([:description, :quantity, :amount, :discount])

          add_field("Vara_#{id}_Verd", format_amount(options[:amount], @fields[mappings[:currency]]))
          add_field("Vara_#{id}_Fjoldi", options[:quantity] || "1")

          add_field("Vara_#{id}_Lysing", options[:description]) if options[:description]
          add_field("Vara_#{id}_Afslattur", options[:discount] || '0')

          @products ||= []
          @products << id.to_i
        end

        def signature
          raise ArgumentError, "Security number not set" unless @security_number
          parts = [@security_number, @fields['Adeinsheimild']]
          @products.sort.uniq.each do |id|
            parts.concat(["Vara_#{id}_Fjoldi", "Vara_#{id}_Verd", "Vara_#{id}_Afslattur"].collect{|e| @fields[e]})
          end if @products
          parts.concat(%w(VefverslunID Tilvisunarnumer SlodTokstAdGjaldfaera SlodTokstAdGjaldfaeraServerSide Gjaldmidill).collect{|e| @fields[e]})
          Digest::MD5.hexdigest(parts.compact.join(''))
        end

        def form_fields
          product(1, :amount => @amount, :description => @order) if Array(@products).empty?
          @fields[mappings[:success_text]] ||= DEFAULT_SUCCESS_TEXT
          @fields.merge('RafraenUndirskrift' => signature)
        end

        def format_amount(amount, currency)
          OffsitePayments::CURRENCIES_WITHOUT_FRACTIONS.include?(currency) ? amount.to_f.round : sprintf("%.2f", amount)
        end
      end

      module ResponseFields
        def success?
          status == 'Completed'
        end
        alias :complete? :success?

        def test?
          @options[:test]
        end

        def item_id
          params['Tilvisunarnumer']
        end
        alias :order :item_id

        def transaction_id
          params['VefverslunSalaID']
        end

        def currency
          nil
        end

        def status
          "Completed" if acknowledge
        end

        def received_at
          Time.parse(params['Dagsetning'].to_s)
        end

        def gross
          "%0.2f" % params['Upphaed'].to_s.sub(',', '.')
        end

        def card_type
          params['Kortategund']
        end

        def card_last_four
          params['KortnumerSidustu']
        end

        def authorization_number
          params['Heimildarnumer']
        end

        def transaction_number
          params['Faerslunumer']
        end

        def customer_name
          params['Nafn']
        end

        def customer_address
          params['Heimilisfang']
        end

        def customer_zip
          params['Postnumer']
        end

        def customer_city
          params['Stadur']
        end

        def customer_country
          params['Land']
        end

        def customer_email
          params['Tolvupostfang']
        end

        def customer_comment
          params['Athugasemdir']
        end

        def password
          @options[:credential2]
        end

        def acknowledge(authcode = nil)
          password ? Digest::MD5.hexdigest("#{password}#{order}") == params['RafraenUndirskriftSvar'] : true
        end
      end

      class Notification < OffsitePayments::Notification
        include ResponseFields
      end

      class Return < OffsitePayments::Return
        include ResponseFields
      end
    end
  end
end