forked from stripe/example-mobile-backend
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathweb.rb
182 lines (165 loc) · 5.85 KB
/
web.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
require 'sinatra'
require 'stripe'
require 'dotenv'
require 'json'
require 'encrypted_cookie'
Dotenv.load
Stripe.api_key = ENV['STRIPE_TEST_SECRET_KEY']
use Rack::Session::EncryptedCookie,
:secret => 'replace_me_with_a_real_secret_key' # Actually use something secret here!
def log_info(message)
puts "\n" + message + "\n\n"
return message
end
get '/' do
status 200
return log_info("Great, your backend is set up. Now you can configure the Stripe example apps to point here.")
end
post '/ephemeral_keys' do
authenticate!
begin
key = Stripe::EphemeralKey.create(
{customer: params["customerId"]},
{stripe_version: params["api_version"]}
)
rescue Stripe::StripeError => e
status 402
return log_info("Error creating ephemeral key: #{e.message}")
end
content_type :json
status 200
key.to_json
end
post '/charge' do
authenticate!
# Get the credit card details submitted
payload = params
if request.content_type.include? 'application/json' and params.empty?
payload = Sinatra::IndifferentHash[JSON.parse(request.body.read)]
end
source = payload[:source]
customer = payload[:customer_id] || @customer.id
# Create the charge on Stripe's servers - this will charge the user's card
begin
charge = Stripe::Charge.create(
:amount => payload[:amount], # this number should be in cents
:currency => "aud",
:customer => customer,
:source => source,
:description => "Example Charge",
:metadata => {
:order_id => '5278735C-1F40-407D-933A-286E463E72D8',
}.merge(payload[:metadata] || {}),
)
rescue Stripe::StripeError => e
status 402
return log_info("Error creating charge: #{e.message}")
end
status 200
return log_info("Charge successfully created")
end
def authenticate!
# This code simulates "loading the Stripe customer for your current session".
# Your own logic will likely look very different.
return @customer if @customer
if session.has_key?(:customer_id)
customer_id = session[:customer_id]
begin
@customer = Stripe::Customer.retrieve(customer_id)
rescue Stripe::InvalidRequestError
end
else
begin
@customer = Stripe::Customer.create(
:description => 'mobile SDK example customer',
:metadata => {
# Add our application's customer id for this Customer, so it'll be easier to look up
:my_customer_id => '72F8C533-FCD5-47A6-A45B-3956CA8C792D',
},
)
rescue Stripe::InvalidRequestError
end
session[:customer_id] = @customer.id
end
@customer
end
# This endpoint is used by the Obj-C and Android example apps to create a charge.
post '/create_charge' do
# Create the charge on Stripe's servers
begin
charge = Stripe::Charge.create(
:amount => params[:amount], # this number should be in cents
:currency => "usd",
:source => params[:source],
:description => "Example Charge",
:metadata => {
:order_id => '5278735C-1F40-407D-933A-286E463E72D8',
}.merge(params[:metadata] || {}),
)
rescue Stripe::StripeError => e
status 402
return log_info("Error creating charge: #{e.message}")
end
status 200
return log_info("Charge successfully created")
end
# This endpoint is used by the mobile example apps to create a PaymentIntent.
# https://stripe.com/docs/api#create_payment_intent
# Just like the `/create_charge` endpoint, a real implementation would include controls
# to prevent misuse
post '/create_intent' do
begin
intent = Stripe::PaymentIntent.create(
:allowed_source_types => ['card'],
:amount => params[:amount],
:currency => params[:currency] || 'usd',
:description => params[:description] || 'Example PaymentIntent charge',
:metadata => {
:order_id => '5278735C-1F40-407D-933A-286E463E72D8',
}.merge(params[:metadata] || {}),
)
rescue Stripe::StripeError => e
status 402
return log_info("Error creating payment intent: #{e.message}")
end
log_info("Payment Intent successfully created")
status 200
return {:intent => intent.id, :secret => intent.client_secret}.to_json
end
# This endpoint responds to webhooks sent by Stripe. To use it, you'll need
# to add its URL (https://{your-app-name}.herokuapp.com/stripe-webhook)
# in the webhook settings section of the Dashboard.
# https://dashboard.stripe.com/account/webhooks
post '/stripe-webhook' do
json = JSON.parse(request.body.read)
# Retrieving the event from Stripe guarantees its authenticity
event = Stripe::Event.retrieve(json["id"])
source = event.data.object
# For sources that require additional user action from your customer
# (e.g. authorizing the payment with their bank), you should use webhooks
# to create a charge after the source becomes chargeable.
# For more information, see https://stripe.com/docs/sources#best-practices
WEBHOOK_CHARGE_CREATION_TYPES = ['bancontact', 'giropay', 'ideal', 'sofort', 'three_d_secure']
if event.type == 'source.chargeable' && WEBHOOK_CHARGE_CREATION_TYPES.include?(source.type)
begin
charge = Stripe::Charge.create(
:amount => source.amount,
:currency => source.currency,
:source => source.id,
:customer => source.metadata["customer"],
:description => "Example Charge",
:metadata => {
:order_id => '5278735C-1F40-407D-933A-286E463E72D8',
}.merge(source.metadata || {}),
)
rescue Stripe::StripeError => e
return log_info("Error creating charge: #{e.message}")
end
# After successfully creating a charge, you should complete your customer's
# order and notify them that their order has been fulfilled (e.g. by sending
# an email). When creating the source in your app, consider storing any order
# information (e.g. order number) as metadata so that you can retrieve it
# here and use it to complete your customer's purchase.
end
status 200
end