11

我正在尝试修改过滤器中的 http 响应并收到以下异常

java.lang.IllegalStateException:org.eclipse.jetty.servlets 的 javax.servlet.ServletResponseWrapper.getOutputStream(ServletResponseWrapper.java:142) 的 org.eclipse.jetty.server.Response.getOutputStream(Response.java:657) 的 WRITER。 org.eclipse.jetty.servlet.ServletHolder.handle(ServletHolder.java:643) 的 ProxyServlet.service(ProxyServlet.java:414) org.eclipse.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1331)在 com.cisco.vsx.node.proxy.http.RegexFilter.doFilter(RegexFilter.java:36)

我正在使用 SelectChannelSelector 和 ProxyServlet.Transparent 代理。

下面是测试类的片段

ServletContextHandler context = new ServletContextHandler(ServletContextHandler.SESSIONS);
context.setContextPath("/");

ProxyServlet.Transparent p1 = new ProxyServlet.Transparent("/proxy",
    "www.cisco.com", 80);
ServletHolder servletHolder = new ServletHolder(p1);
context.addServlet(servletHolder, "/proxy/*"); 
context.addFilter(new FilterHolder(RegexFilter.class), "/*", null);

server.setHandler(context);

server.start();
server.join();

这是过滤器类的代码

PrintWriter out = response.getWriter();
CharResponseWrapper wrapper = new CharResponseWrapper((HttpServletResponse) response);

chain.doFilter(request, wrapper);

String html = wrapper.toString();
if (regex != null && response.getContentType() != null 
        && response.getContentType().startsWith("text/html")) {
    Matcher matcher = regex.matcher(html);
    Map<Integer, Integer> matches = new LinkedHashMap<Integer, Integer>();
    while (matcher.find()) {
        int start = matcher.start(1);
        System.out.println("START" + start);
        int end = matcher.end(1);
        System.out.println("END" + end);
        matches.put(start, end - start);
    }
    StringBuffer sb = new StringBuffer();
    int start = 0;
    for (int startIndex : matches.keySet()) {
        String str = html.substring(start, startIndex) + "/proxy/";
        sb.append(str);
        start = startIndex + matches.get(startIndex);
    }
    html = sb.toString();
}

response.setContentLength(html.getBytes().length);
out.write(html);

不知道哪里出了问题。

4

1 回答 1

15

您的 Jetty Response 可以采用两种(技术上是三种)不同的模式。一种是writer模式,一种是streaming模式(第三种基本是未定模式)。

如果您调用getWriter()“未决定”的响应,则将其置于编写器模式,这是无法逆转的。如果稍后尝试在流模式下使用此响应(通过调用getOutputStream()),则会抛出您看到的异常。

要解决此问题,请不要在编写器模式下使用此响应,而是在 OutputStream 上“做你的事”。如果您稍后(在 doChain 之后)访问作者,您会得到逆异常

java.lang.IllegalStateException:流

反而。

于 2015-07-09T08:03:31.283 回答