From af35e15b28bb3e41d3a138ed82508fc13879f985 Mon Sep 17 00:00:00 2001 From: shij Date: Tue, 19 Nov 2024 11:43:48 +0800 Subject: [PATCH] feat(payment-stripe): support stripe invoice --- .../payment-stripe/src/core/stripe-base.ts | 264 +++++++++++++----- .../providers/payment-stripe/src/index.ts | 2 + .../payment-stripe/src/services/index.ts | 1 + .../src/services/stripe-invoice.ts | 18 ++ .../payment-stripe/src/types/index.ts | 1 + 5 files changed, 216 insertions(+), 70 deletions(-) create mode 100644 packages/modules/providers/payment-stripe/src/services/stripe-invoice.ts diff --git a/packages/modules/providers/payment-stripe/src/core/stripe-base.ts b/packages/modules/providers/payment-stripe/src/core/stripe-base.ts index a210e4e9d1622..b1e4a69f58ccb 100644 --- a/packages/modules/providers/payment-stripe/src/core/stripe-base.ts +++ b/packages/modules/providers/payment-stripe/src/core/stripe-base.ts @@ -73,87 +73,196 @@ abstract class StripeBase extends AbstractPaymentProvider { async initiatePayment( data: CreatePaymentProviderSession ): Promise { - const intentRequestData = this.getPaymentIntentOptions() + const { currency_code, amount, token, context } = data const { billing_address, shipping_address, email, customer, payment_description, + invoice, + invoiceItems, ...metadata - } = data.context - const { currency_code, amount, token } = data - - const description = (payment_description ?? - this.options_?.paymentDescription) as string - - const intentRequest: Stripe.PaymentIntentCreateParams = { - description, - amount: getSmallestUnit(amount, currency_code), - currency: currency_code, - payment_method: token, - confirm: !!token, - shipping: - shipping_address || customer - ? { - address: { - city: shipping_address?.city ?? undefined, - country: shipping_address?.country_code ?? undefined, - line1: shipping_address?.address_1 ?? undefined, - line2: shipping_address?.address_2 ?? undefined, - postal_code: shipping_address?.postal_code ?? undefined, - state: shipping_address?.province ?? undefined, - }, - name: `${customer?.first_name ?? ""} ${ - customer?.last_name ?? "" - }`.trim(), - phone: shipping_address?.phone ?? undefined, + } = context + + const customerName = `${customer?.first_name ?? ""} ${ + customer?.last_name ?? "" + }`.trim() + const customerPhone = customer?.phone ?? "" + const intentOptions = this.getPaymentIntentOptions() + + let stripeCustomerId: string | undefined + let shipping: + | { + address: Stripe.AddressParam + name: string + phone?: string + } + | undefined + + if (customer?.metadata?.stripe_id) { + stripeCustomerId = customer.metadata.stripe_id as string + } else if (this.options_.createCustomer || invoice) { + try { + if (customer?.email) { + const list = await this.stripe_.customers.list({ + email: customer.email, + limit: 100, + }) + for (let i = 0; i < list.data.length; i++) { + const stripeCustomer = list.data[i] + if ( + customer.id && + customer.id === stripeCustomer.metadata.medusa_id + ) { + stripeCustomerId = stripeCustomer.id + if ( + (stripeCustomer.name ?? "") !== customerName || + (stripeCustomer.phone ?? "") !== customerPhone + ) { + await this.stripe_.customers.update(stripeCustomerId, { + name: customerName, + phone: customerPhone, + }) + } + break } - : undefined, - metadata: metadata as Stripe.MetadataParam, - capture_method: this.options_.capture ? "automatic" : "manual", - expand: ["latest_charge", "payment_method"], - ...intentRequestData, + } + } + + if (!stripeCustomerId) { + const stripeCustomer = await this.stripe_.customers.create({ + email: customer?.email, + name: customerName, + phone: customerPhone, + metadata: customer?.id && { medusa_id: customer.id }, + }) + stripeCustomerId = stripeCustomer.id + } + } catch (e) { + return this.buildError( + "An error occurred in initiatePayment when creating a Stripe customer", + e + ) + } } - const automaticPaymentMethods = this.options_?.automaticPaymentMethods - if (automaticPaymentMethods) { - intentRequest.automatic_payment_methods = - typeof automaticPaymentMethods === "boolean" - ? { enabled: true } - : automaticPaymentMethods + if (shipping_address || customer) { + shipping = { + address: { + city: shipping_address?.city ?? undefined, + country: shipping_address?.country_code ?? undefined, + line1: shipping_address?.address_1 ?? undefined, + line2: shipping_address?.address_2 ?? undefined, + postal_code: shipping_address?.postal_code ?? undefined, + state: shipping_address?.province ?? undefined, + }, + name: customerName, + phone: shipping_address?.phone || customerPhone, + } } - if (customer?.metadata?.stripe_id) { - intentRequest.customer = customer.metadata.stripe_id as string - } else if (this.options_.createCustomer) { - let stripeCustomer + if (invoice) { try { - stripeCustomer = await this.stripe_.customers.create({ - email, + let stripeInvoice = await this.stripe_.invoices.create({ + ...(typeof invoice === "boolean" ? {} : invoice), + auto_advance: false, + currency: currency_code, + customer: stripeCustomerId, + shipping_details: shipping, + payment_settings: { + payment_method_types: + intentOptions.payment_method_types as Stripe.InvoiceCreateParams.PaymentSettings.PaymentMethodType[], + }, }) + + if (invoiceItems) { + await Promise.all( + (invoiceItems as Stripe.InvoiceItemCreateParams[]).map((params) => + this.stripe_.invoiceItems.create({ + ...params, + customer: stripeCustomerId!, + invoice: stripeInvoice.id, + currency: currency_code, + amount: + params.amount && + getSmallestUnit(params.amount, currency_code), + unit_amount: + params.unit_amount && + getSmallestUnit(params.unit_amount, currency_code), + unit_amount_decimal: + params.unit_amount_decimal && + getSmallestUnit(params.unit_amount_decimal, currency_code) + + "", + }) + ) + ) + } + + stripeInvoice = await this.stripe_.invoices.finalizeInvoice( + stripeInvoice.id, + { auto_advance: false } + ) + + const intent = await this.stripe_.paymentIntents.update( + stripeInvoice.payment_intent as string, + { + shipping, + metadata: metadata as Stripe.MetadataParam, + description: + payment_description ?? this.options_.paymentDescription, + setup_future_usage: intentOptions.setup_future_usage, + expand: ["latest_charge", "payment_method", "invoice"], + } + ) + return { + ...(await this.buildResponse(intent)), + data: intent as unknown as Record, + context: intent.metadata, + } } catch (e) { return this.buildError( - "An error occurred in initiatePayment when creating a Stripe customer", + "An error occurred in initiatePayment during the creation of the stripe invoice", e ) } + } else { + const createParams: Stripe.PaymentIntentCreateParams = { + ...intentOptions, + amount: getSmallestUnit(amount, currency_code), + currency: currency_code, + payment_method: token, + confirm: !!token, + customer: stripeCustomerId, + shipping, + metadata: metadata as Stripe.MetadataParam, + description: payment_description ?? this.options_.paymentDescription, + expand: ["latest_charge", "payment_method"], + capture_method: + typeof this.options_.capture === "boolean" + ? this.options_.capture + ? "automatic" + : "manual" + : intentOptions.capture_method ?? "automatic", + automatic_payment_methods: + this.options_.automaticPaymentMethods === true + ? { enabled: true } + : { enabled: false, ...this.options_.automaticPaymentMethods }, + } - intentRequest.customer = stripeCustomer.id - } - - try { - const intent = await this.stripe_.paymentIntents.create(intentRequest) - return { - ...(await this.buildResponse(intent)), - data: intent as unknown as Record, - context: intent.metadata, + try { + const intent = await this.stripe_.paymentIntents.create(createParams) + return { + ...(await this.buildResponse(intent)), + data: intent as unknown as Record, + context: intent.metadata, + } + } catch (e) { + return this.buildError( + "An error occurred in initiatePayment during the creation of the stripe payment intent", + e + ) } - } catch (e) { - return this.buildError( - "An error occurred in initiatePayment during the creation of the stripe payment intent", - e - ) } } @@ -164,7 +273,7 @@ abstract class StripeBase extends AbstractPaymentProvider { try { const intent = await this.stripe_.paymentIntents.confirm(id, { payment_method: data.token, - expand: ["latest_charge", "payment_method"], + expand: ["latest_charge", "payment_method", "invoice"], }) return { ...(await this.buildResponse(intent)), @@ -182,11 +291,24 @@ abstract class StripeBase extends AbstractPaymentProvider { async cancelPayment( paymentSessionData: PaymentProviderSessionResponse["data"] ): Promise { - const { id } = paymentSessionData as unknown as Stripe.PaymentIntent + const { id, invoice } = + paymentSessionData as unknown as Stripe.PaymentIntent try { - const intent = await this.stripe_.paymentIntents.cancel(id, { - expand: ["latest_charge", "payment_method"], - }) + let intent: Stripe.PaymentIntent + + if (invoice) { + await this.stripe_.invoices.voidInvoice( + typeof invoice === "string" ? invoice : invoice.id + ) + intent = await this.stripe_.paymentIntents.retrieve(id, { + expand: ["latest_charge", "payment_method", "invoice"], + }) + } else { + intent = await this.stripe_.paymentIntents.cancel(id, { + expand: ["latest_charge", "payment_method"], + }) + } + return { ...(await this.buildResponse(intent)), data: intent as unknown as Record, @@ -262,7 +384,7 @@ abstract class StripeBase extends AbstractPaymentProvider { : undefined, }) const intent = await this.stripe_.paymentIntents.retrieve(id, { - expand: ["latest_charge", "payment_method"], + expand: ["latest_charge", "payment_method", "invoice"], }) return { ...(await this.buildResponse(intent)), @@ -280,7 +402,7 @@ abstract class StripeBase extends AbstractPaymentProvider { const { id } = paymentSessionData as unknown as Stripe.PaymentIntent try { const intent = await this.stripe_.paymentIntents.retrieve(id, { - expand: ["latest_charge", "payment_method"], + expand: ["latest_charge", "payment_method", "invoice"], }) return { ...(await this.buildResponse(intent)), @@ -307,6 +429,8 @@ abstract class StripeBase extends AbstractPaymentProvider { email, customer, payment_description, + invoice, + invoiceItems, ...metadata } = context if (payment_description) { @@ -351,7 +475,7 @@ abstract class StripeBase extends AbstractPaymentProvider { } const intent = await this.stripe_.paymentIntents.update(id, { ...updateParams, - expand: ["latest_charge", "payment_method"], + expand: ["latest_charge", "payment_method", "invoice"], }) return { ...(await this.buildResponse(intent)), @@ -372,7 +496,7 @@ abstract class StripeBase extends AbstractPaymentProvider { if (event.data.object.object === "payment_intent") { intent = await this.stripe_.paymentIntents.retrieve( event.data.object.id, - { expand: ["latest_charge", "payment_method"] } + { expand: ["latest_charge", "payment_method", "invoice"] } ) } else if (event.data.object.object === "charge") { if (!event.data.object.payment_intent) { @@ -382,7 +506,7 @@ abstract class StripeBase extends AbstractPaymentProvider { } intent = await this.stripe_.paymentIntents.retrieve( event.data.object.payment_intent as string, - { expand: ["latest_charge", "payment_method"] } + { expand: ["latest_charge", "payment_method", "invoice"] } ) } else if (event.data.object.object === "invoice") { // TODO diff --git a/packages/modules/providers/payment-stripe/src/index.ts b/packages/modules/providers/payment-stripe/src/index.ts index 5d7614763e44c..f22decae1c8f7 100644 --- a/packages/modules/providers/payment-stripe/src/index.ts +++ b/packages/modules/providers/payment-stripe/src/index.ts @@ -6,6 +6,7 @@ import { StripeIdealService, StripeProviderService, StripePrzelewy24Service, + StripeInvoiceService, } from "./services" const services = [ @@ -15,6 +16,7 @@ const services = [ StripeIdealService, StripeProviderService, StripePrzelewy24Service, + StripeInvoiceService, ] const providerExport: ModuleProviderExports = { diff --git a/packages/modules/providers/payment-stripe/src/services/index.ts b/packages/modules/providers/payment-stripe/src/services/index.ts index 6a3ccaeb5a56d..f59b9cd422f2c 100644 --- a/packages/modules/providers/payment-stripe/src/services/index.ts +++ b/packages/modules/providers/payment-stripe/src/services/index.ts @@ -4,3 +4,4 @@ export { default as StripeGiropayService } from "./stripe-giropay" export { default as StripeIdealService } from "./stripe-ideal" export { default as StripeProviderService } from "./stripe-provider" export { default as StripePrzelewy24Service } from "./stripe-przelewy24" +export { default as StripeInvoiceService } from "./stripe-invoice" diff --git a/packages/modules/providers/payment-stripe/src/services/stripe-invoice.ts b/packages/modules/providers/payment-stripe/src/services/stripe-invoice.ts new file mode 100644 index 0000000000000..7877f11d1648a --- /dev/null +++ b/packages/modules/providers/payment-stripe/src/services/stripe-invoice.ts @@ -0,0 +1,18 @@ +import StripeBase from "../core/stripe-base" +import { PaymentIntentOptions, PaymentProviderKeys } from "../types" + +class InvoiceProviderService extends StripeBase { + static PROVIDER = PaymentProviderKeys.INVOICE + + constructor(_, options) { + super(_, options) + } + + get paymentIntentOptions(): PaymentIntentOptions { + return { + payment_method_types: ["card"], + } + } +} + +export default InvoiceProviderService diff --git a/packages/modules/providers/payment-stripe/src/types/index.ts b/packages/modules/providers/payment-stripe/src/types/index.ts index edcd6330034ba..6b37564e3a4d6 100644 --- a/packages/modules/providers/payment-stripe/src/types/index.ts +++ b/packages/modules/providers/payment-stripe/src/types/index.ts @@ -51,4 +51,5 @@ export const PaymentProviderKeys = { GIROPAY: "stripe-giropay", IDEAL: "stripe-ideal", PRZELEWY_24: "stripe-przelewy24", + INVOICE: "stripe-invoice", }