전자정부표준프레임워크 폼 검증(Form validation)하기
전자정부표준프레임워크는 폼 검증에 Jakarta Commons Validator를 사용하고, 이것을 Spring Framework와 연동하기 위해서 spring-modules-validation 을 사용합니다.
Jakarta Commons Validator는 각종 검증 규칙을 xml 파일로 작성합니다. 이 규칙에 따른 검증은 Java 클래스 및 Javascript로도 제공되어 서버측 검증과 클라이언트측 검증 모두에서 사용되어 집니다.
이글의 예제는 전자정부표준프레임워크의 eGovFrame Web Project에서 생성된 샘플 소스를 사용하여 테스트 해 봅니다.
1. 예제용 프로젝트 만들기
- 메뉴에서 "File -> New -> egovFrame Web Project"를 선택합니다.
- 다음 정보로 프로젝트를 생성합니다.(나머지는 기본값 사용)
Project name: egov_validation
Target Runtime: Apache Tomcat 8.0
Dynamic Web Module version: 2.5
Group Id: egov_validation
- Generate Example에 체크하여예제 코드를 만듭니다.
2. 전자정부표준프레임워크 샘플에는 기본적으로 Jakarta Commons Validator와 spring-modules-validation 라이브러리가 포함되어 있습니다.
- 전자정부표준프레임워크가 아니라 스프링프레임워크에 적용이 필요하다면 아래 의존성을 pom.xml 파일에 적용하면 추가하면 되겠습니다.
<dependency>
<groupId>commons-validator</groupId>
<artifactId>commons-validator</artifactId>
<version>1.4</version>
</dependency>
<dependency>
<groupId>org.springmodules</groupId>
<artifactId>spring-modules-validation</artifactId>
<version>0.9</version>
</dependency>
3. Validation기능 설정하기
- src/main/resources/egovframework/spring/context-validator.xml 파일에서 검증 기능을 수행하는 빈을 생성합니다.
<bean id="beanValidator" class="org.springmodules.validation.commons.DefaultBeanValidator">
<property name="validatorFactory" ref="validatorFactory"/>
</bean>
<bean id="validatorFactory" class="org.springmodules.validation.commons.DefaultValidatorFactory">
<property name="validationConfigLocations">
<list>
<value>/WEB-INF/config/egovframework/validator/validator-rules.xml</value>
<value>/WEB-INF/config/egovframework/validator/validator.xml</value>
</list>
</property>
</bean>
- DefaultValidatorFactory는 사용되는 모든 규칙을 가지고 있는 validator-rules.xml 파일과 입력폼에서 규칙이 어떻게 적용되는지 정보를 가지고 있는 validator.xml 파일을 받아서 필요한 Commons Validator들의 인스턴스를 생성합니다.
- DefaultBeanValidator는 validatorFactory로 부터 가져온 인스턴스를 이용해서 실제 검증을 수행합니다.
- 폼별 검증 방법을 가지고 있는 validator.xml 파일은 폼별로 여러개로 나눠 만들 수 있습니다. 그럴경우 다음과 같이 지정할 수있습니다.(클래스패스 아래 여러개의 설정파일을 로드하도록 지정한 예입니다.)
<value>classpath:/egovframework/validator/validator-rules.xml</value>
<value>classpath:/egovframework/validator/**/*.xml</value>
4. validator-rules.xml 설정하기
- 이 파일을 사용되어질 수 있는 모든 검증 규칙을 정의해둔 파일 입니다. 아래는 필수값 체크 예입니다.
<validator name="required"
classname="org.springmodules.validation.commons.FieldChecks"
method="validateRequired"
methodParams="java.lang.Object,
org.apache.commons.validator.ValidatorAction,
org.apache.commons.validator.Field,
org.springframework.validation.Errors"
msg="errors.required">
<javascript><![CDATA[
function validateRequired(form) {
...
}
]]>
</javascript>
</validator>
4.1. <validator>의 속성
- name : 검증 규칙을 나타냅니다.
- classname : 검증을 수행하는 클래스명 입니다.
- method : classname으로 지정된 클래스에서 검증을 수행하는 메서드명 입니다.
- methodParams : 검증을 수행하는 메서드의 파라미터 입니다.
- msg : 에러 메세지의 key 값입니다.
- javascript : 클라이언트측 검증을 수행할 자바스크립트 입니다. method 속성에 지정된 것과 같은 함수명입니다.
4.2. 제공되는 검증 규칙
- FieldCheck 클래스 이름은 다음과 같습니다.
* springmodules : org.springmodules.validation.commons.FieldChecks
* egovframework : egovframework.rte.ptl.mvc.validation.RteFieldChecks
이름 | 클래스 | 메서드 | 기능 |
---|---|---|---|
required | springmodules | validateRequired | 필수값 체크 |
requiredif | springmodules | validateRequiredIf | 다른 필드의 값에 따른 체크 |
validwhen | springmodules | validateValidWhen | 다른 필드의 값에 따른 체크 |
minlength | springmodules | validateMinLength | 최소 길이 체크 |
maxlength | springmodules | validateMaxLength | 최대 길이 체크 |
mask | springmodules | validateMask | 정규식으로 체크 |
byte | springmodules | validateByte | -128 <= 값 <= 128 체크 |
short | springmodules | validateShort | -32768 <= 값 <= 32767 체크 |
integer | springmodules | validateInteger | -2147483648 <= 값 <= 2147483647 체크 |
long | springmodules | validateLong | Long값 체크 |
float | springmodules | validateFloat | Float값 체크 |
double | springmodules | validateDouble | Double값 체크 |
date | springmodules | validateDate | 날짜 체크 |
range | springmodules | validateRange | 범위 체크 |
intRange | springmodules | validateIntRange | 정수범위 체크 |
floatRange | springmodules | validateFloatRange | 실수범위 체크 |
creditCard | springmodules | validateCreditCard | 신용카드번호 체크 |
springmodules | validateEmail | 이메일 체크 | |
ihidnum | egovframework | validateIhIdNum | 주민번호 체크 |
korean | egovframework | validateKorean | 한글인지 체크 |
htmltag | egovframework | validateHtmlTag | HTML태그인지 체크 |
- 검증에 사용되는 클래스와 메서드는 org.springmodules.validation.commons.FieldChecks와 이 클래스의 메서드들 입니다. 이 클래스는 내부적으로 Commons Validator 인스턴스를 사용하도록 되어 있습니다.
- Commons Validator는 계속 업데이트 되고 있는데, springmodules-validation은 개발이 중단된것 같습니다. Commons Validator에 추가된 기능을 사용하기 위해서는 springmodules의 FieldChecks 클래스를 상속해서 추가된 기능을 넣을 수 있습니다.
5. validator.xml 설정하기
- validator.xml 파일은 검증할 폼의 필드들과 검증 규칙을 연결하는 정보를 가지는 파일 입니다.
<form name="sampleVO">
<field property="name" depends="required">
<arg0 key="title.sample.name" />
</field>
<field property="description" depends="required">
<arg0 key="title.sample.description" />
</field>
<field property="regUser" depends="required">
<arg0 key="title.sample.regUser" />
</field>
</form>
5.1. <form> 태그의 name 속성
- validator.xml 파일 안의 form name 은 HTML의 <form> 태그의 name 속성과 관계가 없습니다.
- 서버측 검증은 컨트롤러에서 submit된 값이 맵핑되는 클래스 이름이 됩니다.(SampleVO sampleVO)
- 클라이언트측 검증은 jsp 파일의 <valdator:javascript> 태그의 formName 속성의 값이 됩니다.
// 컨트롤러 java 파일
@RequestMapping(value = "/addSample.do", method = RequestMethod.POST)
public String addSample(SampleVO sampleVO, BindingResult bindingResult, Model model) throws Exception {
// Server-Side Validation
beanValidator.validate(sampleVO, bindingResult);
// jsp 파일
<validator:javascript formName="sampleVO" staticJavascript="false" xhtml="true" cdata="false"/>
- 그러므로 validator.xml의 <form name="sampleVO"> 속성, 컨트롤러의 메소드 public String addSample(SampleVO sampleVO), jsp의 <validator:javascript formName="sampleVO"/> 태그의 formName 속성
세 가지를 일치시켜야 서버측과 클라이언트측의 검증이 모두 동작합니다.
5.2. <field> 태그
- peroperty 속성은 폼 데이터가 맵핑되는 클래스의 이름과, HTML 폼의 엘리먼트의 이름과 같습니다.
- depnds 속성은 validator-rules.xml에 정의되어 있는 적용될 규칙을 지정합니다. 여러개의 규칙을 지정하기 위해서는 콤마(,)로 분리하여 여러개를 지정할 수 있습니다.
5.3 <arg0> 태그
- 필드에 대한 라벨에 해당하는 내용을 지정합니다. 메세지 프로퍼티의 키를 지정할 수 도 있고 직접 값을 입력할 수도 있습니다.
- 예로 <arg0 key="이름"/>로 지정되어 있는 필수 입력필드에 입력하지 않으면, "이름은 필수 입력입니다." 라는 에러메세지가 보여지게 됩니다.
6. 서버측 검증
6.1. 컨트롤러 (Controller)에서 검증하기
- 서버측 검증은 주로 컨트롤러에서 하게 됩니다.
@Controller
public class EgovSampleController {
/** Validator */
@Resource(name = "beanValidator")
protected DefaultBeanValidator beanValidator;
@RequestMapping(value = "/addSample.do", method = RequestMethod.POST)
public String addSample(SampleVO sampleVO, BindingResult bindingResult, Model model) throws Exception {
// 서버측 검증
beanValidator.validate(sampleVO, bindingResult);
if(bindingResult.hasErrors()) {
model.addAttribute("sampleVO", sampleVO);
return "sample/egovSampleRegister";
}
...
}
- context-validator.xml에서 정의한 beanValidator가 검증에 사용되어집니다.
beanValidator.validate(samplevO, bindingResult);
- 검증 결과는 bindingResult 객체에 저장되어 집니다. 에러가 있으면 다시 입력페이지를 보여줍니다.
if(bindingResult.hasErrors()) {}
6.2. 검증 결과를 JSP페이지에 보여주기
<tr>
<td class="tbtd_caption"><label for="name"><spring:message code="title.sample.name" /></label></td>
<td class="tbtd_content">
<form:input path="name" maxlength="30" cssClass="txt"/>
<form:errors path="name" /></td>
</tr>
- 스프링 <form:errors> 태그로 오류 메세지가 보여지게 됩니다.
6.3. 에러메세지 등록하기
- 에러 메세지는 src/main/resources/egovframework/message/message-common.properties 파일에 등록합니다. 다국어 메세지 프로퍼티 파일 입니다.
errors.required={0} 은 필수 입력값입니다.
errors.minlength={0} 은 {1}자 이상 입력해야 합니다.
errors.maxlength={0} 은 {1}자 이상 입력할수 없습니다.
6.4. 실행결과 입니다.
7. 클라이언트측 검증
7.1. validator.jsp 추가하기
- src/main/webapp/WEB-INF/jsp/egovframework/example/cmmn/validator.jsp 파일 입니다.
<%@ page language="java" contentType="javascript/x-javascript" %>
<%@ taglib prefix="validator" uri="http://www.springmodules.org/tags/commons-validator" %>
<validator:javascript dynamicJavascript="false" staticJavascript="true"/>
- 이 파일은 validator-rules.xml 파일에 있는 모든 자바스크립트 함수를 모아둔 .js 파일과 같이 동작합니다.
- /validator.do 파일로 맵핑하는 컨트롤러를 만들어도 되고, 이 예제에서는 <mvc:view-controller>를 이용한 정적 맵핑을 사용합니다.
- src/main/webapp/WEB-INF/config/egovframework/springmvc/dispatcher-servlet.xml 파일입니다.
<mvc:view-controller path="/cmmn/validator.do" view-name="cmmn/validator"/>
7.2. jsp 파일에 taglib 및 javascrip 추가하기
- src/main/webapp/WEB-INF/jsp/egovframework/example/sample/egovSampleRegister.jsp 파일입니다.
<!-- valitor-rules.xml 파일의 모든 자바스크립트 라이브러리 입니다. -->
<script type="text/javascript" src="<c:url value='/cmmn/validator.do'/>"></script>
<!-- validator.xml의 sampleVO 를 검증하기위한 스크립트를 생성합니다. -->
<validator:javascript formName="sampleVO" staticJavascript="false" xhtml="true" cdata="false"/>
<script type="text/javaScript" language="javascript" defer="defer">
<!--
/* 글 등록 function - 등록시 검증을 수행합니다. */
function fn_egov_save() {
frm = document.detailForm;
if(!validateSampleVO(frm)) {
return;
} else {
frm.action = "<c:url value="${registerFlag == 'create' ? '/addSample.do' : '/updateSample.do'}"/>";
frm.submit();
}
}
-->
</script>
7.3 클라이언트측 검증 실행 결과 입니다.