测试开发专题:spring-boot自定义返回参数校验错误信息

测试轩      2022-04-04     366

关键词:

之前两篇文章 Spring-boot自定义参数校验注解如何在spring-boot中进行参数校验,我们介绍了,参数校验以及如何自定义参数校验注解,但是当传递参数出错时,只是把错误信息打印到了控制台,合理的做法是应该把校验的错误信息返回给前端,告知用户那里有问题,下面就这一步内容进行说明。

请求body参数

上篇文章 Spring-boot自定义参数校验注解的最后,在控制台打印了校验出错的信息

出错的异常类是MethodArgumentNotValidException,那如果想要自定义异常的返回,就需要在全局的异常处理器中针对这种异常进行处理。

在这篇文章 spring-boot自定义异常返回中,我们说了如何进行自定义异常的返回,参数校验的错误信息返回依然按照此方式进行处理,在全局异常处理类中定义异常处理方法:

@ExceptionHandler(value = MethodArgumentNotValidException.class)
@ResponseStatus(HttpStatus.BAD_REQUEST)
@ResponseBody
public UnifyResponse handlerBeanValidationException(HttpServletRequest request,
                                                    MethodArgumentNotValidException ex) {
    String requestUri = request.getRequestURI();
    String method = request.getMethod();

    List<ObjectError> errors = ex.getBindingResult().getAllErrors();
    return UnifyResponse.builder()
            .code(5000)
            .message(formatError(errors))
            .requestUri(method + " " + requestUri)
            .build();
}

private String formatError(List<ObjectError> errors) {
    StringBuilder builder = new StringBuilder();
    errors.forEach(error -> builder.append(error.getDefaultMessage()).append(";"));
    return builder.toString();
}

我们来对上面的代码进行一下解释:

  • 因为这个处理方法只是针对MethodArgumentNotValidException这个异常进行处理,所以@ExceptionHandler(value = MethodArgumentNotValidException.class)这里指定
  • @ResponseStatus(HttpStatus.BAD_REQUEST),所有的参数校验错误都是一类的,状态码设置为HttpStatus.BAD_REQUEST,也就是code等于400,当然也可以定义为其他的,按照自己业务需求定义就好,可以参考这篇文章 spring-boot自定义异常返回里关于自定义状态码的部分。
  • @ResponseBody,因为这个异常处理方法要返回自定义的对象,所以要使用这个注解,不然spring-boot是不会对自定义对象进行序列化的
  • List<ObjectError> errors = ex.getBindingResult().getAllErrors()进行参数校验的时候,可能多个参数都有问题,我们希望能够有问题的参数的错误信息全部都返回回去,所以这里要获取所有的错误。

回顾一下参数的定义,对这里有疑惑的同学可以看一下这篇文章Spring-boot自定义参数校验注解

@Builder
@Getter
@Setter
@PasswordEqual(min = 5, message = "密码和确认密码不一样")
public class UserDto {

    private int userId;

    @Length(min = 2, max = 10, message = "用户名长度必须在2-10的范围内")
    private String username;

    private String password;

    private String confirmPassword;
}

接下来我们定再定义一个简单的接口,当传参出错时看异常处理方法能否按照定义的那样返回错误信息

@RequestMapping("/v2/user/create")
public UserDto createUser(@RequestBody @Validated  UserDto userDto){
    return userDto;
}

我们先来构造一个密码和确认密码不一致的情况

file

可以看到定义的错误信息被返回,而且状态码和自定义的code都是符合设计的,接下来我们再看一下多个参数错误的场景:

file

上面的场景中,用户名是不符合要求的,密码和确认密码也不一样,所以会产生两条错误信息,将其拼接到一起,返回给前端。

之前讨论的都是body里提交的参数,接下来我们看下路径参数或者查询参数校验出错时的处理

查询参数和路径参数

我们先定义两个接口一个是路径参数查询信息,一个是通过查询参数查询信息

@GetMapping("/v2/user/info")
public UserDto getUserInfo(@RequestParam @Length(min = 2, max = 5, message = "用户名长度必须在2-5的范围")
                                       String username){
    return UserDto.builder()
            .userId(1000)
            .username(username)
            .build();
}

@GetMapping("/v2/user/{username}")
public UserDto getUserInfoV2(@PathVariable @Length(min = 2, max = 5, message = "用户名长度必须在2-5的范围") String username){
    return UserDto.builder()
            .userId(2000)
            .username(username)
            .build();
}

然后我们访问这两接口,当发生错误时,看看他们会不会进入上文定义的异常处理方法中:

file

很明显,并没有进入上文定义的异常处理方法中,而是进入了handleException这个异常方法当中,这个算是个兜底的异常处理方法。

看一下控制台的输出:

file

这里抛出了ConstraintViolationException异常,这个异常我们并没有定制对应的异常处理函数,下面我们就来写一下:

@ExceptionHandler(ConstraintViolationException.class)
@ResponseStatus(HttpStatus.BAD_REQUEST)
@ResponseBody
public UnifyResponse handlerConstraintViolationException(HttpServletRequest request, ConstraintViolationException ex){
    String requestUri = request.getRequestURI();
    String method = request.getMethod();

    Set<ConstraintViolation<?>> errors = ex.getConstraintViolations();
    return UnifyResponse.builder()
            .code(6000)
            .message(formatConstraintException(errors))
            .requestUri(method + " " + requestUri)
            .build();
}

private String formatConstraintException(Set<ConstraintViolation<?>> constraintViolations){
    StringBuilder builder = new StringBuilder();
    constraintViolations.forEach(constraintViolation -> builder.append(constraintViolation.getMessage()));
    return builder.toString();
}

整体来说异常处理和上文几乎是一样的,只是获取错误message的方式不一样而已,我们再请求一下:

file

至此参数校验的错误message自定义返回,都完成了。

本文链接:https://www.immortalp.com/articles/2020/05/16/1589623786527.html
欢迎大家去 我的博客 瞅瞅,里面有更多关于测试实战的内容哦!!

带有spring-boot的自定义sql中的Liquibase参数

】带有spring-boot的自定义sql中的Liquibase参数【英文标题】:Liquibaseparametersincustomsqlwithspring-boot【发布时间】:2019-04-1910:16:51【问题描述】:我在jdk11上有一个spring-boot应用程序,使用maven,具有以下liquibase依赖项:<dependency><... 查看详情

在 Spring Boot MVC 测试中自定义 bean

...【发布时间】:2017-06-2101:34:51【问题描述】:我正在使用Spring-boot1.4.0.RELEASE。已按照建议构建了一个MVCint测试https://docs.spring.io/spring-boot/docs/current/reference/html/boot-feature 查看详情

微信公众号开发总结(代码片段)

...响,在开发过程中,我们一般不采用注册的公众号来部署测试,而是采用公众平台测试号公众平台测试号访问地址在测试号页面中,我们需要获取或操作一下几个相关的配置查看测试号的APPID和APPSECRET给该测试号设置自定义菜单... 查看详情

接口测试

转自要做接口测试,需要找开发获取接口文档,并明确一下几点:  1、被测接口的地址  2、接口参数,以及各个参数的说明  3、必要的http头与http体(http头是可以自定义的,可以用来校验是否是自己人访问)  4、接口返... 查看详情

《springsecurity框架专题》-03实现自定义登录界面(代码片段)

...1.3.其他页面2.SpringSecurity相关配置2.1.配置认证信息3.登录测试4.关闭csrf拦截5.csrf防护5.1.CsrfFilter源码查看5.2.在认证页面携带token请求6.注销前面通过入门案例介绍,我们发现在SpringSecurity中如果我们没有使用自定义的登录界面,那... 查看详情

企业级spring-boot案例-自定义springbootstarter(代码片段)

...ctories3.7发布自定义starter3.8测试自定义starter3.8.1添加config-spring-boot-starter依赖3.8.2配置application.yml3.8.3编写ConfigInfoController.java3.8.4启动测试4.SpringBootStarter原理4.1自动配置原理4.1.1自动配置类的获取与注入4.1.2自动配置的过程4.2SpringBo... 查看详情

spring-boot使用嵌入式容器,那怎么配置自定义filter呢

Listener、Filter和Servlet是JavaWeb开发过程中常用的三个组件,其中Filter组件的使用频率最高,经常被用来做简单的权限处理、请求头过滤和防止XSS攻击等。如果我们使用的是传统的SpringMVC进行开发,那么只需要在Tomcat的web.xml文件中... 查看详情

后端开发总结:代码健壮性:容错处理+测试(代码片段)

代码健壮性:容错处理+测试1.1客户端接收云端请求容错处理1.2参数校验1.3测试1.1客户端接收云端请求容错处理几种错误情况没有请求通,没有返回。服务端错误,对应的就是http的状态码,比如500请求通了,... 查看详情

项目实战:自定义参数解析器+策略模式实现异步通知返回参数的处理

查看详情

阶段一-01.万丈高楼,地基首要-第3章用户登录注册模块开发-3-3自定义响应数据结构

...api这一层。我们在进行maven安装的时候,也会运行这里的测试类我们也可以通过maven去忽略这里为了简便,我们直接把这几行代码注释、掉。再次进行maveninstall启动api测试类postman测试。返回了200,也就是我们的请求成功了。HttpSta... 查看详情

PyQt4:创建返回参数的自定义对话框

】PyQt4:创建返回参数的自定义对话框【英文标题】:PyQt4:Createacustomdialogthatreturnsparameters【发布时间】:2011-04-2222:20:08【问题描述】:我正在尝试向我当前的GUI添加一个自定义对话框,可以启动该对话框供用户设置一些参数。理... 查看详情

自定义参数校验以及统一处理结果集(代码片段)

...目前后端是否分离都是非常必要的,方便对接接口的开发人员更加清晰地知道这个接口的调用是否成功(不能仅仅简单地看返回值是否为null就判断成功与否,因为有些接口的设计就是如此),使用一个状态码... 查看详情

loadrunner-获取返回值和自定义参数(参数运算)

实例:手机端操作,A新增了一条事件(返回结果:事件id,例如:1),A这时需要获取新增产生的事件id,并作为参数进行传递,才能将这条事件上报给B(返回结果:事件id不变,步骤id等于事件id加1),B需要获取上报返回的事... 查看详情

如何创建自定义注解拆分请求参数并收集返回结果?

】如何创建自定义注解拆分请求参数并收集返回结果?【英文标题】:Howtocreateacustomisationannotationforsplittingrequestparamandcollectreturnresult?【发布时间】:2019-03-1911:23:27【问题描述】:我有一个方法params是一个大于50000个项目的列表;... 查看详情

spring-boot 单元测试获取应用程序属性

】spring-boot单元测试获取应用程序属性【英文标题】:spring-bootunittestinggetapplicationproperties【发布时间】:2019-01-2407:55:45【问题描述】:我是spring-boot新手,目前正在尝试开发一个kafka生产者我想测试使用属性文件中定义的值的方... 查看详情

spring boot中自定义验证的参数化junit测试

】springboot中自定义验证的参数化junit测试【英文标题】:Parameterizedjunittestofcustomvalidationinspringboot【发布时间】:2021-09-0507:00:58【问题描述】:我在springboot中有这个验证器,当整数不在1和3之间时会给出错误,并且我正在使用addCo... 查看详情

ios开发:uinavigationcontroller自定义返回按钮,系统导航支持侧滑返回(代码片段)

...手势,那么又区别于原生系统,改变了用户的操作习惯在开发过程中,你是否也遇到了这样的困惑呢?希望这篇文章可以解决你的困惑!自定义返回按钮自定义一个导航的基类,继承于UINavigationController@interfaceFQBaseNavigationControlle... 查看详情

spring 发生事件后如何使用自定义参数启动@Bean 进行测试?

】spring发生事件后如何使用自定义参数启动@Bean进行测试?【英文标题】:Howtostarta@Beanwithcustomparametersafteraneventhadhappenedwithspringfortests?【发布时间】:2021-11-2619:59:52【问题描述】:我正在为使用R2dbc的项目添加带有TestContainers框架... 查看详情