Spring Boot - 处理用BindException包装的异常

时间:2018-02-14 11:31:34

标签: java spring-boot exception-handling

我正在寻找一种方法来处理在将请求参数绑定到DTO字段期间抛出的自定义异常。

我在Spring Boot应用程序中有一个cantroller,如下所示

@GetMapping("/some/url")
public OutputDTO filterEntities(InputDTO inputDTO) {
    return service.getOutput(inputDTO);
}

输入DTO的字段很少,其中一个是枚举类型

public class InputDTO {

    private EnumClass enumField;
    private String otherField;

    /**
     * more fields
     */
}

用户将以某种方式点击网址

localhost:8081/some/url?enumField=wrongValue&otherField=anyValue

现在,如果用户为enumField发送了错误的值,我想用特定的消息抛出我的CustomException。枚举实例创建和抛出异常的过程在binder

中实现
@InitBinder
public void initEnumClassBinder(final WebDataBinder webdataBinder) {
    webdataBinder.registerCustomEditor(
            EnumClass.class,
            new PropertyEditorSupport() {
                @Override
                public void setAsText(final String text) throws IllegalArgumentException {
                    try {
                        setValue(EnumClass.valueOf(text.toUpperCase()));
                    } catch (Exception exception) {
                        throw new CustomException("Exception while deserializing EnumClass from " + text, exception);
                    }
                }
            }
    );
}

问题是当抛出异常时,无法用

处理它
@ExceptionHandler(CustomException.class)
public String handleException(CustomException exception) {
    // log exception
    return exception.getMessage();
}

Spring使用BindException包装初始异常。该实例包含我的初始错误消息,但与其他文本连接,这对我来说是多余的。我不认为解析和子串化该消息是好的......

我错过了什么吗?从最初获取消息的正确方法是什么 CustomException在这里?

1 个答案:

答案 0 :(得分:1)

在使用@ExceptionHandler带注释的方法进入控制器方法之前,您将无法处理抛出的异常。 Spring在进入控制器之前通过注册DefaultHandlerExceptionResolver extends AbstractHandlerExceptionResolver处理程序来处理这些异常。 这是BindingException的情况,当Spring无法绑定请求参数以匹配您的InputDTO对象时,抛出该异常。 您可以做的是注册您自己的处理程序(创建实现ComponentHandlerExceptionResolver接口的Ordered,在处理错误时将其赋予最高优先级,并根据需要处理异常。 您还必须注意BindException,因为它包装了您的自定义异常CustomException.class

import java.io.IOException;

import javax.servlet.http.HttpServletRequest; 
import javax.servlet.http.HttpServletResponse;

import org.slf4j.Logger; import org.slf4j.LoggerFactory; 
import org.springframework.core.Ordered; 
import org.springframework.stereotype.Component; 
import org.springframework.validation.BindException; 
import org.springframework.validation.ObjectError; 
import org.springframework.web.servlet.HandlerExceptionResolver; 
import org.springframework.web.servlet.ModelAndView;


import yourpackage.CustomException;

@Component() 
public class BindingExceptionResolver implements HandlerExceptionResolver, Ordered {
    private static final Logger logger = LoggerFactory.getLogger(BindingExceptionResolver.class);

    public BindingExceptionResolver() {
    }

    private ModelAndView handleException(ObjectError objectError, HttpServletResponse response){
        if (objectError == null) return null;
        try {
            if(objectError.contains(CustomException.class)) {
                CustomException ex = objectError.unwrap(CustomException.class);
                logger.error(ex.getMessage(), ex);
                return handleCustomException(ex, response);
            }
        } catch (Exception e) {
            logger.error(e.getMessage(), e);
        }
        return null;
    }

    protected ModelAndView handleCustomException(CustomException ex, HttpServletResponse response) throws IOException {
        response.sendError(HttpServletResponse.SC_BAD_REQUEST, ex.getMessage());
        return new ModelAndView();
    }

    @Override
    public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) {
        try {
            if (ex instanceof org.springframework.validation.BindException) {
                BindException be = (BindException) ex;
                logger.debug("Binding exception in {} :: ({}) :: ({})=({})", be.getObjectName(), be.getBindingResult().getTarget().getClass(), be.getFieldError().getField(), be.getFieldError().getRejectedValue());
                return be.getAllErrors().stream()
                    .filter(o->o.contains(Exception.class))
                    .map(o -handleException(o, response))
                    .filter(mv -mv !=null)
                    .findFirst().orElse(null);
            }
        } catch (Exception handlerException) {
            logger.error("Could not handle exception", handlerException); 
        }
        return null;
    }


    @Override
    public int getOrder() {
        return Integer.MIN_VALUE;
    }

}

希望有帮助