forked from stripe-ruby-mock/stripe-ruby-mock
-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathcharges.rb
203 lines (168 loc) · 7.41 KB
/
charges.rb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
module StripeMock
module RequestHandlers
module Charges
def Charges.included(klass)
klass.add_handler 'post /v1/charges', :new_charge
klass.add_handler 'get /v1/charges', :get_charges
klass.add_handler 'get /v1/charges/search', :search_charges
klass.add_handler 'get /v1/charges/((?!search).*)', :get_charge
klass.add_handler 'post /v1/charges/(.*)/capture', :capture_charge
klass.add_handler 'post /v1/charges/(.*)/refund', :refund_charge
klass.add_handler 'post /v1/charges/(.*)/refunds', :refund_charge
klass.add_handler 'post /v1/charges/(.*)', :update_charge
end
def new_charge(route, method_url, params, headers = {})
stripe_account = headers && headers[:stripe_account] || Stripe.api_key
if headers && headers[:idempotency_key]
params[:idempotency_key] = headers[:idempotency_key]
if charges.any?
original_charge = charges.values.find { |c| c[:idempotency_key] == headers[:idempotency_key]}
return charges[original_charge[:id]] if original_charge
end
end
id = new_id('ch')
if params[:source]
if params[:source].is_a?(String)
# if a customer is provided, the card parameter is assumed to be the actual
# card id, not a token. in this case we'll find the card in the customer
# object and return that.
if params[:customer]
params[:source] = get_card(customers[stripe_account][params[:customer]], params[:source])
else
params[:source] = get_card_or_bank_by_token(params[:source])
end
elsif params[:source][:id]
raise Stripe::InvalidRequestError.new("Invalid token id: #{params[:source]}", 'card', http_status: 400)
end
elsif params[:customer]
customer = customers[stripe_account][params[:customer]]
if customer && customer[:default_source]
params[:source] = get_card(customer, customer[:default_source])
end
end
ensure_required_params(params)
bal_trans_params = { amount: params[:amount], source: id, application_fee: params[:application_fee] }
balance_transaction_id = new_balance_transaction('txn', bal_trans_params)
charges[id] = Data.mock_charge(
params.merge :id => id,
:balance_transaction => balance_transaction_id)
charge = charges[id].clone
if params[:expand] == ['balance_transaction']
charge[:balance_transaction] =
balance_transactions[balance_transaction_id]
end
charge
end
def update_charge(route, method_url, params, headers)
route =~ method_url
id = $1
charge = assert_existence :charge, id, charges[id]
allowed = allowed_params(params)
disallowed = params.keys - allowed
if disallowed.count > 0
raise Stripe::InvalidRequestError.new("Received unknown parameters: #{disallowed.join(', ')}" , '', http_status: 400)
end
charges[id] = Util.rmerge(charge, params)
end
def get_charges(route, method_url, params, headers)
params[:offset] ||= 0
params[:limit] ||= 10
clone = charges.clone
if params[:customer]
clone.delete_if { |k,v| v[:customer] != params[:customer] }
end
Data.mock_list_object(clone.values, params)
end
SEARCH_FIELDS = [
"amount",
"currency",
"customer",
"payment_method_details.card.brand",
"payment_method_details.card.exp_month",
"payment_method_details.card.exp_year",
"payment_method_details.card.fingerprint",
"payment_method_details.card.last4",
"status",
].freeze
def search_charges(route, method_url, params, headers)
require_param(:query) unless params[:query]
results = search_results(charges.values, params[:query], fields: SEARCH_FIELDS, resource_name: "charges")
Data.mock_list_object(results, params)
end
def get_charge(route, method_url, params, headers)
route =~ method_url
charge_id = $1 || params[:charge]
charge = assert_existence :charge, charge_id, charges[charge_id]
charge = charge.clone
if params[:expand] == ['balance_transaction']
balance_transaction = balance_transactions[charge[:balance_transaction]]
charge[:balance_transaction] = balance_transaction
end
charge
end
def capture_charge(route, method_url, params, headers)
route =~ method_url
charge = assert_existence :charge, $1, charges[$1]
if params[:amount]
refund = Data.mock_refund(
:balance_transaction => new_balance_transaction('txn'),
:id => new_id('re'),
:amount => charge[:amount] - params[:amount]
)
add_refund_to_charge(refund, charge)
end
if params[:application_fee]
charge[:application_fee] = params[:application_fee]
end
charge[:captured] = true
charge
end
def refund_charge(route, method_url, params, headers)
charge = get_charge(route, method_url, params, headers)
new_refund(
route,
method_url,
params.merge(:charge => charge[:id]),
headers
)
end
private
def ensure_required_params(params)
if params[:amount].nil?
require_param(:amount)
elsif params[:currency].nil?
require_param(:currency)
elsif non_integer_charge_amount?(params)
raise Stripe::InvalidRequestError.new("Invalid integer: #{params[:amount]}", 'amount', http_status: 400)
elsif non_positive_charge_amount?(params)
raise Stripe::InvalidRequestError.new('Invalid positive integer', 'amount', http_status: 400)
elsif params[:source].nil? && params[:customer].nil?
raise Stripe::InvalidRequestError.new('Must provide source or customer.', nil, http_status: nil)
end
end
def non_integer_charge_amount?(params)
params[:amount] && !params[:amount].is_a?(Integer)
end
def non_positive_charge_amount?(params)
params[:amount] && params[:amount] < 1
end
def allowed_params(params)
allowed = [:description, :metadata, :receipt_email, :fraud_details, :shipping, :destination]
# This is a workaround for the way the Stripe API sends params even when they aren't modified.
# Stipe will include those params even when they aren't modified.
allowed << :fee_details if params.has_key?(:fee_details) && params[:fee_details].nil?
allowed << :source if params.has_key?(:source) && params[:source].empty?
if params.has_key?(:refunds) && (params[:refunds].empty? ||
params[:refunds].has_key?(:data) && params[:refunds][:data].nil?)
allowed << :refunds
end
if params.has_key?(:payment_method_details) && (params[:payment_method_details].empty? ||
params[:payment_method_details].has_key?(:card) && (params[:payment_method_details][:card].empty? ||
params[:payment_method_details][:card].has_key?(:checks) && params[:payment_method_details][:card][:checks].empty?))
allowed << :payment_method_details
end
allowed
end
end
end
end