請求引數校驗,在實際的應用中很常見,網上的文章大部分提供的使用註解的方式做引數校驗。本文主要介紹 Spring Webflux Function Endpoint 使用 Spring Validation 來校驗請求的引數。使用上一篇文章的示例來演示。
使用步驟如下:
1.建立校驗器 Validator
2.運用校驗器
3.丟擲異常,返回 http status 400 錯誤
PersonValidator.java
package com.example.springbootdemo.webflux.restful;
import org.springframework.stereotype.Component;
import org.springframework.validation.Errors;
import org.springframework.validation.ValidationUtils;
import org.springframework.validation.Validator;
@Component
public class PersonValidator implements Validator {
@Override
public boolean supports(Class<?> clazz) {
return Person.class.isAssignableFrom(clazz);
}
// 校驗引數的方法
@Override
public void validate(Object o, Errors errors) {
ValidationUtils.rejectIfEmpty(errors, "name", "name.required");
ValidationUtils.rejectIfEmpty(errors, "age", "age.required");
Person p = (Person) o;
if (p.getAge() != null && p.getAge() < 0) {
errors.rejectValue("age", "negative.value");
} else if (p.getAge() != null && p.getAge() > 200) {
errors.rejectValue("age", "too.old");
}
}
}
校驗器在 savePerson
方法中的使用
@Slf4j
@Component
public class PersonHandler {
@Autowired
private PersonRepository repository;
@Autowired
private PersonValidator validator;
public Mono<ServerResponse> savePerson(ServerRequest request) {
Mono<Person> personMono = request.bodyToMono(Person.class).doOnNext(this::validate);
return ServerResponse.ok().contentType(MediaType.APPLICATION_JSON)
.body(this.repository.savePerson(personMono), Void.class);
}
public void validate(Person person) {
Errors errors = new BeanPropertyBindingResult(person, Person.class.getName());
validator.validate(person, errors);
if (errors.hasErrors()) {
// 丟擲 http status 400 異常
throw new ServerWebInputException(errors.toString());
}
}
// .... 省略
}
請求效果:
使用 Spring 官方文件提供的示例不會丟擲 http code 400 錯誤,返回的是http code 為 200。
接下來,我們來看一下Validator
介面中的兩個方法 supports
和 validate
supports(Class)
: 判斷當前的校驗器用指定的類上。validate(Object, org.springframework.validation.Errors)
: 校驗給定的物件,如果出現錯誤,就給Errors
註冊Error
資訊。
另外,Spring 還提供了非常好用的 ValidationUtils
的工具類,提供了靜態的方法
rejectIfEmpty
rejectIfEmptyOrWhitespace
全域性異常的使用
@Configuration
@Slf4j
public class GlobalErrorConfig {
private ObjectMapper objectMapper = new ObjectMapper();
@Bean
@Order(-2)
public WebExceptionHandler exceptionHandler() {
return (ServerWebExchange serverWebExchange, Throwable t) -> {
DataBuffer dataBuffer = serverWebExchange.getResponse().bufferFactory().allocateBuffer();
Result result = new Result();
if (t instanceof ServerWebInputException) {
ServerWebInputException exception = (ServerWebInputException) t;
result.setCode(exception.getStatus().value());
result.setMessage(exception.getReason());
} else {
result.setCode(HttpStatus.INTERNAL_SERVER_ERROR.value());
result.setMessage(HttpStatus.INTERNAL_SERVER_ERROR.toString());
}
try {
dataBuffer.write(objectMapper.writeValueAsBytes(result));
} catch (JsonProcessingException e) {
log.error(NestedExceptionUtils.buildMessage("write error", e));
}
ServerHttpResponse response = serverWebExchange.getResponse();
response.setRawStatusCode(result.getCode());
return response.writeWith(Mono.just(dataBuffer));
};
}
}
Result.java
import lombok.Getter;
import lombok.Setter;
@Getter
@Setter
public class Result {
private Integer code;
private String message;
}
請求效果:
至此,Webflux 的Function Endpoint 的引數校驗的使用結束了。
參考: