Class ProductRowConverter
- Author:
- Kelly Tisdell (ktisdell)
-
Nested Class Summary
Modifier and TypeClassDescriptionstatic final class
Properties that can be provided in a product option column to configure an option.static final class
-
Field Summary
Modifier and TypeFieldDescriptionstatic final String
InBatchRequest.BatchContext.getAdditionalContextMap()
, we expect a nested map under this key to contain prefetched products by their external ID.static final String
InBatchRequest.BatchContext.getAdditionalContextMap()
, we expect a nested map under this key to contain prefetched products by their ID.Fields inherited from class com.broadleafcommerce.common.dataimport.conversion.AbstractRowConverter
COMPLEX_COLUMN_HEADER_PREFIX_DELIMITER
-
Constructor Summary
ConstructorDescriptionProductRowConverter
(com.fasterxml.jackson.databind.ObjectMapper mapper, com.broadleafcommerce.common.extension.TypeFactory typeFactory, com.broadleafcommerce.common.dataimport.util.IdResolver idResolver) -
Method Summary
Modifier and TypeMethodDescriptionboolean
canConvert
(@NonNull com.broadleafcommerce.common.dataimport.messaging.BatchRecord record, com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) com.broadleafcommerce.common.dataimport.util.ConversionUtils.ConversionResponse<Product>
convert
(Object parent, @NonNull com.broadleafcommerce.common.dataimport.messaging.BatchRecord record, com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) protected com.broadleafcommerce.data.tracking.core.type.OperationType
determineOperationType
(@NonNull Product product, boolean productAlreadyExistsInDatastore, @NonNull com.broadleafcommerce.common.dataimport.messaging.BatchRecord record, com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) Determine the effective operation type for the given batch record and product instance.extractSpecialColumns
(@NonNull Map<String, String> mutableRow, com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) protected String
getAttributeName
(Map<String, String> propertyMap) protected com.broadleafcommerce.common.dataimport.util.IdResolver
protected com.fasterxml.jackson.databind.ObjectMapper
protected Product
getPrefetchedProduct
(com.broadleafcommerce.common.dataimport.messaging.BatchRecord batchRecord, Map<String, String> batchRecordRow, com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) Checks theBatchRequest.BatchContext.getAdditionalContextMap()
to see if there is already a pre-fetched product matching thebatchRecord
.protected com.broadleafcommerce.common.extension.TypeFactory
protected VariantRowConverter
Lazy injection to avoid circular dependency issues.protected void
initializeData
(@NonNull Product product, @NonNull com.broadleafcommerce.common.dataimport.messaging.BatchRecord record, @NonNull Map<String, String> mutableRow, @NonNull com.broadleafcommerce.data.tracking.core.type.OperationType operation, com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) Initialize the data from the row onto the product.instantiateOrGetPrefetchedProduct
(@NonNull com.broadleafcommerce.common.dataimport.messaging.BatchRecord batchRecord, @NonNull Map<String, String> mutableRow, com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) We expect theCompleteProductImportBatchHandler
to have pre-fetched products by various identifier fields if supplied in the original rows.protected Product
modifyProduct
(@NonNull Product product, @NonNull com.broadleafcommerce.common.dataimport.messaging.BatchRecord record, com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) Hook point to allow implementers to modify the details of the product or return a different product instance.protected void
parseAndSetAttributes
(@NonNull Product product, @NonNull com.broadleafcommerce.data.tracking.core.type.OperationType operationType, @NonNull com.broadleafcommerce.common.dataimport.messaging.BatchRecord record, @NonNull Map<String, String> specialColumns, com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) protected void
parseAndSetCurrency
(@NonNull Product product, @NonNull com.broadleafcommerce.data.tracking.core.type.OperationType operationType, @NonNull com.broadleafcommerce.common.dataimport.messaging.BatchRecord record, @NonNull Map<String, String> mutableRow, com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) Currency is not complex but gets special treatment because we should not mix different currencies in a single Catalog without using the Pricing Service.protected void
parseAndSetFulfillmentFlatRates
(@NonNull Product product, @NonNull com.broadleafcommerce.data.tracking.core.type.OperationType operationType, @NonNull com.broadleafcommerce.common.dataimport.messaging.BatchRecord record, @NonNull Map<String, String> mutableRow, com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) protected void
parseAndSetKeywords
(@NonNull Product product, @NonNull com.broadleafcommerce.data.tracking.core.type.OperationType operationType, @NonNull com.broadleafcommerce.common.dataimport.messaging.BatchRecord record, @NonNull Map<String, String> specialColumns, com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) protected void
parseAndSetOptions
(@NonNull Product product, @NonNull com.broadleafcommerce.data.tracking.core.type.OperationType operationType, @NonNull com.broadleafcommerce.common.dataimport.messaging.BatchRecord record, @NonNull Map<String, String> mutableRow, com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) The syntax we expect for theProductRowConverter.ProductProperties.OPTIONS_HEADER
column is a series of individual option values separated byRowUtils.VALUE_SEPARATOR
.protected String
resolveProductId
(@NonNull Product product, @NonNull com.broadleafcommerce.common.dataimport.messaging.BatchRecord record, @NonNull com.broadleafcommerce.data.tracking.core.type.OperationType operation, com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) protected void
setProductPricingKeyIfNeeded
(Product product, com.broadleafcommerce.common.dataimport.messaging.BatchRecord productRecord, com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context, com.broadleafcommerce.data.tracking.core.type.OperationType operationType) protected void
setProductTypeBasedOnRecord
(Product product, com.broadleafcommerce.common.dataimport.messaging.BatchRecord productRecord, com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context, com.broadleafcommerce.data.tracking.core.type.OperationType operationType) SetProduct.getProductType()
based on whether or not the givenBatchRecord
for product has any variants.protected void
setProductUriIfNeeded
(Product product, com.broadleafcommerce.common.dataimport.messaging.BatchRecord productRecord, com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context, com.broadleafcommerce.data.tracking.core.type.OperationType operationType) void
setVariantRowConverter
(VariantRowConverter variantRowConverter) Lazy injection to avoid circular dependency issues.toKeyAttributePair
(String[] keyValueArray) protected org.springframework.data.util.Pair<String,
FulfillmentFlatRate> toKeyFulfillmentFlatRatePair
(String[] keyValueArray)
-
Field Details
-
BATCH_CONTEXT_PREFETCHED_PRODUCTS_BY_ID_MAP
InBatchRequest.BatchContext.getAdditionalContextMap()
, we expect a nested map under this key to contain prefetched products by their ID.- See Also:
-
BATCH_CONTEXT_PREFETCHED_PRODUCTS_BY_EXTERNAL_ID_MAP
InBatchRequest.BatchContext.getAdditionalContextMap()
, we expect a nested map under this key to contain prefetched products by their external ID.- See Also:
-
-
Constructor Details
-
ProductRowConverter
public ProductRowConverter(com.fasterxml.jackson.databind.ObjectMapper mapper, com.broadleafcommerce.common.extension.TypeFactory typeFactory, com.broadleafcommerce.common.dataimport.util.IdResolver idResolver)
-
-
Method Details
-
convert
@NonNull public com.broadleafcommerce.common.dataimport.util.ConversionUtils.ConversionResponse<Product> convert(@Nullable Object parent, @NonNull @NonNull com.broadleafcommerce.common.dataimport.messaging.BatchRecord record, @Nullable com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) - Specified by:
convert
in classcom.broadleafcommerce.common.dataimport.conversion.AbstractRowConverter<com.broadleafcommerce.common.dataimport.util.ConversionUtils.ConversionResponse<Product>>
-
canConvert
public boolean canConvert(@NonNull @NonNull com.broadleafcommerce.common.dataimport.messaging.BatchRecord record, @Nullable com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) - Specified by:
canConvert
in classcom.broadleafcommerce.common.dataimport.conversion.AbstractRowConverter<com.broadleafcommerce.common.dataimport.util.ConversionUtils.ConversionResponse<Product>>
-
determineOperationType
protected com.broadleafcommerce.data.tracking.core.type.OperationType determineOperationType(@NonNull @NonNull Product product, boolean productAlreadyExistsInDatastore, @NonNull @NonNull com.broadleafcommerce.common.dataimport.messaging.BatchRecord record, @Nullable com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) Determine the effective operation type for the given batch record and product instance. This can help inform decisions on how to instantiate/map fields.- Parameters:
product
- the resolved product instanceproductAlreadyExistsInDatastore
- whether the product was determined to already exist in the datastore. SeeinstantiateOrGetPrefetchedProduct(BatchRecord, Map, BatchContext)
record
- the original batch recordcontext
- the batch context- Returns:
- the effective operation type for the given batch record
-
resolveProductId
protected String resolveProductId(@NonNull @NonNull Product product, @NonNull @NonNull com.broadleafcommerce.common.dataimport.messaging.BatchRecord record, @NonNull @NonNull com.broadleafcommerce.data.tracking.core.type.OperationType operation, @Nullable com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) -
parseAndSetKeywords
protected void parseAndSetKeywords(@NonNull @NonNull Product product, @NonNull @NonNull com.broadleafcommerce.data.tracking.core.type.OperationType operationType, @NonNull @NonNull com.broadleafcommerce.common.dataimport.messaging.BatchRecord record, @NonNull @NonNull Map<String, String> specialColumns, @Nullable com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) -
parseAndSetAttributes
protected void parseAndSetAttributes(@NonNull @NonNull Product product, @NonNull @NonNull com.broadleafcommerce.data.tracking.core.type.OperationType operationType, @NonNull @NonNull com.broadleafcommerce.common.dataimport.messaging.BatchRecord record, @NonNull @NonNull Map<String, String> specialColumns, @Nullable com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) -
toKeyAttributePair
-
parseAndSetOptions
protected void parseAndSetOptions(@NonNull @NonNull Product product, @NonNull @NonNull com.broadleafcommerce.data.tracking.core.type.OperationType operationType, @NonNull @NonNull com.broadleafcommerce.common.dataimport.messaging.BatchRecord record, @NonNull @NonNull Map<String, String> mutableRow, @Nullable com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) The syntax we expect for theProductRowConverter.ProductProperties.OPTIONS_HEADER
column is a series of individual option values separated byRowUtils.VALUE_SEPARATOR
.Each option value consists of a few properties separated by
RowUtils.PROPERTY_SEPARATOR
. Assignment of property values is done withRowUtils.VALUE_ASSIGNMENT_SEPARATOR
.For convenience, some properties can be omitted in certain scenarios with an implied default. Here are a few examples of option values:
label::Label for Option and Attribute Choice;required::true;type::Size
label::Label for Option and Attribute Choice;type::Size
-> implies 'required' is falselabel::Size
-> implies 'required' is false, and 'type' will be set to match 'label' since it matches one ofDefaultAttributeChoiceType
Size
-> implies 'required' is false, and both 'label' and 'type' will be set to match the given value since it matches one ofDefaultAttributeChoiceType
Note that 'type' is only optional in situations where the given 'label' value matches (ignoring case) one of
DefaultAttributeChoiceType
. This prevents inadvertent creation/usage of random types. If custom types need to be used, they should be explicitly provided.Furthermore,
AttributeChoice.getAllowedValues()
are set on each option from the values found inCompleteProductImportBatchHandler.RecordTypes.VARIANT
dependent records of the productrecord
.- Parameters:
product
- the product to set the parsed options onoperationType
- the determined operation typerecord
- the original product batch recordmutableRow
- the row from which to obtain the options columncontext
- the batch context
-
parseAndSetCurrency
protected void parseAndSetCurrency(@NonNull @NonNull Product product, @NonNull @NonNull com.broadleafcommerce.data.tracking.core.type.OperationType operationType, @NonNull @NonNull com.broadleafcommerce.common.dataimport.messaging.BatchRecord record, @NonNull @NonNull Map<String, String> mutableRow, @Nullable com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) Currency is not complex but gets special treatment because we should not mix different currencies in a single Catalog without using the Pricing Service. Therefore, this method will ensure that there is no currency mismatch issue. By default, this is actually a no-op becauseJpaProduct.preToMe(ContextInfo, Object)
already handles setting the currency.- Parameters:
product
- the product to set the parsed currency onoperationType
- the determined operation typerecord
- the original product batch recordmutableRow
- the row from which to obtain the currency columncontext
- the batch context
-
parseAndSetFulfillmentFlatRates
protected void parseAndSetFulfillmentFlatRates(@NonNull @NonNull Product product, @NonNull @NonNull com.broadleafcommerce.data.tracking.core.type.OperationType operationType, @NonNull @NonNull com.broadleafcommerce.common.dataimport.messaging.BatchRecord record, @NonNull @NonNull Map<String, String> mutableRow, @Nullable com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) -
toKeyFulfillmentFlatRatePair
protected org.springframework.data.util.Pair<String,FulfillmentFlatRate> toKeyFulfillmentFlatRatePair(String[] keyValueArray) -
instantiateOrGetPrefetchedProduct
protected org.springframework.data.util.Pair<Boolean,Product> instantiateOrGetPrefetchedProduct(@NonNull @NonNull com.broadleafcommerce.common.dataimport.messaging.BatchRecord batchRecord, @NonNull @NonNull Map<String, String> mutableRow, @Nullable com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) We expect theCompleteProductImportBatchHandler
to have pre-fetched products by various identifier fields if supplied in the original rows. Any products that were found in the datastore should have been populated intoBatchRequest.BatchContext.getAdditionalContextMap()
.If a product is present in this map, we will use this existing instance.
If a product is not present in this map, we will simply instantiate a new instance.
- Parameters:
batchRecord
- the originalBatchRecord
mutableRow
- the row from the batch record (usually with any special columns pre-removed viaextractSpecialColumns(Map, BatchContext)
)context
- theBatchRequest.BatchContext
, which should have any pre-fetched products inBatchRequest.BatchContext.getAdditionalContextMap()
- Returns:
- a
Pair
containing a boolean describing whether the product was found in the pre-fetched map, as well as the product instance itself
-
getPrefetchedProduct
@Nullable protected Product getPrefetchedProduct(com.broadleafcommerce.common.dataimport.messaging.BatchRecord batchRecord, Map<String, String> batchRecordRow, @Nullable com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) Checks theBatchRequest.BatchContext.getAdditionalContextMap()
to see if there is already a pre-fetched product matching thebatchRecord
.- Parameters:
batchRecord
- the batch record for which to find the corresponding pre-fetched product instance (if exists)batchRecordRow
- the row from the batch record (pre-processed if necessary)context
- the batch context which should contain pre-fetched products set byCompleteProductImportBatchHandler
- Returns:
- the pre-fetched product instance if found, or
null
-
modifyProduct
protected Product modifyProduct(@NonNull @NonNull Product product, @NonNull @NonNull com.broadleafcommerce.common.dataimport.messaging.BatchRecord record, @Nullable com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) Hook point to allow implementers to modify the details of the product or return a different product instance.This MUST return a non-null value. By default, it returns the provided value.
- Parameters:
product
- the product that has been created populated with data from the rowrecord
- the record that represents the row that was mapped to the productcontext
- the batch context
-
initializeData
protected void initializeData(@NonNull @NonNull Product product, @NonNull @NonNull com.broadleafcommerce.common.dataimport.messaging.BatchRecord record, @NonNull @NonNull Map<String, String> mutableRow, @NonNull @NonNull com.broadleafcommerce.data.tracking.core.type.OperationType operation, @Nullable com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) Initialize the data from the row onto the product. Set additional data as needed. For example, set ID and activeStartDate.- Parameters:
product
- the target product onto which we set data.record
- the original BatchRecordmutableRow
- the mutable row from the BatchRecord with entries removed that cannot be handled by reflectionoperation
- the operation; typically CREATE or UPDATEcontext
- the BatchContext
-
extractSpecialColumns
protected Map<String,String> extractSpecialColumns(@NonNull @NonNull Map<String, String> mutableRow, @Nullable com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context) - Overrides:
extractSpecialColumns
in classcom.broadleafcommerce.common.dataimport.conversion.AbstractRowConverter<com.broadleafcommerce.common.dataimport.util.ConversionUtils.ConversionResponse<Product>>
-
setProductTypeBasedOnRecord
protected void setProductTypeBasedOnRecord(Product product, com.broadleafcommerce.common.dataimport.messaging.BatchRecord productRecord, @Nullable com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context, com.broadleafcommerce.data.tracking.core.type.OperationType operationType) Set
Product.getProductType()
based on whether or not the givenBatchRecord
for product has any variants. If it does,Product.getProductType()
is set toVARIANT_BASED
. Otherwise,STANDARD
. This way, theProduct.getProductType()
wouldn't need to be explicitly declared.The
Product.getProductType()
should only be set if theBatchRecord.getOperation()
isCREATE
, sinceProduct.getProductType()
should never be changed once it's set.This method will set the product type to null if the operation is anything but create. This will cause the eventual update operation to skip updating the field.
- Parameters:
product
- product created from theBatchRecord
productRecord
-BatchRecord
for the productcontext
-BatchRequest.BatchContext
-
setProductUriIfNeeded
protected void setProductUriIfNeeded(Product product, com.broadleafcommerce.common.dataimport.messaging.BatchRecord productRecord, @Nullable com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context, com.broadleafcommerce.data.tracking.core.type.OperationType operationType) -
setProductPricingKeyIfNeeded
protected void setProductPricingKeyIfNeeded(Product product, com.broadleafcommerce.common.dataimport.messaging.BatchRecord productRecord, @Nullable com.broadleafcommerce.common.dataimport.messaging.BatchRequest.BatchContext context, com.broadleafcommerce.data.tracking.core.type.OperationType operationType) -
getAttributeName
-
getMapper
protected com.fasterxml.jackson.databind.ObjectMapper getMapper() -
getTypeFactory
protected com.broadleafcommerce.common.extension.TypeFactory getTypeFactory() -
getIdResolver
protected com.broadleafcommerce.common.dataimport.util.IdResolver getIdResolver() -
getVariantRowConverter
Lazy injection to avoid circular dependency issues. -
setVariantRowConverter
Lazy injection to avoid circular dependency issues.
-