/* DHtmlLayoutPortlet.java Purpose: Description: History: Wed Jan 11 13:59:27 2006, Created by tomyeh Copyright (C) 2006 Potix Corporation. All Rights Reserved. {{IS_RIGHT This program is distributed under LGPL Version 2.1 in the hope that it will be useful, but WITHOUT ANY WARRANTY. }}IS_RIGHT */ package org.zkoss.zk.ui.http; import java.util.Map; import java.util.HashMap; import java.io.Writer; import java.io.IOException; import javax.servlet.ServletContext; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import javax.portlet.GenericPortlet; import javax.portlet.PortletSession; import javax.portlet.PortletException; import javax.portlet.RenderRequest; import javax.portlet.RenderResponse; import javax.portlet.PortletPreferences; import org.zkoss.lang.Classes; import org.zkoss.lang.Exceptions; import org.zkoss.lang.Library; import org.zkoss.mesg.Messages; import org.zkoss.util.logging.Log; import org.zkoss.web.Attributes; import org.zkoss.web.portlet.Portlets; import org.zkoss.web.portlet.RenderHttpServletRequest; import org.zkoss.web.portlet.RenderHttpServletResponse; import org.zkoss.zk.mesg.MZk; import org.zkoss.zk.ui.WebApp; import org.zkoss.zk.ui.Session; import org.zkoss.zk.ui.Desktop; import org.zkoss.zk.ui.Page; import org.zkoss.zk.ui.Execution; import org.zkoss.zk.ui.Richlet; import org.zkoss.zk.ui.UiException; import org.zkoss.zk.ui.util.DesktopRecycle; import org.zkoss.zk.ui.sys.UiFactory; import org.zkoss.zk.ui.sys.WebAppCtrl; import org.zkoss.zk.ui.sys.SessionCtrl; import org.zkoss.zk.ui.sys.SessionsCtrl; import org.zkoss.zk.ui.sys.RequestInfo; import org.zkoss.zk.ui.sys.PageRenderPatch; import org.zkoss.zk.ui.impl.RequestInfoImpl; import org.zkoss.zk.ui.metainfo.PageDefinition; import org.zkoss.zk.ui.metainfo.PageDefinitions; /** * The portlet used to process the request for a ZUML page. * *

Notes:

* * *

To patch the rendering result of a ZK portlet, you can implement * {@link PageRenderPatch} (and specified it in {@link org.zkoss.zk.ui.sys.Attributes#PORTLET_RENDER_PATCH_CLASS}). * @author tomyeh */ public class DHtmlLayoutPortletFix extends GenericPortlet { private static final Log log = Log.lookup(DHtmlLayoutPortletFix.class); /** The parameter or attribute to specify the path of the ZUML page. */ private static final String ATTR_PAGE = "zk_page"; /** The parameter or attribute to specify the path of the richlet. */ private static final String ATTR_RICHLET = "zk_richlet"; /** The default page. */ private String _defpage; public void init() throws PortletException { _defpage = getPortletConfig().getInitParameter(ATTR_PAGE); } protected void doView(RenderRequest request, RenderResponse response) throws PortletException, IOException { //try parameter first and then attribute boolean bRichlet = false; String path = request.getParameter(ATTR_PAGE); if (path == null) { path = (String)request.getAttribute(ATTR_PAGE); if (path == null) { PortletPreferences prefs = request.getPreferences(); path = prefs.getValue(ATTR_PAGE, null); if (path == null) { path = request.getParameter(ATTR_RICHLET); bRichlet = path != null; if (!bRichlet) { path = (String)request.getAttribute(ATTR_RICHLET); bRichlet = path != null; if (!bRichlet) { path = prefs.getValue(ATTR_RICHLET, null); bRichlet = path != null; if (!bRichlet) path = _defpage; } } } } } final Session sess = getSession(request); if (!SessionsCtrl.requestEnter(sess)) { handleError(sess, request, response, path, null, Messages.get(MZk.TOO_MANY_REQUESTS)); return; } SessionsCtrl.setCurrent(sess); try { if (!process(sess, request, response, path, bRichlet)) handleError(sess, request, response, path, null, null); } catch (Throwable ex) { handleError(sess, request, response, path, ex, null); } finally { SessionsCtrl.requestExit(sess); SessionsCtrl.setCurrent((Session)null); } } /** Returns the session. */ private Session getSession(RenderRequest request) throws PortletException { final WebApp wapp = getWebManager().getWebApp(); final PortletSession psess = request.getPortletSession(); final Session sess = SessionsCtrl.getSession(wapp, psess); return sess != null ? sess: SessionsCtrl.newSession(wapp, psess, request); } /** Process a portlet request. * @return false if the page is not found. * @since 3.0.0 */ protected boolean process(Session sess, RenderRequest request, RenderResponse response, String path, boolean bRichlet) throws PortletException, IOException { // Class org.zkoss.lang.D is missing in ZK 6.0.1 release build //if (D.ON && log.debugable()) log.debug("Creates from "+path); final WebManager webman = getWebManager(); final WebApp wapp = webman.getWebApp(); final WebAppCtrl wappc = (WebAppCtrl)wapp; final HttpServletRequest httpreq = new RenderHttpServletRequestFix(RenderHttpServletRequest.getInstance(request)); final HttpServletResponse httpres = RenderHttpServletResponse.getInstance(response); final ServletContext svlctx = (ServletContext)wapp.getNativeContext(); final DesktopRecycle dtrc = wapp.getConfiguration().getDesktopRecycle(); Desktop desktop = dtrc != null ? DesktopRecycles.beforeService(dtrc, svlctx, sess, httpreq, httpres, path): null; try { if (desktop != null) { //recycle final Page page = Utils.getMainPage(desktop); if (page != null) { final Execution exec = new ExecutionImpl(svlctx, httpreq, httpres, desktop, page); fixContentType(response); wappc.getUiEngine() .recycleDesktop(exec, page, response.getWriter()); } else desktop = null; //something wrong (not possible; just in case) } if (desktop == null) { desktop = webman.getDesktop(sess, httpreq, httpres, path, true); if (desktop == null) //forward or redirect return true; final RequestInfo ri = new RequestInfoImpl( wapp, sess, desktop, httpreq, PageDefinitions.getLocator(wapp, path)); ((SessionCtrl)sess).notifyClientRequest(true); final Page page; final PageRenderPatch patch = getRenderPatch(); final Writer out = patch.beforeRender(ri); final UiFactory uf = wappc.getUiFactory(); if (uf.isRichlet(ri, bRichlet)) { final Richlet richlet = uf.getRichlet(ri, path); if (richlet == null) return false; //not found page = WebManager.newPage(uf, ri, richlet, httpres, path); final Execution exec = new ExecutionImpl(svlctx, httpreq, httpres, desktop, page); fixContentType(response); wappc.getUiEngine().execNewPage(exec, richlet, page, out != null ? out: response.getWriter()); } else if (path != null) { final PageDefinition pagedef = uf.getPageDefinition(ri, path); if (pagedef == null) return false; //not found page = WebManager.newPage(uf, ri, pagedef, httpres, path); final Execution exec = new ExecutionImpl(svlctx, httpreq, httpres, desktop, page); fixContentType(response); wappc.getUiEngine().execNewPage(exec, pagedef, page, out != null ? out: response.getWriter()); } else return true; //nothing to do if (out != null) patch.patchRender(ri, page, out, response.getWriter()); } } finally { if (dtrc != null) DesktopRecycles.afterService(dtrc, desktop); } return true; //success } private static PageRenderPatch getRenderPatch() { if (_prpatch != null) return _prpatch; synchronized (DHtmlLayoutPortletFix.class) { if (_prpatch != null) return _prpatch; final PageRenderPatch patch; final String clsnm = Library.getProperty( org.zkoss.zk.ui.sys.Attributes.PORTLET_RENDER_PATCH_CLASS); if (clsnm == null) { patch = new PageRenderPatch() { public Writer beforeRender(RequestInfo reqInfo) { return null; } public void patchRender(RequestInfo reqInfo, Page page, Writer result, Writer out) throws IOException { } }; } else { try { patch = (PageRenderPatch)Classes.newInstanceByThread(clsnm); } catch (ClassCastException ex) { throw new UiException(clsnm+" must implement "+PageRenderPatch.class.getName()); } catch (Throwable ex) { throw UiException.Aide.wrap(ex, "Unable to instantiate"); } } return _prpatch = patch; } } private static PageRenderPatch _prpatch; private static void fixContentType(RenderResponse response) { //Bug 1548478: content-type is required for some implementation (JBoss Portal) if (response.getContentType() == null) response.setContentType("text/html;charset=UTF-8"); } /** Returns the layout servlet. */ private final WebManager getWebManager() throws PortletException { final WebManager webman = (WebManager)getPortletContext().getAttribute(WebManager.ATTR_WEB_MANAGER); if (webman == null) throw new PortletException("The Layout Servlet not found. Make sure is specified for "+DHtmlLayoutServlet.class.getName()); return webman; } /** Handles exception being thrown when rendering a page. * @throws PortletException the exception being throw. If null, it means the page * is not found. */ private void handleError(Session sess, RenderRequest request, RenderResponse response, String path, Throwable err, String msg) throws PortletException, IOException { if (err != null) { //Bug 1714094: we have to handle err, because Web container //didn't allow developer to intercept errors caused by inclusion final String errpg = sess.getWebApp().getConfiguration() .getErrorPage(sess.getDeviceType(), err); if (errpg != null) { try { request.setAttribute("javax.servlet.error.message", Exceptions.getMessage(err)); request.setAttribute("javax.servlet.error.exception", err); request.setAttribute("javax.servlet.error.exception_type", err.getClass()); request.setAttribute("javax.servlet.error.status_code", new Integer(500)); if (process(sess, request, response, errpg, false)) return; //done log.warning("The error page not found: "+errpg); } catch (IOException ex) { //eat it (connection off) } catch (Throwable ex) { log.warning("Failed to load the error page: "+errpg, ex); } } if (msg == null) msg = Messages.get(MZk.PAGE_FAILED, new Object[] {path, Exceptions.getMessage(err), Exceptions.formatStackTrace(null, err, null, 6)}); } else { if (msg == null) msg = path != null ? Messages.get(MZk.PAGE_NOT_FOUND, new Object[] {path}): Messages.get(MZk.PORTLET_PAGE_REQUIRED); } final Map attrs = new HashMap(); attrs.put(Attributes.ALERT_TYPE, "error"); attrs.put(Attributes.ALERT, msg); Portlets.include(getPortletContext(), request, response, "~./html/alert.dsp", attrs, Portlets.OVERWRITE_URI); //Portlets doesn't support PASS_THRU_ATTR yet (because //protlet request will mangle attribute name) } }