Klarna Docs - How to integrate Express checkout
How to integrate Express checkout

Follow this guide to integrate Express checkout.

16 min read

Currently there are two (2) supported integration path for Klarna Express Checkout:

  1. version New integration method, Express Checkout has just been added to the new Klarna Web SDK
  2. version Previous integration method using Klarna Payments library.

With the new integration method, you’ll get access to the latest Express Checkout features such as Shipping Options, which allows you to let customers choose their preferred shipping method right within Express Checkout.

If you are integrating Express Checkout for the first time, it is recommended you follow the latest integration path available to get access to the latest features available.

Alternatively, If you’ve been using the previous integration with the Klarna Payments library, don’t worry—it’s still supported and might be a better fit depending on your needs.

Currently, only the previous integration method supports the multi-step flow, but don’t worry; this feature will soon be available in the new Klarna Web SDK too.

The simplest and fastest way to start using Express checkout is to use the minimal template below.

Please refer to the Before you start article for instructions on how to obtain your CLIENT_ID .

HTML
<html>
<head>
  <!-- 
      1. Add SDK with `defer` attribute 
      to ensure DOM creation was finished. 
  -->
  <script defer src="https://js.klarna.com/web-sdk/v2/klarna.js">
  </script>


Learn more about each of the steps.

Integrate Klarna's functionality by adding the SDK to your web page. You can find the library at this URL: https://js.klarna.com/web-sdk/v2.js.

Important: Only include this script once per page to avoid any side-effects.

HTML
<script defer
  src="https://js.klarna.com/web-sdk/v2/klarna.js"
>
</script>

Within your page's <head> tag, add the following script to initialize the Klarna Web SDK.

Make sure to place the SDK interactions within the KlarnaSDKCallback function - this guarantees that the SDK is fully loaded and ready to use.

JAVASCRIPT
<script>
  window.KlarnaSDKCallback = async function () {
    const klarna = await Klarna.init({
      clientId: "klarna_live_client_elZGI1B5dHBIRWcjZrNldnbEVj[...]"
    });
  }
</script>

Web SDK provides the button method that allows to create and mount the Express checkout button programmatically.

JAVASCRIPT
<script>
  window.KlarnaSDKCallback = async function () {
    const klarnaExpressCheckout = klarna.Payment.button({ 
        id: "klarna-payment-button",
        // style the button according to what the partner provided
        shape: "default",
        theme: "default"
    });
    klarnaExpressCheckout.mount("#kec-button-container");
  }

If the button method is called more than once with the configuration object, it will create multiple button instances. If you do not need more than one button, please make sure the button method is only called once.

If you do need multiple buttons make sure you provide an id attribute to each button instance.

In Express checkout, a Payment Request represents a specific payment session. To initiate a payment, you'll need to create a Payment Request when a user clicks the "Pay with Klarna" button.

To create a basic Payment Request, you'll need to provide the following information:

  • Payment Amount: The total amount to be charged.
  • Currency: The currency code for the transaction.

To enable specific features like shipping address collection, shipping option selection, and customer information collection, you can configure the Payment Request with additional settings. Please refer to the "Additional Features" section for more details.

JAVASCRIPT
<script>
  window.KlarnaSDKCallback = async function () {
    // Handle click event
    klarnaExpressCheckout.on("click", (paymentRequest) => {
      return paymentRequest.initiate({
        "paymentAmount": 1000,
        "currency": "EUR",
        "supplementaryPurchaseData": {
          "lineItems": [
            {
ParameterType and description
currencyrequiredType:string, Currency in ISO 4217 format.
Description: Supported currencies: AUD, EUR, CAD, CZK, DKK, HUF, MXN, NZD, NOK, PLN, RON, SEK, CHF, GBP, USD.
paymentAmountrequiredType: integer
Description:Total amount of a one-off purchase, including tax and any available discounts. The value should be in non-negative minor units. Eg: 25 Dollars should be 2500. This is the amount that Klarna will charge the customer for a one-off purchase. Should not include any amounts for subscriptions.
lineItemsType:object
Description: Detailed line item information of the purchase. This data is used for fraud detection and customer communication.
customerType:object
Description:This represents who the customer is according to the merchant. These data points may be used by Klarna to simplify sign-up and during fraud assessment, they will not be used for underwriting and will not be persisted on created Payment Transactions.

Once the user completes the payment, you’ll receive a confirmation_token upon a successful transaction. Use this token to confirm the payment and complete the process.

On the frontend, you can listen for the update event. When the customer successfully completes the purchase flow, the payment request’s status will change to PENDING_CONFIRMATION, and you can then retrieve the confirmation_token.

JAVASCRIPT
klarna.Payment.on("update", async (paymentRequest) => {
  // The customer has successfully completed the payment flow and
  // the payment request can be confirmed using the confirmation token.
  if (paymentRequest.state == "PENDING_CONFIRMATION") {
    // The confirmation token is available in paymentRequest.stateContext.confirmationToken
    // Use it to confirm the payment request server side.
  }
});

Example stateContext value

JSON
{
  "paymentRequestId": "krn:payment:eu1:request:64be490f...",
  "state": "PENDING_CONFIRMATION",
  "stateContext": {
      "state": "PENDING_CONFIRMATION",
      "shipping": {
          "recipient": {
              "givenName": "Bob",
              "familyName": "Test",
              "email": "generated_email_de+1715945998626@example.com",

To confirm a payment request, provide the confirmation_token that was returned after the successful payment flow. Confirmation is done via the Klarna Payments endpoint, which is idempotent - meaning you can call it multiple times with the same confirmation_token and receive the same response each time.

Use the following URLs:

On this request, you must provide the payment_confirmation_token returned after a successful payment flow.

Use the credentials from your current integration or obtain new ones from Klarna's Partner portal.

klarna_api_key AND klarna_partner_account

The request body follows the same format as the previously defined payment request. The data included in the confirmation call must match the information provided to the Express Checkout.

Payments Confirmation PATH parameters

ParameterType and description
account_idrequiredType: string (AccountIdentifier)
Description: Identifier of the Account the action is done on behalf of Example: krn:partner:global:account:live:LWT2XJSE
payment_confirmation_tokenrequiredType: string (PaymentConfirmationToken) <= 255 characters
Description: The payment confirmation token of the Payment Request to confirmExample: krn:payment:eu1:confirmation-token:e15432a5-ebcc-45bc-944c-e61399db597b

Payments Confirmation body

application/jsonapplication/json

ParameterType and description
currencyrequiredType: string^[A-Za-z]{3}$
Description: Currency of the payment amount in 3-letter ISO 4217 currency code. Must be identical to the value in the payment request creation.
payment_amountrequiredType: string^[A-Za-z]{3}$
Description: Currency of the payment amount in 3-letter ISO 4217 currency code. Must be identical to the value in the payment request creation.
payment_transaction_referenceType: string (PaymentTransactionReference) [ 1 .. 255 ] characters
Description: Reference to the payment or equivalent resource created on your side. This will be exposed in the Payment Transaction webhooks for the purpose of correlating your resource with the Klarna Payment Transaction.
supplementary_purchase_dataAdditional information that provide more detailed information about the transaction, which helps reduce the risk of fraud and enhances transparency.
supplementary_purchase_data.shippingType: Array of objects (Shipping)
Description: Confirmation of shipping details if not collected by Klarna. If the details passed here do not match the details passed in the payment request the confirmation will be rejected.
supplementary_purchase_data.shipping.recipientrequiredType: Objects
Description: The recipient of the shipment.
supplementary_purchase_data.shipping.recipient.given_namerequiredType: string (GivenName) [ 1 .. 99 ] characters
Description: The given name of the person
supplementary_purchase_data.shipping.recipient.family_namerequiredType: string (FamilyName) [ 1 .. 99 ] characters
Description: The family name of the person
supplementary_purchase_data.shipping.recipient.emailType: string <email> (Email) (?:[a-z0-9!#$%&'*+/=?^_`{|}~-]+(?:\.[a-z0-9!#$%&'*+/=?^_`{|}~-]+)*|"(?:[\x01-\x08\x0b\x0c\x0e-\x1f\x21\x23-\x5b\x5d-\x7f]|\\[\x01-\x09\x0b\x0c\x0e-\x7f])*")@(?:(?:[a-z0-9](?:[a-z0-9-]*[a-z0-9])?\.)+[a-z0-9](?:[a-z0-9-]*[a-z0-9])?|\[(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?|[a-z0-9-]*[a-z0-9]:(?:[\x01-\x08\x0b\x0c\x0e-\x1f\x21-\x5a\x53-\x7f]|\\[\x01-\x09\x0b\x0c\x0e-\x7f])+)\])
Description: E-mail address
supplementary_purchase_data.shipping.recipient.phoneType: string (PhoneLocal) <= 99 characters ^(?=.*\d)[+\d\s().-]{1,99}$
Description: Phone number in local format
supplementary_purchase_data.shipping.recipient.attentionType: string [ 1 .. 99 ] characters
Description: Attention string for the recipient
supplementary_purchase_data.shipping.addressThe shipping address as provided by the customer at the merchant web site.
supplementary_purchase_data.shipping.address.street_addressrequiredType: string [ 1 .. 99 ] characters
Description: Street address, line 1
supplementary_purchase_data.shipping.address.street_address2Type: string [ 1 .. 99 ] characters
Description: Street address, line 2
supplementary_purchase_data.shipping.address.postal_codeType: string [ 1 .. 10 ] characters
Description: Postal code formatted according to country
supplementary_purchase_data.shipping.address.cityType: string [ 1 .. 99 ] characters
Description: City/town
supplementary_purchase_data.shipping.address.regionType: string [ 1 .. 99 ] characters
Description: State/county/province/region formatted according to country. Mandatory for US and AU market. Validations according to ISO 3166-2 format, e.g. OH, NJ, etc.
supplementary_purchase_data.shipping.address.countryType: string ^[A-Za-z]{2}$
Description: Country in ISO 3166-1 alpha-2 format.
Allowed countries: AD, AE, AF, AG, AI, AL, AM, AN, AO, AQ AR, AS, AT, AU, AW, AX, AZ, BA, BB, BD BE, BF, BG, BH, BI, BJ, BL, BM, BN, BO BQ, BR, BS, BT, BV, BW, BY, BZ, CA, CC CD, CF, CG, CH, CI, CK, CL, CM, CN, CO CR, CU, CV, CW, CX, CY, CZ, DE, DJ, DK DM, DO, DZ, EC, EE, EG, EH, ER, ES, ET FI, FJ, FK, FM, FO, FR, GA, GB, GD, GE GF, GG, GH, GI, GL, GM, GN, GP, GQ, GR GS, GT, GU, GW, GY, HK, HM, HN, HR, HT HU, ID, IE, IL, IM, IN, IO, IQ, IR, IS IT, JE, JM, JO, JP, KE, KG, KH, KI, KM KN, KP, KR, KW, KY, KZ, LA, LB, LC, LI LK, LR, LS, LT, LU, LV, LY, MA, MC, MD ME, MF, MG, MH, MK, ML, MM, MN, MO, MP MQ, MR, MS, MT, MU, MV, MW, MX, MY, MZ NA, NC, NE, NF, NG, NI, NL, NO, NP, NR NU, NZ, OM, PA, PE, PF, PG, PH, PK, PL PM, PN, PR, PS, PT, PW, PY, QA, RE, RO RS, RU, RW, SA, SB, SC, SD, SE, SG, SH SI, SJ, SK, SL, SM, SN, SO, SR, SS, ST SV, SX, SY, SZ, TC, TD, TF, TG, TH, TJ TK, TL, TM, TN, TO, TR, TT, TV, TW, TZ UA, UG, UM, US, UY, UZ, VA, VC, VE, VG VI, VN, VU, WF, WS, XK, YE, YT, ZA, ZM ZW

Example payment confirmation body

JSON
{
  "currency": "USD",
  "payment_amount": 2000,
  "payment_transaction_reference": "partner-transaction-reference-4567",
  "supplementary_purchase_data": {
    "shipping": [
      {
        "recipient": {
          "given_name": "John",
          "family_name": "Doe",

Confirmation response

JSON
{
  "payment_request_id": "krn:payment:eu1:request:552603c0-fe8b-4ab1-aacb-41d55fafbdb4",
  "state": "AUTHORIZED",
  "state_expires_at": "2024-01-01T15:00:00Z",
  "state_context": {
    "payment_transaction_id": "krn:payment:eu1:transaction:6debe89e-98c0-486e-b7a5-08a4f6df94b0",
  },
  "expires_at": "2024-01-02T13:00:00Z",
  "created_at": "2024-01-01T12:00:00Z",
  "updated_at": "2024-01-01T13:00:00Z"

payment_transaction_idpayment_transaction_id has a krn format , for example krn:payment:eu1:request:ccbaba75-e287-414e-98ef-34ce53975ee7.

The UUID is the order id that you should use with the Order management API.

To enable additional KEC features like Shipping Address Collection, Shipping Options, and Customer Information Collection, include the PaymentRequestConfig object when creating the payment request.

The following sections explain how to enable and manage each feature. Note that you can activate all features simultaneously if needed.

Defines whether the integrator will receive Klarna user account data at the end of the purchase flow, as well as which specific data points will be provided. Please ensure you have a valid legal basis under GDPR or other relevant regulations to request and handle customer identity information.

To enable this feature, include the config property in the object passed to the initiate function, specifying requestCustomerProfile with the list of required customer data fields.

The following customer information is available:

  • profile:email
  • profile:name
  • profile:phone
  • profile:billing_address
  • profile:locale
  • profile:national_identification
  • profile:date_of_birth
JAVASCRIPT
klarnaExpressCheckout.on("click", (paymentRequest) => {
  return paymentRequest.initiate({
    "paymentAmount": 1000,
    "currency": "EUR",
    "lineItems": [
      {
        name: 'Product 1',
        quantity: 1,
        totalAmount: 1000,
        totalTaxAmount: 100

Example state context value with customer data

JSON
{
  "paymentRequestId": "krn:payment:eu1:request:ccb46f6a...",
  "state": "PENDING_CONFIRMATION",
  "stateContext": {
      "shipping": {},
      "paymentConfirmationToken": "krn:payment:eu1:confirmation-token:24b36821...",
      "klarnaCustomer": {
          "customerProfile": {
              "givenName": "Bob",
              "familyName": "Test",

KEC can handle shipping address collection on behalf of the integrator. To enable this feature, include the PaymentRequestConfig object with the attribute "requestShippingData": ["SHIPPING_ADDRESS"].

If you need to restrict the countries to which you ship, include the allowedShippingCountries attribute with a list of permitted countries. By default, all Klarna-supported countries are available unless specified otherwise.

JAVASCRIPT
klarnaExpressCheckout.on('click', (paymentRequest) => {
  return paymentRequest.initiate(
    {
      paymentAmount: 1000,
      currency: 'EUR',
      supplementaryPurchaseData: {
        lineItems: [
          {
            name: 'Product 1',
            quantity: 1,

KEC can handle shipping address and shipping options collection on behalf of the integrator. To enable this feature, include the PaymentRequestConfig object with the attribute "requestShippingData": ["SHIPPING_ADDRESS", "SHIPPING_OPTION"].

Note that shipping options cannot be collected without enabling the shipping address feature.

If you need to restrict the countries where you ship, include the allowedShippingCountries attribute with a list of allowed countries. By default, all Klarna-supported countries are available unless specified otherwise.

JAVASCRIPT
klarnaExpressCheckout.on('click', (paymentRequest) => {
  return paymentRequest.initiate(
    {
      paymentAmount: 100,
      currency: 'EUR',
      supplementaryPurchaseData: {
        lineItems: [
          {
            name: 'Product 1',
            quantity: 1,

To enable proper functionality of the shipping options integration, you, as the integrator, need to handle two event callbacks: shippingaddresschange and shippingoptionselect. These events allow you to communicate available shipping options after an address is selected, enabling the customer to choose from the provided options.

The shippingaddresschange event is triggered each time a customer updates their shipping address during the purchase flow. You should respond by returning a list of available shipping options for the selected address. If no response is received within 5 seconds, the purchase flow will terminate with an error.

Shipping address returned has the following format.

Shipping address

JSON
{
  "streetAddress": "123 Main St",
  "streetAddress2": "Apt 4B",
  "city": "Springfield",
  "postalCode": "12345",
  "region": "IL",
  "country": "USA",
  "familyName": "Doe",
  "givenName": "John"
}

The shipping options returned must follow this format.

Shipping options

JSON
[
  {
    "shippingOptionReference": "shipping-option-1",
    "amount": 500,
    "displayName": "Standard shipping",
    "description": "1 - 3 working days",
    "shippingType": "TO_DOOR"
  },
  {
    "shippingOptionReference": "shipping-option-2",

The shippingoptionselect event is triggered every time a customer selects a different shipping option. You should return an updated order data here. If nothing was returned in 5 seconds, the purchase flow will end in an error state.

Load the Klarna payments JavaScript library when the cart page or a product detail page is loaded. Ensure the library is included only once to prevent conflicts.

HTML
<script defer
  src="https://x.klarnacdn.net/kp/lib/v1/api.js">
</script>

Then, implement the klarnaAsyncCallback function where you initialize Express checkout. Implement the klarnaAsyncCallback before importing the library. This way you ensure it will be invoked when the library is loaded. Within klarnaAsyncCallback, include the logic to:

  1. Initialize Klarna’s JavaScript SDK providing your client identifier as client_id. Get your client_id from the Merchant portal. Refer to the Before you start article for instructions.
  2. Load the button in a chosen container using a load() function. To help debug any issues that occur when loading the button, implement the [ load() callback].
  3. Handle the on_click event, in which you start the payment authorization process by calling the authorize() function. Keep the following best practices in mind:
  4. When initiating the authorization, make sure that the format of the orderPayload is the same as the body of the request to create a Klarna payments session.
  5. Always invoke the authorize() callback that you receive in on_click.
  6. Avoid having multiple nested asynchronous calls before invoking authorization.

The order payload object can contain all information allowed in the Klarna payments API, for example, merchant references, merchant URLs, and extra merchant data.

Here’s an example of code that initializes the Klarna JavaScript library and renders the Express checkout button on a cart or a product detail page:

HTML
<script defer
  src="https://x.klarnacdn.net/kp/lib/v1/api.js">
</script>

<script>
  window.klarnaAsyncCallback = function () {
    window.Klarna.Payments.Buttons.init({
      client_id: 'klarna_client_test...',
    }).load(
    {

The table below lists the attributes of the load() function’s configuration object.

ParameterDescription
containerrequiredcontainer represents the location where you want the Express checkout button to be displayed.
In this attribute, you can specify either:
  • a CSS selector, for example, #my-component-id, .my-component-class
  • an element-type object directly, for example, document.createElement('div')
on_clickrequiredThe function passed in this attribute will be executed when the Express checkout button is clicked. It will receive the authorize() function, which has to be invoked to start the Express checkout flow.
The authorize() function acts similarly to authorize() in a standard [ Klarna payments] integration.
themeThe color theme of the button.
The possible values are default, light, and dark. If the value isn't specified, default is used.
shapeThe shape of the button.
The possible values are default, rect, and pill. If the value isn't specified, default is used.
localeThe language of the button text. If not specified, the browser's language will be used.

The value of locale passed in the authorize() function’s configuration object defines the language of the button text. On the other hand, the value of locale passed to authorize() inside the orderPayload defines the language of the purchase flow.

Learn more about locale formats in Klarna APIs.

The table below lists the attributes for the authorize() function’s configuration object.

ParametersDescription
collect_shipping_addressInforms Express checkout whether you need the customer's shipping address from Klarna.
The default value is false.
auto_finalizeAllows you to specify whether the authorization should automatically be finalized when the customer clicks the Express button.
In one-step Express checkout, set auto_finalize to true. Otherwise, the purchase won’t be authorized automatically.
The default value is true. If you omit this attribute, the order will be authorized automatically.

If the authorization is successful, you will receive the authorization_token from the client-side authorize() response, in the authorization callback, or by getting the payment details from Klarna payments API.

Here’s an example of a response from the client-side authorize() call (one-step checkout):

JSON
{
  "show_form": true,
  "approved": true,
  "finalize_required": false,
  "authorization_token": "1eddf502-f3a0-45bf-b1fd-f2e3a2758200",
  "session_id": "e4b81ca2-0aae-4c16-bcb2-29a0a088a35b", 
  "collected_shipping_address": { // if collect_shipping_address was set to true
    "attention": "Attn",
    "city": "London",
    "country": "GB",

This example shows a response from the authorization callback.

JSON
{
  "authorization_token": "1eddf502-f3a0-45bf-b1fd-f2e3a2758200",
  "session_id": "e4b81ca2-0aae-4c16-bcb2-29a0a088a35b",
  "merchant_reference1": "" // if provided
  "merchant_reference2": "" // if provided
}

For multi-step checkout, the authorization token is not passed here. If the merchant requires this to pre-populate the checkout, as a workaround the merchant can use GET session call to retrieve the customer's shipping address.

Once you have the authorization_token, create an order. When creating the order, make sure the shipping address provided in the API request matches the collected shipping address returned alongside the authorization_token. . .