Доброго времени суток, Хабр!
Спустя некоторое время решил вновь написать сюда и поделиться своим опытом. На этот раз статья будет о том, как кастомизировать стандартные валидаторы, и вызывать их там, где нам будет нужно, используя Spring — аспекты. Ну а сподвигло меня на написание — практически отсутствие подобной информации, особенно на русском.
// будет работать только для методов
@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Validate {}
@Validate
public SomeBankRequest requestToBankRequest(Request<T> request) {
SomeBankRequest bankRequest = ...;
...
// Преобразуем реквест в реквест для конкретного банка
...
return bankRequest;
}
@Aspect
@Component
public class ValidationAspect {
private final Validator validator;
// Автовайрим наш валидатор
public ValidationAspect(Validator validator) {
this.validator = validator;
}
// Перехватываем все точки вхождения нашей аннотации
// @Validate и объект возвращаемый помеченным ей методом
@AfterReturning(pointcut = "@annotation(api.annotations.Validate)", returning = "result")
public void validate(JoinPoint joinPoint, Object result) {
// Вызываем валидацию для объекта
Set<ConstraintViolation<Object>> violations = validator.validate(result);
// Если сэт будет пустым, значит валидация прошла успешно, иначе в сэте будет // вся информация о полях не прошедших валидацию
if (!violations.isEmpty()) {
StringBuilder builder = new StringBuilder();
// берём нужную нам инфу и создаём из неё подходящее сообщение, проходя по // сэту
violations.forEach(violation -> builder
.append(violation.getPropertyPath())
.append("[" + violation.getMessage() + "],"));
throw new IllegalArgumentException("Invalid values for fields: " + builder.toString());
}
}
}
Set<ConstraintViolation<Object>>
— если коротко — сэт классов с различной информацией о валидируемых полях и ошибках. В случае если ошибок не будет, то и сэт будет пустым. Дальше просто проходимся по сэту и создаём сообщение об ошибке, со всеми полями не прошедшими валидацию и выбрасываем экзепшн.
violation.getPropertyPath() - возвращает название поля
violation.getMessage() - конкретное сообщение, почему данное поле не прошло валидацию
К сожалению, не доступен сервер mySQL