Replace setup_intents class from active merchat v1.i98 with a decorator that will adapt the version in v1.119

This commit is contained in:
Luis Ramos
2021-03-30 22:37:35 +01:00
committed by Matt-Yorkley
parent d3ef281e66
commit 4057445562
2 changed files with 86 additions and 277 deletions

View File

@@ -1,277 +0,0 @@
# frozen_string_literal: true
# Here we bring commit 823faaeab0d6d3bd75ee037ec894ab7c9d95d3a9 from ActiveMerchant v1.98.0
# This class integrates with the new StripePaymentIntents API
# This can be removed once we upgrade to ActiveMerchant v1.98.0
require 'active_support/core_ext/hash/slice'
module ActiveMerchant #:nodoc:
module Billing #:nodoc:
# This gateway uses the current Stripe
# {Payment Intents API}[https://stripe.com/docs/api/payment_intents].
# For the legacy API, see the Stripe gateway
class StripePaymentIntentsGateway < StripeGateway
ALLOWED_METHOD_STATES = %w[automatic manual].freeze
ALLOWED_CANCELLATION_REASONS = %w[duplicate fraudulent requested_by_customer abandoned].freeze
CREATE_INTENT_ATTRIBUTES =
%i[description statement_descriptor receipt_email save_payment_method].freeze
CONFIRM_INTENT_ATTRIBUTES =
%i[receipt_email return_url save_payment_method setup_future_usage off_session].freeze
UPDATE_INTENT_ATTRIBUTES =
%i[description statement_descriptor receipt_email setup_future_usage].freeze
DEFAULT_API_VERSION = '2019-05-16'
def create_intent(money, payment_method, options = {})
post = {}
add_amount(post, money, options, true)
add_capture_method(post, options)
add_confirmation_method(post, options)
add_customer(post, options)
add_payment_method_token(post, payment_method, options)
add_metadata(post, options)
add_return_url(post, options)
add_connected_account(post, options)
add_shipping_address(post, options)
setup_future_usage(post, options)
CREATE_INTENT_ATTRIBUTES.each do |attribute|
add_whitelisted_attribute(post, options, attribute)
end
commit(:post, 'payment_intents', post, options)
end
def show_intent(intent_id, options)
commit(:get, "payment_intents/#{intent_id}", nil, options)
end
def confirm_intent(intent_id, payment_method, options = {})
post = {}
add_payment_method_token(post, payment_method, options)
CONFIRM_INTENT_ATTRIBUTES.each do |attribute|
add_whitelisted_attribute(post, options, attribute)
end
commit(:post, "payment_intents/#{intent_id}/confirm", post, options)
end
def create_payment_method(payment_method, options = {})
post = {}
post[:type] = 'card'
post[:card] = {}
post[:card][:number] = payment_method.number
post[:card][:exp_month] = payment_method.month
post[:card][:exp_year] = payment_method.year
post[:card][:cvc] = payment_method.verification_value if payment_method.verification_value
commit(:post, 'payment_methods', post, options)
end
def update_intent(money, intent_id, payment_method, options = {})
post = {}
post[:amount] = money if money
add_payment_method_token(post, payment_method, options)
add_payment_method_types(post, options)
add_customer(post, options)
add_metadata(post, options)
add_shipping_address(post, options)
add_connected_account(post, options)
UPDATE_INTENT_ATTRIBUTES.each do |attribute|
add_whitelisted_attribute(post, options, attribute)
end
commit(:post, "payment_intents/#{intent_id}", post, options)
end
def authorize(money, payment_method, options = {})
create_intent(money,
payment_method,
options.merge!(confirm: true, capture_method: 'manual'))
end
def purchase(money, payment_method, options = {})
create_intent(money,
payment_method,
options.merge!(confirm: true, capture_method: 'automatic'))
end
def capture(money, intent_id, options = {})
post = {}
post[:amount_to_capture] = money
add_connected_account(post, options)
commit(:post, "payment_intents/#{intent_id}/capture", post, options)
end
def void(intent_id, options = {})
post = {}
if ALLOWED_CANCELLATION_REASONS.include?(options[:cancellation_reason])
post[:cancellation_reason] = options[:cancellation_reason]
end
commit(:post, "payment_intents/#{intent_id}/cancel", post, options)
end
def refund(money, intent_id, options = {})
intent = commit(:get, "payment_intents/#{intent_id}", nil, options)
charge_id = intent.params.dig('charges', 'data')[0].dig('id')
super(money, charge_id, options)
end
# Note: Not all payment methods are currently supported by the
# {Payment Methods API}[https://stripe.com/docs/payments/payment-methods]
# Current implementation will create
# a PaymentMethod object if the method is a token or credit card
# All other types will default to legacy Stripe store
def store(payment_method, options = {})
params = {}
post = {}
# If customer option is provided, create a payment method and attach to customer id
# Otherwise, create a customer, then attach
# if payment_method.is_a?(StripePaymentToken) ||
# payment_method.is_a?(ActiveMerchant::Billing::CreditCard)
add_payment_method_token(params, payment_method, options)
if options[:customer]
customer_id = options[:customer]
else
post[:validate] = options[:validate] unless options[:validate].nil?
post[:description] = options[:description] if options[:description]
post[:email] = options[:email] if options[:email]
customer = commit(:post, 'customers', post, options)
customer_id = customer.params['id']
# return the stripe response if expected customer id is not present
return customer if customer_id.nil?
end
commit(:post,
"payment_methods/#{params[:payment_method]}/attach",
{ customer: customer_id }, options)
# else
# super(payment, options)
# end
end
def unstore(identification, options = {}, deprecated_options = {})
if identification.include?('pm_')
_, payment_method = identification.split('|')
commit(:post, "payment_methods/#{payment_method}/detach", nil, options)
else
super(identification, options, deprecated_options)
end
end
private
def add_whitelisted_attribute(post, options, attribute)
post[attribute] = options[attribute] if options[attribute]
post
end
def add_capture_method(post, options)
capture_method = options[:capture_method].to_s
post[:capture_method] = capture_method if ALLOWED_METHOD_STATES.include?(capture_method)
post
end
def add_confirmation_method(post, options)
confirmation_method = options[:confirmation_method].to_s
if ALLOWED_METHOD_STATES.include?(confirmation_method)
post[:confirmation_method] = confirmation_method
end
post
end
def add_customer(post, options)
customer = options[:customer].to_s
post[:customer] = customer if customer.start_with?('cus_')
post
end
def add_return_url(post, options)
return unless options[:confirm]
post[:confirm] = options[:confirm]
post[:return_url] = options[:return_url] if options[:return_url]
post
end
def add_payment_method_token(post, payment_method, options)
return if payment_method.nil?
if payment_method.is_a?(ActiveMerchant::Billing::CreditCard)
p = create_payment_method(payment_method, options)
payment_method = p.params['id']
end
if payment_method.is_a?(StripePaymentToken)
post[:payment_method] = payment_method.payment_data['id']
elsif payment_method.is_a?(String)
if payment_method.include?('|')
customer_id, payment_method_id = payment_method.split('|')
token = payment_method_id
post[:customer] = customer_id
else
token = payment_method
end
post[:payment_method] = token
end
end
def add_payment_method_types(post, options)
payment_method_types = options[:payment_method_types] if options[:payment_method_types]
return if payment_method_types.nil?
post[:payment_method_types] = Array(payment_method_types)
post
end
def setup_future_usage(post, options = {})
if %w(on_session off_session).include?(options[:setup_future_usage])
post[:setup_future_usage] = options[:setup_future_usage]
end
if options[:off_session] && options[:confirm] == true
post[:off_session] = options[:off_session]
end
post
end
def add_connected_account(post, options = {})
return unless transfer_data = options[:transfer_data]
post[:transfer_data] = {}
if transfer_data[:destination]
post[:transfer_data][:destination] = transfer_data[:destination]
end
post[:transfer_data][:amount] = transfer_data[:amount] if transfer_data[:amount]
post[:on_behalf_of] = options[:on_behalf_of] if options[:on_behalf_of]
post[:transfer_group] = options[:transfer_group] if options[:transfer_group]
post[:application_fee_amount] = options[:application_fee] if options[:application_fee]
post
end
def add_shipping_address(post, options = {})
return unless shipping = options[:shipping]
post[:shipping] = {}
post[:shipping][:address] = {}
post[:shipping][:address][:line1] = shipping[:address][:line1]
post[:shipping][:address][:city] = shipping[:address][:city] if shipping[:address][:city]
if shipping[:address][:country]
post[:shipping][:address][:country] = shipping[:address][:country]
end
post[:shipping][:address][:line2] = shipping[:address][:line2] if shipping[:address][:line2]
if shipping[:address][:postal_code]
post[:shipping][:address][:postal_code] = shipping[:address][:postal_code]
end
post[:shipping][:address][:state] = shipping[:address][:state] if shipping[:address][:state]
post[:shipping][:name] = shipping[:name]
post[:shipping][:carrier] = shipping[:carrier] if shipping[:carrier]
post[:shipping][:phone] = shipping[:phone] if shipping[:phone]
post[:shipping][:tracking_number] = shipping[:tracking_number] if shipping[:tracking_number]
post
end
end
end
end

View File

@@ -0,0 +1,86 @@
# frozen_string_literal: true
ActiveMerchant::Billing::StripeGateway::StripePaymentIntentsGateway.class_eval do
def create_intent(money, payment_method, options = {})
post = {}
add_amount(post, money, options, true)
add_capture_method(post, options)
add_confirmation_method(post, options)
add_customer(post, options)
add_payment_method_token(post, payment_method, options)
add_metadata(post, options)
add_return_url(post, options)
add_connected_account(post, options)
add_shipping_address(post, options)
setup_future_usage(post, options)
CREATE_INTENT_ATTRIBUTES.each do |attribute|
add_whitelisted_attribute(post, options, attribute)
end
commit(:post, 'payment_intents', post, options)
end
def capture(money, intent_id, options = {})
post = {}
post[:amount_to_capture] = money
add_connected_account(post, options)
commit(:post, "payment_intents/#{intent_id}/capture", post, options)
end
def refund(money, intent_id, options = {})
intent = commit(:get, "payment_intents/#{intent_id}", nil, options)
charge_id = intent.params.dig('charges', 'data')[0].dig('id')
super(money, charge_id, options)
end
# Note: Not all payment methods are currently supported by the
# {Payment Methods API}[https://stripe.com/docs/payments/payment-methods]
# Current implementation will create
# a PaymentMethod object if the method is a token or credit card
# All other types will default to legacy Stripe store
def store(payment_method, options = {})
params = {}
post = {}
# If customer option is provided, create a payment method and attach to customer id
# Otherwise, create a customer, then attach
# if payment_method.is_a?(StripePaymentToken) ||
# payment_method.is_a?(ActiveMerchant::Billing::CreditCard)
add_payment_method_token(params, payment_method, options)
if options[:customer]
customer_id = options[:customer]
else
post[:validate] = options[:validate] unless options[:validate].nil?
post[:description] = options[:description] if options[:description]
post[:email] = options[:email] if options[:email]
customer = commit(:post, 'customers', post, options)
customer_id = customer.params['id']
# return the stripe response if expected customer id is not present
return customer if customer_id.nil?
end
commit(:post,
"payment_methods/#{params[:payment_method]}/attach",
{ customer: customer_id }, options)
# else
# super(payment, options)
# end
end
private
def add_connected_account(post, options = {})
return unless transfer_data = options[:transfer_data]
post[:transfer_data] = {}
if transfer_data[:destination]
post[:transfer_data][:destination] = transfer_data[:destination]
end
post[:transfer_data][:amount] = transfer_data[:amount] if transfer_data[:amount]
post[:on_behalf_of] = options[:on_behalf_of] if options[:on_behalf_of]
post[:transfer_group] = options[:transfer_group] if options[:transfer_group]
post[:application_fee_amount] = options[:application_fee] if options[:application_fee]
post
end
end