Skip to content

An AppBuilder starter kit that provides the scaffolding for checkout integration patterns with Adobe Commerce.

License

Notifications You must be signed in to change notification settings

adobe/commerce-checkout-starter-kit

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

6 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Adobe Commerce checkout starter kit

Welcome to home of Adobe Commerce checkout starter kit.

This starter kit is designed to help you get started with building custom checkout experiences for Adobe Commerce. Its goal is to showcase how to use Adobe Commerce Extensibility in combination with Adobe App Builder to build custom checkout experiences.

References

How to use this repo

  • See DEVELOPMENT.md for information about the prerequisites and how to set up and run the project locally.
  • See CICD.md for information about the CI/CD setup.
  • See EDS.md for information about the Edge Delivery Service(EDS) Storefront integration.

Prerequisites

Before starting with the starter kit, ensure that your Adobe Commerce installation meets the following prerequisites:

Install Out-of-Process Payment Extensions (OOPE) Module in Adobe Commerce

To enable out-of-process payment methods in your Commerce instance, install the magento/module-out-of-process-payment-methods in your Commerce instance. This module enables out-of-process payment functionalities. Execute the following command using Composer:

composer require magento/module-out-of-process-payment-methods --with-dependencies

Install Commerce Eventing Module in Adobe Commerce

The Commerce Eventing module is crucial for handling events within Adobe Commerce and has been included in the core since Adobe Commerce version 2.4.6. Ensure your installation is up-to-date, especially if you are using this starter kit, which requires at least version 1.10.0 of the Commerce Eventing module:

composer update magento/commerce-eventing --with-dependencies

For Adobe Commerce versions 2.4.4 or 2.4.5, the Adobe I/O Events for Adobe Commerce module will need to be installed manually. Follow the instructions provided in the Adobe I/O Events installation documentation.

Project structure

Configurations

All configurations align with the guidelines found on the App Builder Configuration Files page. In addition to the configurations mentioned there, this starter kit requires the following additional configurations:

events.config.yaml

The events.config.yaml file is used to define the event providers and their metadata.

Field Description
label Label of the event provider.
provider_metadata Metadata of the event provider (e.g., dx_commerce_events for the commerce event provider).
description Description of the event provider.
docs_url Documentation URL for the event provider.
events_metadata List of event metadata to register. Not required for Commerce event provider as it will be done by event subscription.
subscription Required only for the commerce event provider. List of commerce events to subscribe to.
Payload specifications can be found here.

Scripts

A set of scripts has been provided to help you get started with the project. You can find them in the scripts/ and they can be run using npm run <script-name>.

configure-events

The configure-events script configures the Adobe I/O Events integration for your project with a single command.

It performs the following actions:

  1. It reads the event providers specification from the events.config.yaml file and synchronizes the event providers and their metadata.
    • Note that the labels of the event providers defined in the specification are suffixed with the Adobe I/O Runtime namespace to ensure uniqueness across the projects of the organization.
  2. The script also updates the AIO_EVENTS_PROVIDERMETADATA_TO_PROVIDER_MAPPING environment variable with the latest provider metadata.

To run the configure-events script, ensure that your project configuration (.aio file) includes the following:

  • Organization ID: project.org.id
  • IMS Organization ID: project.org.ims_org_id
  • Project ID: project.id
  • Workspace ID: project.workspace.id

Additionally, the script uses the following environment variables:

  • AIO_runtime_namespace: The Adobe I/O Runtime namespace used as suffix for the Adobe I/O Events provider label.
  • AIO_EVENTS_PROVIDERMETADATA_TO_PROVIDER_MAPPING: (Optional) Existing provider metadata to provider mapping.

Note that event providers deletion is not supported by the script. If you need to delete an event provider, you can do it through AIO cli with the following command aio event provider delete <provider-id>.

configure-commerce-events

The configure-commerce-events script configures the commerce event provider for your Commerce instance.

It reads dx_commerce_events event provider specification from the events.config.yaml and .env files, and performs the following actions:

  1. It configures commerce eventing in the Commerce instance.
    • If the Commerce instance has already been configured with a different provider, the script will return an error to prevent overriding another project's configuration.
  2. It subscribes to the required commerce events.

To run the script, ensure you have set the followings up:

  1. You have the commerce eventing module installed in your commerce instance.
  2. Make sure you have already set up the Adobe Commerce HTTP Client to authenticate with the commerce instance.
  3. Ensure that your events.config.yaml and .env files are correctly configured with the commerce event provider specification.
    • The event provider needs to be created in advance, which you can do by running the configure-events script.
    • If you already have a commerce event provider, please ensure that
      • events.config.yaml file matches the existing provider metadata.
      • The environment variable AIO_EVENTS_PROVIDERMETADATA_TO_PROVIDER_MAPPING contains the commerce event provider id.
  4. Additionally, the script requires the following environment variables, which will be used to update the values in Stores > Configuration > Adobe Services > Adobe I/O Events > Commerce events:
    • COMMERCE_ADOBE_IO_EVENTS_MERCHANT_ID: The merchant ID of the commerce instance.
    • COMMERCE_ADOBE_IO_EVENTS_ENVIRONMENT_ID: The environment ID of the commerce instance.

Note that this script must be completed before deploying the application for event registration.

create-payment-methods

The create-payment-methods script is used to create payment methods in Adobe Commerce. It reads the payment methods configuration from the payment-methods.yaml file and creates the payment methods in Adobe Commerce.

To run the create-payment-methods script, ensure that the Adobe Commerce HTTP Client is configured.

Use cases

3rd party events processing

3rd party systems usually offer a way to subscribe to events that are emitted when certain actions are performed. For example, with a payment gateway we may subscribe to Authorization, Capture or Refund events.

Adobe I/O Events can be used to offload the events processing which requires to configure an Event Provider. The configure-events script provided in this project can be used to manage the 3rd party event providers required for your integration. See the Scripts/configure-events section for more information.

Once the event provider is configured, the 3rd party events can be published and a consumer can be registered to process them accordingly.

(3rd party system) -> (Adobe I/O Events) -> (AppBuilder app)

Publication

There are different options on how to publish these events with an AppBuilder app, depending on the flexibility of the 3rd party system.

The options are ordered by preference.

Directly from 3rd party system

The best option is to ingest the events directly from the 3rd party system. This is the most efficient way to process events but the source system has to be adapted to send the events to Adobe I/O Events.

(3rd party system) -> (Event provider) -> (Consumer runtime action)

The Events Publishing API is described in the Adobe I/O Events documentation.

Note this example is not exemplified in this project since it depends on source system details.

Publication using an action

If the 3rd party system does not support sending events to Adobe I/O Events, it usually supports registering a webhook that should be called when an event occurs. Additionally, the 3rd party system may be configured to use an authentication mechanism in the webhook (basic auth, OAuth, etc.) so that only authorized requests are accepted.

(3rd party system) -> (Consumer runtime action) -> (Event provider) -> (Consumer runtime action)

This use case is implemented in the actions/3rd-party-events/publish.js action.

Note that for implementing this use case correctly the action should receive the OAUTH_* environment variables to be able to retrieve an access token to publish in the event provider. This configuration is done by specifying the env vars in the .env file and setting them as app.config.yaml.

Consumption

Consumption of events can be done using webhooks where the action is registered as a consumer of the event provider.

An example of a consumer is the actions/3rd-party-events/consume.js action which is registered declaratively as a Webhook in app.config.yaml. Note that in this config file, the value used in the provider_metadata field is specified in the AIO_EVENTS_PROVIDERMETADATA_TO_PROVIDER_MAPPING environment variable so the registration can know to which provider the action should be registered.

AIO cli provides an interactive command to register the webhook and the action as a consumer of the event provider:

aio app add event

Extended documentation about how to implement a consumer action and register it as a webhook can be found in the AppBuilder Applications with Adobe I/O Events documentation

See also Adobe I/O Events Webhook FAQ which contains interesting information about how to handle event consumption (state of registration, retries, debugging).

Payment flow: Obtain order details from Adobe Commerce using the masked cart id

To understand the payment flow, we need to consider the following steps:

  1. It all starts on the frontend. When checkout is completed, the frontend sends the masked cart ID to the payment gateway.
  2. The payment gateway then sends a request to the AppBuilder application with the masked cart ID, as this is the only information it has about the order. This request could be a webhook or an event.
  3. The AppBuilder application uses the Adobe Commerce HTTP Client to retrieve the order details using the masked cart ID. To facilitate this, the starter kit provides the method getOrderByMaskedCartId in the Adobe Commerce HTTP Client.

sequence.png

Payment methods: Validate payment info

Since the checkout process and the payment of the order is expected to be done in a headless way, the Commerce instance has to ensure that the payment has succeeded and the order can be placed.

In order to ingest the payment gateway specific information in the payment process, it is expected that the checkout process uses setPaymentMethodOnCart mutation in combination with payment_method.additional_data field in order to persist all the information required to validate later the payment once the order is placed.

setPaymentMethodOnCart(
  input: {
    cart_id: $cartId
    payment_method: {
      code: $code
      additional_data: [
        {
          key: 'sessionId',
          value: '86A76C95-8F56-4922-B226-636533C06708',
        },
        {
          key: 'status',
          value: 'DONE',
        },
      ]
    }
  }
) {
  cart {
    selected_payment_method {
      code
      title
    }
  }
}

With this information persisted, a webhook can be configured with the help of Adobe Commerce Webhooks so every time there's an order placed a synchronous call is dispatched to the AppBuilder application implementing the payment method to validate the payment.

In order to register a webhook, go to the Adobe Commerce Admin > System > Webhooks and create a new webhook with the following configuration:

Hook Settings
  Webhook Method: observer.sales_order_place_before
  Webhook Type: before
  Batch Name validate_payment
  Hook Name: oope_payment_methods_sales_order_place_before
  URL: https://yourappbuilder.runtime.adobe.io/api/v1/web/commerce-checkout-starter-kit/validate-payment
  Active: Yes
  Method: POST

Hook Fields
  Field: payment_method Source: order.payment.method
  Field: payment_additional_information Source: order.payment.additional_information

Hook Rules
  Field: payment_method Value: yourpaymentmethodcode Operator: equal

Additionally, you can enable webhook signature generation according to Webhooks signature verification

See the action implemented in actions/payment-methods/validate-payment.js for an example of how to receive the request and validate the payment according to the payment gateway needs.

Adobe Commerce HTTP Client

adobe-commerce.js provides a set of methods to interact with the Adobe Commerce instance. The client is built using the Adobe Commerce HTTP Client, which is a wrapper around the Adobe Commerce REST API.

To utilize the Adobe Commerce HTTP Client, update COMMERCE_BASE_URL=<commerce_instance_url> in the .env file, and complete the authentication setup.

Authentication

Depending on your Adobe Commerce setup, there are 2 options to authenticate and communicate with the App Builder:

  1. Configure Adobe Identity Management Service (IMS)
  2. Configure Commerce Integration

It's important to know that if commerce integration is detected, it will have precedence over IMS Auth. However, if none of them is detected or configured, than client instantiation will directly fail.

Option 1. Configure Adobe Identity Management Service (IMS)

To proceed with this authentication, some previous setup needs to be done.

  1. Configure IMS for Commerce following the steps in Configure the Commerce Admin Integration with Adobe ID.

  2. Create new IMS credentials through the Adobe Developer Console. To do so, add a new service of type API in the workspace. From the list of API's, select I/O Management API and follow the steps shown by the wizard. On completion, all credentials will be generated.

  3. Add Technical Account to Commerce Admin

    1. Ensure that the technical account associated with the server-to-server credentials is added to the Commerce Admin with the appropriate permissions. If not, you can add it using Admin User Creation Guide.

    2. When associating the user, make sure to find your actual Technical Account email as a part of generated IMS credentials with following pattern: @techacct.adobe.com and use that value in the Email field shown in the following image: img.png

    3. When selecting the user role from the User Roletab shown in the previous image, make sure to select the Administrators to have all the necessary permissions.

Finally, copy the generated credentials (client id, client secret, technical account id, technical account email) to the .env file in the root of the project as following:

OAUTH_CLIENT_ID=<client id>
OAUTH_CLIENT_SECRETS=<client secret>
OAUTH_TECHNICAL_ACCOUNT_ID=<technical account id>
OAUTH_TECHNICAL_ACCOUNT_EMAIL=<technical account email>
OAUTH_SCOPES=<scope>
OAUTH_IMS_ORG_ID=<img org>

Option 2. Configure Commerce Integration

This option also enables us to communicate with the platform. It requires some setup as following:

  1. Create a new Adobe Commerce Integration by following this guide.
  2. Copy the integration details (consumer key, consumer secret, access token, and access token secret) to the .env file in the root of the project.
    COMMERCE_CONSUMER_KEY=<key>
    COMMERCE_CONSUMER_SECRET=<secret>
    COMMERCE_ACCESS_TOKEN=<access token>
    COMMERCE_ACCESS_TOKEN_SECRET=<access token secret>
    

Debugging of requests

From now, you can also debug and see some customized logs using the LOG_LEVEL environment variable. If this variable is set, logs from different phases of the commerce client instantiation will be shown with detailed information.

Commerce API Methods

To call the Commerce REST endpoints, initialize the Adobe Commerce Client as follows:

const { getAdobeCommerceClient } = require('../lib/adobe-commerce');
const commerceClient = await getAdobeCommerceClient(process.env);

Create a new OOPE payment method

createOopePaymentMethod creates a new out-of-process payment method with the necessary details such as code, title, and configuration.

Payload parameters:

Parameter Type Description
code String Unique identifier for the payment method.
title String Display name of the payment method.
description String Description of the payment method.
active Boolean Status indicating if the method is active.
backend_integration_url String URL for backend integration, which is an app builder URL
stores Array List of store codes that payment method is available
order_status String Initial order status when using this method. Default is pending
countries Array List of countries where the method is available.
currencies Array Currencies supported by the payment method.
custom_config Array Custom configuration settings for payment methods

Example usage:

try {
  const createResponse = await commerceClient.createOopePaymentMethod({
    code: 'method-1',
    title: 'Method 1',
    description: 'Description for Method 1',
    active: true,
    backend_integration_url: 'https://example.com',
    stores: ['store-1', 'store-2'],
    order_status: 'processing',
    countries: ['US', 'ES'],
    currencies: ['USD', 'EUR'],
    custom_config: [{ key: 'key1', value: 'value1' }],
  });

  if (!createResponse.success) {
    return errorResponse(createResponse.statusCode, 'Failed to create payment method');
  }

  console.log('Created payment method:', createResponse.message);
} catch (error) {
  return errorResponse(HTTP_INTERNAL_ERROR, 'Error occurred while creating payment method');
}

Example response:

{
  "success": true,
  "message": {
    "id": 3,
    "code": "method-1",
    "title": "Method 1",
    "description": "Description for Method 1",
    "active": true,
    "backend_integration_url": "http://example.com",
    "stores": ["store-1", "store-2"],
    "order_status": "processing",
    "countries": ["ES", "US"],
    "currencies": ["EUR", "USD"],
    "custom_config": [
      {
        "key1": "value1"
      }
    ]
  }
}

List all payment methods

getOopePaymentMethods retrieves the list of all out of process payment methods in the Adobe Commerce instance.

Example usage:

try {
  const listResponse = await commerceClient.getOopePaymentMethods();
  if (!listResponse.success) {
    return errorResponse(listResponse.statusCode, 'Failed to list payment methods');
  }
  console.log('List of payment methods:', listResponse.message);
} catch (error) {
  return errorResponse(HTTP_INTERNAL_ERROR, 'Error occurred while listing payment methods');
}

Example response:

{
  "success": true,
  "message": [
    {
      "id": 1,
      "code": "method-1",
      "title": "Method one",
      "active": true,
      "backend_integration_url": "http://oope-payment-method.pay/event",
      "stores": [],
      "order_status": "complete",
      "countries": [],
      "currencies": [],
      "custom_config": []
    }
  ]
}

Get an OOPE payment method by code

getOopePaymentMethod retrieves one out of process payment method by code from the Adobe Commerce instance.

Payload parameters:

Parameter Type Description
code String Unique identifier for the payment method.

Example usage:

try {
  const getResponse = await commerceClient.getOopePaymentMethod('method-1');
  if (!getResponse.success) {
    return errorResponse(getResponse.statusCode, 'Failed to retrieve payment method');
  }
  console.log('Retrieved payment method details:', getResponse.message);
} catch (error) {
  return errorResponse(HTTP_INTERNAL_ERROR, 'Error occurred while retrieving payment method');
}

Example response:

{
  "success": true,
  "message": {
    "id": 2,
    "code": "method-1",
    "title": "Method one",
    "active": true,
    "backend_integration_url": "http://oope-payment-method.pay/event",
    "stores": ["default"],
    "order_status": "complete",
    "countries": ["ES", "US"],
    "currencies": ["EUR", "USD"],
    "custom_config": [
      {
        "key": "can_refund",
        "value": "true"
      }
    ]
  }
}

Retrieve an order by masked cart ID

getOrderByMaskedCartId retrieves order details from the Adobe Commerce instance using a masked cart ID. This is typically used when the app builder application receives a webhook or event from the payment gateway. This method uses the Adobe Commerce API order search criteria.

Payload parameters:

Parameter Type Description
maskedCartId String The cart ID from the payment method webhook or event.

Example usage:

try {
  const orderResponse = await commerceClient.getOrderByMaskedCartId(maskedCartId);
  if (!orderResponse.success) {
    const errMsg =
      orderResponse.statusCode === HTTP_NOT_FOUND
        ? 'Order not found for the given maskedCartId.'
        : 'Unexpected error getting order by maskedCartId';
    return errorResponse(orderResponse.statusCode, errMsg);
  }
  console.log('Order details:', orderResponse.message);
} catch (error) {
  return errorResponse(HTTP_INTERNAL_ERROR, 'Failed to fetch order due to an unexpected error');
}

CI/CD

To read about continuous integration and continuous delivery for any application built using App builder visit CI/CD for App Builder Applications.

In addition, to help with the implementation, workflow samples are also provided under workflow-samples. To understand these workflows, visit CICD.md file in this project.

About

An AppBuilder starter kit that provides the scaffolding for checkout integration patterns with Adobe Commerce.

Resources

License

Code of conduct

Security policy

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Contributors 4

  •  
  •  
  •  
  •