mirror of
https://github.com/openfoodfoundation/openfoodnetwork
synced 2026-01-24 20:36:49 +00:00
86 lines
2.2 KiB
Ruby
86 lines
2.2 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
# When directing a customer to Stripe to authorize the payment, we specify a
|
|
# redirect_url that Stripe should return them to. When checking out, it's
|
|
# /checkout; for admin payments and subscription payemnts it's the order url.
|
|
#
|
|
# This class confirms that the payment intent matches what's in our database,
|
|
# marks the payment as complete, and removes the cvv_response_message field,
|
|
# which we use to indicate that authorization is required. It also completes the
|
|
# Order, if appropriate.
|
|
|
|
class ProcessPaymentIntent
|
|
class Result
|
|
attr_reader :error
|
|
|
|
def initialize(ok:, error: "")
|
|
@ok = ok
|
|
@error = error
|
|
end
|
|
|
|
def ok?
|
|
@ok
|
|
end
|
|
end
|
|
|
|
def initialize(payment_intent, order)
|
|
@payment_intent = payment_intent
|
|
@order = order
|
|
@last_payment = OrderPaymentFinder.new(order).last_payment
|
|
end
|
|
|
|
def call!
|
|
return Result.new(ok: false) unless valid?
|
|
return Result.new(ok: true) if already_processed?
|
|
|
|
# Moves the order to competed state, which calls #process_payments! (and #purchase!)
|
|
# This completes the payment via Stripe and sets the payment's state to completed if successful
|
|
OrderWorkflow.new(order).next
|
|
|
|
if last_payment.reload.completed?
|
|
last_payment.mark_as_processed
|
|
|
|
Result.new(ok: true)
|
|
else
|
|
Result.new(ok: false, error: I18n.t("payment_could_not_complete"))
|
|
end
|
|
|
|
rescue Stripe::StripeError => e
|
|
Result.new(ok: false, error: e.message)
|
|
end
|
|
|
|
private
|
|
|
|
attr_reader :order, :payment_intent, :last_payment
|
|
|
|
def valid?
|
|
order.present? && matches_last_payment? && ready_for_capture?
|
|
end
|
|
|
|
def ready_for_capture?
|
|
payment_intent_status == 'requires_capture'
|
|
end
|
|
|
|
def already_processed?
|
|
payment_intent_status == 'succeeded'
|
|
end
|
|
|
|
def payment_intent_status
|
|
@payment_intent_status ||= Stripe::PaymentIntentValidator.new.
|
|
call(payment_intent, stripe_account_id).
|
|
status
|
|
end
|
|
|
|
def matches_last_payment?
|
|
last_payment&.state == "pending" && last_payment&.response_code == payment_intent
|
|
end
|
|
|
|
def stripe_account_id
|
|
StripeAccount.find_by(enterprise_id: preferred_enterprise_id).stripe_user_id
|
|
end
|
|
|
|
def preferred_enterprise_id
|
|
last_payment.payment_method.preferred_enterprise_id
|
|
end
|
|
end
|