Tuesday, 21 June 2016

Performance testing - Store front REST APIs - Identify duplicate and redundant REST APIs

If you are lost while finding all the REST requests made from a store page ( like product page, category page or home page ) and looking for a simple tool to debug the performance issues on WCS Store front, then this post is for you.
In just 3 steps, you will be able to identify all the REST requests made from a store page right in the store itself by looking at the store page. No more enabling trace / debugging through trace files / going through JSP code. The store page while displaying the details, will also display all the REST API calls, along with parameters passed to those REST calls and the time taken by each REST request and the JSP from where this REST request is made.

Download the patch that matches your WCS version and follow the steps outlined in Instructions.txt
  1. Performance Test Patch  

Once you have deployed the code, you should be able to see the results on the store front in Footer like below:

JSP == /AuroraStorefrontAssetStore/Widgets/Header/Header.jsp variableName = categoryHierarchy URL = http://localhost:80/search/resources/store/10201/categoryview/@top Total Time elapsed = 16 millSeconds
Paramters {catalogId=[10052], responseFormat=[json], langId=[-1], contractId=[10005], depthAndLimit=[11,11]}


JSP == /AuroraStorefrontAssetStore/Widgets/Header/Header.jsp variableName = categoryHierarchy URL = http://localhost:80/search/resources/store/10201/categoryview/@top Total Time elapsed = 16 millSeconds
Paramters {catalogId=[10052], responseFormat=[json], langId=[-1], contractId=[10005], depthAndLimit=[11,11]}


JSP == /Widgets_701/com.ibm.commerce.store.widgets.ContentRecommendation/ContentRecommendation.jsp variableName = queryRemoteWidgetsConfigResult URL = store/{storeId}/configuration/{uniqueId} Total Time elapsed = 1 millSeconds
Paramters {}


JSP == /Widgets_701/com.ibm.commerce.store.widgets.ContentRecommendation/ContentRecommendation.jsp variableName = queryRemoteWidgetsConfigResult URL = store/{storeId}/configuration/{uniqueId} Total Time elapsed = 1 millSeconds
Paramters {}

Special Thanks to Siddharth for helping with this idea.

Thursday, 24 September 2015

SEO PAGE DEF OVERRIDE



Assume in ItemDisplay.jsp you are displaying title and meta-desc for items using the
below wcf:getData tag.
<wcf:getData type="com.ibm.commerce.seo.facade.datatypes.SEOXXXType" var="seoType" expressionBuilder="getPageDefinitionByCatEntryIdOrPageName">
               <wcf:param name="defaultPageName" value="ITEM_PAGE" />
               <wcf:param name=”catentry_id” value=${param.catentry_id}   
</wcf:getData>

At server side, the title and meta-desc is identified using the following flow chart:
  
Case 1: No specific overrides defined for this catEntry  in SEOPAGEDEFOVR table 
Result: Will use the defaultPageName to get the TMD details. Get the SEOPAGEDEF_ID
for pageName = ITEM_PAGE defined for this store in SEOPAGEDEF table and then get the
details from the lang specific table. 

Case 2: Specific overrides defined for this catEntry. In this case there will be an entry in
the SEOPAGEDEFOVR for this catEntry.
Result: Get the SEOPAGEDEF_ID defined for this catEntry in the SEOPAGEDEFOVR
table and ignore the defaultPageName=ITEM_PAGE parameter. Use this SEOPAGEDEF_ID
and get the details from lang specific table 

Case 3: Parent of this catEntry enforcing its definition for all its children. In this case, 
there will be an entry in SEOPAGEDEFOVR for the parent catEntry
with APPLY_TO_CHILD marked as true. And this catEntry will have no entries.
Result: Get the SEOPAGEDEF_ID defined for the parent catentry and then get details
as in case 2.

When we need TMD for catEntries or categories, we always search for TMD with
predefined pageName, based on the catentry type, say PRODUCT_PAGE, ITEM_PAGE,
KIT_PAGE, BUNDLE_PAGE. If an override is present in SEOPAGEDEFOVR table
for the catentry or catgroup, then that TMD will be displayed. If no overrides are present
for the catentry, then an entry in SEOPAGEDEFOVR with catentry_id = 0 will be searched.
Always there will be one entry in SEOPAGEDEFOVR table with catentry_id set to 0 and
this entry will be treated as default entry. So even overrides can use same PageName
and hence PageName is not unique index in SEOPAGEDEF table.

Sunday, 31 May 2015

WCS Solr framework - REST Services

Starting with FEP-7, WCS provided REST Services to interact with Solr Server directly without going through BOD / Component Services layer. This post explores the new search framework in detail and touches upon various customization points provided by the framework to implement customer requirements.

The best way to start learning WCS Solr framework is to first get yourself familiarized with basics of Solr. Solr mainly consists of 2 parts -
  1.  Indexing the data.
  2. Querying the index to get the desired results.
In this article we will look at how to query the index without worrying too much about the steps involved in building the index. So it's better to get familiar with Solr edisMax (parser used by WCS) query syntax. At the minimum you should understand the usage of: q, fq, fl, qf, facet, bq parameters. (You can install Solr Admin UI which provides a intuitive user interface to query the index and learn more about the query syntax. More info can be found in knowledge center:
http://www-01.ibm.com/support/knowledgecenter/SSZLC2_7.0.0/com.ibm.commerce.admin.doc/concepts/csdsearchperf_dup.htm?lang=en )

Some resources which might be useful to get started are:
Once you are comfortable with Solr basic query syntax you are good to start with WCS - Solr framework. Let us consider a simple example of searching for keyword 'Apple'.

How do you write a DB query to select products whose name or short description contains keyword 'Apple' ?

Lets start with a basic query :
Select * from catentry where catentry_id in (select catentry_id from catentdesc where language_id = -1 and ( name like '%Apple%' or shortdescription like '%Apple%'))

Then add store constraint:
and member_id = (select member_id from storeent where storeent_id = 10001))


To return only product beans and ignore item beans, add below constraint:
and catenttype_id = ‘ProductBean

To return only published catalog entries, add below constraint:
and catentdesc.published = 1

To return products belonging to current sales catalog, the constraint will be:
join catgpenrel where catentry_id in ( ) and catalog_id = ( )

What happens if we want to return products within certain price range, belonging to certain category and want the result set to be sorted by price or relevance ? The query starts getting more and more complicated and difficult to manage.

Let us see how the Solr query looks in this case.

You will start with a simple query using 'q' parameter:
q = "Apple" 

And then tell Solr to search in name and short description fields. Also products where searchTerm appears in name are more relevant than products where search term appears in shortDescription field.
qf=name^10.0 shortDescription^5.0

Add filter queries to return only published products ( ignore item Beans ) belonging to current catalog and store.
fq=catalog_id:"10052"
fq=storeent_id:("10001")
fq=published:1
fq=-catenttype_id_ntk_cs:ItemBean

Return only few selected fields.
fl=catentry_id,storeent_id,buyable,partNumber_ntk

Also return facets based on category and brand name 
facet=true
facet.field=parentCatgroup_id_search
facet.field=mfName_ntk_cs 
f.mfName_ntk_cs.facet.limit=21 
f.mfName_ntk_cs.facet.mincount=1
f.mfName_ntk_cs.facet.sort=count

Add spell check 
spellcheck=true
spellcheck.count=5 
spellcheck.onlyMorePopular=false
spellcheck.accuracy=0.3
spellcheck.alternativeTermCount=5 
spellcheck.maxResultsForSuggest=3
spellcheck.q=apple

Add some meta data around pagination, debugging: 
start=0
rows=50
timeAllowed=15000 
defType=edismax
echoHandler=true
echoParams=all
degug = true

As you can see Solr provides you a simple yet powerful query syntax to query the index and organize the results the way you want. You can set the fields to query, set the fields to retrieve, boost the results based on field names, add sorting, spell check, highlighting features to result set, get the stats about the result set, control the facets, filter query based on various conditions etc., quite easily.

The final Solr query will look like something below:
fl=catentry_id,partNumber_ntk,name,shortDescription,thumbnail,storeent_id,childCatentry_id,catentry_id,partNumber_ntk,name,shortDescription,thumbnail,storeent_id,childCatentry_id&start=0&rows=4&timeAllowed=15000&defType=edismax&qf=name^10.0 defaultSearch^1.0 categoryname^100.0 shortDescription^5.0 name_suggest^1.0 shortDesc_suggest^1.0&pf=name^10.0 defaultSearch^1.0 categoryname^100.0 shortDescription^5.0 name_suggest^1.0 shortDesc_suggest^1.0&ps=100&mm=1&tie=0.1&tie=0.1&wt=json&json.nl=map&q="apple"&fq=catalog_id:"10052"&fq=storeent_id:("10001")&fq=published:1&fq=-(catenttype_id_ntk_cs:ItemBean AND parentCatentry_id:[* TO *])

Once the final Solr Query is built, it will be executed by SolrRESTSearchExpressionProcessor class. But who is responsible for building this Solr Query ? The entire query is NOT built by a single java class. Instead the query is split into separate logical parts like q, fq, fl, qf etc., and each part of the query is built by a separate java class known as Providers and PreProcessors. The parts of query are then assembled by SolrRESTSearchExpressionProcessor and executed against Solr Server. We will look at the roles of Providers and PreProcessors in detail in coming section.

A sample REST Search request from JSP


As part of the framework, new JSTL Tag wcf:rest was introduced so that store front can easily invoke these REST Services. (The tag also takes care of local binding / remote binding of the services)

Below code snippet shows an example of how to use this tag to invoke productview/bySearchTerm GET REST service:
 
<wcf:rest var="catalogNavigationView" url="${searchHostNamePath}${searchContextPath}/store/${WCParam.storeId}/productview/bySearchTerm/*">
    <wcf:param name="pageSize" value="${pageSize}" />
    <wcf:param name="pageNumber" value="${pageNumber + 1}" />
    <wcf:param name="profileName" value="IBM_findProductsBySearchTerm_Summary" />    <wcf:param name="searchType" value="${searchType}" />
    <wcf:param name="searchTerm" value="${searchTerm}" />
    <wcf:param name="langId" value="${langId}"/>
    <wcf:param name="responseFormat" value="json"/>      
    <wcf:param name="catalogId" value="${WCParam.catalogId}"/>
    <wcf:param name="categoryId" value="${currentCategoryId}" />
    <wcf:param name="manufacturer" value="${newManufacturer}" />
    <wcf:param name="minPrice" value="${WCParam.minPrice}" />
    <wcf:param name="maxPrice" value="${WCParam.maxPrice}" />
</wcf:rest>

Note:
<wcbase:usebean> - To invoke data beans
<wcf:getData> - To invoke GET data component services
<wcf:rest> - To invoke REST services.

Required parameters to build the final Sol Query like searchTerm, pageNumber, pageSize, categoryId, minPrice, maxPrice etc., are passed to the REST Service using <wcf:param> tag.
These external query parameter names are converted into internal names based on the mapping defined in:
workspace\Search\xml\config\com.ibm.commerce.catalog\wc-component.xml

Sample Mapping:
<_config:valuemapping externalName="SearchControlParameterMapping" internalName="SearchControlParameterMapping">
    <_config:valuemap externalValue="searchType" internalValue="_wcf.search.type" />
    <_config:valuemap externalValue="categoryId"  internalValue="_wcf.search.category" />
    <_config:valuemap externalValue="catalogId"    internalValue="_wcf.search.catalog" />
    <_config:valuemap externalValue="storeId"       internalValue="_wcf.search.store.online" />
    <_config:valuemap externalValue="facet"          internalValue="_wcf.search.facet" />
    <_config:valuemap externalValue="facetLimit"   internalValue="_wcf.search.facet.field.limit" />
    <_config:valuemap externalValue="minPrice"    internalValue="_wcf.search.price.minimum" />
    <_config:valuemap externalValue="maxPrice"   internalValue="_wcf.search.price.maximum" />
    <_config:valuemap externalValue="searchTerm" internalValue="_wcf.search.term" />
    <_config:valuemap externalValue="profileName" internalValue="_wcf.search.profile" />
</_config:valuemapping>

A Selection Criteria object will be built using these internal names which will be accessible to all the downstream processors like provider, pre-processor and post-processor

Along with the request, we also pass a special parameter by name "profileName". wc-search.xml (inside workspace\Search\xml\config\com.ibm.commerce.catalog) defines various search profiles which controls how Solr queries are built, the meta-data for the query, the return fields etc., 
A sample profile will look like:

<_config:profile name="IBM_findProductsBySearchTerm_Summary" indexName="CatalogEntry">
    <_config:query>
        <_config:param name="maxRows" value="500" />           
        <_config:provider classname="com.ibm.solr.SolrRESTSearchBasedMerchandisingExpressionProvider"/>
        <_config:provider classname="com.ibm.xxx.solr.SolrRESTSearchTermAssociationExpressionProvider"/>
        <_config:preprocessor classname="com.ibm.xxx.solr.SolrSearchDebugQueryPreprocessor"/>
        <_config:preprocessor classname="com.ibm.xxx.solr.SolrSearchResultFieldQueryPreprocessor"/>

        <_config:field name="name"/>
        <_config:field name="defaultSearch"/>
        <_config:field name="categoryname"/>
        <_config:field name="shortDescription"/>
        <_config:postprocessor classname="com.ibm.xxx.solr.SolrRESTSearchPreviewQueryPostprocessor" />
        <_config:postprocessor classname="com.ibm.solr.SolrRESTSearchExperimentQueryPostprocessor" />
    </_config:query>        
    <_config:result>
        <_config:field name="catentry_id"/>
        <_config:field name="storeent_id"/>
        <_config:field name="buyable"/>
          <_config:field name="mfName_ntk"/>
        <_config:field name="catenttype_id_ntk_cs"/>
        <_config:field name="price_*"/>
        <_config:field name="listprice_*"/>
        <_config:field name="parentCatgroup_id_facet"/>
        <_config:field name="childCatentry_id"/>
    </_config:result>
</_config:profile>
  1. <_config:query> defines a set of providers and pre-processor. Each of these providers and pre-processor will contribute a part of Solr query. SolrRESTSearchExpressionProcessor will build the final query aggregating all the part queries built by these providers and pre-processor and executes it against the Solr index.
  2. <_config:param> is used to add name/value parameters to final Solr Query directly and it can be used to control how query is executed or fine tune the query further.
  3. <confg:provider> and <config:preprocessor> are used to define the providers and pre-processor used while building the query.
  4. <config:field> defines the index field names against which search should be performed. (qf fields)
  5. <config:result> defines the fields to be returned in the result set (fl fields)
  6. You can also use <_config:sort>, <_config:highlight>, <_config:spellCheck> and other configurations to fine tune the Solr Query.
Search framework will retrieve the profileName from query parameters and identifies the suitable profile defined in wc-search.xml. It then invokes all the providers and pre-processors + default providers and pre-processors defined for the profile, to build the Solr Query which will then be executed against the Solr Index. Once Solr returns back the result set, the search framework invokes the post-processors defined against the profile in a sequential order. Each of these post-processors will modify the result set returned by the Solr. ( Add / modify / remove data from the result set ).
Finally the result set will be converted into required responseFormat ( JSON / XML ) and returned back to JSP.

Customizing Solr Services
WCS - Solr framework provides various touch points to customize the existing services or create new services. Some of the customization touch points which we will explore in detail are:
  1. JSP
  2. wc-search.xml
  3. Provider
  4. Pre-Processor
  5. Post-Processor
Customization using JSP
The first step of customization is JSP, where the actual Search REST request is made. You can pass various parameters to control how the search is performed. You can control the amount of data returned by Solr Server by using appropriate profileName. (Use xxx_Summary profiles when you want to retrieve lesser data).
Let us look at how some advanced customization scenarios can be achieved.

Scenario 1: By default MinimumMatch property is set to "1" in wc-search.xml. Which means, when a shopper searches for "red apple" all products which either contains a word "red" or "apple" will be returned. So shopper will see - apples, red dress, red potatoes, red handbags, red ginsing etc., in the result set.
Let us assume that you want to control this parameter value from JSP while doing the search and set it to different values based on some logic.
SolrSearchEDismaxQueryPreProcessor is the PreProcessor class which looks for
 "_wcf.search.edismax.mm" parameter in the selection criteria object and uses the value to set "mm" parameter in the final Solr Query executed against Search Server. So you can pass this parameter from JSP using <wcf:param> tag.

<wcf:rest var="catalogNavigationView" url="${searchHostNamePath}${searchContextPath}/store/${storeId}/productview/bySearchTerm">
    <wcf:param name="pageSize" value="${pageSize}" />
    <wcf:param name="searchTerm" value="${term}" />
    xxx
    xxx
    <wcf:param name="_wcf.search.edismax.mm" value="2"/>

</wcf:rest>

With this param set in request, when you search for "red apple", you will see that results contains only products which matches both the words - red and apple.

You can always update wc-component.xml to map this internal name "_wcf.search.edismax.mm" to a user friendly external name like "mm" and start using "mm" in the JSP file.

Scenario 2: Let us assume that you have defined an attribute "Clearance" and made it searchable and facetable using CMC tooling. Now when someone searches for a product, you would like to take advantage of this attribute and return only those products which has this attribute.

To achieve this first identify the SearchFieldName ( index field name) for the "Clearance" attribute using below query:

select SRCHFIELDNAME from ATTRDICTSRCHCONF where attr_id in (select attr_id from attr where identifier = 'Clearance' and storeent_id = <storeId>);

Lets assume the above query returned SearchFieldName as: ads_f10101.

Search framework provides SolrRESTSearchByCustomExpressionProvider which allows you to add optional query ( OR constraint ), mandatory query ( AND constraint ) and Filter query ( fq ) part to the final Solr query. This custom provider looks for query parameter with name "_wcf.search.expr", "_wcf.search.mandatory.expr" and "_wcf.search.filter.expr". You can directly use these param names in JSP or create a mapping in wc-component.xml as below:

            <_config:valuemap externalValue="minMatch" internalValue="_wcf.search.edismax.mm" />
            <_config:valuemap externalValue="optionalQuery" internalValue="_wcf.search.expr" />
            <_config:valuemap externalValue="mandatoryQuery" internalValue="_wcf.search.mandatory.expr" />
            <_config:valuemap externalValue="filterQuery" internalValue="_wcf.search.filter.expr" />

With this mapping in place you can pass these additional parameters to <wcf:rest> tag in JSP.
// Return products which matches SearchTerm AND which has this attribute set to Clearance.
<wcf:param name="mandatoryQuery" value="ads_f10101_ntk_cs:(\"Clerance\")"/>

OR

// Return products which matches search term OR the products which has this attribute set to Cleranace
<wcf:param name="optionalQuery" value="ads_f10101_ntk_cs:(\"Clerance\")"/>

OR

// Return products which matches search term, filter out the results to return products with attribute Clerarance.
<wcf:param name="filterQuery" value="ads_f10101_ntk_cs:(\"Clerance\")"/>

FilterQuery is similar to ManatoryQuery, but filter Query performs the filter on the returned result set and hence maintains the relevance score better.

So by making use of these additional parameters and appropriate search profiles you can control the modify the solr query to suit your needs.

Customization using wc-search.xml
If needed you can define your own custom profiles and use it in JSP pages while making REST Service calls.
Create custom profiles in wc-search.xml ( in workspace\Search\xml\config\com.ibm.commerce.catalog\ext folder ). 
You can use existing providers, pre-processors, post-processors or create your own custom providers / pre-processors / post-processors and use it in the search profile. You can make use of various other aspects of wc-search.xml like <_config:field>, <_config:param>, <_config:sort> to customize the solr query.

Customization using custom Providers
Sometimes custom scenarios may be complex and you may not be able achieve required features just by using the existing profiles or providers. Passing in mandatoryQuery, optionalQuery or filterQuery may not be sufficient or relevant for some customization scenarios.
If the customization calls for modifying the Solr Query to add additional constraints or boost the result set o modify the query based on certain conditions, then you can always create a custom provider.

Let us look at the following REST service which retrieves product data byId.
http://localhost/search/resources/store/10001/productview/byIds?id=10034&id=10014&id=10906&profileName=IBM_findProductByIds_Summary

When you inspect the result set from this service, you will notice that the products are not in the same order in which the product ids were passed (10034, 10014, 10906). The order is completely random (probably based on the order in which the data was indexed).

Now if you would like to maintain the order of products, then one option is to use the boost query to boost the products as below, so that the sequence order is maintained and id1 is returned first and id3 is returned at the end.


bq=catentry_id:"id1"^10.0 catentry_id:"id2"^9.0 catentry_id:"id3"^8.0

Lets create a provider which will add this query part to the Selection Criteria object, so that it can be consumed by the final ExpressionProcessor while building the Solr Query.

Sample Provider Class






Above listing shows a sample Provider Class which extends from AbstractSolrSearchExpressionProvider and implements SearchExpressionProvider interface.

It overrides invoke method and execute some custom logic to build the boost query string. getControlParameterValue gives you access to the input query parameters. Notice that we are using SearchServiceConstants.CTRL_PARAM_SEARCH_TERM as the key name while retrieving the product ids.

Once the boost search query is built, it is added back to the Selection Criteria object using addControlParameterValue method. The key used in this case is SearchServiceConstants.CTRL_PARAM_SEARCH_INTERNAL_BOOST_QUERY. This is the predefined name which is used by the Expression Processors to build the "bq" Solr Query.

Once the provider is built.. TODO

Customization using custom Pre-processor - TODO


Customization using custom Post-processor - TODO

Tuesday, 28 April 2015

Caching Store level or Site level widgets.

While creating cachespec.xml, decision about widgets can be either :
  1. Do not cache
  2. Cache separately
  3. Consume within parent servlet ( cache along with parent JSP )
Do not cache
Widgets like MiniShopCartWidget, InventoryWidget, DiscountWidget etc will not be cached. An entry in cachespec.xml with attributes do-not-cache and do-not-consume set to true will be sufficient for these widgets

    <cache-entry>
        <class>servlet</class>

        <name>/Widgets_701/xxx/InventoryAvailability.jsp</name>
        <name>/Widgets_701/Common/Discounts/Discounts.jsp</name>
        <name>/Widgets_701/xxx/Discounts.jsp</name>
        <property name="do-not-consume">true</property>
        <property name="do-not-cache">true</property>
            <cache-id>
            <component id="storeId" type="parameter">
                <required>false</required>
            </component>
        </cache-id>   
    </cache-entry>


Cache separately

Widgets like productRecommendations, CategoryRecommendations etc can be cached separately so that it can be reused across layout JSPs. An entry in cachespec.xml with attribute do-not-consume = true will be sufficient for these widgets 

    <cache-entry>
        <class>servlet</class>
        <name>/Widgets_701/xxxx/BreadcrumbTrail.jsp</name>
        <property name="save-attributes">false</property>
        <property name="ignore-get-post">true</property>
        <property name="consume-subfragments">true</property>
        <property name="do-not-consume">true</property> <!--  cache separately.. -->
        <property name="do-not-cache">false</property>

        <cache-id>
             <component id="DC_storeIdentifier" type="attribute">
                 <required>true</required>
            </component>

           ...
           ...
        </cache-id>
   </cache-entry>

  

Emarketing Widgets 

Emarketing spot widgets are handled separately. Since activities associated with eMarketing spots can be dynamic or static, the decision about whether to cache the widget or not should be done dynamically at run time based on the activities associated with widget. So general recommendation is to set do-not-cache = true and do-not-consume = true and let the marketing component decide whether to cache the widget or not and whether to cache separately or consume along with parent servlet.


Also you need to  include the following metaDataGenerator in cachespec.xml. This metaDataGenerator will make appropriate decision about caching and adding dependency Ids

<metadatagenerator>
     com.ibm.commerce.marketing.cache.EMarketingSpotMetaDataGenerator
</metadatagenerator>

Consuming widgets
Widgets like CatalogEntry widgets can be consumed within the parent layout JSP like CategoryDisplayPage or SubCategoryDisplay pages. 
When a widget is consumed by a parent JSP, no entries were made for the consumed JSP in cachespec.xml. But the dependency id's of the consumed JSP were added directly to the parent JSP cache entry in cachespec.xml. So assuming ProductPage imports PriceDisplay.jsp, price:<productId> dependency id was added to ProductPage. Whenever the price of a product changes, price:<productId> invalidation id will be published by server and hence ProdcutPage cache gets invalidated. 

When a Store is not using commerce composer tooling, IT developer will create the ProductPage and cachespec.xml. So IT developer will know that productPage consumes PriceDisplay widget and hence adds price dependency id's to the ProductPage cache entry.

But with commerce composer tool, business user has full control over layout widgets and can pick and choose the widgets which goes into layout. So during Code and UT phase, IT developer will have no idea about widgets present in a page and hence cannot add dependency Id's directly to the parent servlet.

For example, assume ProductPage contains price widget and hence developer adds price dependency id to ProductPage entry. Now for some reason, business user removes price widget from the ProductPageLayout widget and adds inventory widget in that slot. Now what happens when price changes ? The entire productPage gets invalidated, even though it doesn't display price. What happens when inventory changes? The page will still be served from cache and it doesn't display the updated inventory status. We cannot expect business user to take help of developers to modify the cachespec.xml as and when he changes the layout widgets.

The answer to this issue is <wcpgl:pageLayoutWidgetCache/> JSTL tag. When this tag gets executed, it adds the dependency id's to the parent servlet cache entry dynamically.

Follow these 3 high level steps to consume the widget with the parent servlet:
  1. Add <wcfpgl:pageLayoutWidgetCache/> to the main widget JSP.
  2. Make an entry in cachespec.xml for the widget JSP. Add all the cache-ids and dependency ids for the entry
  3. Set do-not-consume = true and do-not-cache = true.
Now if the widget is included in the layout by business user, then during runtime, the widget JSP gets executed. Since we have <wcpgl:pageLayoutWidgetCache/> tag inside the widget JSP, this tag gets executed. This tag reads the dependency ids from the cachespec.xml and sets it on the parent servlet. It also resets do-not-consume and do-not-cache values to false.

If the widget is not included in the layout by business user, then widget JSP doesn't get executed and parent page will not contain any additional dependency IDs.

So far so good. But what happens if someone wants to genuinely set do-not-consume and do-not-cache to true. How does someone tell that the values set in cachespec.xml shouldn't be modified by the included <wcfpgl:pageLayoutWidgetCache/> tag. This is where special dependency id ignoreDoNotConsume comes into picture. So if you want to let the pageLayoutWidgetCache tag to reset the do-not-consume and do-not-cache values and add dependency id's to the parent servlet, then add this dependency tag to the widget cache entry in cachespec.xml

<dependency-id>ignoreDoNotConsume</dependency-id>

 This special tag tells the pageLayoutWidgetCache handler to ignore the do-not-consume setting on cachespec.xml and reset it to false. If this dependency id is not present, then pageLayoutWidgetCache will not do any additional processing and lets the cache component of application server to handle it appropriately.