Создание настраиваемого средства проверки Hibernate на основе существующего средства проверки Hibernate
Я хотел бы создать собственное ограничение @LengthIfNotBlank(min=,max=)
на основе гибернат @Length
ограничение:
@Target( { METHOD, FIELD, ANNOTATION_TYPE })
@Retention(RUNTIME)
@Constraint(validatedBy = LengthIfNotBlankValidator.class)
@Documented
public @interface LengthIfNotBlank {
String message() default "{com.mycompany.LengthIfNotBlank}";
Class<? extends Payload>[] payload() default {};
int max() default Integer.MAX_VALUE;
Class<?>[] groups() default {};
int min() default 0;
}
public class LengthIfNotBlankValidator extends LengthValidator {
@Override
public boolean isValid(String value, ConstraintValidatorContext constraintValidatorContext) {
return StringUtils.isBlank(value) || super.isValid(value, constraintValidatorContext);
}
}
Исключение:
[java] javax.validation.ValidationException: Unable to initialize com.example.constraints.LengthIfNotBlankValidator
[java] Caused by: java.lang.ClassCastException: $Proxy32 cannot be cast to org.hibernate.validator.constraints.Length
ОБНОВИТЬ
Все еще получаю это исключение:-(
public class LengthIfNotBlankValidator extends LengthValidator {
public void initialize(LengthIfNotBlank parameters) {
Map<String,Object> elements = new HashMap<String,Object>();
elements.put("message", parameters.message());
elements.put("payload", parameters.payload());
elements.put("groups", parameters.groups());
elements.put("min", parameters.min());
elements.put("max", parameters.max());
AnnotationDescriptor<Length> descriptor = AnnotationDescriptor.getInstance(Length.class, elements);
Length length = AnnotationFactory.create(descriptor);
super.initialize(length);
}
@Override
public boolean isValid(String value, ConstraintValidatorContext constraintValidatorContext) {
return StringUtils.isBlank(value) || super.isValid(value, constraintValidatorContext);
}
}
2 ответа
Так как типы аннотаций не могут быть унаследованы, вы должны переопределить initialize()
и передать фактический экземпляр типа Length
в super.initialize()
,
Обратите внимание, что вы не можете создать экземпляр типа аннотации напрямую, поэтому вы должны использовать что-то вроде AnnotationFactory
,
Также я не уверен, что компилятор позволит вам сделать это из-за ограничений безопасности типов. Если это не так, используйте композицию вместо наследования (т.е. создайте поле типа LengthValidator
).
Я думаю, вы не должны беспокоиться об этом: Hibernate's LengthValidator
это действительно просто, и что бы вы ни делали, у вас получится больше кода, который, если вы только что заново реализовали код, вот так:
public class LengthIfNotBlankValidator implements ConstraintValidator<LengthIfNotBlank, String> {
private int min;
private int max;
public void initialize(LengthIfNotBlank parameters) {
min = parameters.min();
max = parameters.max();
validateParameters();
}
public boolean isValid(String value, ConstraintValidatorContext constraintValidatorContext) {
if ( StringUtils.isBlank(value) ) {
return true;
}
int length = value.length();
return length >= min && length <= max;
}
private void validateParameters() {
if ( min < 0 ) {
throw new IllegalArgumentException( "The min parameter cannot be negative." );
}
if ( max < 0 ) {
throw new IllegalArgumentException( "The max parameter cannot be negative." );
}
if ( max < min ) {
throw new IllegalArgumentException( "The max cannot be less than the min." );
}
}
}
(Это просто код Hibernate, адаптированный для вашего конкретного случая).
Кроме того, я не думаю, что было бы возможно продлить LengthValidator
при осуществлении ConstraintValidator<LengthIfNotBlank, String>
, что является обязательным для вашего валидатора.
Однако обратите внимание, что если вы хотите повторно использовать непараметризированные валидаторы (или валидаторы с постоянными параметрами), вы можете просто аннотировать свой валидатор повторно используемым (и, возможно, добавить @ReportAsSingleViolation
)