AssetRenderer fails for Journal/Web Content utilising template when embeded in custom portlet due to incorrect class loader
Description
Activity
Show:

Mark Jin February 15, 2012 at 2:39 AM
PASSED Manual Testing following the steps in the description.
Fixed on:
Tomcat 7.0 + MySQL 5. 6.1.x GIT ID: 4166a5e8894ce3c6024bfc549545e5fc74c1320c.
Tomcat 7.0 + MySQL 5. 6.2.x GIT ID: 6535813184f8681e14487b205c17df1caee259d0.

Michael Saechang February 10, 2012 at 2:59 PM
Committed on:
6.1.x GIT ID: 1d3be829596a137ea72b1a2346bce8f4e092267c.
6.2.x GIT ID: 1d3be829596a137ea72b1a2346bce8f4e092267c.

James Falkner January 25, 2012 at 9:17 AM
Able to reproduce this issue in 6.1.0 GA1. Assigning to support-qa.

Robert Kornmesser January 19, 2012 at 6:10 AM
Previewing any webcontent with a non-default vm-template causes this error, too.

Milen Dyankov January 1, 2012 at 6:31 PM
Just ran into this issue while attempting to get article's content in my portlet. I really hope this will get fixed in 6.1 GA. And here is another simple way to reproduce (Liferay 6.1rc1, Tomcat bundle), just add:
in a custom portlet. Then when executed :
Fixed
Details
Assignee
Mark JinMark Jin(Deactivated)Reporter
Richard GibsonRichard GibsonLabels
Branch Version/s
6.1.xBackported to Branch
CommittedFix Priority
5Git Pull Request
Components
Fix versions
Affects versions
Priority
Medium
Details
Details
Assignee

Reporter

Labels
Branch Version/s
6.1.x
Backported to Branch
Committed
Fix Priority
5
Git Pull Request
Components
Fix versions
Affects versions
Priority
Zendesk Support
Linked Tickets
Zendesk Support
Linked Tickets
Zendesk Support

Linked Tickets
Created June 23, 2011 at 4:02 AM
Updated June 24, 2023 at 9:58 AM
Resolved March 12, 2012 at 2:28 PM
Description:
Using the AssetRenderer to display Journal/Web Content in a custom portlet results in a null pointer when the Web Content has an associated Template.
This is because the template parser com.liferay.portal.kernel.templateparser.BaseTransformer attempts to lookup the Template Parser with the following code:
Line 147 of BaseTransformer.java
templateParser =(TemplateParser)InstanceFactory.newInstance(templateParserClassName);
This results in the InstanceFactory getting the class loader from the current thread:
Line 47 of InstanceFactory.java
Thread currentThread = Thread.currentThread();
The class loader for the current thread will be the class loader for the custom portlet, thus the expected classes will not be available and an exception:
com.liferay.portal.kernel.templateparser.TransformException:
java.lang.ClassNotFoundException: com.liferay.portlet.journal.util.VelocityTemplateParser
will be generated.
Fortunately the InstanceFactory.newInstance(..) call can be passed a class loader, so modification of
Line 147 of BaseTransformer.java to:
templateParser=(TemplateParser)InstanceFactory.newInstance(
PortalClassLoaderUtil.getClassLoader(),
templateParserClassName);
i.e. passing in the PortalClassLoader fixes the issue.
Step to reproduce:
1. Create a structure and template for web content.
2. Add an instance of web content using the new structure and template.
3. Write a custom portlet that attempt to display the data using the AssetRenderer in FULL, i.e. AssetRenderer.TEMPLATE_FULL_CONTENT
example code:
<%
AssetRendererFactory af = AssetRendererFactoryRegistryUtil.getAssetRendererFactoryByClassName( JournalArticle.class.getName());
AssetRenderer assetRenderer=af.getAssetRenderer('articleId');
String renderPath=assetRenderer.render(portletRequest, portletResponse, AssetRenderer.TEMPLATE_FULL_CONTENT);
String renderPortletId = AssetRendererFactoryRegistryUtil.getAssetRendererFactoryByClassName( JournalArticle.class.getName()).getPortletId()
<liferay-util:include page="<%=renderPath%>" portletId="<%=renderPortletId %>" />
4. View custom portlet (sometimes it will work on the first view)
5. Refresh page - null pointer will occur in "html\portlet\journal\asset\full_content.jsp" at line 43 as the articleDisplay object is null due to the above exception.