TomEE:对于 warding 请求到另一个 war
TomEE: Forwarding request to a different war
我无法让 TomEE 将我的请求从同一个 ear 文件转发到不同的 webapp。
我更新了 context.xml 启用的 crossContext:
<Context antiResourceLocking="false" privileged="true" crossContext="true">
<!--
<Valve className="org.apache.catalina.valves.RemoteAddrValve"
allow="172.17.0.1" />
-->
<Manager sessionAttributeValueClassNameFilter="java\.lang\.(?:Boolean|Integer|Long|Number|String)|org\.apache\.catalina\.filters\.CsrfPreventionFilter$LruCache(?:$1)?|java\.util\.(?:Linked)?HashMap"/>
</Context>
我转发的 webapp 是一个简单的 Servlet:
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
try {
RequestDispatcher dispatcher;
dispatcher = getServletContext().getContext("/web2").getRequestDispatcher("/webapp2/hello");
dispatcher.forward(req, resp);
} catch (Exception e) {
StringWriter sw = new StringWriter();
PrintWriter pw = new PrintWriter(sw, true);
e.printStackTrace(pw);
resp.sendError(HttpServletResponse.SC_BAD_REQUEST, sw.toString());
}
}
它应该被转发到的 webapp 是一个看起来像这样的 Rest 资源:
package local.gerb;
import javax.ws.rs.*;
@Path("webapp2")
public class HelloResource {
@GET
@Path("hello")
@Produces("text/plain")
public String getMessage() {
return "Rest Never Sleeps";
}
}
我确实通过卷曲特定端点验证了 webapps2 是否正常工作。
但是,当我卷曲应该转发到 webapp2 的 webapp1 时,我得到了 404。奇怪的是,如果我将 Rest Resource 转换为普通 servlet,转发工作正常。
如果你想看我把它推送到 github 的代码:
https://github.com/jstralko/tomee-fwd
我有一个解释如何构建和 运行 这个应用程序的自述文件。它 运行 位于 docker 容器中,因此一切都是独立的并且非常容易构建并且 运行。
我决定调试 TomEE,并在 CXFJAXRSFilter.java 中发现导致问题的代码:
@Override
public void doFilter(final ServletRequest request, final ServletResponse response, final FilterChain chain) throws IOException, ServletException {
if (!HttpServletRequest.class.isInstance(request)) {
chain.doFilter(request, response);
return;
}
final HttpServletRequest httpServletRequest = HttpServletRequest.class.cast(request);
final HttpServletResponse httpServletResponse = HttpServletResponse.class.cast(response);
if (CxfRsHttpListener.TRY_STATIC_RESOURCES) { // else 100% JAXRS
if (servletMappingIsUnderRestPath(httpServletRequest)) {
chain.doFilter(request, response);
return;
}
final InputStream staticContent = delegate.findStaticContent(httpServletRequest, welcomeFiles);
if (staticContent != null) {
chain.doFilter(request, response);
return;
}
}
try {
delegate.doInvoke(
new ServletRequestAdapter(httpServletRequest, httpServletResponse, request.getServletContext()),
new ServletResponseAdapter(httpServletResponse));
} catch (final Exception e) {
throw new ServletException("Error processing webservice request", e);
}
}
导致问题的行:
if (CxfRsHttpListener.TRY_STATIC_RESOURCES) { // else 100% JAXRS
所以我找到了TRY_STATIC_RESOURCES
的定义
public static final boolean TRY_STATIC_RESOURCES = "true".equalsIgnoreCase(SystemInstance.get().getProperty("openejb.jaxrs.static-first", "true"));
所以我将 openejb.jaxrs.static-first 的 system.property 更新为 false,它起作用了。
$> curl 'http://localhost:8080/web1/webapp1?op=foo'
Hello From Webapp2 Rest Resource%
我为我们这些一起玩的人推送了我对 github 存储库的更改。
我无法让 TomEE 将我的请求从同一个 ear 文件转发到不同的 webapp。
我更新了 context.xml 启用的 crossContext:
<Context antiResourceLocking="false" privileged="true" crossContext="true">
<!--
<Valve className="org.apache.catalina.valves.RemoteAddrValve"
allow="172.17.0.1" />
-->
<Manager sessionAttributeValueClassNameFilter="java\.lang\.(?:Boolean|Integer|Long|Number|String)|org\.apache\.catalina\.filters\.CsrfPreventionFilter$LruCache(?:$1)?|java\.util\.(?:Linked)?HashMap"/>
</Context>
我转发的 webapp 是一个简单的 Servlet:
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
try {
RequestDispatcher dispatcher;
dispatcher = getServletContext().getContext("/web2").getRequestDispatcher("/webapp2/hello");
dispatcher.forward(req, resp);
} catch (Exception e) {
StringWriter sw = new StringWriter();
PrintWriter pw = new PrintWriter(sw, true);
e.printStackTrace(pw);
resp.sendError(HttpServletResponse.SC_BAD_REQUEST, sw.toString());
}
}
它应该被转发到的 webapp 是一个看起来像这样的 Rest 资源:
package local.gerb;
import javax.ws.rs.*;
@Path("webapp2")
public class HelloResource {
@GET
@Path("hello")
@Produces("text/plain")
public String getMessage() {
return "Rest Never Sleeps";
}
}
我确实通过卷曲特定端点验证了 webapps2 是否正常工作。
但是,当我卷曲应该转发到 webapp2 的 webapp1 时,我得到了 404。奇怪的是,如果我将 Rest Resource 转换为普通 servlet,转发工作正常。
如果你想看我把它推送到 github 的代码: https://github.com/jstralko/tomee-fwd
我有一个解释如何构建和 运行 这个应用程序的自述文件。它 运行 位于 docker 容器中,因此一切都是独立的并且非常容易构建并且 运行。
我决定调试 TomEE,并在 CXFJAXRSFilter.java 中发现导致问题的代码:
@Override
public void doFilter(final ServletRequest request, final ServletResponse response, final FilterChain chain) throws IOException, ServletException {
if (!HttpServletRequest.class.isInstance(request)) {
chain.doFilter(request, response);
return;
}
final HttpServletRequest httpServletRequest = HttpServletRequest.class.cast(request);
final HttpServletResponse httpServletResponse = HttpServletResponse.class.cast(response);
if (CxfRsHttpListener.TRY_STATIC_RESOURCES) { // else 100% JAXRS
if (servletMappingIsUnderRestPath(httpServletRequest)) {
chain.doFilter(request, response);
return;
}
final InputStream staticContent = delegate.findStaticContent(httpServletRequest, welcomeFiles);
if (staticContent != null) {
chain.doFilter(request, response);
return;
}
}
try {
delegate.doInvoke(
new ServletRequestAdapter(httpServletRequest, httpServletResponse, request.getServletContext()),
new ServletResponseAdapter(httpServletResponse));
} catch (final Exception e) {
throw new ServletException("Error processing webservice request", e);
}
}
导致问题的行:
if (CxfRsHttpListener.TRY_STATIC_RESOURCES) { // else 100% JAXRS
所以我找到了TRY_STATIC_RESOURCES
的定义public static final boolean TRY_STATIC_RESOURCES = "true".equalsIgnoreCase(SystemInstance.get().getProperty("openejb.jaxrs.static-first", "true"));
所以我将 openejb.jaxrs.static-first 的 system.property 更新为 false,它起作用了。
$> curl 'http://localhost:8080/web1/webapp1?op=foo'
Hello From Webapp2 Rest Resource%
我为我们这些一起玩的人推送了我对 github 存储库的更改。