npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2025 – Pkg Stats / Ryan Hefner

@webbers/pay-payments-medusa

v2.2.11

Published

The Pay.nl payment provider plugin for Medusa v2

Readme

Getting Started

Don’t have an account with Pay. yet? Register now!

[!CAUTION] If you have subscribers listening to the order.placed event, update them to listen to the payment.captured event instead. See below.

[!WARNING] This plugin creates orders in Medusa immediately, even if the payment has not yet been captured. If a payment expires, the associated order will be automatically canceled.

This change in flow is required to support asynchronous payment methods (e.g., SprayPay), where payment confirmation can take hours depending on customer input.

Table of Contents

Demo store

Visit the demo store here: https://pay-demo.webbers.com

Pay. payment methods

Card payment methods

  • Mastercard
  • VISA
  • American Express
  • Carte Bancaire
  • Maestro
  • PostePay
  • DanKort
  • Nexi
  • Visa Mastercard

Digital wallets

  • Apple Pay
  • Google Pay

Afterpay methods / Buy now pay later

  • iDEAL IN3
  • Billink
  • SprayPay
  • Riverty
  • Mondu
  • AlmaPAY
  • Klarna

Other

  • PayPal

Recurring payments

  • SEPA Direct Debit
  • Card-on-file / recurring card payments (tokenisation) [Coming soon]

Region specific payment methods

| Payment Method | Region(s) | |-----------------------------------------------|-------------------------------------------------------------------------------------------------| | iDEAL | 🇳🇱 Netherlands | | Bancontact | 🇧🇪 Belgium | | Vipps | 🇳🇴 Norway🇸🇪 Sweden | | Bizum | 🇪🇸 Spain | | Payconiq | 🇧🇪 Belgium🇱🇺 Luxembourg | | Twint | 🇨🇭 Switzerland | | MB Way | 🇵🇹 Portugal | | Pay By Bank | 🇧🇪 Belgium🇩🇪 Germany🇮🇹 Italy🇱🇺 Luxembourg🇳🇱 Netherlands🇪🇸 Spain | | WeChat Pay | 🇨🇳 China |

| Payment Method | Region(s) | |-----------------------------------------------|------------------------------------------| | Alipay Plus | 🇨🇳 China🇭🇰 Hong Kong | | MultiBanco | 🇵🇹 Portugal | | Swish | 🇸🇪 Sweden | | Satispay | 🇮🇹 Italy | | Blik | 🇵🇱 Poland | | EPS | 🇦🇹 Austria | | Przelewy24 | 🇵🇱 Poland | | MobilePay | 🇩🇰 Denmark🇫🇮 Finland |

InStore / SoftPOS payments

Would you like to integrate Pay. (Soft)P0S? Get in touch!

Features

  • Easily Extendable: The modular architecture makes it easy to add support for additional Pay payment methods.
  • Webhook Support: Full support for Pay webhooks for real-time payment status updates.
  • Automatic Capture: Configurable automatic capture of payments.

Prerequisites

  • Medusa server v2.20.1 or later
  • Node.js v20 or later
  • A Pay account and token & secret with payment methods enabled.

Installation

pnpm add @webbers/pay-payments-medusa

Configuration

Add the provider to the @medusajs/payment module in your medusa-config.ts file & add it as plugin:

module.exports = defineConfig({
  projectConfig: {
    // ...
  },
  plugins: [
    // ... other plugins
    '@webbers/pay-payments-medusa'
  ],
  modules: [
    // ... other modules
    {
      resolve: "@medusajs/medusa/payment",
      options: {
        providers: [
          {
            resolve: "@webbers/pay-payments-medusa/providers/pay",
            id: "pay",
            options: {
              paymentDescription: "Your description", // optional
              atCode: process.env.PAY_AT_CODE,
              apiToken: process.env.PAY_API_TOKEN,
              slCode: process.env.PAY_SL_CODE,
              slSecret: process.env.PAY_SL_SECRET,
              returnUrl: process.env.PAY_RETURN_URL,
              testMode: process.env.PAY_TEST_MODE === 'true',
              tguApiUrl: process.env.PAY_TGU_API_URL, // defaults to https://connect.pay.nl/v1
              otherSlCodes: process.env.PAY_OTHER_SL_CODE ? JSON.parse(process.env.PAY_OTHER_SL_CODE) : undefined,
            },
          },
        ]
      }
    }
  ]
})

Configuration Options

[!NOTE] You can get the API token & secret from your Pay dashboard: click Settings > Click sales channel > Copy api tokens

| Option | Description | Default | |----------------|-----------------------------------------------|-------------------------------------------------------------------------------------------------------------------------------------------------------| | atCode | Your Pay AT code | Required | | apiToken | Your Pay API token | Required | | slCode | Your Pay sales channel code | Required | | slSecret | Your Pay sales channel secret | Required | | returnUrl | The URL to return to after payment | Required | | medusaUrl | The URL of your Medusa server | Required | | testMode | Whether to enable test payments | Optional | | tguApiUrl | Pay TGU API Url | Optional, use if you want to use a specific or private TGU, see here. | | otherSlCodes | Your other Pay sales channel code and secrets | Optional, used for webhook signature validation when using multiple Pay. sales channels. Format '{"SL-CODE-X":"secretX","SL-CODE-Y":"secretY"}' |

Environment Variables

Create or update your .env file with the following variables:

PAY_AT_CODE="<your-pay-at-code>"
PAY_API_TOKEN="<your-pay-api-token>"
PAY_SL_CODE="<your-pay-sl-code>"
PAY_SL_SECRET="<your-pay-sl-secret>"
#PAY_TEST_MODE="true"
PAY_EXCHANGE_URL="https://your-store.com/checkout/payment"

Usage

Once installed and configured, the Pay payment methods will be available in your Medusa admin. To enable them, log in to you Medusa Admin, browse to Settings > Regions, add or edit a region and select the desired Pay providers from the dropdown.

Screenshot 2025-03-10 at 14 14 43

Make sure that the selected payment methods are enabled in your Pay origanization settings as well.

Supported Payment Methods

The plugin currently supports the following Pay payment methods:

| Payment Method | Provider ID | |--------------------------------------------------------------------------------------------------------------------------------------|---------------------------------------| | Pay. hosted checkout | pp_pay-hosted-checkout_pay | | CreditcardsMastercard / VISA / American Express/ Carte Bancaire / Maestro / PostePay / DanKort / Nexi / Visa Mastercard | pp_pay-creditcard-group_pay | | Card-on-file / recurring card payments (tokenisation) | Coming soon | | Apple Pay | pp_pay-apple-pay_pay | | Google Pay | pp_pay-google-pay_pay | | iDEAL IN3 | pp_pay-ideal-in3_pay | | Billink | pp_pay-billink_pay | | SprayPay | pp_pay-spraypay_pay | | Riverty | pp_pay-riverty_pay | | Mondu | pp_pay-mondu_pay | | AlmaPAY | pp_pay-almapay_pay | | Klarna | pp_pay-klarna_pay | | PayPal | pp_pay-paypal_pay | | SEPA Direct Debit | pp_pay-direct-debit_pay | | iDEAL | pp_pay-ideal_pay | | Bancontact | pp_pay-bancontact_pay | | Vipps | Get in touch | | Bizum | Get in touch | | Payconiq | pp_pay-payconiq_pay | | Twint | pp_pay-twint_pay | | MB Way | Get in touch | | Pay by Bank | pp_pay-paybybank_pay | | WeChat Pay | pay-wechatpay | | AliPay Plus | Get in touch | | MultiBanco | Get in touch | | Swish | Get in touch | | Satispay | Get in touch | | Blik | pp_pay-blik_pay | | EPS | pp_pay-eps_pay | | Przelewy24 | pp_przelewy24_pay | | MobilePAY | pp_pay-mobilepay_pay | | SoftPOS | pp_pay-softpos_pay | | Gift Card | pp_pay-giftcard_pay |

Client-Side Integration

To integrate with your storefront, you'll need to implement the payment flow according to Pay's and Medusa's documentation. Here's a basic example:

  1. Create a payment session in your checkout flow
  2. Redirect the customer to the Pay payment page
  3. Handle the webhook notifications to update the payment status

Example integration using the Medusa Next.js Starter:

https://github.com/user-attachments/assets/742ee261-5e41-4e33-9a72-faf1a424fc52

Duplicate cart endpoint

[!TIP] Use the duplicate cart endpoint in your storefront

When a customer cancels a payment or returns to the storefront without completing the Pay. checkout, a new duplicate cart should automatically be created. This allows the customer to easily start a new transaction without losing the items they had selected.

API Route: GET /store/carts/:id/duplicate

Alter your storefront retrieve cart function(s) and check if the returned cart.completed_at value is set. If so request a new cart with the duplicate cart endpoint and update cart id in cookies accordingly.

The duplicate cart endpoint is idempotent, so it can be called multiple times with the same cart id.

Adding payment method icons

  1. Download the latest payment images from here: https://github.com/paynl/payment-images
  2. Add these to your storefront public assets
  3. In your checkout, create the mapping from the provider id to the icon.
  4. You can also utilize the exported payPaymentMethods from this plugin to find the corresponding ID.
  5. I.e. const paymentMethodData = payPaymentMethods.find(method => `pp_${method.value}_pay` === provider_id)

Extending the Plugin

To add one of the missing Pay payment methods, create a new service in src/providers/Pay/services that extends the PayBase class:

import {PaymentMethod} from "@Pay/api-client";
import PayBase from "../core/Pay-base";
import {PaymentOptions, PaymentProviderKeys} from "../types";

class PayNewMethodService extends PayBase {
  static identifier = "Pay-new-method";

  get paymentCreateOptions(): PaymentOptions {
    return {
      method: PaymentMethod.newMethod,
    };
  }
}

export default PayNewMethodService;

Make sure to replace new method with the actual Pay payment method ID.

Export your new service from src/providers/Pay/services/index.ts. Then add your new service to the list of services in src/providers/Pay/index.ts.

We will be working on providing all the available Pay. options in the near future.

Medusa v1 Support

Searching for support for Medusa v1, we have a legacy plugin available. Get in touch