Jetty Filter修改响应 - java.lang.IllegalStateException:WRITER

时间:2012-11-28 18:58:41

标签: servlets jetty servlet-filters

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

  

java.lang.IllegalStateException:WRITER at   org.eclipse.jetty.server.Response.getOutputStream(Response.java:657)     在   javax.servlet.ServletResponseWrapper.getOutputStream(ServletResponseWrapper.java:142)     在   org.eclipse.jetty.servlets.ProxyServlet.service(ProxyServlet.java:414)     在   org.eclipse.jetty.servlet.ServletHolder.handle(ServletHolder.java:643)     在   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);

不确定哪里出错了。

1 个答案:

答案 0 :(得分:13)

您的Jetty响应可以是两种(技术上为三种)不同的模式。一种是写入模式,另一种是流模式(第三种基本上是未定模式)。

如果您在未定的&#39;上致电getWriter()回复,你把它放在作家模式,这是不可逆转的。如果以后尝试在流模式下使用此响应(通过调用getOutputStream()),则会抛出您看到的异常。

要解决此问题,请不要在作家模式下使用此响应,并且“做你自己的事情”。而是在OutputStream上。如果您稍后访问作者(在doChain之后),您将获得逆异常

  

java.lang.IllegalStateException:STREAM

代替。