The module used for importing/syncing Types into a commercetools project. It also provides utilities for generating update actions based on the comparison of a Type against a TypeDraft.
Use the ClientConfigurationUtils which apply the best practices for ProjectApiRoot
creation.
To create ClientCredentials
which are required for creating a client please use the ClientCredentialsBuilder
provided in java-sdk-v2 Client OAUTH2 package
If you have custom requirements for the client creation, have a look into the Important Usage Tips.
final ClientCredentials clientCredentials =
new ClientCredentialsBuilder()
.withClientId("client-id")
.withClientSecret("client-secret")
.withScopes("scopes")
.build();
final ProjectApiRoot apiRoot = ClientConfigurationUtils.createClient("project-key", clientCredentials, "auth-url", "api-url");
The following fields are required to be set in, otherwise, they won't be matched by sync:
Draft | Required Fields | Note |
---|---|---|
TypeDraft | key |
Also, the types in the target project are expected to have the key fields set. |
After the projectApiRoot
is set up, a TypeSyncOptions
should be built as follows:
// instantiating a TypeSyncOptions
final TypeSyncOptions typeSyncOptions = TypeSyncOptionsBuilder.of(projectApiRoot).build();
SyncOptions
is an object which provides a place for users to add certain configurations to customize the sync process.
Available configurations:
A callback that is called whenever an error event occurs during the sync process. Each resource executes its own error-callback. When the sync process of a particular resource runs successfully, it is not triggered. It contains the following context about the error-event:
- sync exception
- type draft from the source
- type of the target project (only provided if an existing type could be found)
- the update-actions, which failed (only provided if an existing type could be found)
final Logger logger = LoggerFactory.getLogger(TypeSync.class);
final TypeSyncOptions typeSyncOptions = TypeSyncOptionsBuilder
.of(projectApiRoot)
.errorCallback((syncException, draft, type, updateActions) ->
logger.error(new SyncException("My customized message"), syncException)).build();
A callback is called whenever a warning event occurs during the sync process. Each resource executes its own warning-callback. When the sync process of a particular resource runs successfully, it is not triggered. It contains the following context about the warning message:
- sync exception
- type draft from the source
- type of the target project (only provided if an existing type could be found)
final Logger logger = LoggerFactory.getLogger(TypeSync.class);
final TypeSyncOptions typeSyncOptions = TypeSyncOptionsBuilder
.of(projectApiRoot)
.warningCallback((syncException, draft, type) ->
logger.warn(new SyncException("My customized message"), syncException)).build();
During the sync process, if a target type and a type draft are matched, this callback can be used to intercept the update request just before it is sent to the commercetools platform. This allows the user to modify update actions array with custom actions or discard unwanted actions. The callback provides the following information :
- type draft from the source
- type from the target project
- update actions that were calculated after comparing both
// Example: Ignore update actions that remove field definition
final TriFunction<
List<TypeUpdateAction>, TypeDraft, Type, List<TypeUpdateAction>>
beforeUpdateTypeCallback =
(updateActions, newTypeDraft, oldType) -> updateActions.stream()
.filter(updateAction -> !(updateAction instanceof TypeRemoveFieldDefinitionAction))
.collect(Collectors.toList());
final TypeSyncOptions typeSyncOptions =
TypeSyncOptionsBuilder.of(projectApiRoot).beforeUpdateCallback(beforeUpdateTypeCallback).build();
During the sync process, if a type draft should be created, this callback can be used to intercept the create request just before it is sent to the commercetools platform. It contains the following information :
- type draft that should be created
Please refer to example in product sync document.
A number that could be used to set the batch size with which types are fetched and processed,
as types are obtained from the target project on commercetools platform in batches for better performance. The algorithm accumulates up to batchSize
resources from the input list, then fetches the corresponding types from the target project on the commecetools platform in a single request. Playing with this option can slightly improve or reduce processing speed. If it is not set, the default batch size is 50 for type sync.
final TypeSyncOptions typeSyncOptions =
TypeSyncOptionsBuilder.of(projectApiRoot).batchSize(30).build();
In the service classes of the commercetools-sync-java library, we have implemented an in-memory LRU cache to store a map used for the reference resolution of the library. The cache reduces the reference resolution based calls to the commercetools API as the required fields of a resource will be fetched only one time. These cached fields then might be used by another resource referencing the already resolved resource instead of fetching from commercetools API. It turns out, having the in-memory LRU cache will improve the overall performance of the sync library and commercetools API. which will improve the overall performance of the sync and commercetools API.
Playing with this option can change the memory usage of the library. If it is not set, the default cache size is 10.000
for type sync.
final TypeSyncOptions typeSyncOptions =
TypeSyncOptionsBuilder.of(projectApiRoot).cacheSize(5000).build();
After all the aforementioned points in the previous section have been fulfilled, to run the sync:
// instantiating a type sync
final TypeSync typeSync = new TypeSync(typeSyncOptions);
// execute the sync on your list of types
CompletionStage<TypeSyncStatistics> syncStatisticsStage = typeSync.sync(typeDrafts);
The result of completing the syncStatisticsStage
in the previous code snippet contains a TypeSyncStatistics
which contains all the stats of the sync process; which includes a report message, the total number of updated, created,
failed, processed types and the processing time of the last sync batch in different time units and in a
human-readable format.
final TypeSyncStatistics stats = syncStatisticsStage.toCompletebleFuture().join();
stats.getReportMessage();
/*"Summary: 2000 types were processed in total (1000 created, 995 updated, 5 failed to sync)."*/
Note The statistics object contains the processing time of the last batch only. This is due to two reasons:
- The sync processing time should not take into account the time between supplying batches to the sync.
- It is not known by the sync which batch is going to be the last one supplied.
-
If two matching
fieldDefinition
s (old and new) on the matchingtype
s (old and new) have a differentFieldType
, the sync will remove the existingfieldDefinition
and then add a newfieldDefinition
with the newFieldType
. -
The
fieldDefinition
with missingfieldType
(isnull
) will not be synced.
Make sure to read the Important Usage Tips for optimal performance.
A utility method provided by the library to compare a Type
with a new TypeDraft
and results in a list of type update actions.
List<TypeUpdateAction> updateActions = TypeSyncUtils.buildActions(type, typeDraft, typeSyncOptions);
Utility methods provided by the library to compare the specific fields of a Type
and a new TypeDraft
, and in turn builds
the update action. One example is the buildChangeNameUpdateAction
which compares names:
Optional<TypeUpdateAction> updateAction = TypeUpdateActionUtils.buildChangeNameUpdateAction(oldType, typeDraft);
More examples of those utils for different types can be found here.
The type-sync uses the JVM-SDK-V2, therefore ensure you Install JVM SDK module commercetools-sdk-java-api
with
any HTTP client module. The default one is commercetools-http-client
.
<!-- Sample maven pom.xml -->
<properties>
<commercetools.version>LATEST</commercetools.version>
</properties>
<dependencies>
<dependency>
<groupId>com.commercetools.sdk</groupId>
<artifactId>commercetools-http-client</artifactId>
<version>${commercetools.version}</version>
</dependency>
<dependency>
<groupId>com.commercetools.sdk</groupId>
<artifactId>commercetools-sdk-java-api</artifactId>
<version>${commercetools.version}</version>
</dependency>
</dependencies>
For client creation use ClientConfigurationUtils which apply the best practices for ProjectApiRoot
creation.
If you have custom requirements for the client creation make sure to replace SphereClientFactory
with ApiRootBuilder
as described in this Migration Document.
As models and update actions have changed in the JVM-SDK-V2 the signature of SyncOptions is different. It's constructor now takes a ProjectApiRoot
as first argument. The callback functions are signed with TypeDraft
, Type
and TypeUpdateAction
from package com.commercetools.api.models.type.*
Note: Type
UpdateAction<Type>
has changed toTypeUpdateAction
. Make sure you create and supply a specific TypeUpdateAction inbeforeUpdateCallback
. For that you can use the library-utilities or use a JVM-SDK builder (see also):
// Example: Create a type update action to change name taking the 'newName' of the typeDraft
final Function<LocalizedString, TypeUpdateAction> changeNameBeforeUpdateAction =
(newName) -> TypeChangeNameAction.builder().name(newName).build();
// Add the change name action to the list of update actions before update is executed
final TriFunction<
List<TypeUpdateAction>, TypeDraft, Type, List<TypeUpdateAction>>
beforeUpdateTypeCallback =
(updateActions, newTypeDraft, oldType) -> {
final TypeUpdateAction beforeUpdateAction =
changeNameBeforeUpdateAction.apply(newTypeDraft.getName());
updateActions.add(beforeUpdateAction);
return updateActions;
};
The type-sync expects a list of TypeDraft
s to process. If you use java-sync-library to sync your types from any external system into a commercetools platform project you have to convert your data into CTP compatible TypeDraft
type. This was done in previous version using DraftBuilder
s.
The V2 SDK do not have inheritance for DraftBuilder
classes but the differences are minor and you can replace it easily. Here's an example:
// TypeDraftBuilder in v1 takes parameters 'key', 'name' and a set of 'resourceTypeIds'
final TypeDraft typeDraft =
TypeDraftBuilder
.of("type-key", ofEnglish("name"), ResourceTypeIdsSetBuilder.of().addCategories().build())
.build();
// TypeDraftBuilder in v2. 'resourceTypeIds' is a list
TypeDraftBuilder.of()
.key("type-key")
.name(LocalizedString.ofEnglish("name"))
.resourceTypeIds(ResourceTypeId.CATEGORY)
.build();
For more information, see the Guide to replace DraftBuilders.
If you sync types between different commercetools projects you have to transform Type
into TypeDraft
.
However, if you need to query Types
from a commercetools project instead of passing TypeQuery
s to a sphereClient
, create (and execute) requests directly from the apiRoot
.
Here's an example:
// SDK v1: TypeQuery to fetch all types
final TypeQuery query = TypeQuery.of();
final PagedQueryResult<Type> pagedQueryResult = sphereClient.executeBlocking(query);
// SDK v2: Create and execute query to fetch all types in one line
final TypePagedQueryResponse result = apiRoot.types().get().executeBlocking().getBody();
Read more about querying resources.
On any other needs to migrate your project using jvm-sdk-v2 please refer to it's Migration Guide.