背景
我有一个 java.lang.Thread,它在特定时间在 Web 服务器(JBoss 或 WebSphere)上的应用程序内运行,无需人工交互,它所做的只是发送电子邮件。电子邮件的内容与我们在 Web View 中用作显示的 JSP (/jsp/Report.jsp) 的内容类似。
我不想重复相同的工作或将 JSP 更改为两者都可以访问的静态类,而是想从线程内部获取 JSP 运行的内容并将其放入电子邮件中进行发送。
我通过使用“web.xml”中的监听器获得了当前的ServletContext。我当前在线程中的 JSP 调用如下:
servletContext.getRequestDispatcher("/jsp/Report.jsp").include(dummyRequest, dummyResponse);
请求/响应类基本上是这样创建的:
final HttpServletRequest dummyRequest = new HttpServletRequest() { .... }
final HttpServletResponse dummyResponse = new HttpServletResponse() { .... }
我打算通过 dummyRequest 为 JSP 设置附加属性(类),例如“dummyRequest.setAttribute(name, value)”。
每当我进行调用时,都会出现异常,因为虚拟请求/响应是 HttpServletResponse/HttpServletRequest 的匿名类。
WebSphere 应用程序服务器 7.0.0.17:
java.lang.RuntimeException: SRV.8.2: RequestWrapper objects must extend ServletRequestWrapper or HttpServletRequestWrapper
JBoss AS 7.1.1:
java.lang.ClassCastException: my.test.thread$1$2 incompatible with javax.servlet.ServletRequestWrapper
如果没有原始请求/响应,我无法创建 HttpServletResponseWrapper/HttpServletRequestWrapper。
<小时/>问题
那么...是否可以使用上下文从 Web 应用程序的线程内部获取 JSP 的内容?
如果是这样,我该怎么做?
<小时/>更新
这是我用于测试的代码:link
<小时/>研究
我现在开始深入研究服务器的源代码,试图了解发生了什么。
JBoss
在 ApplicationDispatcher 中,“forward”不执行任何操作,因为请求中未设置“DISPATCHER_TYPE”属性(见 processRequest 方法)。这对于“包含”来说不是必需的。
我在“incude”中遇到的关于不兼容类型的问题位于“ApplicationFilterFactory.createFilterChain”内部。 Request 对象不是它正在寻找的正确类,在 JBoss 的情况下是“org.apache.catalina.connector.Request”或“org.apache.catalina.connector.RequestFacade”。除非请求与这些类型之一匹配,否则它根本不会继续。
所以当我使用以下请求时:
final HttpServletRequest dummyRequest = new org.apache.catalina.connector.RequestFacade(new org.apache.catalina.connector.Request() { ... });
它成功运行并从线程内部返回 JSP 的结果。
Websphere
我无法在 Websphere 上产生相同的结果。
Websphere 需要一个“com.ibm.ws.webcontainer.srt.SRTConnectionContextImpl”实例,然后将 ServletContext 操作为其原始类“com.ibm.wsspi.webcontainer.facade.ServletContextFacade”,但随后我陷入困境“com.ibm.ws.webcontainer.srt.SRTServletRequest$SRTServletRequestHelper”内出现“access$200”空指针异常,这使得我看起来像是在以某种方式破坏了 Java。
java.lang.NullPointerException at com.ibm.ws.webcontainer.srt.SRTServletRequest$SRTServletRequestHelper.access$200(SRTServletRequest.java:2629)
这是我当前的代码:
SRTConnectionContext n = new SRTConnectionContextImpl();
(((SRTServletRequest) (n.getRequest())).getRequestContext())
.setCurrWebAppBoundary((WebApp) ((ServletContextFacade) context)
.getIServletContext());
servletContext.getRequestDispatcher("/jsp/Report.jsp").include(n.getRequest(), n.getResponse());
结束
希望有人能够找到一种方法在 Websphere 上实现这一目标。
从我对源代码的查看来看,除非我缺少一个辅助方法,否则如果没有服务器自己的特定请求类文件,您就无法运行包含/转发。甚至请求包装器也被解包到它们的基类,这就是为什么我总是在有或没有包装器的情况下得到类转换异常。
如果没有一种更干净的、非特定于服务器类的服务器类从线程内部获取 JSP 结果的方法,那么这可能是我最初问题的答案,无论它看起来多么困惑。
最佳答案
because the dummy request/response is an anonymous class of HttpServletResponse/HttpServletRequest.
没有。你会得到类广播异常,因为它们是不同的容器,需要包装器。你的代码正在提供请求/响应。
我们不太清楚您到底在哪里调用来创建虚拟 HttpServletRequest/Response.. 看起来像是从您调用的位置...您实际上需要实例化 ServletRequestWrapper/responseWapper 对象,如果您有句柄,则在其上设置您的请求,然后使用它。
也许这会有所帮助 http://hc.apache.org/从 jsp/response 中读取 html 并从中创建电子邮件。
关于java - 如何对 JSP 和通过单独进程发送的电子邮件使用相同的内容?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28077357/