На основе аннотации Spring bean валидация
Я изучаю подход, основанный на аннотации, для проверки Spring beans с помощью spring modules. В этот учебник в качестве примера используется bean (getters and seters опущены):
public final class User {
@NotBlank
@Length(max = 80)
private String name;
@NotBlank
@Email
@Length(max = 80)
private String email;
@NotBlank
@Length(max = 4000)
private String text;
}
Сообщение об ошибке, которое используется, если определенное правило проверки не оспаривается, должно следовать этому формату:
bean-class.bean-propery[validation-rule]=Validation Error message
Примеры для класса, показанного выше, включают в себя:
User.email[not.blank]=Please enter your e-mail address.
User.email[email]=Please enter a valid e-mail address.
User.email[length]=Please enter no more than {2} characters.
Тот факт, что ключи сообщения содержат имя класса, представляет собой пару проблем:
- Если класс переименован, необходимо также изменить клавиши сообщения
-
Если у меня есть другой класс (например, Person) с свойством электронной почты, которое проверено идентично с User.email, мне нужно дублировать сообщения, например.
Person.email [not.blank] = Пожалуйста, введите свой адрес электронной почты.
Person.email [email] = Пожалуйста, введите действительный адрес электронной почты.
Person.email [length] = Введите не более {2} символов.
Фактически, в документации утверждается, что можно настроить сообщение по умолчанию для определенного правила (например, @Email) следующим образом:
email=email address is invalid
Это сообщение по умолчанию должно использоваться, если не удается найти сообщение bean для конкретного правила. Однако, мой опыт заключается в том, что это просто не работает.
Альтернативным механизмом предотвращения дублирования сообщений является передача ключа сообщения об ошибке в аннотацию правила. Например, предположим, что я определил следующее сообщение об ошибке по умолчанию для правила @Email
badEmail=Email address is invalid
Это сообщение должно использоваться, если я аннотирую соответствующее свойство следующим образом:
@Email(errorCode="badEmail")
private String email;
Однако я попробовал это, и снова, он просто не работает. Кто-нибудь нашел способ избежать дублирования сообщений об ошибках при использовании этой рамки проверки?
Ответы
Ответ 1
Я быстро просмотрел API BeanValidator, и похоже, что вы можете попробовать свойство errorCodeConverter.
Вам нужно будет реализовать собственный ErrorCodeConverter или использовать одну из предоставленных реализаций?
....
<bean id="validator" class="org.springmodules.validation.bean.BeanValidator"
p:configurationLoader-ref="configurationLoader"
p:errorCodeConverter-ref="errorCodeConverter" />
<bean id="errorCodeConverter" class="contact.MyErrorCodeConverter" />
....
Примечание: ConfigurationLoader - это еще один bean, определенный в XML файле конфигурации, используемом в учебнике
Преобразователь примеров:
package contact;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springmodules.validation.bean.converter.ErrorCodeConverter;
public class MyErrorCodeConverter implements ErrorCodeConverter {
private Log log = LogFactory.getLog(MyErrorCodeConverter.class);
@Override
public String convertPropertyErrorCode(String errorCode, Class clazz, String property) {
log.error(String.format("Property %s %s %s", errorCode, clazz.getClass().getName(), property));
return errorCode; // <------ use the errorCode only
}
@Override
public String convertGlobalErrorCode(String errorCode, Class clazz) {
log.error(String.format("Global %s %s", errorCode, clazz.getClass().getName()));
return errorCode;
}
}
Теперь свойства должны работать:
MyEmailErrorCode=Bad email
class Foo {
@Email(errorCode="MyEmailErrorCode")
String email
}
Ответ 2
Spring У проверки есть ErrorCodeConverter, который делает это:
org.springmodules.validation.bean.converter.KeepAsIsErrorCodeConverter
Когда это используется, пакет ресурсов будет проверяться на следующие коды:
[errorCode.commandBeanName.fieldName, errorCode.fieldName, errorCode.fieldClassName, errorCode]
- errorCode - это фактический код ошибки проверки, например. not.blank, электронная почта.
- commandBeanName совпадает с именем ключа модели, которое ссылается на
форма поддержки bean.
- fieldName - это имя поля.
- fieldClassName - это имя класса поля, например. java.lang.String, java.lang.Integer
Так, например, если у меня есть bean, на который ссылаются в модели ключом "formBean", а поле emailAddress типа java.lang.String не содержит адрес электронной почты, что вызывает сообщение errorCode. Рамка проверки будет пытаться разрешить следующие коды сообщений:
[email.formBean.emailAddress, email.emailAddress, email.java.lang.String, email]
Если код errorCode заменяется кодом errorCode "badEmail" следующим образом:
@email (ERRORCODE = "badEmail" )
Коды сообщений, которые будет проверять инфраструктура, будут следующими:
[badEmail.formBean.emailAddress, badEmail.emailAddress, badEmail.java.lang.String, badEmail]
Я хотел бы предложить сохранить errodCode одинаково. Таким образом, одно сообщение может использоваться для всех полей, которые связаны с этим кодом ошибок. Если вам нужно быть более конкретным с сообщением для определенного поля, вы можете добавить сообщение в пакеты ресурсов с кодом errorCode.commandBeanName.field.
Ответ 3
Добавьте следующий beans в ваш applicationContext.xml
файл.
<bean id="configurationLoader"
class="org.springmodules.validation.bean.conf.loader.annotation.AnnotationBeanValidationConfigurationLoader" />
<!-- Use the error codes as is. Don't convert them to <Bean class name>.<bean field being validated>[errorCode]. -->
<bean id="errorCodeConverter"
class="org.springmodules.validation.bean.converter.KeepAsIsErrorCodeConverter"/>
<!-- shortCircuitFieldValidation = true ==> If the first rule fails on a field, no need to check
other rules for that field -->
<bean id="validator" class="org.springmodules.validation.bean.BeanValidator"
p:configurationLoader-ref="configurationLoader"
p:shortCircuitFieldValidation="true"
p:errorCodeConverter-ref="errorCodeConverter"/>