Recurring payments
Recurring payments can be used to charge customers on a regular basis or to offer automatic top-ups with credits-based services. Since the amount for a recurring payment is variable, other innovative use-cases are possible as well. Like doing a partial prepayment and later a final recurring payment, for more expensive products. Or charging a customer via your application manually, for orders by phone.
Recurring payments happen in the background. The customer goes through the payment steps only once, for the first payment.
Please note that you can not use recurring payments in our Web app. It's only possible to use our API (direct or indirect, e.g. with a partner) for this.
Reducing the risk of chargebacks
To reduce the risk of chargebacks, itβs recommended to communicate how often and how much the customer will be charged as clearly as possible. We suggest notifying the customer a couple of days in advance of the next payment, for example by sending them an email.
How to get started
In the following sections we explain the following topics.
- Setting up the first payment
- Charging immediately on-demand
- Charging periodically with subscriptions
- How do webhooks for subscriptions work?
For more information on how to test recurring payments, refer to our guide for testing the Mollie API.
Enable Recurring on PayPalIf you have connected your PayPal account to Mollie before January 2025, you might have to reconnect your account in the Mollie Web app to enable all recurring functionality.
Setting up the first payment
In order to get started with recurring payments you need to require the customerβs consent through a first payment. Itβs similar to a regular payment, but the customer is shown information about your organization, and the customer needs to complete the payment with the account or card that will be used for recurring charges in the future. After the first payment is completed successfully, the customerβs account or card will immediately be chargeable on-demand, or periodically through subscriptions.
- Create a unique customer using the Customers API. If you are using Mollie Connect, make sure you have the permission customers.write in order to create a customer.
curl -X POST https://api.mollie.com/v2/customers \
    -H "Authorization: Bearer test_dHar4XY7LxsDOtmnkVtjNVWXLSlXsM" \
    -H "Content-Type: application/json" \
    -d "{\"name\":\"Customer A\",\"email\":\"[email protected]\"}"<?php
$mollie = new \Mollie\Api\MollieApiClient();
$mollie->setApiKey("test_dHar4XY7LxsDOtmnkVtjNVWXLSlXsM");
$customer = $mollie->customers->create([
    "name" => "Customer A",
    "email" => "[email protected]"
]);- Save the customerβs idin your database. You need it for the next step.
- Create a payment (or order) for the customer by specifying the customerIdand setting thesequenceTypeparameter tofirst. If you're using the Orders API, those parameters must be included in thepaymentsection of the Create Order payload.
Authorise a first payment for subscription paymentsFor credit card and PayPal payments, you can create a payment with a zero amount. No money will then be debited from the card or account when doing the first payment. If you intend to start a subscription, you can use this payment as the first installment, e.g. by charging the amount you will periodically charge. Next, you can set up a subscription which starts after the first period. See also the startDate field in the Subscriptions API.
curl -X POST https://api.mollie.com/v2/payments \
    -H "Authorization: Bearer test_dHar4XY7LxsDOtmnkVtjNVWXLSlXsM" \
    -H "Content-Type: application/json" \
    -d \
    "{
        \"amount\": {\"currency\":\"EUR\", \"value\":\"0.01\"},
        \"customerId\": \"cst_Ok2DlrJe5\",
        \"sequenceType\": \"first\",
        \"description\": \"First payment\",
        \"redirectUrl\": \"https://webshop.example.org/order/12345/\",
        \"webhookUrl\": \"https://webshop.example.org/payments/webhook/\"
    }"<?php
$mollie = new \Mollie\Api\MollieApiClient();
$mollie->setApiKey("test_dHar4XY7LxsDOtmnkVtjNVWXLSlXsM");
$payment = $mollie->payments->create([
    "amount" => [
        "currency" => "EUR",
        "value" => "0.01"
    ],
    "customerId" => "cst_Ok2DlrJe5",
    "sequenceType" => "first",
    "description" => "First payment",
    "redirectUrl" => "https://webshop.example.org/order/12345/",
    "webhookUrl" => "https://webshop.example.org/payments/webhook/"
]);- Redirect the customer to the _links.checkout.hrefto complete the first payment. Make sure to use an HTTPGETredirect.
- Once completed there will be a customer mandate that you can access via the Mandates API. Depending on the payment method that was used to create the first payment, a specific mandate will be created. See the table below.
| Payment Method | Mandate | 
|---|---|
| creditcard(incl Apple Pay and Google Pay) | creditcard | 
| paypal | paypal | 
| belfius | directdebit | 
| bacs**Make sure the first payment value is 0.00 | bacs | 
| bancontact | directdebit | 
| eps | directdebit | 
| ideal | directdebit | 
| kbc | directdebit | 
| paybybank | directdebit | 
| trustly | directdebit | 
If you want to use a payment method that creates a
directdebitmandate, make sure to also enable the SEPA Direct Debit payment method in your Mollie profile.
Charging immediately on-demand
Now that the customer has given their consent, itβs possible to perform a recurring payment on-demand. Instead of the regular payment with a redirectUrl, a recurring payment happens in the background without a browser session, i.e. without the customer going through payments steps. You can create a recurring payment with the sequenceType set to recurring when creating a payment with the Payments API.
In order to do recurring payments, SEPA Direct Debit or credit card has to be activated on your profile.
- Make sure the customer has valid mandates. Find out using the Mandates API.
curl -X GET https://api.mollie.com/v2/customers/cst_4qqhO89gsT/mandates \
    -H "Authorization: Bearer test_dHar4XY7LxsDOtmnkVtjNVWXLSlXsM"<?php
$mollie = new \Mollie\Api\MollieApiClient();
$mollie->setApiKey("test_dHar4XY7LxsDOtmnkVtjNVWXLSlXsM");
$mandates = $mollie->customers->get("cst_4qqhO89gsT")->mandates();- If thereβs at least one mandate with a statusset tovalidthen continue.
- Set the sequenceTypeparameter torecurringand pass in themandateIdto charge the customer on-demand.
curl -X POST https://api.mollie.com/v2/payments \
    -H "Authorization: Bearer test_dHar4XY7LxsDOtmnkVtjNVWXLSlXsM" \
    -H "Content-Type: application/json" \
    -d \
    "{
        \"amount\": {\"currency\": \"EUR\", \"value\": \"10.00\"},
        \"customerId\": \"cst_Ok2DlrJe5\",
        \"mandateId\": \"mdt_h7Id8SjfJ\",
        \"sequenceType\": \"recurring\",
        \"description\": \"Background payment\",
        \"webhookUrl\": \"https://webshop.example.org/payments/webhook/\"
    }"<?php
$mollie = new \Mollie\Api\MollieApiClient();
$mollie->setApiKey("test_dHar4XY7LxsDOtmnkVtjNVWXLSlXsM");
$payment = $mollie->payments->create([
    "amount" => [
        "currency" => "EUR",
        "value" => "10.00"
    ],
    "customerId" => "cst_Ok2DlrJe5",
    "sequenceType" => "recurring",
    "description" => "Background payment",
    "webhookUrl" => "https://webshop.example.org/payments/webhook/"
]);- Like regular payments your webhook is called for retrieving status updates.
Charging periodically with subscriptions
For simple regular recurring payments with constant amounts, you can create subscriptions with the Subscriptions API. Subscription payments will be spawned automatically at the specified frequency, and will show up in your Web app.
- Make sure the customer has a pending or valid mandate using the Mandates API.
curl -X GET https://api.mollie.com/v2/customers/cst_4qqhO89gsT/mandates \
    -H "Authorization: Bearer test_dHar4XY7LxsDOtmnkVtjNVWXLSlXsM"<?php
$mollie = new \Mollie\Api\MollieApiClient();
$mollie->setApiKey("test_dHar4XY7LxsDOtmnkVtjNVWXLSlXsM");
$mandates = $mollie->customers->get("cst_4qqhO89gsT")->mandates();- Continue if thereβs a mandate with its statusbeing eitherpendingorvalid, otherwise set up a _first _payment for the customer first.
- Create the subscription using the Create subscription endpoint.
curl -X POST https://api.mollie.com/v2/customers/cst_Ok2DlrJe5/subscriptions \
    -H "Authorization: Bearer test_dHar4XY7LxsDOtmnkVtjNVWXLSlXsM" \
    -H "Content-Type: application/json" \
    -d \
    "{
        \"amount\": {\"currency\":\"EUR\", \"value\":\"25.00\"},
        \"times\": 4,
        \"interval\": \"3 months\",
        \"description\": \"Quarterly payment\",
        \"webhookUrl\": \"https://webshop.example.org/subscriptions/webhook/\"
    }"<?php
$mollie = new \Mollie\Api\MollieApiClient();
$mollie->setApiKey("test_dHar4XY7LxsDOtmnkVtjNVWXLSlXsM");
$subscription = $mollie->subscriptions->createForId("cst_4qqhO89gsT", [
    "amount" => [
        "currency" => "EUR",
        "value" => "25.00"
    ],
    "times" => 4,
    "interval" => "3 months",
    "description" => "Quarterly payment",
    "webhookUrl" => "https://webshop.example.org/subscriptions/webhook/"
]);- In the above example the customer is charged β¬25.00 for 4 times every 3 months, starting today.
- The webhook URL will be triggered for every payment to communicate any status updates.
Refer to the documentation of the API client you are using for more examples.
If you have a monthly subscription, and the charge date does not exist like in February, we will charge on the last date of the month.
Example: you charge every month on the 29th, in February we will charge on the 28th.
How do webhooks for subscriptions work?
When using our Subscriptions API to charge a customer periodically, new payments are created by Mollie every time the customer is charged. We will call your webhook as usual for these payments. The only difference is, the payment ID will not be known by your system yet when we call the webhook to report the paymentβs status.
With normal payments you know the payment ID, because youβve received this when creating the payment. With subscriptions you do not know the payment ID in advance. You will receive a webhook call with a payment ID that you have likely never seen before.
The payment object will, however, contain a subscriptionId field that contains the subscription ID you received when the subscription was created. This allows you to recognize where the payment belongs to.
We do not provide webhooks specifically for status changes of a Subscription itself.
Failing or disputed subscription payments
As with any other recurring payment, subscription payments can fail or be disputed via a chargeback. Generally, unsuccessful payments will not affect your subscription, but there are some exceptions. Here are the two most important things to know:
Mollie will not retry the failed or disputed payment If your subscription payment does not succeed, Mollie will not make another attempt. The next charge date will be updated regardless of the result, and the next payment will be created following the normal interval set by you.
Some chargeback/failure reasons will cancel your subscription As stated above, Mollie will generally not cancel your subscription when a payment fails. However, there are some reasons that can lead to cancellation. It varies whether the subscription will be canceled immediately or after a few occurrences of the same issue. The following reasons may cause your subscription to be canceled:
| Reason | Reason code | Cancels subscription... | 
|---|---|---|
| Account identifier incorrect (i.e. invalid IBAN) | AC01 | Immediately | 
| Account closed | AC04 | Immediately | 
| Account blocked for Direct Debit | AC06 | Immediately | 
| No Mandate | MD01 | After 3 times | 
| Return of funds requested by end customer | MD06 | After 3 times | 
| Debtor deceased | MD07 | Immediately | 
| Refusal by the Debtor | MS02 | After 3 times | 
| Reason not specified | MS03 | After 3 times | 
| Specific Service offered by the Debtor Bank | SL01 | After 3 times | 
Be aware!If your subscription has a fixed mandate set (
mandateIdfield), cancelling the subscription based on the reasons above will also revoke the mandate itself.
Updated 2 months ago