Class: Spree::CreditCard

Inherits:
Base
  • Object
show all
Defined in:
app/models/spree/credit_card.rb

Constant Summary collapse

CARD_TYPES =
{
  visa: /^4\d{12}(\d{3})?(\d{3})?$/,
  master: /^(5[1-5]\d{4}|677189|222[1-9]\d{2}|22[3-9]\d{3}|2[3-6]\d{4}|27[01]\d{3}|2720\d{2})\d{10}$/,
  discover: /^(6011|65\d{2}|64[4-9]\d)\d{12}|(62\d{14})$/,
  american_express: /^3[47]\d{13}$/,
  diners_club: /^3(0[0-5]|[68]\d)\d{11}$/,
  jcb: /^35(28|29|[3-8]\d)\d{12}$/,
  switch: /^6759\d{12}(\d{2,3})?$/,
  solo: /^6767\d{12}(\d{2,3})?$/,
  dankort: /^5019\d{12}$/,
  maestro: /^(5[06-8]|6\d)\d{10,17}$/,
  forbrugsforeningen: /^600722\d{10}$/,
  laser: /^(6304|6706|6709|6771(?!89))\d{8}(\d{4}|\d{6,7})?$/
}

Instance Attribute Summary collapse

Instance Method Summary collapse

Methods inherited from Base

page, spree_base_scopes

Methods included from Preferences::Preferable

#clear_preferences, #default_preferences, #defined_preferences, #get_preference, #has_preference!, #has_preference?, #preference_default, #preference_type, #set_preference

Instance Attribute Details

#encrypted_dataObject

Returns the value of attribute encrypted_data.



16
17
18
# File 'app/models/spree/credit_card.rb', line 16

def encrypted_data
  @encrypted_data
end

#importedObject

Returns the value of attribute imported.



16
17
18
# File 'app/models/spree/credit_card.rb', line 16

def imported
  @imported
end

#numberObject

Returns the value of attribute number.



15
16
17
# File 'app/models/spree/credit_card.rb', line 15

def number
  @number
end

#track_dataString

ActiveMerchant::Billing::CreditCard added this accessor used by some gateways. More info: github.com/spree/spree/issues/6209

Returns or sets the track data for the card

Returns:

  • (String)


38
39
40
# File 'app/models/spree/credit_card.rb', line 38

def track_data
  @track_data
end

#verification_valueObject

Returns the value of attribute verification_value.



16
17
18
# File 'app/models/spree/credit_card.rb', line 16

def verification_value
  @verification_value
end

Instance Method Details

#actionsObject



107
108
109
# File 'app/models/spree/credit_card.rb', line 107

def actions
  %w{capture void credit}
end

#can_capture?(payment) ⇒ Boolean

Indicates whether its possible to capture the payment

Returns:

  • (Boolean)


112
113
114
# File 'app/models/spree/credit_card.rb', line 112

def can_capture?(payment)
  payment.pending? || payment.checkout?
end

#can_credit?(payment) ⇒ Boolean

Indicates whether its possible to credit the payment. Note that most gateways require that the payment be settled first which generally happens within 12-24 hours of the transaction.

Returns:

  • (Boolean)


123
124
125
# File 'app/models/spree/credit_card.rb', line 123

def can_credit?(payment)
  payment.completed? && payment.credit_allowed > 0
end

#can_void?(payment) ⇒ Boolean

Indicates whether its possible to void the payment.

Returns:

  • (Boolean)


117
118
119
# File 'app/models/spree/credit_card.rb', line 117

def can_void?(payment)
  !payment.failed? && !payment.void?
end

#cc_type=(type) ⇒ Object

cc_type is set by jquery.payment, which helpfully provides different types from Active Merchant. Converting them is necessary.



77
78
79
80
81
82
83
84
85
# File 'app/models/spree/credit_card.rb', line 77

def cc_type=(type)
  self[:cc_type] = case type
  when 'mastercard', 'maestro' then 'master'
  when 'amex' then 'american_express'
  when 'dinersclub' then 'diners_club'
  when '' then try_type_from_number
  else type
  end
end

#display_numberObject

Show the card number, with all but last 4 numbers replace with “X”. (XXXX-XXXX-XXXX-4338)



103
104
105
# File 'app/models/spree/credit_card.rb', line 103

def display_number
  "XXXX-XXXX-XXXX-#{last_digits}"
end

#expiry=(expiry) ⇒ Object



55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
# File 'app/models/spree/credit_card.rb', line 55

def expiry=(expiry)
  return unless expiry.present?

  self[:month], self[:year] =
  if expiry.match(/\d{2}\s?\/\s?\d{2,4}/) # will match mm/yy and mm / yyyy
    expiry.delete(' ').split('/')
  elsif match = expiry.match(/(\d{2})(\d{2,4})/) # will match mmyy and mmyyyy
    [match[1], match[2]]
  end
  if self[:year]
    self[:year] = "20#{ self[:year] }" if self[:year] / 100 == 0
    self[:year] = self[:year].to_i
  end
  self[:month] = self[:month].to_i if self[:month]
end

#first_nameObject

ActiveMerchant needs first_name/last_name because we pass it a Spree::CreditCard and it calls those methods on it. Looking at the ActiveMerchant source code we should probably be calling #to_active_merchant before passing the object to ActiveMerchant but this should do for now.



134
135
136
# File 'app/models/spree/credit_card.rb', line 134

def first_name
  name.to_s.split(/[[:space:]]/, 2)[0]
end

#has_payment_profile?Boolean

Returns:

  • (Boolean)


127
128
129
# File 'app/models/spree/credit_card.rb', line 127

def has_payment_profile?
  gateway_customer_profile_id.present? || gateway_payment_profile_id.present?
end

#last_nameObject



138
139
140
# File 'app/models/spree/credit_card.rb', line 138

def last_name
  name.to_s.split(/[[:space:]]/, 2)[1]
end

#set_last_digitsObject



87
88
89
90
91
# File 'app/models/spree/credit_card.rb', line 87

def set_last_digits
  number.to_s.gsub!(/\s/,'')
  verification_value.to_s.gsub!(/\s/,'')
  self.last_digits ||= number.to_s.length <= 4 ? number : number.to_s.slice(-4..-1)
end

#to_active_merchantObject



142
143
144
145
146
147
148
149
150
151
# File 'app/models/spree/credit_card.rb', line 142

def to_active_merchant
  ActiveMerchant::Billing::CreditCard.new(
    number: number,
    month: month,
    year: year,
    verification_value: verification_value,
    first_name: first_name,
    last_name: last_name,
  )
end

#try_type_from_numberObject



93
94
95
96
# File 'app/models/spree/credit_card.rb', line 93

def try_type_from_number
  numbers = number.delete(' ') if number
  CARD_TYPES.find{|type, pattern| return type.to_s if numbers =~ pattern}.to_s
end

#verification_value?Boolean

Returns:

  • (Boolean)


98
99
100
# File 'app/models/spree/credit_card.rb', line 98

def verification_value?
  verification_value.present?
end