@Valid 어노테이션을 사용하면 사용자의 요청을 검증할 수 있다.
간단하게 검증에 대해서 알아보고 좀 더 복잡한 검증을 처리해보자!
implementation 'org.springframework.boot:spring-boot-starter-validation'
@Size
@NotNull
@NotEmpty
@NotBlank
@Past
@Future
@FutureOrPresent
@Max
@Min
@Pattern
@Valid
@Email
public class PortfolioInputItemDTO {
@NotNull(message = "Stock name must not be null")
private String stockName;
@Min(value = 0, message = "Weight must be at least 0")
@Max(value = 1, message = "Weight must be at most 1")
@NotNull(message = "weight must not be null")
private float weight;
}
다음의 과정에서 Handler Mapping 단계에서 @Valid 또는 @Validated가 붙은 파라미터가 있는 경우 유효성 검사
public class PortfolioInputItemDTO {
private List<Stock> stocks;
//Stock(String name, float weight)
}
- Annotation을 정의한다.
- 검증식을 만든다.
- 사용하기
public class PortfolionputDTO {
private LocalDate startDate;
private LocalDate endDate;
private List<PortfolioInputItemDTO> portfolioInputItemDTOList;
}
import jakarta.validation.Constraint;
import jakarta.validation.Payload;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
@Constraint(validatedBy = PortfolioInputValidator.class)
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
public @interface ValidPortfolioInput {
String message() default "Valid Error";
Class<?>[] groups() default {};
Class<? extends Payload>[] payload() default {};
}
import com.chan.stock_portfolio_backtest_api.dto.PortfolioInputItemDTO;
import com.chan.stock_portfolio_backtest_api.dto.PortfolionputDTO;
import jakarta.validation.ConstraintValidator;
import jakarta.validation.ConstraintValidatorContext;
import java.time.LocalDate;
import java.util.List;
public class PortfolioInputValidator implements ConstraintValidator<ValidPortfolioInput, PortfolionputDTO> {
@Override
public boolean isValid(PortfolionputDTO portfolionputDTO, ConstraintValidatorContext constraintValidatorContext) {
constraintValidatorContext.disableDefaultConstraintViolation();
if (portfolionputDTO == null) {
return true;
}
//startDate < endDate valid
LocalDate startDate = portfolionputDTO.getStartDate();
LocalDate endDate = portfolionputDTO.getEndDate();
if (startDate != null && endDate != null) {
if (!startDate.isBefore(endDate)) {
constraintValidatorContext
.buildConstraintViolationWithTemplate("endDate must be after the startDate")
.addPropertyNode("endDate")
.addConstraintViolation();
return false;
}
}
//weight sum == 1 valid
List<PortfolioInputItemDTO> portfolioInputItemDTOList = portfolionputDTO.getPortfolioInputItemDTOList();
float weightSum = 0;
for (PortfolioInputItemDTO i : portfolioInputItemDTOList) {
weightSum += i.getWeight();
}
if (weightSum != 1) {
constraintValidatorContext
.buildConstraintViolationWithTemplate("weight sum is might be 1")
.addPropertyNode("portfolioInputItemDTOList")
.addConstraintViolation();
return false;
}
return true;
}
}
if (!startDate.isBefore(endDate)) {
constraintValidatorContext
.buildConstraintViolationWithTemplate("endDate must be after the startDate")
.addPropertyNode("endDate")
.addConstraintViolation();
return false;
}
@ValidPortfolioInput
public class PortfolionputDTO {
@DateTimeFormat(pattern = "yyyy-MM-dd")
@NotNull(message = "Start date must not be null")
private LocalDate startDate;
@DateTimeFormat(pattern = "yyyy-MM-dd")
@NotNull(message = "End date must not be null")
private LocalDate endDate;
@Valid
private List<PortfolioInputItemDTO> portfolioInputItemDTOList;
}