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 存储库的更改。