spring - validation (유효성 검사) 하기
Validation을 해보자
웹 개발을 하다 보면 서버에 들어온 요청이 서버에서 요구하는 스펙에 잘 맞게 들어왔는지 검사해야 할 필요가 있습니다.
예를 들어 회원가입을 할 때 이름은 필수로 들어와야 한다던지, 나이는 0보다 커야 한다던지 같은 것 입니다.
물론 이런 식으로 들어온 요청에 대해 검사를 할 수도 있을 것입니다.
//이름이 비어있으면 exception을 던진다.
if (request.getName() == null) {
throw Exception;
}
// 나이가 0보다 작으면 exception을 던진다.
if (request.getAge() < 0) {
throw Exception;
}
하지만 점점 커지는 웹 어플리케이션 에서 위처럼 요청에 대한 검사를 하다 보면 필드가 늘어남에 따라 코드의 대부분이 유효성 검사 코드로 뒤덮일 것입니다.
스프링에서는 이런 상황을 방지하기 위해서 손쉽게 Validation을 하기 위 전략 중 하나로 Hibernate Bean Validator를 사용할 수 있습니다.
Hibernate Bean Validator는 유효성 검사를 해야 하는 필드에 @NotEmpty, @NotNull, @Max와 같은 어노테이션을 붙여줌으로써 유효성 검사를 진행할 수 있습니다.
Bean Validator 사용 하기
먼저 Hibernate bean validator에 대한 의존성 설정을 해야 합니다.
pom.xml
<dependency>
<groupId>org.hibernate</groupId>
<artifactId>hibernate-validator</artifactId>
<version>6.1.5.Final</version>
</dependency>
스프링 부트 + gradle를 사용할 경우는 다음과 같이 해줍니다.
dependencies {
implementation 'org.springframework.boot:spring-boot-starter-validation'
// 기타 의존성
}
예시를 위해 간단한 회원가입 시 사용할 Dto 클래스를 만들어 보았습니다.
public class MemberDto {
private String realName;
private String password;
private String nickName;
private int age;
}
회원 가입을 위해서는 [이름, 비밀번호, 닉네임, 나이]가 필요합니다.
@RestController
public class MemberController {
@PostMapping("/member")
public String createMember(@RequestBody MemberDto memberDto) {
System.out.println("memberDto.getRealName() = " + memberDto.getRealName());
System.out.println("memberDto.getPassword() = " + memberDto.getPassword());
System.out.println("memberDto.getNickName() = " + memberDto.getNickName());
System.out.println("memberDto.getAge() = " + memberDto.getAge());
// 비지니스 로직이 들어가는 자리.
return "성공!";
}
}
간단하게 "/member"에 MemberDto를 RequestBody로 받아보았습니다.
아래 보이는 것처럼 Postman을 통해 요청을 보내 보니 요청이 제대로 도달한 것을 알 수 있습니다.
하지만 여기서 RealName이나 Password를 보내지 않으면 어떻게 될까요?
Password를 빼고 보내도 Password가 null로 찍힐 뿐 요청을 처리하는데 아무 문제가 없었습니다.
하지만 우리는 요청으로 받는 Request의 Body에 RealName과 Password가 Null이면 요청을 거부하고 예외 처리를 해주고 싶습니다.
그럴 때는 아래와 같이 RequestBody를 받는 부분 앞에 @Valid 어노테이션을 붙여줍니다.
그리고 Dto 클래스에 가서 Non-Nullable 하게 만들고 싶은 필드에 @NotNull 어노테이션을 달아줍니다.
public class MemberDto {
@NotNull
private String realName;
@NotNull
private String password;
private String nickName;
private int age;
}
이 상태에서 다시 Password를 빼고 요청을 보내보면!?
클라이언트에서 요청을 잘못했다는 400 status code와 함께 에러 리스폰스를 응답해 준 것을 볼 수 있습니다.
서버 쪽에는 아래와 같이 MethodArgumentNotValidException이 발생하게 됩니다.
Resolved [org.springframework.web.bind.MethodArgumentNotValidException:
Validation failed for argument [0] in public java.lang.String
garden.practice.validation.MemberController.createMember
(garden.practice.validation.MemberDto):
[Field error in object 'memberDto' on field 'password': rejected value [null];
codes [NotNull.memberDto.password,NotNull.password,NotNull.java.lang.String,NotNull];
arguments [org.springframework.context.support.DefaultMessageSourceResolvable: codes [memberDto.password,password];
arguments []; default message [password]]; default message [널이어서는 안됩니다]] ]
우리는 이런 에러를 exception handler를 통해 응답에 메세지를 담아 넘겨주는 등의 행위를 할 수 있을 것입니다.
Bean Validator 에는 @NotNull 어노테이션뿐 아니라 @NotEmpty, @Max, @Min, @Length와 같은 여러 가지 어노테이션이 존재합니다.
어떤 어노테이션이 존재하는지는 공식 문서 (docs.jboss.org/hibernate/stable/validator/reference/en-US/html_single/#section-builtin-constraints) 에 가면 자세히 나와 있습니다.
또한 javax.validation.constraints 패키지에 가면 어노테이션의 구성을 볼 수 도 있으니 궁금하신 분들은 찾아보시길 바랍니다.
이렇게 Bean Validation의 사용법에 대해 알아보았습니다.