泽西岛…如何记录所有异常,但仍然调用ExceptionMappers


问题内容

我有点束缚…要我的蛋糕,也要吃。

我想记录我的应用程序引发的 所有 异常。因此,如果有人输入错误的URL,我想将堆栈跟踪记录到SLF4J。

因此,您可能会想,“嘿,这很简单,只需实现一个异常映射器并记录异常。”所以我做了:

public class RestExceptionMapper implements ExceptionMapper<java.lang.Exception> {
    private static final Logger log = LoggerFactory.getLogger(RestExceptionMapper.class);

    /**
     * {@inheritDoc}
     */
    @Override
    public Response toResponse(Exception exception) {
        log.error("toResponse() caught exception", exception);
        return null;
    }
}

如果您这样做,则当有人输入错误的URL时,他们会看到500错误,而不是404错误。有人猜测返回null会在链处理程序中传播异常,但是Jersey并没有这样做。实际上,它提供的信息很少,为什么会选择一个处理程序而不是另一个处理程序…

有没有人遇到过这个问题,您是如何解决的?


问题答案:

要返回正确的http状态代码,您的异常映射器可能看起来像这样:

@Provider
public class RestExceptionMapper implements ExceptionMapper<Throwable>
{
    private static final Logger log = LoggerFactory.getLogger(RestExceptionMapper.class);

    @Override
    public Response toResponse(Throwable exception)
    {
        log.error("toResponse() caught exception", exception);

        return Response.status(getStatusCode(exception))
                .entity(getEntity(exception))
                .build();
    }

    /*
     * Get appropriate HTTP status code for an exception.
     */
    private int getStatusCode(Throwable exception)
    {
        if (exception instanceof WebApplicationException)
        {
            return ((WebApplicationException)exception).getResponse().getStatus();
        }

        return Response.Status.INTERNAL_SERVER_ERROR.getStatusCode();
    }

    /*
     * Get response body for an exception.
     */
    private Object getEntity(Throwable exception)
    {
        // return stack trace for debugging (probably don't want this in prod...)
        StringWriter errorMsg = new StringWriter();
        exception.printStackTrace(new PrintWriter(errorMsg));
        return errorMsg.toString();            
    }
}

也听起来您对级联异常映射器感兴趣,但是根据规范,这是不可能的:

JAX-RS 2.0规范,第4.4章

“异常映射提供程序将已检查的或运行时异常映射到Response实例。异常映射提供程序实现ExceptionMapper接口,并且可以使用@Provider进行注释以进行自动发现。在选择异常映射提供程序来映射异常时,必须使用实现泛型类型是异常的最接近超类的提供程序。

当资源类或提供程序方法引发异常时,存在一个异常映射提供程序,则匹配提供程序用于获取Response实例。处理结果响应,就像Web资源方法已返回响应一样,请参阅第3.3.3节。特别是,必须使用第6章中定义的ContainerResponse过滤器链来处理映射的Response。

为避免潜在的无限循环,在处理请求及其相应的响应期间必须使用单个异常映射器。JAX-
RS实现不得在处理先前从异常映射的响应时尝试映射抛出的异常。相反,必须按照3.3.4节中的步骤3和4所述处理此异常。”