- Get Started
- Product
- Resources
- Tools & SDKs
- Framework
- Reference
- Get Started
- Product
- Resources
- Tools & SDKs
- Framework
- Reference
How to Create a Payment Provider
In this document, you’ll learn how to create a Payment Provider to be used with the Payment Module.
1. Create Module Directory#
Start by creating a new directory for your module. For example, src/modules/my-payment
.
2. Create the Payment Provider Service#
Create the file src/modules/my-payment/service.ts
that holds the module's main service. It must extend the AbstractPaymentProvider
class imported from @medusajs/framework/utils
:
constructor#
The constructor allows you to access resources from the module's container using the first parameter, and the module's options using the second parameter.
Example
1import { AbstractPaymentProvider } from "@medusajs/framework/utils"2import { Logger } from "@medusajs/framework/types"3 4type Options = {5 apiKey: string6}7 8type InjectedDependencies = {9 logger: Logger10}11 12class MyPaymentProviderService extends AbstractPaymentProvider<Options> {13 protected logger_: Logger14 protected options_: Options15 // assuming you're initializing a client16 protected client17 18 constructor(19 container: InjectedDependencies,20 options: Options21 ) {22 super(container, options)23 24 this.logger_ = container.logger25 this.options_ = options26 27 // TODO initialize your client28 }29 // ...30}31 32export default MyPaymentProviderService
Type Parameters
TConfig
objectOptionalParameters
cradle
Record<string, unknown>identifier#
Each payment provider has a unique identifier defined in its class. The provider's ID
will be stored as pp_{identifier}_{id}
, where {id}
is the provider's id
property in the medusa-config.ts
.
Example
validateOptions#
This method validates the options of the provider set in medusa-config.ts
.
Implementing this method is optional. It's useful if your provider requires custom validation.
If the options aren't valid, throw an error.
Example
Parameters
options
Record<any, any>Returns
void
voidmedusa-config.ts
.
Implementing this method is optional. It's useful if your provider requires custom validation.
If the options aren't valid, throw an error.capturePayment#
This method is used to capture a payment. The payment is captured in one of the following scenarios:
- The authorizePayment method returns the status
captured
, which automatically executed this method after authorization. - The merchant requests to capture the payment after its associated payment session was authorized.
- A webhook event occurred that instructs the payment provider to capture the payment session. Learn more about handing webhook events in this guide.
In this method, use the third-party provider to capture the payment.
Example
1// other imports...2import {3 PaymentProviderError,4 PaymentProviderSessionResponse,5} from "@medusajs/framework/types"6 7class MyPaymentProviderService extends AbstractPaymentProvider<8 Options9> {10 async capturePayment(11 paymentData: Record<string, unknown>12 ): Promise<PaymentProviderError | PaymentProviderSessionResponse["data"]> {13 const externalId = paymentData.id14 15 try {16 // assuming you have a client that captures the payment17 const newData = await this.client.capturePayment(externalId)18 19 return {20 ...newData,21 id: externalId22 }23 } catch (e) {24 return {25 error: e,26 code: "unknown",27 detail: e28 }29 }30 }31 32 // ...33}
Parameters
paymentData
Record<string, unknown>data
property of the payment. Make sure to store in it
any helpful identification for your third-party integration.Returns
Promise
Promise<Record<string, unknown> | PaymentProviderError>The new data to store in the payment's data
property, or an error object.
Promise
Promise<Record<string, unknown> | PaymentProviderError>data
property, or an error object.authorizePayment#
This method authorizes a payment session. When authorized successfully, a payment is created by the Payment Module which can be later captured using the capturePayment method.
Refer to this guide to learn more about how this fits into the payment flow and how to handle required actions.
To automatically capture the payment after authorization, return the status captured
.
Example
1// other imports...2import {3 PaymentProviderError,4 PaymentProviderSessionResponse,5 PaymentSessionStatus6} from "@medusajs/framework/types"7 8class MyPaymentProviderService extends AbstractPaymentProvider<9 Options10> {11 async authorizePayment(12 paymentSessionData: Record<string, unknown>,13 context: Record<string, unknown>14 ): Promise<15 PaymentProviderError | {16 status: PaymentSessionStatus17 data: PaymentProviderSessionResponse["data"]18 }19 > {20 const externalId = paymentSessionData.id21 22 try {23 // assuming you have a client that authorizes the payment24 const paymentData = await this.client.authorizePayment(externalId)25 26 return {27 data: {28 ...paymentData,29 id: externalId30 },31 status: "authorized"32 }33 } catch (e) {34 return {35 error: e,36 code: "unknown",37 detail: e38 }39 }40 }41 42 // ...43}
Parameters
paymentSessionData
Record<string, unknown>data
property of the payment session. Make sure to store in it
any helpful identification for your third-party integration.context
Record<string, unknown>cart_id
property indicating the ID of the associated cart.Returns
Promise
Promise<PaymentProviderError | object>Either an object of the new data to store in the created payment's data
property and the
payment's status, or an error object. Make sure to set in data
anything useful to later retrieve the session.
Promise
Promise<PaymentProviderError | object>data
property and the
payment's status, or an error object. Make sure to set in data
anything useful to later retrieve the session.cancelPayment#
This method cancels a payment.
Example
1// other imports...2import {3 PaymentProviderError,4 PaymentProviderSessionResponse,5} from "@medusajs/framework/types"6 7class MyPaymentProviderService extends AbstractPaymentProvider<8 Options9> {10 async cancelPayment(11 paymentData: Record<string, unknown>12 ): Promise<PaymentProviderError | PaymentProviderSessionResponse["data"]> {13 const externalId = paymentData.id14 15 try {16 // assuming you have a client that cancels the payment17 const paymentData = await this.client.cancelPayment(externalId)18 } catch (e) {19 return {20 error: e,21 code: "unknown",22 detail: e23 }24 }25 }26 27 // ...28}
Parameters
paymentData
Record<string, unknown>data
property of the payment. Make sure to store in it
any helpful identification for your third-party integration.Returns
Promise
Promise<Record<string, unknown> | PaymentProviderError>An error object if an error occurs, or the data received from the integration.
Promise
Promise<Record<string, unknown> | PaymentProviderError>initiatePayment#
This method is used when a payment session is created. It can be used to initiate the payment in the third-party session, before authorizing or capturing the payment later.
Example
1// other imports...2import {3 PaymentProviderError,4 PaymentProviderSessionResponse,5} from "@medusajs/framework/types"6 7class MyPaymentProviderService extends AbstractPaymentProvider<8 Options9> {10 async initiatePayment(11 context: CreatePaymentProviderSession12 ): Promise<PaymentProviderError | PaymentProviderSessionResponse> {13 const {14 amount,15 currency_code,16 context: customerDetails17 } = context18 19 try {20 // assuming you have a client that initializes the payment21 const response = await this.client.init(22 amount, currency_code, customerDetails23 )24 25 return {26 ...response,27 data: {28 id: response.id29 }30 }31 } catch (e) {32 return {33 error: e,34 code: "unknown",35 detail: e36 }37 }38 }39 40 // ...41}
Parameters
context
CreatePaymentProviderSessionReturns
Promise
Promise<PaymentProviderError | PaymentProviderSessionResponse>An object whose data
property is set in the created payment session, or an error
object. Make sure to set in data
anything useful to later retrieve the session.
Promise
Promise<PaymentProviderError | PaymentProviderSessionResponse>data
property is set in the created payment session, or an error
object. Make sure to set in data
anything useful to later retrieve the session.deletePayment#
This method is used when a payment session is deleted, which can only happen if it isn't authorized, yet.
Use this to delete or cancel the payment in the third-party service.
Example
1// other imports...2import {3 PaymentProviderError,4 PaymentProviderSessionResponse,5} from "@medusajs/framework/types"6 7class MyPaymentProviderService extends AbstractPaymentProvider<8 Options9> {10 async deletePayment(11 paymentSessionData: Record<string, unknown>12 ): Promise<13 PaymentProviderError | PaymentProviderSessionResponse["data"]14 > {15 const externalId = paymentSessionData.id16 17 try {18 // assuming you have a client that cancels the payment19 await this.client.cancelPayment(externalId)20 } catch (e) {21 return {22 error: e,23 code: "unknown",24 detail: e25 }26 }27 }28 29 // ...30}
Parameters
paymentSessionData
Record<string, unknown>data
property of the payment session. Make sure to store in it
any helpful identification for your third-party integration.Returns
Promise
Promise<Record<string, unknown> | PaymentProviderError>An error object or the response from the third-party service.
Promise
Promise<Record<string, unknown> | PaymentProviderError>getPaymentStatus#
This method gets the status of a payment session based on the status in the third-party integration.
Example
1// other imports...2import {3 PaymentSessionStatus4} from "@medusajs/framework/types"5 6class MyPaymentProviderService extends AbstractPaymentProvider<7 Options8> {9 async getPaymentStatus(10 paymentSessionData: Record<string, unknown>11 ): Promise<PaymentSessionStatus> {12 const externalId = paymentSessionData.id13 14 try {15 // assuming you have a client that retrieves the payment status16 const status = await this.client.getStatus(externalId)17 18 switch (status) {19 case "requires_capture":20 return "authorized"21 case "success":22 return "captured"23 case "canceled":24 return "canceled"25 default:26 return "pending"27 }28 } catch (e) {29 return "error"30 }31 }32 33 // ...34}
Parameters
paymentSessionData
Record<string, unknown>data
property of the payment session. Make sure to store in it
any helpful identification for your third-party integration.Returns
Promise
Promise<PaymentSessionStatus>The payment session's status.
Promise
Promise<PaymentSessionStatus>refundPayment#
This method refunds an amount of a payment previously captured.
Example
1// other imports...2import {3 PaymentProviderError,4 PaymentProviderSessionResponse,5} from "@medusajs/framework/types"6 7class MyPaymentProviderService extends AbstractPaymentProvider<8 Options9> {10 async refundPayment(11 paymentData: Record<string, unknown>,12 refundAmount: number13 ): Promise<14 PaymentProviderError | PaymentProviderSessionResponse["data"]15 > {16 const externalId = paymentData.id17 18 try {19 // assuming you have a client that refunds the payment20 const newData = await this.client.refund(21 externalId,22 refundAmount23 )24 25 return {26 ...newData,27 id: externalId28 }29 } catch (e) {30 return {31 error: e,32 code: "unknown",33 detail: e34 }35 }36 }37 38 // ...39}
Parameters
paymentData
Record<string, unknown>data
property of the payment. Make sure to store in it
any helpful identification for your third-party integration.refundAmount
numberReturns
Promise
Promise<Record<string, unknown> | PaymentProviderError>The new data to store in the payment's data
property, or an error object.
Promise
Promise<Record<string, unknown> | PaymentProviderError>data
property, or an error object.retrievePayment#
Retrieves the payment's data from the third-party service.
Example
1// other imports...2import {3 PaymentProviderError,4 PaymentProviderSessionResponse,5} from "@medusajs/framework/types"6 7class MyPaymentProviderService extends AbstractPaymentProvider<8 Options9> {10 async retrievePayment(11 paymentSessionData: Record<string, unknown>12 ): Promise<13 PaymentProviderError | PaymentProviderSessionResponse["data"]14 > {15 const externalId = paymentSessionData.id16 17 try {18 // assuming you have a client that retrieves the payment19 return await this.client.retrieve(externalId)20 } catch (e) {21 return {22 error: e,23 code: "unknown",24 detail: e25 }26 }27 }28 29 // ...30}
Parameters
paymentSessionData
Record<string, unknown>data
property of the payment. Make sure to store in it
any helpful identification for your third-party integration.Returns
Promise
Promise<Record<string, unknown> | PaymentProviderError>An object to be stored in the payment's data
property, or an error object.
Promise
Promise<Record<string, unknown> | PaymentProviderError>data
property, or an error object.updatePayment#
Update a payment in the third-party service that was previously initiated with the initiatePayment method.
Example
1// other imports...2import {3 UpdatePaymentProviderSession,4 PaymentProviderError,5 PaymentProviderSessionResponse,6} from "@medusajs/framework/types"7 8class MyPaymentProviderService extends AbstractPaymentProvider<9 Options10> {11 async updatePayment(12 context: UpdatePaymentProviderSession13 ): Promise<PaymentProviderError | PaymentProviderSessionResponse> {14 const {15 amount,16 currency_code,17 context: customerDetails,18 data19 } = context20 const externalId = data.id21 22 try {23 // assuming you have a client that updates the payment24 const response = await this.client.update(25 externalId,26 {27 amount,28 currency_code,29 customerDetails30 }31 )32 33 return {34 ...response,35 data: {36 id: response.id37 }38 }39 } catch (e) {40 return {41 error: e,42 code: "unknown",43 detail: e44 }45 }46 }47 48 // ...49}
Parameters
context
UpdatePaymentProviderSessionReturns
Promise
Promise<PaymentProviderError | PaymentProviderSessionResponse>An object whose data
property is set in the updated payment session, or an error
object. Make sure to set in data
anything useful to later retrieve the session.
Promise
Promise<PaymentProviderError | PaymentProviderSessionResponse>data
property is set in the updated payment session, or an error
object. Make sure to set in data
anything useful to later retrieve the session.listPaymentMethods#
List the payment methods associated with the context (eg. customer) of the payment provider, if any.
Example
1// other imports...2import {3 PaymentProviderContext,4 PaymentProviderError,5 PaymentMethodResponse6 PaymentProviderSessionResponse,7} from "@medusajs/framework/types"8 9class MyPaymentProviderService extends AbstractPaymentProvider<10 Options11> {12 async listPaymentMethods(13 context: PaymentProviderContext14 ): Promise<PaymentMethodResponse> {15 const {16 customer,17 } = context18 const externalCustomerId = customer.metadata.stripe_id19 20 try {21 // assuming you have a client that updates the payment22 const response = await this.client.listPaymentMethods(23 {customer: externalCustomerId}24 )25 26 return response.map((method) => ({27 id: method.id,28 data: method29 }))30 } catch (e) {31 return {32 error: e,33 code: "unknown",34 detail: e35 }36 }37 }38 39 // ...40}
Parameters
context
PaymentProviderContextThe context for which the payment methods are listed. Usually the customer should be provided.
context
PaymentProviderContextReturns
Promise
Promise<PaymentMethodResponse[]>An object whose payment_methods
property is set to the data returned by the payment provider.
Promise
Promise<PaymentMethodResponse[]>payment_methods
property is set to the data returned by the payment provider.getWebhookActionAndData#
This method is executed when a webhook event is received from the third-party payment provider. Use it to process the action of the payment provider.
Learn more in this documentation
Example
1// other imports...2import {3 BigNumber4} from "@medusajs/framework/utils"5import {6 ProviderWebhookPayload,7 WebhookActionResult8} from "@medusajs/framework/types"9 10class MyPaymentProviderService extends AbstractPaymentProvider<11 Options12> {13 async getWebhookActionAndData(14 payload: ProviderWebhookPayload["payload"]15 ): Promise<WebhookActionResult> {16 const {17 data,18 rawData,19 headers20 } = payload21 22 try {23 switch(data.event_type) {24 case "authorized_amount":25 return {26 action: "authorized",27 data: {28 session_id: (data.metadata as Record<string, any>).session_id,29 amount: new BigNumber(data.amount as number)30 }31 }32 case "success":33 return {34 action: "captured",35 data: {36 session_id: (data.metadata as Record<string, any>).session_id,37 amount: new BigNumber(data.amount as number)38 }39 }40 default:41 return {42 action: "not_supported"43 }44 }45 } catch (e) {46 return {47 action: "failed",48 data: {49 session_id: (data.metadata as Record<string, any>).session_id,50 amount: new BigNumber(data.amount as number)51 }52 }53 }54 }55 56 // ...57}
Parameters
data
objectThe webhook event's data
data
objectReturns
Promise
Promise<WebhookActionResult>The webhook result. If the action
's value is captured
, the payment is captured within Medusa as well.
If the action
's value is authorized
, the associated payment session is authorized within Medusa.
Promise
Promise<WebhookActionResult>action
's value is captured
, the payment is captured within Medusa as well.
If the action
's value is authorized
, the associated payment session is authorized within Medusa.3. Create Module Definition File#
Create the file src/modules/my-payment/index.ts
with the following content:
This exports the module's definition, indicating that the MyPaymentProviderService
is the module's service.
4. Use Module#
To use your Payment Module Provider, add it to the providers
array of the Payment Module in medusa-config.ts
:
5. Test it Out#
Before you use your payment provider, enable it in a region using the Medusa Admin.
Then, go through checkout to place an order. Your payment provider is used to authorize the payment.