Spring Boot自定义http错误响应?

mat*_*sev 12 java rest error-handling spring-boot

如果Spring Boot Web应用程序中发生异常,如何自定义响应状态代码和响应正文中的数据?

我创建了一个Web应用程序,如果由于某些错误的内部状态而发生意外情况,则会引发自定义异常.因此,触发错误的请求的响应主体看起来像:

HTTP/1.1 500 Internal Server Error
{
    "timestamp": 1412685688268,
    "status": 500,
    "error": "Internal Server Error",
    "exception": "com.example.CustomException",
    "message": null,
    "path": "/example"
}
Run Code Online (Sandbox Code Playgroud)

现在,我想更改状态代码并在响应正文中设置字段.在我脑海中浮现的一个解决方案是:

@ControllerAdvice
class CustomResponseEntityExceptionHandler extends ResponseEntityExceptionHandler {

    @ExceptionHandler
    @ResponseStatus(HttpStatus.BAD_REQUEST)
    @ResponseBody
    ErrorMessage handleBadCredentials(CustomException e) {
        return new ErrorMessage("Bad things happened");
    }
}

@XmlRootElement
public class ErrorMessage(
    private String error;

    public ErrorMessage() {
    }

    public ErrorMessage(String error) {
        this.error = error;
    }

    public String getError() {
        return error;
    }

    public void setError(String error) {
        this.error = error;
    }
)
Run Code Online (Sandbox Code Playgroud)

然而,这创造了(如所怀疑的)完全不同的反应:

HTTP/1.1 400 Bad Request
{
    "error": "Bad things happened"
}
Run Code Online (Sandbox Code Playgroud)

小智 13

正如@zeroflagL所提到的,Spring Boot制作了"标准"错误响应体org.springframework.boot.autoconfigure.web.DefaultErrorAttributes.与您的需求类似,我想利用所有这些,但只是增加了一些由我的一些例外提供的"类型"字段.

我通过实施一个Component分类来做到这一点 DefaultErrorAttributes.Spring Boot自动将其拾起并使用我的而不是默认值.

@Component
public class ExtendedErrorAttributes extends DefaultErrorAttributes {
    @Override
    public Map<String, Object> getErrorAttributes(RequestAttributes requestAttributes, boolean includeStackTrace) {
        final Map<String, Object> errorAttributes = super.getErrorAttributes(requestAttributes, includeStackTrace);

        final Throwable error = super.getError(requestAttributes);
        if (error instanceof TypeProvider) {
            final TypeProvider typeProvider = (TypeProvider) error;
            errorAttributes.put("type", typeProvider.getTypeIdentifier());
        }

        return errorAttributes;
    }
}
Run Code Online (Sandbox Code Playgroud)

有了这个,我得到了一个增强的JSON响应体,例如

{
  "timestamp": 1488058582764,
  "status": 429,
  "error": "Too Many Requests",
  "exception": "com.example.ExternalRateLimitException",
  "message": "DAILY_LIMIT: too many requests",
  "path": "/api/lookup",
  "type": "DAILY_LIMIT"
}
Run Code Online (Sandbox Code Playgroud)

  • 到目前为止(可能从Spring Boot 2开始),您需要扩展org.springframework.boot.web.servlet.error.DefaultErrorAttributes,它在参数中具有org.springframework.web.context.request.WebRequest而不是RequestAttributes。 (2认同)

mat*_*sev 7

可以使用HttpServletResponse.sendError(int)方法更改http响应状态代码,例如

@ExceptionHandler
void handleIllegalArgumentException(IllegalArgumentException e, HttpServletResponse response) throws IOException {
    response.sendError(HttpStatus.BAD_REQUEST.value());
}
Run Code Online (Sandbox Code Playgroud)

或者,@ExceptionHandler如果您有两个或更多异常来生成相同的响应状态,则可以在注释中声明异常类型:

@ExceptionHandler({IllegalArgumentException.class, NullPointerException.class})
void handleBadRequests(HttpServletResponse response) throws IOException {
    response.sendError(HttpStatus.BAD_REQUEST.value());
}
Run Code Online (Sandbox Code Playgroud)

更多信息可以在我的博客文章中找到.

  • 值得注意的是,这仅设置响应状态代码;响应消息的其余部分保持不变。虽然这并不完全是提问者所要求的,但这正是我自己的用例所需要的。 (2认同)