This document gives an overview of the characteristics and common features of the API.
The commercetools Composable Commerce HTTP API offers an interface for programmatic access to the data stored and associated functionality. While you can use the API directly for experimentation or custom integration solutions, we recommend you look at the available client libraries and SDKs, which provide a more comfortable development experience.
Regions
These Regions are completely isolated from each other, and no data is transferred between them.
User accounts
Hosts
The Composable Commerce HTTP API is hosted at the following URLs:
Region | API URL |
---|---|
North America (Google Cloud, Iowa) | https://api.us-central1.gcp.commercetools.com/ |
North America (AWS, Ohio) | https://api.us-east-2.aws.commercetools.com/ |
Europe (Google Cloud, Belgium) | https://api.europe-west1.gcp.commercetools.com/ |
Europe (AWS, Frankfurt) | https://api.eu-central-1.aws.commercetools.com/ |
Australia (Google Cloud, Sydney) | https://api.australia-southeast1.gcp.commercetools.com/ |
Prepend all requests to endpoints with these URLs.
https://api.{region}.commercetools.com
, replace the {region}
placeholder with the actual value.api.sphere.io
(Google Cloud, Belgium), api.commercetools.com
(Google Cloud, Belgium), and api.commercetools.co
(Google Cloud, Iowa) remain intact as aliases, we strongly encourage changing your configuration the new hostname structure.Disallowed requests
Requests to the HTTP API are blocked if any of the following conditions are true:
- The combination of request URL and headers is longer than about 15KB.
- The request method does not allow a body, but the request has one.
- The request contains an upgrade header.
- The HTTP version is unknown.
We strongly recommend using the host and domain names in your applications instead of hardcoding any IP address of the API hosts. IP addresses can change at any time without prior notice.
UTF-8
All text content storage, processing, and API interaction use UTF-8 encoding.
JSON
application/json
payloads. To ensure that payloads are encoded correctly, use Content-type: application/json
in your HTTP header when sending POST
requests.When consuming a JSON response, ignore any unrecognized fields in JSON objects of responses, or deal with them such that it does not cause the client application to crash. The API assumes clients behave properly and considers the addition of new fields to be backward-compatible for existing clients. In addition, client applications should not rely on the presence of undocumented fields. Fields returned by an undocumented endpoint can be renamed or removed at any time without prior notice.
Identifier
id
field that uniquely identifies a resource. The ID is generated by the Composable Commerce API when creating a resource.key
, which also uniquely identifies a resource. The key
identifier is always assigned to a resource by the client.key
field to identify resources. You must assign a key
to resources to update them by import.Resource updates
Partial updates
To update a resource, use one or more update actions with the update endpoint of the target resource. Each resource has a list of applicable update actions.
You can combine multiple update actions in a single request, provided they modify the same resource. Update actions must be ordered based on the sequence in which they should be processed. For example, if you are making changes to a Product and want to publish the changes right away, you must add the publish update action at the end of the update action list.
All update actions within a single update request operate as a cohesive unit. This means that either all update actions succeed collectively, or if any one action fails, the entire set of update actions is rejected. As a result, you should always consider whether to combine specific update actions in a single request or spread them across multiple requests.
The request representation (patch format) of an update request is as follows:
Field | Type | Required/Optional | Description |
---|---|---|---|
id | String | Optional if part of the URL | ID of the target resource. |
version | Number | Required if the target resource is versioned | Expected current version of the resource on which to apply the changes. |
actions | Array | Required | One or more resource-specific update actions. |
actions
array are treated as additional fields and ignored by the API. In particular, no error is sent back.Some resource updates or state transitions might not be compatible with each other within a single request. In these cases, dedicated endpoints might be provided to handle specific updates or state transitions individually.
The following example shows a request that contains two update actions:
{
"id": "a136fd9e-bc29-4d05-813f-350b248aefd8",
"version": 1,
"actions": [
{
"action": "setTitle",
"title": "My title"
},
{
"action": "setText",
"text": "My text"
}
]
}
Update guarantees
The API provides different update guarantees for different kinds of requests.
Strong consistency
Each resource offers read-after-write consistency. The read-after-write consistency implies that when you update an entity, wait for the success response, and then read the same entity for your changes to be visible.
If a server-side problem occurs, indicated by a 500 Internal Server Error HTTP response, the requested action may still successfully complete after the error is returned. If you receive this error, you should verify the status of the requested resource.
When new aggregates are being created there is no general solution to verify the status of the resource. Querying for a supplied unique identifier can help. For example, if you are creating an Order, you could query using the Order number that you supplied during the creation request.
Eventual consistency
Eventual Consistency
guarantee. It means that after a delay, what you can read reflects all the past updates.These delays can vary depending on the amount of data to update. We try to keep these delays as short as possible.
Optimistic concurrency control
version
attribute.
When sending (partial updates) to these resources, send the expected version of a resource as a part of the request.After making an update, the HTTP response body contains the new version of the resource. An API client must wait for the latest version before sending follow-up requests.
Background processes and other events can also change a resource's version without any requests sent from an API client directly to the resource. Do not rely on a consistently incremented version number.
ETag
and If-Match
HTTP headers
for the purpose of optimistic concurrency control.Security
404
error, including an HTML body response on every attempt.It is not necessary to support the full set of SSL/TLS features listed here (or stop supporting other features), especially the cipher suites, as long as one valid combination exists. The TLS handshake includes a "negotiation phase" that chooses the best combination that both client and server support.
- Ciphersuites:
ECDHE-ECDSA-AES256-GCM-SHA384:ECDHE-RSA-AES256-GCM-SHA384:ECDHE-ECDSA-CHACHA20-POLY1305:ECDHE-RSA-CHACHA20-POLY1305:ECDHE-ECDSA-AES128-GCM-SHA256:ECDHE-RSA-AES128-GCM-SHA256:ECDHE-ECDSA-AES256-SHA384:ECDHE-RSA-AES256-SHA384:ECDHE-ECDSA-AES128-SHA256:ECDHE-RSA-AES128-SHA256
- Versions: TLS 1.2, TLS 1.3
Query features
Single resource query response
Query responses for single resources only return a single, top-level JSON object.
PagedQueryResult
limit Int | Number of results requested. Default: 20 Minimum: 0 Maximum: 500 |
offset Int | Number of elements skipped. Default: 0 Minimum: 0 Maximum: 10000 |
count Int | Actual number of results returned. |
total Int | Total number of results matching the query.
This number is an estimation that is not strongly consistent.
This field is returned by default.
For improved performance, calculating this field can be deactivated by using the query parameter withTotal=false .
When the results are filtered with a Query Predicate, total is subject to a limit. |
results Array of BaseResource | The resources matching the query predicate. Each query endpoint returns resources of its specific type. |
meta Object | Object containing supplementary information about the results. |
Sorting
sort
query parameter.For sorting based on values of standard fields of resources the provided parameter value must be a valid sort expression of the form:
<field name> <sort direction>
<field name>.<locale> <sort direction>
custom.fields.<field name> <sort direction>
ASC
. The allowed sort paths are typically listed on the
specific query endpoints.Find below some examples of sort expressions:
key asc
name.en asc
custom.fields.trackingNumber desc
sort
parameters, they are combined into a composed sort where the results are first sorted
by the first expression, followed by equal values being sorted according to the second expression, and so on.The sorting is case-sensitive.
Pagination
limit
and offset
query parameters, allowing clients to request a certain page of the query result.Limit
limit
query parameter. Allowed is a value between 0 and 500
. The default limit
on most query endpoints is 20.Offset
offset
query parameter.
offset
defines the number of elements skipped, it is not a page number.
The default value is 0, indicating that no results should be skipped.
The maximum offset is 10 000
.Total
total
is limited to the maximum offset
when the results are filtered with a Query Predicate.total
field in the PagedQueryResult should be deactivated by passing false
in the withTotal
query parameter whenever possible.Iterate over all elements
When iterating over all elements from an endpoint (for example to modify or export all Products), using offset is not suitable to page through results:
offset
has an upper limit (see above), so exports can later become incomplete when the number of resources grows.- When an element is inserted while iterating over pages, a request with the next offset value might return already processed elements.
- Using large offsets can lead to slower response times.
offset
, you should follow this alternative approach instead:- Query for a limited number of results in which the elements are sorted by ID. Do not request the
total
field by passingwithTotal=false
. - To retrieve the next chunk of elements, iterate with the same query by filtering for IDs greater than the last ID retrieved from the previous step.
- Repeat step 2 as long as the number of returned results equals the requested limit.
The following pseudocode demonstrates the approach using chunks of 100 elements at each iteration:
lastId = null
continue = true
while (continue) {
if (lastId == null)
response = client.get(endpoint + "?withTotal=false&limit=100&sort=id+asc")
else
response = client.get(endpoint + "?withTotal=false&limit=100&sort=id+asc&where=id%3E" + lastId) // "id%3E" is "id >" encoded
results = response.results
continue = (results.size == 100)
lastId = results.last.id
// do something with the result
}
Resource timestamps
createdAt
and the lastModifiedAt
timestamps.
Those timestamps are automatically set by the server on creation or modification of the resource. They can be used as predicates and for sorting in queries.Reference Expansion
Reference Expansion can be used when creating, updating, querying, and deleting these resources.
Expansion of references does not cause requests to fail even if not all of the references can be expanded (for example, because the referenced resource cannot be found). The server will try to expand as much as possible but never beyond what the client requested. It is up to the client to decide which referenced resources it deems critical for completing a specific use-case, for example, by either treating it as an error or by just not rendering/returning the missing information.
Endpoints supporting Reference Expansion
expand
query parameter, which can be used to specify one or more expansion paths. For specifying several expansion paths in the same request, use the expand
query parameter several times with different parameter values.Expansion paths
Expansion paths define the full path to the reference in a resource. Expansion path format:
refFieldName
- expands a reference field namedrefFieldName
located at the root of the resourcerefArrayFieldName[*]
- expands all references contained in the array of references namedrefArrayFieldName
refArrayFieldName[0]
- expands the first reference in the array of references namedrefArrayFieldName
objFieldName.refFieldName
- expands the referencerefFieldName
contained in the objectobjFieldName
objArrayFieldName[*].refFieldName
- expands the referencerefFieldName
in all objects contained in the arrayobjArrayFieldName
objNestedArrayFieldName[*][*].refFieldName
- expands the referencerefFieldName
in all objects contained in the nested arrayobjNestedArrayFieldName
obj
field within the expansion path. For example, to expand the Customer Group of a Customer in a ShoppingList, use customer.customerGroup
.GET https://api.{region}.commercetools.com/<project-key>/shopping-lists/<example-shopping-list-id>?expand=customer.customerGroup
Examples for Reference Expansion
Product Type and Tax Category on a Product
Example-Product
representation that contains two references: one to an Example-Product-Type
and one to an Example-TaxCategory
. Without Reference Expansion you will get such a representation as response from the endpoints:{
"id": "<example-product-id>",
"version": 4,
"productType": {
"typeId": "product-type",
"id": "<example-product-type-id>"
},
...
"taxCategory": {
"typeId": "tax-category",
"id": "<example-tax-category-id>"
},
...
}
obj
field as shown in the following JSON snippet:{
"id": "<example-product-id>",
"version": 4,
"productType": {
"typeId": "product-type",
"id": "<example-product-type-id>"
"obj": {
"id": "<example-product-type-id>",
"version": 4,
"name": "Example Product Type",
"description": "example Product Type to showcase Reference Expansion",
"classifier": "Complex",
"attributes": [
....
],
},
...
"taxCategory": {
"typeId": "tax-category",
"id": "<tax-category-id>"
"obj": {
"id": "<tax-category-id>",
"version": 2,
"name": "Standard tax category",
"rates": [
{
"name": "5% US",
"amount": 0.05,
"includedInPrice": false,
"country": "US",
"id": "zJJ5KIGH"
},
{
"name": "19% MwSt",
"amount": 0.19,
"includedInPrice": true,
"country": "DE",
"id": "893f_nXJ"
}
],
...
}
Example Product
introduced before - including both expansion paths: productType
and taxCategory
- would look like this:GET https://api.{region}.commercetools.com/<project-key>/products/<example-product-id>?expand=productType&expand=taxCategory
Categories on a Product
masterData.current.categories[*]
.The query would thus, look like this:
GET https://api.{region}.commercetools.com/<project-key>/products/<example-product-id>?expand=masterData.current.categories%5B*%5D
masterData.current.categories[*].parent
as expansion path.ReferenceType Attributes on a Product
masterVariant
and one for the other variants
on the Product) to get the expanded objects on all the Product Variants. The example below specifies the expansion path for the current
ProductCatalogData, but it works similarly for the staged
catalog data.masterData.current.masterVariant.attributes[*].value
masterData.current.variants[*].attributes[*].value
GET https://api.{region}.commercetools.com/<project-key>/products/<example-product-id>?expand=masterData.current.masterVariant.attributes%5B*%5D.value&expand=masterData.current.variants%5B*%5D.attributes%5B*%5D.value
value
field in the expansion path with a wildcard [*]
, like so:masterData.current.masterVariant.attributes[*].value[*]
masterData.current.variants[*].attributes[*].value[*]
Line Item States on an Order
lineItems[*].state[*].state
.GET https://api.{region}.commercetools.com/<project-key>/orders/<example-order-id>?expand=lineItems%5B*%5D.state%5B*%5D.state
ReferenceType Custom Fields
custom.fields.nameOfCustomField
.GET https://api.{region}.commercetools.com/<project-key>/categories/<example-category-id>?expand=custom.fields.nameOfCustomField
custom.fields.nameOfCustomFieldSet[*]
GET https://api.{region}.commercetools.com/<project-key>/categories/<example-category-id>?expand=custom.fields.nameOfCustomFieldSet%5B*%5D
Include custom data by expanding Custom Objects
value
field, which can contain any JSON value. You can associate data stored in Custom Objects to Products and other resources and retrieve this data using reference expansion on the reference type Attribute or Custom Field. With this approach, you can query for the custom data associated to the resource without the need for Custom Objects scopes.key-value-document
. To expand this Attribute, see ReferenceType Attributes on a Product.key-value-document
. You can then assign values for this Custom Field to entities using the custom
field in the draft, or the Set Custom Type update action. To expand this Custom Field, see ReferenceType Custom Fields.Cleanup of old data
Automatic cleanup
To maintain optimal performance, an automatic clean-up task runs in the background. This task deletes resources after a set number of inactive days; however, the deletion timer resets with each modification to the resource.
While default retention periods exist for each affected resource type, you have the flexibility to modify these durations based on your specific needs. For example, Carts have a default retention period of 90 days. You might decide to keep the 90-day default for anonymous Carts, but then override the value for individual Carts of logged-in customers, allowing them to persist for longer. Similarly, you could keep the 360-day default for the removal of Shopping Lists but set a lengthier duration for Shopping Lists that have been publicly shared.
The following table details the default retention period for each affected resource type and how the defaults can be adjusted.
Resource type | Default retention period | Configure the retention period |
---|---|---|
Refresh tokens | 200 days | To change the default expiration time, configure the refresh token validity for each API Client. |
Carts | 90 days | To change the Project default, use the Change Carts configuration update action. To change the default for individual resources, use the Set DeleteDaysAfterLastModification update action. |
Shopping Lists | 360 days | To change the Project default, use the Change ShoppingLists configuration update action. To change the default for individual resources, use the Set DeleteDaysAfterLastModification update action. |
Messages | 15 days, if enabled | To change the Project default, use the Change Messages Configuration update action. |
Create carts and anonymous sessions only when necessary
Creating a new cart or an anonymous session on the first page visit can result in a very high number of resources, for example, if a stateless web crawler acts like a new Customer on every request or if a lot of visitors drop off after one or two page views. To avoid creating unused resources, only create them when needed (for example, when the visitor is adding a Product to the cart).
Correlation ID
correlation IDs are strings that may only contain alphanumeric characters, underscores, and hyphens and have a length of 8 to 256 characters. UUIDs can be used to ensure uniqueness.
X-Correlation-ID
is used to propagate the correlation ID on HTTP requests and responses. If a client does not provide a correlation ID for an HTTP request, the Composable Commerce API generates one.When contacting support about a specific request, include the correlation ID, as it helps us locate the request more easily.
Client logging BETA
createdAt
, createdBy
BETA, lastModifiedAt
, and lastModifiedBy
BETA fields.createdBy
BETA field is present on resources created after 1 February 2019. The lastModifiedBy
BETA field is present on resources created and/or updated after 1 February 2019.Events tracked
createdBy
BETA, createdAt
, lastModifiedBy
BETA, and lastModifiedAt
fields are added.lastModifiedBy
BETA and lastModifiedAt
fields are updated. However, if the updated resource does not differ from the current resource, the action might be skipped and these fields are not updated. An API Client update refers to any modifications made through a client application interacting with either the HTTP or GraphQL API, or the Merchant Center.attributedTo
field of either the createdBy
or lastModifiedBy
fields.lastModifiedBy
BETA field is not updated. In some cases, only the lastModifiedAt
field is updated.
For example, activating a Product Discount updates the Price of a Product. As this update is triggered as a background process, and not by an API call or the Merchant Center, the Price update is not tracked.Information in the fields
createdBy
BETA and lastModifiedBy
BETA fields do not contain any personally identifiable information.
However, they can contain an external user ID, reference to the Customer ID, or identifier for an anonymous session in optional fields. The fields themselves are JSON objects. For more information, see the CreatedBy and LastModifiedBy common types.External user IDs
X-External-User-ID
HTTP header. It can be useful for tracking changes made by users in an external service. For example, if you do not use the Merchant Center or the API authorization flows, using the X-External-User-ID
HTTP header can provide more information in client logging fields than might otherwise be available.externalUserId
field returns information passed in this header, and is present on most representations that are passed to your front-end applications. Do not pass personal information, such as user email addresses to the X-External-User-ID
header.X-External-User-ID
header, it is your responsibility to encrypt any information passed to the header or be GDPR and security-compliant.external_user_id:{externalId}
in the scope. This ID is then used in the externalUserId
field.