打造Tapestry5中的智能的错误页面。

我们一般需要为生产和开发环境准备两套error page。 tapestry在开发环境下的error page做得非常漂亮。 非常详细, 但是在生产环境下就不能暴露太多的信息。 而且error page的外观也需要定制。 这时候tapestry默认的error page就不行了。 所以我们期望的是在开发的时候我们需要tapestry自带的error page, 而在生产环境下我们需要定制自己的error page。

http://wiki.apache.org/tapestry/Tapestry5ExceptionPage
上面这篇文章告诉我们怎样替换掉tapestry自带的error page。 但是我们需要更灵活的方式。

OK, 下面开始写代码。 我们在AppModule class 添加以下代码段。

    // handle RedirectException
    public static RequestExceptionHandler decorateRequestExceptionHandler(
        final Object delegate, final Response response,
        final RequestPageCache requestPageCache, final LinkFactory linkFactory,
        final ComponentClassResolver resolver,
        @Symbol(SymbolConstants.PRODUCTION_MODE) final
        boolean productionMode, final PageResponseRenderer renderer
    	)
    {
        return new RequestExceptionHandler()
        {
            public void handleRequestException(Throwable exception) throws IOException
            {
                // check if wrapped
                Throwable cause = exception;
                if (exception.getCause() instanceof RedirectException)
                {
                    cause = exception.getCause();
                }

                //Better way to check if the cause is RedirectException. Sometimes it's wrapped pretty deep..
                int i = 0;
                while(true){
                    if(cause == null || cause instanceof RedirectException || i > 1000){
                        break;
                    }
                    i++;
                    cause = cause.getCause();
                }

                // check for redirect
                if (cause instanceof RedirectException)
                {
                    // check for class and string
                    RedirectException redirect = (RedirectException)cause;
                    URL url = redirect.getUrl();
                    if (url != null) {
                    	response.sendRedirect(url.toString());
                    	return ;
                    }
                    Link pageLink = redirect.getPageLink();
                    if (pageLink == null)
                    {
                        // handle Class (see ClassResultProcessor)
                        String pageName = redirect.getMessage();
                        Class<?> pageClass = redirect.getPageClass();
                        if (pageClass != null)
                        {
                            pageName = resolver.resolvePageClassNameToPageName(pageClass.getName());
                        }

                        // handle String (see StringResultProcessor)
                        Page page = requestPageCache.get(pageName);
                        pageLink = linkFactory.createPageLink(page, false);
                    }

                    // handle Link redirect
                    if (pageLink != null)
                    {
                        response.sendRedirect(pageLink.toRedirectURI());
                        return;
                    }
                }
               if (productionMode) {
                	Page page = requestPageCache.get("ProductionExceptionReport");
                	ExceptionReporter rootComponent = (ExceptionReporter) page.getRootComponent();
                	rootComponent.reportException(exception);
                	renderer.renderPageResponse(page);
                }else {
	                // no redirect so pass on the exception
	                ((RequestExceptionHandler)delegate).handleRequestException(exception);
                }
            }
        };
    }


我们只要关注64至72行的代码。 64行以上的代码是实现Tapestry4中的RedirectException作用的。我们可以只看64至72行的代码就行了。 这段代码很简单。 如果是productionMode(我们运行时servlet container时加了-Dtapestry.production-mode 参数)的话, 我们去拿"ProductionExceptionReport"页面。 然后调用reportException方法。 最后转到ProductionExceptionReport页面。 如果不是production mode的话,我们使用tapestry默认的RequestExceptionHandler去处理这个异常。 这样tapestry就使用自带的异常页面了。

import java.util.List;

import org.apache.tapestry5.Asset;
import org.apache.tapestry5.annotations.Property;
import org.apache.tapestry5.ioc.annotations.Inject;
import org.apache.tapestry5.ioc.services.ExceptionAnalysis;
import org.apache.tapestry5.ioc.services.ExceptionAnalyzer;
import org.apache.tapestry5.ioc.services.ExceptionInfo;
import org.apache.tapestry5.services.ExceptionReporter;

public class ProductionExceptionReport implements ExceptionReporter {
	
	@Property(write=false)
	private Throwable exception;
	
    @Inject
    private ExceptionAnalyzer analyzer;
    
    @Property
    private List<ExceptionInfo> stack;    

	public void reportException(Throwable exception) {
		this.exception = exception;
        ExceptionAnalysis analysis = analyzer.analyze(exception);
        stack = analysis.getExceptionInfos();		
		sendExceptionByEmail();
	}

	private void sendExceptionByEmail() {
		System.out.println(".......................Send Email ..............");
	}

	@Override
	public Asset[] getCsses() {
		return new Asset[] {getIndexCssAsset()};
	}

}


ProductionExceptionReport页面实现了tapestry5的ExceptionReporter 接口。 这里要注意的是
ExceptionReporter rootComponent = (ExceptionReporter)page.getRootComponent();

你不能把它cast成ProductionExceptionReporter, 这是由tapestry5的classloader超成的。 tapestry5文档中有说明。 你可以在reportException方法中做些事情。 比如发邮件通知管理员。

下面是个简单的error page的模板页面。
<?xml version="1.0" encoding="UTF-8"?>  
 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN"  
    "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">  
<t:layout home="false" errorPage="true" showAddNewSite="false" showSideBar="false" title="literal:Error"  xmlns:t="http://tapestry.apache.org/schema/tapestry_5_0_0.xsd">
	<font color="red">有错误发生,请联系管理员。</font>
	<br/>
	<a t:type="pagelink" t:page="Index">重新开始</a>
</t:layout>


OK。大工告成!

在你开发的环境中启动参数中加上-Dtapestry.production-mode=false


参考:
http://wiki.apache.org/tapestry/Tapestry5ExceptionPage
http://tapestry.apache.org/tapestry5/cookbook/exceptions.html

你可能感兴趣的:(apache,xml,servlet,IOC,tapestry)