Broadleaf Microservices
  • v1.0.0-latest-prod

Comprehensive Products

Overview

A Comprehensive Product is a data structure that roughly mirrors a com.broadleafcommerce.catalog.domain.product.Product from Catalog Services and provides constituent references to things like Variants, Inventory, Prices, and Translations.

Endpoints

There is currently one endpoint associated with Comprehensive Product processing:

  • /data-exchange/comprehensive-product (POST)

This endpoint handles all necessary creates/updates/deletes, so there are no PUT/PATCH/DELETE endpoints.

Data Model

The following is the payload for the request and the response for a call to either of the endpoints defined, above. Errors should not be passed in as part of a request and are ignored. Rather, errors are set on the response. The following class representations generally include everything defined in the DataExchangeServices. However, it should be stressed that most of these entities make use of a @JsonAnyGetter and @JsonAnySetter to allow additonal, complete data to be set. This data is generally defined by the resource services (e.g. CatalogServices) that is being called behind the scenes and to which this data is being passed along. Note that DataExchangeServices, by default, does not perform data validation. As a result, it is important that the client pass the correct data, generally according to the projection domain expectation of the resource tier service (e.g. CatalogServices). For example, ComprehensiveProduct#productType expects an enumerated value. These values are defined in the CatalogServices and include STANDARD, VARIANT_BASED, BUNDLE, MERCHANDISING_PRODUCT, SELECTOR. There are additional fields, not explicitly defined by the entities in DataExchangeServices, that are available on Product in CatalogServices. For example, keywords - a List of Strings; terms - a List of ProductTerm instances, and attributes - a Map of Attribute. Anything that is stored on the product as a field in the DB or as a JSON structure stored as a String field in the DB is a candidate for a pass-through field. If something requires a separate call to process (e.g. Variants), then those entity relationships should be explicitly defined and the processing should be defined by an extension and registration of a AbstractBatchItemHandler.

@Data
public class ProductList<T extends ComprehensiveProduct> implements Serializable, Iterable<T> {
    @JsonAlias({"products", "content"})
    private List<T> products = new ArrayList<>();

    @JsonAlias({"errors"})
    private List<DataExchangeError> errors;
}

The following is an abstract base class for most items that require independent processing and has its own ID. The newlyCreated indicates if the item was created in the context of this request (otherwise indicating that there should be no child entities to fetch or replace since this is a new entity).

@Data
@EqualsAndHashCode(callSuper = true)
public abstract class EntityExchangeObject extends AbstractExchangeObject {
    /**
     * The ID of the entity.
     */
    private String id;

    @JsonIgnore
    private boolean newlyCreated = false;
}

The following is an abstract base class for most items that require independent processing. Each of these has a UUID that represents its correlationId. This also includes a parentCorrelationId that gets set during processing. This is the correlationId of the parent object. There is a Map<String,Object> that is used by the @JsonAnySetter and @JsonAnyGetter to get/set unknown or otherwise unmapped fields. This allows for easy pass-through of data without needing to extend ComprehensiveProduct to send basic fields through to CatalogServices.

@Data
@JsonInclude(JsonInclude.Include.NON_EMPTY)
@ToString(exclude = {"customProperties"})
public abstract class AbstractExchangeObject implements Serializable, ExchangeObject {
    @JsonIgnore
    private String parentCorrelationId;

    @JsonIgnore
    private String correlationId = UUID.randomUUID().toString();

    /**
     * Any custom properties from extensions that should just be passed through.
     */
    private Map<String, Object> customProperties = new HashMap<>();

    @JsonIgnore
    private boolean newlyCreated = false;

    @JsonAnySetter
    public void setCustomProperty(String key, Object value) {
        customProperties.put(key, value);
    }

    @JsonAnyGetter
    public Map<String, Object> getCustomProperties() {
        return customProperties;
    }
}

ComprehensiveProduct is the top-level entity in the context of this service.

@Data
@EqualsAndHashCode(callSuper = true)
@JsonInclude(JsonInclude.Include.NON_NULL)
public class ComprehensiveProduct extends EntityExchangeObject implements Translatable {
    private String name;

    private String description;

    private String sku;

    private String upc;

    private String externalId;

    private Boolean online;

    private String inventoryType;

    private Boolean availableOnline;

    private Boolean searchable;

    private String productType;

    private String businessType;

    private String pricingKey;

    private List<Translation> translations;

    private List<ProductAsset> assets;

    private List<Variant> variants;

    private List<SkuInventory> skuInventories;

    private List<Prices> prices;

    private List<ProductTag> productTags;
}
@Data
@EqualsAndHashCode(callSuper = true)
@JsonInclude(JsonInclude.Include.NON_NULL)
@JsonView({RequestView.class, ResponseView.class})
public class Translation extends AbstractExchangeObject {
    private String id;

    private String entityField;

    @JsonView(RequestView.class)
    private String entityId;

    private String entityType;

    private Locale locale;

    private Object value;
}
@Data
@EqualsAndHashCode(callSuper = true)
@JsonInclude(JsonInclude.Include.NON_NULL)
@JsonIgnoreProperties(ignoreUnknown = true)
public class ProductAsset extends ExchangeEntityAsset implements Translatable {
    private String id;

    private String productId;

    private Boolean primary;

    private Boolean sorted;

    private List<Translation> translations;
}
@EqualsAndHashCode(callSuper = true)
@Data
public class Variant extends EntityExchangeObject implements Translatable {
    private String sku;

    private String upc;

    private String externalId;

    private String productId;

    private String name;

    private List<Translation> translations;

    private List<ProductTag> productTags;
}
@EqualsAndHashCode(callSuper = true)
@Data
public class SkuInventory extends EntityExchangeObject implements Serializable {
    @JsonAlias({"inventoryLocationId", "inventoryLocationContextId"})
    private String inventoryLocationContextId;

    private SkuRef skuRef;
}
Note
SkuInventory mirrors com.broadleafcommerce.inventory.domain.SkuInventory from InventoryServices and does not explicitly define properties like quantityOnHand, quantityAvailable, futureQuantityReserved, safetyStock, etc. These should be passed along, and they will be forwarded to the InventoryServices.
@Data
@EqualsAndHashCode(callSuper = true)
@JsonInclude(JsonInclude.Include.NON_NULL)
public class Prices extends AbstractExchangeObject {
    private String priceListId;

    private List<PriceData> priceData;
}
@Data
@JsonInclude(JsonInclude.Include.NON_NULL)
public class PriceData implements Serializable {
    private String id;

    private String priceListId;

    private String priceListName;

    private PriceableTargetRef target;

    private MonetaryAmount price;

    /**
     * Any custom properties from extensions that should just be passed through.
     */
    private Map<String, Object> customProperties = new HashMap<>();

    @JsonAnySetter
    public void setCustomProperty(String key, Object value) {
        customProperties.put(key, value);
    }

    @JsonAnyGetter
    public Map<String, Object> getCustomProperties() {
        return customProperties;
    }
}
@Data
@JsonInclude(Include.NON_NULL)
public class PriceableTargetRef implements Serializable {
    private String targetId;

    private String targetType;

    private String vendorRef;
}
@Data
@EqualsAndHashCode(callSuper = true)
public class ProductTag extends EntityExchangeObject {

    private AdvancedTag tag;

    private IdHolder product;

    private IdHolder variant;
}
@Data
public class AdvancedTag {

    private String id;

    private String name;

    /**
     * Any custom properties from extensions that should just be passed through.
     */
    private Map<String, Object> customProperties = new HashMap<>();

    @JsonAnySetter
    public void setCustomProperty(String key, Object value) {
        customProperties.put(key, value);
    }

    @JsonAnyGetter
    public Map<String, Object> getCustomProperties() {
        return customProperties;
    }
}
@Data
public static class IdHolder {
    private String id;

    /**
     * Any custom properties from extensions that should just be passed through.
     */
    private Map<String, Object> customProperties = new HashMap<>();

    @JsonAnySetter
    public void setCustomProperty(String key, Object value) {
        customProperties.put(key, value);
    }

    @JsonAnyGetter
    public Map<String, Object> getCustomProperties() {
        return customProperties;
    }
}

Additional Properties

  • catalog-provider.url (default https://localhost:9461)

  • catalog-provider.read-products-path (default /products)

  • catalog-provider.create-product-path (default /products)

  • catalog-provider.update-product-path (default /products/{productId})

  • catalog-provider.read-product-by-id-path (default /products/{productId})

  • catalog-provider.read-variants-path (default /products/{productId}/variants)

  • catalog-provider.read-variant-by-id-path (default /products/{productId}/variants/{variantId})

  • catalog-provider.create-variant-path (default /products/{productId}/variants)

  • catalog-provider.update-variant-path (default /products/{productId}/variants/{variantId})

  • catalog-provider.delete-variant-path (default /products/{productId}/variants/{variantId})

  • catalog-provider.read-assets-path (default /products/{productId}/assets)

  • catalog-provider.create-asset-path (default /products/{productId}/assets)

  • catalog-provider.update-asset-path (default /products/{productId}/assets/{assetId})

  • catalog-provider.save-product-translations-path (default /products/{productId}/translations/{locale})

  • catalog-providersave-variant-translations-path (default /products/{productId}/variants/{variantId}/translations/{locale})

  • catalog-provider.save-asset-translations-path (default /products/{productId}/assets/{assetId}/translations/{locale})

  • inventory-provider.url (default https://localhost:9461)

  • inventory-provider.read-sku-inventory-path (default /sku-inventory)

  • inventory-provider.create-sku-inventory-path (default /sku-inventory)

  • inventory-provider.update-sku-inventory-path (default /sku-inventory/{id})

  • pricing-provider.url (default https://localhost:9461)

  • pricing-provider.read-price-data-path (default /price-data/{id})

  • pricing-provider.create-price-data-path (default /price-data)

  • pricing-provider.update-price-data-path (default /price-data/{id})

  • pricing-provider.delete-price-data-path (default /price-data/{id})

  • pricing-provider.bulk-read-price-data-path (default /price-data/bulk)

  • pricing-provider.bulk-delete-price-data-pth (default /price-data/bulk)

  • advanced-tag-provider.url (default https://localhost:9461)

  • advanced-tag-provider.read-advanced-tags-path (default /tags)

  • advanced-tag-provider.create-advanced-tag-path (default /tags)

  • advanced-tag-provider.update-advanced-tag-path (default /tags/{id})

  • advanced-tag-provider.delete-advanced-tag-path (default /tags/{id})

  • advanced-tag-provider.save-tag-translations-path (default /tags/{id}/translations/{locale})

  • advanced-tag-provider.product-tag-path (default /products/{productId}/tags)

  • advanced-tag-provider.update-product-tag-path (default /products/{productId}/tags/{tagId})

  • advanced-tag-provider.delete-product-tag-path (default /products/{productId}/tags/{tagId})

  • advanced-tag-provider.variant-tag-path (default /products/{productId}/variants/{variantId}/tags)

  • advanced-tag-provider.update-variant-tag-path (default /products/{productId}/variants/{variantId}/tags/{tagId})

  • advanced-tag-provider.delete-variant-tag-path (default /products/{productId}/variants/{variantId}/tags/{tagId})

Security

The security policy required to access these endpoints has a root of COMPREHENSIVE_PRODUCT.