2020-12-23 15:29:56 +01:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
# Helpers and utilities for interactions with the Stripe payment gateway
|
|
|
|
class StripeService
|
|
|
|
class << self
|
|
|
|
|
|
|
|
# Create the provided PaymentSchedule on Stripe, using the Subscription API
|
2020-12-30 10:16:39 +01:00
|
|
|
def create_stripe_subscription(payment_schedule, setup_intent_id)
|
2020-12-23 16:25:24 +01:00
|
|
|
stripe_key = Setting.get('stripe_secret_key')
|
2020-12-23 15:29:56 +01:00
|
|
|
first_item = payment_schedule.ordered_items.first
|
|
|
|
|
2020-12-29 18:55:00 +01:00
|
|
|
case payment_schedule.scheduled_type
|
|
|
|
when Reservation.name
|
|
|
|
subscription = payment_schedule.scheduled.subscription
|
|
|
|
reservable_stp_id = payment_schedule.scheduled.reservable&.stp_product_id
|
|
|
|
when Subscription.name
|
|
|
|
subscription = payment_schedule.scheduled
|
|
|
|
reservable_stp_id = nil
|
|
|
|
else
|
|
|
|
raise InvalidSubscriptionError
|
|
|
|
end
|
|
|
|
|
|
|
|
handle_wallet_transaction(payment_schedule)
|
|
|
|
|
2020-12-23 16:25:24 +01:00
|
|
|
# setup intent (associates the customer and the payment method)
|
|
|
|
intent = Stripe::SetupIntent.retrieve(setup_intent_id, api_key: stripe_key)
|
2020-12-23 15:29:56 +01:00
|
|
|
# subscription (recurring price)
|
|
|
|
price = create_price(first_item.details['recurring'],
|
2020-12-28 18:30:03 +01:00
|
|
|
subscription.plan.stp_product_id,
|
2020-12-23 15:29:56 +01:00
|
|
|
nil, monthly: true)
|
|
|
|
# other items (not recurring)
|
2020-12-28 18:30:03 +01:00
|
|
|
items = subscription_invoice_items(payment_schedule, subscription, first_item, reservable_stp_id)
|
2020-12-23 15:29:56 +01:00
|
|
|
|
|
|
|
stp_subscription = Stripe::Subscription.create({
|
|
|
|
customer: payment_schedule.invoicing_profile.user.stp_customer_id,
|
2021-02-22 11:28:52 +01:00
|
|
|
cancel_at: (payment_schedule.ordered_items.last.due_date + 3.day).to_i,
|
2020-12-23 15:29:56 +01:00
|
|
|
add_invoice_items: items,
|
2020-12-30 10:16:39 +01:00
|
|
|
coupon: payment_schedule.coupon&.code,
|
2020-12-23 15:29:56 +01:00
|
|
|
items: [
|
|
|
|
{ price: price[:id] }
|
2020-12-23 16:25:24 +01:00
|
|
|
],
|
|
|
|
default_payment_method: intent[:payment_method]
|
|
|
|
}, { api_key: stripe_key })
|
2020-12-23 15:29:56 +01:00
|
|
|
payment_schedule.update_attributes(stp_subscription_id: stp_subscription.id)
|
|
|
|
end
|
|
|
|
|
2020-12-30 10:16:39 +01:00
|
|
|
def create_stripe_coupon(coupon_id)
|
|
|
|
coupon = Coupon.find(coupon_id)
|
2021-01-04 16:26:44 +01:00
|
|
|
stp_coupon = { id: coupon.code }
|
2020-12-30 10:16:39 +01:00
|
|
|
if coupon.type == 'percent_off'
|
|
|
|
stp_coupon[:percent_off] = coupon.percent_off
|
|
|
|
elsif coupon.type == 'amount_off'
|
2021-05-26 13:38:16 +02:00
|
|
|
stp_coupon[:amount_off] = stripe_amount(coupon.amount_off)
|
2020-12-30 10:16:39 +01:00
|
|
|
stp_coupon[:currency] = Setting.get('stripe_currency')
|
|
|
|
end
|
|
|
|
|
2021-01-04 16:26:44 +01:00
|
|
|
stp_coupon[:duration] = coupon.validity_per_user == 'always' ? 'forever' : 'once'
|
2020-12-30 10:16:39 +01:00
|
|
|
stp_coupon[:redeem_by] = coupon.valid_until.to_i unless coupon.valid_until.nil?
|
|
|
|
stp_coupon[:max_redemptions] = coupon.max_usages unless coupon.max_usages.nil?
|
|
|
|
|
|
|
|
Stripe::Coupon.create(stp_coupon, api_key: Setting.get('stripe_secret_key'))
|
|
|
|
end
|
|
|
|
|
2021-05-26 13:38:16 +02:00
|
|
|
def stripe_amount(amount)
|
|
|
|
currency = Setting.get('stripe_currency')
|
|
|
|
return amount / 100 if zero_decimal_currencies.any? { |s| s.casecmp(currency).zero? }
|
|
|
|
|
|
|
|
amount
|
|
|
|
end
|
|
|
|
|
2020-12-23 15:29:56 +01:00
|
|
|
private
|
|
|
|
|
2020-12-28 18:30:03 +01:00
|
|
|
def subscription_invoice_items(payment_schedule, subscription, first_item, reservable_stp_id)
|
2020-12-23 15:29:56 +01:00
|
|
|
second_item = payment_schedule.ordered_items[1]
|
|
|
|
|
|
|
|
items = []
|
|
|
|
if first_item.amount != second_item.amount
|
|
|
|
unless first_item.details['adjustment']&.zero?
|
|
|
|
# adjustment: when dividing the price of the plan / months, sometimes it forces us to round the amount per month.
|
|
|
|
# The difference is invoiced here
|
|
|
|
p1 = create_price(first_item.details['adjustment'],
|
2020-12-28 18:30:03 +01:00
|
|
|
subscription.plan.stp_product_id,
|
2020-12-23 15:29:56 +01:00
|
|
|
"Price adjustment for payment schedule #{payment_schedule.id}")
|
|
|
|
items.push(price: p1[:id])
|
|
|
|
end
|
|
|
|
unless first_item.details['other_items']&.zero?
|
|
|
|
# when taking a subscription at the same time of a reservation (space, machine or training), the amount of the
|
|
|
|
# reservation is invoiced here.
|
|
|
|
p2 = create_price(first_item.details['other_items'],
|
|
|
|
reservable_stp_id,
|
|
|
|
"Reservations for payment schedule #{payment_schedule.id}")
|
|
|
|
items.push(price: p2[:id])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
items
|
|
|
|
end
|
|
|
|
|
|
|
|
def create_price(amount, stp_product_id, name, monthly: false)
|
|
|
|
params = {
|
2021-05-26 13:38:16 +02:00
|
|
|
unit_amount: stripe_amount(amount),
|
2020-12-23 15:29:56 +01:00
|
|
|
currency: Setting.get('stripe_currency'),
|
|
|
|
product: stp_product_id,
|
|
|
|
nickname: name
|
|
|
|
}
|
|
|
|
params[:recurring] = { interval: 'month', interval_count: 1 } if monthly
|
|
|
|
|
2020-12-23 16:25:24 +01:00
|
|
|
Stripe::Price.create(params, api_key: Setting.get('stripe_secret_key'))
|
2020-12-23 15:29:56 +01:00
|
|
|
end
|
2020-12-29 18:55:00 +01:00
|
|
|
|
|
|
|
def handle_wallet_transaction(payment_schedule)
|
|
|
|
return unless payment_schedule.wallet_amount
|
|
|
|
|
|
|
|
customer_id = payment_schedule.invoicing_profile.user.stp_customer_id
|
|
|
|
Stripe::Customer.update(customer_id, { balance: -payment_schedule.wallet_amount }, { api_key: Setting.get('stripe_secret_key') })
|
|
|
|
end
|
2021-05-26 13:38:16 +02:00
|
|
|
|
|
|
|
# @see https://stripe.com/docs/currencies#zero-decimal
|
|
|
|
def zero_decimal_currencies
|
|
|
|
%w[BIF CLP DJF GNF JPY KMF KRW MGA PYG RWF UGX VND VUV XAF XOF XPF]
|
|
|
|
end
|
2020-12-23 15:29:56 +01:00
|
|
|
end
|
|
|
|
end
|