如何在不向用户显示堆栈跟踪的情况下处理servlet过滤器中的错误状态?

Nat*_*dly 6 java jetty resteasy servlet-filters

我正在开发一个Jetty/RESTEasy应用程序.如果我WebApplicationException(myResponse)从我的一个REST端点抛出一个,它会将给定的响应发送给客户端.

当筛选器检测到错误时,我想要相同的行为:

  1. 它应该停止继续执行,并且
  2. 它应该给用户一个明确的,JSON格式的错误,不包括堆栈跟踪.

显然,只需returndoFilter方法中写入响应流和工作.但这不适用于其他方法doFilter.

抛出任何异常将满足条件#1但我还没有想出一个理智的方式来满足条件#2.(你可以在底部看到我最好的尝试.)

作为感知他的回答解释,WebApplicationExceptions的处理就像在一个过滤器的情况下任何其他异常,因此给用户带来不错的丑陋堆栈跟踪.

那么,总结一下我的问题:

  • serveltt容器有什么相同的throw new WebApplicationException(Response)
  • 也许更重要的是,其他java项目如何处理这个?

我在一个过滤器中有这个代码并且它可以工作,但我更喜欢一个更优雅的解决方案,它自动应用于所有过滤器:

public void doFilter(final ServletRequest   request, final ServletResponse response, final FilterChain chain) throws IOException, ServletException {
    try {
        doFilterOrThrow(request, response, chain);
    } catch (WebApplicationException e) {
        Response res = e.getResponse();
        ((HttpServletResponse) response).sendError(res.getStatus(), (String) res.getEntity());
    }
}
Run Code Online (Sandbox Code Playgroud)

Per*_*ion 7

你提到的具体处理的Web应用程序异常仅一个JAX-RS的容器,其中,顺便说一句,是的范围内定义一样的东西作为一个Servlet容器.

Web过滤器由Servlet容器处理,该容器不知道或不关心同一应用程序服务器中是否存在JAX-RS容器.它也不知道或不关心Web应用程序异常.因此,当您从过滤器中抛出WAE时,它将被视为与任何其他异常相同(带有堆栈跟踪的服务器错误,或者如果您在Web应用程序中设置了一个预配置的错误页面).

在我看来,如果您通过过滤器向客户端指示错误,您可以直接写入响应流.但是,如果您尝试利用一些现有的JAX-RS逻辑,那么(RESTEasy特定的)解决方案是将请求标记为过滤器中的错误,然后使用提供程序类在JAX-RS中生成WAE.例:

@WebFilter(urlPatterns = "*")
public class ForwardingFilter implements Filter {

    @Override
    public void destroy() {
        return;
    }

    @Override
    public void doFilter(final ServletRequest request,
            final ServletResponse response, final FilterChain chain)
            throws IOException, ServletException {
        // Add an error response to be processed by the JAX-RS container.
        // This would obviously be based on some condition.
        request.setAttribute("errorResponse",
                Response.status(500).entity("Didn't work out!").build());
        chain.doFilter(request, response);
    }

    @Override
    public void init(FilterConfig arg0) throws ServletException {
        return;
    }
}

@Provider
@ServerInterceptor
@HeaderDecoratorPrecedence
@RequestScoped
public class ForwardingHandlerProvider implements PreProcessInterceptor {

    @Override
    public ServerResponse preProcess(final HttpRequest request,
            final ResourceMethod method) throws Failure,
            WebApplicationException {
        final Response errorResponse = (Response) request
                .getAttribute("errorResponse");
        if (errorResponse != null)
            throw new WebApplicationException(errorResponse);
        return null;
    }
}
Run Code Online (Sandbox Code Playgroud)

由于提供程序存在于JAX-RS域中,因此Web应用程序异常将根据JAX-RS规范的第3.3.4节的规则进行处理,并在客户端获得所需的响应.

*编辑:*

最重要的是,没有标准的Java EE规定方式(当前)以集中方式处理servlet异常,类似于JAX-RS中可用的方式.既然你正在使用JBoss/RestEASY,你可以利用JBoss Seam Catch库来获得非常接近的结果.

@HandlesExceptions
public class ExceptionHandler {
    public void handleServletException(
            final @Handles @WebRequest CaughtException<ServletException> caught,
            @Context final HttpServletResponse response) {
        try {
            response.sendError(500, "An error occured");
        } catch (final IOException ioe) {
            System.err.println("Dumb IO Exception: " + ioe);
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

上面说明了一个异常处理程序,如Seam Catch文档中所述.请注意,图书馆目前处于巨大变化之中,因此您只想将其作为最后的手段使用.