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:

src/modules/my-payment/service.ts
1import { AbstractPaymentProvider } from "@medusajs/framework/utils"2
3type Options = {4  apiKey: string5}6
7class MyPaymentProviderService extends AbstractPaymentProvider<8  Options9> {10  // TODO implement methods11}12
13export default MyPaymentProviderService

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.

NoteA module's options are passed when you register it in the Medusa application.

Example

Code
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

TConfigobjectOptional
The type of the provider's options passed as a second parameter.

Parameters

cradleRecord<string, unknown>
The module's container cradle used to resolve resources.

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

Code
1class MyPaymentProviderService extends AbstractPaymentProvider<2  Options3> {4  static identifier = "my-payment"5  // ...6}

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

Code
1class MyPaymentProviderService extends AbstractPaymentProvider<Options> {2  static validateOptions(options: Record<any, any>) {3    if (!options.apiKey) {4      throw new MedusaError(5        MedusaError.Types.INVALID_DATA,6        "API key is required in the provider's options."7      )8    }9  }10}

Parameters

optionsRecord<any, any>
The provider's options.

Returns

voidvoid
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.

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

Code
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

paymentDataRecord<string, unknown>
The data property of the payment. Make sure to store in it any helpful identification for your third-party integration.

Returns

PromisePromise<Record<string, unknown> | PaymentProviderError>
The new data to store in the payment's 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

Code
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

paymentSessionDataRecord<string, unknown>
The data property of the payment session. Make sure to store in it any helpful identification for your third-party integration.
contextRecord<string, unknown>
The context in which the payment is being authorized. For example, in checkout, the context has a cart_id property indicating the ID of the associated cart.

Returns

PromisePromise<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.

cancelPayment#

This method cancels a payment.

Example

Code
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

paymentDataRecord<string, unknown>
The data property of the payment. Make sure to store in it any helpful identification for your third-party integration.

Returns

PromisePromise<Record<string, unknown> | PaymentProviderError>
An error object if an error occurs, or the data received from the integration.

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

Code
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

contextCreatePaymentProviderSession
The details of the payment session and its context.

Returns

PromisePromise<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.

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

Code
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

paymentSessionDataRecord<string, unknown>
The data property of the payment session. Make sure to store in it any helpful identification for your third-party integration.

Returns

PromisePromise<Record<string, unknown> | PaymentProviderError>
An error object or the response from the third-party service.

getPaymentStatus#

This method gets the status of a payment session based on the status in the third-party integration.

Example

Code
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

paymentSessionDataRecord<string, unknown>
The data property of the payment session. Make sure to store in it any helpful identification for your third-party integration.

Returns

PromisePromise<PaymentSessionStatus>
The payment session's status.

refundPayment#

This method refunds an amount of a payment previously captured.

Example

Code
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

paymentDataRecord<string, unknown>
The data property of the payment. Make sure to store in it any helpful identification for your third-party integration.
refundAmountnumber
The amount to refund.

Returns

PromisePromise<Record<string, unknown> | PaymentProviderError>
The new data to store in the payment's data property, or an error object.

retrievePayment#

Retrieves the payment's data from the third-party service.

Example

Code
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

paymentSessionDataRecord<string, unknown>
The data property of the payment. Make sure to store in it any helpful identification for your third-party integration.

Returns

PromisePromise<Record<string, unknown> | PaymentProviderError>
An object to be stored in the payment's data property, or an error object.

updatePayment#

Update a payment in the third-party service that was previously initiated with the initiatePayment method.

Example

Code
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

contextUpdatePaymentProviderSession
The details of the payment session and its context.

Returns

PromisePromise<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.

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

Code
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

dataobject
The webhook event's data

Returns

PromisePromise<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.

3. Create Module Definition File#

Create the file src/modules/my-payment/index.ts with the following content:

src/modules/my-payment/index.ts
1import MyPaymentProviderService from "./service"2import { 3  ModuleProvider, 4  Modules5} from "@medusajs/framework/utils"6
7export default ModuleProvider(Modules.PAYMENT, {8  services: [MyPaymentProviderService],9})

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:

medusa-config.ts
1module.exports = defineConfig({2  // ...3  modules: [4    {5      resolve: "@medusajs/medusa/payment",6      options: {7        providers: [8          {9            resolve: "./src/modules/my-payment",10            id: "my-payment",11            options: {12              // provider options...13              apiKey: "..."14            }15          }16        ]17      }18    }19  ]20})

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.


Useful Guides#

Was this page helpful?