728x90
반응형

Spring boot 에서 Validation은 보통 Entity와 Request body의 정보가 제대로 들어왔는지 확인하는 경우에 사용한다.

예를 들면, 아래와 같이 Request body의 Variable 의 값이 Null이거나 Blank 일 때를 확인해서 이러한 경우에는 Client에게 Error Reponse 를 보내거나, log 상에 에러를 발생시키게 된다.

@Getter
@Setter
@Accessors(chain = true)
@AllArgsConstructor
@NoArgsConstructor
@ToString
@Schema(description = "Session Creation Request")
public static class SessionCreateRequest {
    @Schema(example = "1L")
    @NotNull(message = "{account.id.empty}")
    private Long accountId;
    @Schema(example = "{\"orderCount\":1}")
    @NotBlank(message = "{session.value.empty}")
    private String values;
}

 

이러한 Validation 작업은 Server의 완성도를 높여줄 수 있고 Client에게 Request 상의 문제를 알려줄 수 있기 때문에 굉장히 유용한 기능이라고 할 수 있다. 셋팅은 완료하고 나면 간단하게 Annotation을 붙이면 체크를 해주니 사용성 또한 아주 간단하다. 

나중에 포스팅을 하겠지만, Custom annotation을 만들어서 체크도 할 수 있다. 

 

그럼 설정방법에 대해서 알아보자

 

Dependency 설정

Spring boot 에서 Validation Dependency를 제공한다. 아래를 pom.xml에 추가하도록 하자.

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-validation</artifactId>
</dependency>

 

Configuration setting

Validation Configuration

아주 기본적인 Configuration이다. Configuration이 없어도 Default로 셋팅이 되지만 Validation Message Source를 넣고자 Configuration file을 만들었다. Validation Message Source는 Validation error가 발생되었을 때 그 정보를 정의해둔 메세지 형태로 Client로 보내고자 할 때 사용한다. 아래에 설정 방법을 추가하도록 하겠다. 

@Configuration
public class ValidationConfig {
    private final MessageSource validationMessageSource;

    @Bean
    public LocalValidatorFactoryBean getValidator() {
        LocalValidatorFactoryBean bean = new LocalValidatorFactoryBean();
        bean.setValidationMessageSource(validationMessageSource);

        return bean;
    }
}

Message Accessor Configuration

Validation Message configuration은 ReloadableResourceBundleMessageSource를 생성하여 몇가지 옵션을 제공한다. 

@Configuration
public class MessageConfig {
    private static final String VALIDATION_MESSAGE_SOURCE_PATH = "classpath:/messages/validation_message";

    @Bean
    public MessageSource validationMessageSource() {
        ReloadableResourceBundleMessageSource messageSource = new ReloadableResourceBundleMessageSource();
        messageSource.setBasename(VALIDATION_MESSAGE_SOURCE_PATH);
        messageSource.setDefaultEncoding("UTF-8");

        return messageSource;
    }

    @Bean("validationMessageSourceAccessor")
    public MessageSourceAccessor validationMessageSourceAccessor() {
        return new MessageSourceAccessor(validationMessageSource());
    }
}

Basename은 Message 목록의 Path를 지정한다. 아래와 같이 Validation에는 message 옵션이 붙는데 Validation이 실패할 경우 보여줄 메세지를 지정한다. `{}`안에 있는 것이 key이며 파일에서 아래의 key를 찾아 메세지를 매핑하게 된다. 

@NotBlank(message = "{session.value.empty}")

Message source 를 생성한 후 그것에 접근할 수 있는 Accessor를 Bean에 등록한다. 

이렇게 설정할 경우에는 메세지를 더 효율적으로 관리할 수 있다. 

 

Message properties 파일

위의 설정의 Basename으로 들어간 파일의 경로에 message.properties 파일이 있어야 한다. 아래와 같이 key값에 대응되는 값을 가지고 있어야 한다. 한가지 주의할 점은 Unit test를 만들 경우에는 아래의 파일이 test/resources에도 있어야 한다. 

# JwtToken
account.id.empty=Account id is empty.
token.value.empty=Jwt token is empty.

# Session
session.value.empty=Session value is empty.

 

@Valid annotation

Vadilation check를 위한 Annotaiton을 붙였다라고 해서 Validation이 되지 않는다. 원하는 Request Body에 `@Valid` annotation을 붙여줘야 한다. 아래 Contoller source를 보면 `@RequestBody` 앞에 `@Valid` annotation이 붙은 것을 볼 수 있다. 

@PostMapping("/session")
public SessionDTO.SessionInfo createSession(@Valid @RequestBody SessionDTO.SessionCreateRequest request) {
    Session session = modelMapper.map(request, Session.class);
    Session createdSession = sessionService.create(session);

    return SessionDTO.SessionInfo.from(createdSession, modelMapper);
}

 

Validation Result

Validation 설정을 완료하고 테스트를 해보면 Validation error가 발생하였을 경우에는 아래와 같은 메세지가 나오게 된다.

{
    "timestamp": "2022-01-19T19:53:48.832+00:00",
    "status": 400,
    "error": "Bad Request",
    "trace": "org.springframework.web.bind.MethodArgumentNotValidException: Validation failed for argument [0] in public io.coolexplorer.session.dto.JwtTokenDTO$JwtTokenInfo io.coolexplorer.session.controller.JwtTokenController.createToken(io.coolexplorer.session.dto.JwtTokenDTO$JwtTokenCreateRequest): [Field error in object 'jwtTokenCreateRequest' on field 'jwtToken': rejected value []; codes [NotBlank.jwtTokenCreateRequest.jwtToken,NotBlank.jwtToken,NotBlank.java.lang.String,NotBlank]; arguments [org.springframework.context.support.DefaultMessageSourceResolvable: codes [jwtTokenCreateRequest.jwtToken,jwtToken]; arguments []; default message [jwtToken]]; default message [Jwt token is empty.]] \n\tat org.springframework.web.servlet.mvc.method.annotation.RequestResponseBodyMethodProcessor.resolveArgument(RequestResponseBodyMethodProcessor.java:141)\n\tat org.springframework.web.method.support.HandlerMethodArgumentResolverComposite.resolveArgument(HandlerMethodArgumentResolverComposite.java:121)\n\tat org.springframework.web.method.support.InvocableHandlerMethod.getMethodArgumentValues(InvocableHandlerMethod.java:179)\n\tat org.springframework.web.method.support.InvocableHandlerMethod.invokeForRequest(InvocableHandlerMethod.java:146)\n\tat org.springframework.web.servlet.mvc.method.annotation.ServletInvocableHandlerMethod.invokeAndHandle(ServletInvocableHandlerMethod.java:117)\n\tat org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.invokeHandlerMethod(RequestMappingHandlerAdapter.java:895)\n\tat org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.handleInternal(RequestMappingHandlerAdapter.java:808)\n\tat org.springframework.web.servlet.mvc.method.AbstractHandlerMethodAdapter.handle(AbstractHandlerMethodAdapter.java:87)\n\tat org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:1067)\n\tat org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:963)\n\tat org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:1006)\n\tat org.springframework.web.servlet.FrameworkServlet.doPost(FrameworkServlet.java:909)\n\tat javax.servlet.http.HttpServlet.service(HttpServlet.java:681)\n\tat org.springframework.web.servlet.FrameworkServlet.service(FrameworkServlet.java:883)\n\tat javax.servlet.http.HttpServlet.service(HttpServlet.java:764)\n\tat org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:227)\n\tat org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:162)\n\tat org.apache.tomcat.websocket.server.WsFilter.doFilter(WsFilter.java:53)\n\tat org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:189)\n\tat org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:162)\n\tat org.springframework.web.filter.CharacterEncodingFilter.doFilterInternal(CharacterEncodingFilter.java:201)\n\tat org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:117)\n\tat org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:189)\n\tat org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:162)\n\tat org.springframework.web.filter.RequestContextFilter.doFilterInternal(RequestContextFilter.java:100)\n\tat org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:117)\n\tat org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:189)\n\tat org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:162)\n\tat org.springframework.web.filter.FormContentFilter.doFilterInternal(FormContentFilter.java:93)\n\tat org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:117)\n\tat org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:189)\n\tat org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:162)\n\tat org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:197)\n\tat org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:97)\n\tat org.apache.catalina.authenticator.AuthenticatorBase.invoke(AuthenticatorBase.java:540)\n\tat org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:135)\n\tat org.apache.catalina.valves.ErrorReportValve.invoke(ErrorReportValve.java:92)\n\tat org.apache.catalina.core.StandardEngineValve.invoke(StandardEngineValve.java:78)\n\tat org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:357)\n\tat org.apache.coyote.http11.Http11Processor.service(Http11Processor.java:382)\n\tat org.apache.coyote.AbstractProcessorLight.process(AbstractProcessorLight.java:65)\n\tat org.apache.coyote.AbstractProtocol$ConnectionHandler.process(AbstractProtocol.java:895)\n\tat org.apache.tomcat.util.net.NioEndpoint$SocketProcessor.doRun(NioEndpoint.java:1732)\n\tat org.apache.tomcat.util.net.SocketProcessorBase.run(SocketProcessorBase.java:49)\n\tat org.apache.tomcat.util.threads.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1191)\n\tat org.apache.tomcat.util.threads.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:659)\n\tat org.apache.tomcat.util.threads.TaskThread$WrappingRunnable.run(TaskThread.java:61)\n\tat java.base/java.lang.Thread.run(Thread.java:829)\n",
    "message": "Validation failed for object='jwtTokenCreateRequest'. Error count: 1",
    "errors": [
        {
            "codes": [
                "NotBlank.jwtTokenCreateRequest.jwtToken",
                "NotBlank.jwtToken",
                "NotBlank.java.lang.String",
                "NotBlank"
            ],
            "arguments": [
                {
                    "codes": [
                        "jwtTokenCreateRequest.jwtToken",
                        "jwtToken"
                    ],
                    "arguments": null,
                    "defaultMessage": "jwtToken",
                    "code": "jwtToken"
                }
            ],
            "defaultMessage": "Jwt token is empty.",
            "objectName": "jwtTokenCreateRequest",
            "field": "jwtToken",
            "rejectedValue": "",
            "bindingFailure": false,
            "code": "NotBlank"
        }
    ],
    "path": "/api/v1/token"
}

 

728x90
반응형
728x90
반응형

Maven Jgitflow plugin 은 Maven command를 통해서 Git flow를 수행하는 Plugin이다. Git flow의 간단한 설명은 아래 문서를 참고하자. 

2022.01.18 - [CI&CD] - Git-flow 란?

 

Git-flow 란?

Giflow is an alternative Git branching model that involves the use of feature branches and multiple primary branches. Git flow는 Git branching model로서 Git으로 관리되는 Branch 들의 관리방법을 정의..

corono.tistory.com

 

Maven Jgit flow는 아래와 같은 Feature를 가지고 있다. 

Features

  • starting a release - creates a release branch and updates pom(s) with release versions 
    • Release branch를 생성하고 Pom 파일에 Release version을 업데이트 한다.
  • finishing a release - runs a maven build (deploy or install), merges the release branch, updates pom(s) with development versions
    • Maven build를 수행하고, Release branch를 mater로 머지하고 develop branch에 Snapshot version을 업데이트 한다.
  • starting a hotfix - creates a hotfix branch and updates pom(s) with hotfix versions
    • Hotfix branch를 생성하고 Pom 파일에 Hotfix version을 업데이트 한다.
  • finishing a hotfix - runs a maven build (deploy or install), merges the hotfix branch, updates pom(s) with previous versions
    • Maven build를 수행하고, Hotfix branch를 mater로 머지하고 버전을 업데이트 한다.
  • starting a feature - creates a feature branch
    • Feature branch를 생성한다.
  • finishing a feature - merges the feature branch
    • Feature branch를 머지한다.

 

이 문서에서는 많이 사용하는 `Starting a release` 와 `Finishing a release` 에 대해서 알아보도록 하겠다. 

 

Plugin 추가

우선, Plugin을 설치해보자. Pom 파일에 추가한다.

<properties>
	<!-- Jgit Configuration -->
    <git.user> </git.user>
    <git.password> </git.password>
</properties>

<plugin>
    <groupId>external.atlassian.jgitflow</groupId>
    <artifactId>jgitflow-maven-plugin</artifactId>
    <version>1.0-m5.1</version>
    <configuration>
        <!-- For specific plugin configuration options,
        see the Goals documentation page -->
        <username>${git.user}</username>
        <password>${git.password}</password>
        <flowInitContext>
            <masterBranchName>main</masterBranchName>
            <developBranchName>develop</developBranchName>
        </flowInitContext>
        <autoVersionSubmodules>true</autoVersionSubmodules>
        <enableSshAgent>true</enableSshAgent>
        <noDeploy>true</noDeploy>
        <pushReleases>true</pushReleases>
        <pullDevelop>true</pullDevelop>
        <pullMaster>true</pullMaster>
        <scmCommentPrefix>[RELEASE] </scmCommentPrefix>
    </configuration>
</plugin>

Plugin 이외에 몇가지 Configuration을 추가하였다. 

  • username / password - Git repository의 접속 Auth 정보를 입력한다. 나의 경우에는 Command line parameter로 계정 정보를 넘겨 줄 것이기 때문에 Properties에 실제 Auth 정보를 넣지 않았다. 
  • flowInitContext - master branch와 develop branch의 이름을 넣어준다. Github이 최근 master branch 이름이 main으로 변경되어서 main으로 바꿔 주었다.
  • autoVersionSubmodules - 자동으로 버전의 입력시켜준다. False로 설정할 경우 사용자가 버전을 명시해 주어야 한다. 
  • enableSshAgent - SSH agent를 Enable 한다.
  • noDeploy - Deploy는 하지 않는다. 
  • pushReleases - Release branch를 remote stream에 push 한다. 
  • pullDevelop - Jgitflow가 초기화 되었을 때 develop branch를 pull 한다.
  • pullMaster - Jgitflow가 초기화 되었을 때 master branch를 pull한다.
  • scmCommentPrefix - Jgitflow 동작 중 변경사항을 저장시에 Prefix를 붙인다. (Release만 사용할 것이기 때문에 [Release]를 붙이도록 하였다. 

이렇게 추가를 하고 pom.xml을 다시 Reload를 하면 끝이다. 

 

Maven Command 설정

Release start

$ -Dgit.user=<username> -Dgit.password=<password> jgitflow:release-start -Dmaven.test.skip=true -Dmaven.javadoc.skip=true

 

 

Release finish

$ -Dgit.user=<username> -Dgit.password=<password> jgitflow:release-finish -Dmaven.test.skip=true -Dmaven.javadoc.skip=true

 

실행결과

아래와 같이 Github repository에 Release 가 되었고 Tagging이 되었다. 

 

728x90
반응형
728x90
반응형

https://github.com/ozimov/embedded-redis

 

GitHub - ozimov/embedded-redis: Redis embedded server

Redis embedded server. Contribute to ozimov/embedded-redis development by creating an account on GitHub.

github.com

Embedded Redis를 사용하는 목적은 개발시에 실제 Redis 설치없이 Unit test를 작성하기 위함이다. 

 

초기 Spring boot Redis dependency를 테스트 하기 위해서 Redis 설치를 하였지만 Unit test를 실제 Redis에 연결해서 할 수 없으니, Embedded Redis를 설치하였다. 

 

2022.01.12 - [Infrastructure/redis] - Redis 설치하기 in kubernetes

 

Redis 설치하기 in kubernetes

$ helm install -f values.yaml my-release bitnami/redis​ $ helm repo add bitnami https://charts.bitnami.com/bitnami 현재 구현하고 있는 Service의 경우 Redis를 사용해야 하기 때문에 Redis는 local에 설..

corono.tistory.com

 

그럼 Embedded Redis를 이용한 Unit test는 어떻게 작성하는지 알아보자. 

 

Embedded Redis를 Unit test에서 사용하는 범위는 Repository Test로 한정할 예정이다. Embedded Redis의 목적에 맞게 한정시킬 예정이다.Controller, Service의 Unit test는 Mockito를 이용하여 Mocking 해서 작성하면 된다. 

 

Test 용 RedisConfiguration 추가

TestRedisConfig class는 Emnbedded Redis server를 Test가 시작할 때 start 시키고, Test가 완료될 때 stop 시킨다. 

 

Application-test.yaml 생성

Test 용 appilcation-test.yaml을 만들고 Embedded Redis로 접속할 수 있는 정보를 입력하였다. 

 

JwtTokenRepository.java 파일

이 파일은 현재까지 나의 프로젝트에서 작성된 JwtTokenRepository.java 파일이다. 기본적은 save(), getById(), delete()등의 Method는 기본적으로 생성이 되므로 포함되어 있지 않다. 

 

JwtTokenRepositoryTest.java 파일

기본 CRUD에 대한 Unit test를 추가하였다. 

TestJwtTokenBuilder.java 파일

Unit test를 작성하다보면 반복적으로 Dummy data를 생성하는 경우가 있다. Unit test가 가지는 불편함 중에 하나이다. 이 부분을 간단히 하기 위해서 별도의 Static Class를 만들어서 간단한 호출을 통해서 사용할 수 있도록 하였다. 

 

728x90
반응형
728x90
반응형

Spring Project를 진행하다 보면 `SLF4J: Class path contains multiple SLF4J bindings.` 라는 Warning 메세지를 볼 수 있다. 

SLF4J: Class path contains multiple SLF4J bindings.
SLF4J: Found binding in [jar:file:/Users/kimseunghwan/.m2/repository/ch/qos/logback/logback-classic/1.2.9/logback-classic-1.2.9.jar!/org/slf4j/impl/StaticLoggerBinder.class]
SLF4J: Found binding in [jar:file:/Users/kimseunghwan/.m2/repository/org/slf4j/slf4j-simple/1.7.32/slf4j-simple-1.7.32.jar!/org/slf4j/impl/StaticLoggerBinder.class]
SLF4J: See http://www.slf4j.org/codes.html#multiple_bindings for an explanation.
SLF4J: Actual binding is of type [ch.qos.logback.classic.util.ContextSelectorStaticBinder]

 

이런 Warning 메세지는 Dependecy 내부에서 같은 Dependecy지만 다른 버전을 Dependency 내부에서 참조를 하고 있으면 발생하게 된다. 이를 해결하기 위해서는 아래와 같은 방법을 사용하면 된다. 

 

Dependecy tree 확인

현재 어떤 Dependecy가 충돌되는 Dependency를 참조하고 있는지 모르기 때문에 `mvn dependecy:tree`로 현재 내가 사용하고 있는 Dependecy를 확인한다. 

$ mvn dependecy:tree

IDE에서 확인하는 방법 (Intellij)

 

나의 경우에는 아래와 같이 Spring boot 와 Embedded-redis에서 충돌이 발생하고 있다. 

[INFO] io.coolexplorer:spring-boot-session:war:0.0.1-SNAPSHOT
[INFO] +- org.springframework.boot:spring-boot-starter-web:jar:2.6.2:compile
[INFO] |  +- org.springframework.boot:spring-boot-starter:jar:2.6.2:compile
[INFO] |  |  +- org.springframework.boot:spring-boot-starter-logging:jar:2.6.2:compile
[INFO] |  |  |  +- ch.qos.logback:logback-classic:jar:1.2.9:compile
[INFO] |  |  |  |  \- ch.qos.logback:logback-core:jar:1.2.9:compile
[INFO] |  |  |  +- org.apache.logging.log4j:log4j-to-slf4j:jar:2.17.0:compile
[INFO] |  |  |  |  \- org.apache.logging.log4j:log4j-api:jar:2.17.0:compile
[INFO] |  |  |  \- org.slf4j:jul-to-slf4j:jar:1.7.32:compile
...
[INFO] \- it.ozimov:embedded-redis:jar:0.7.3:test
[INFO]    +- com.google.guava:guava:jar:21.0:test
[INFO]    +- commons-io:commons-io:jar:2.5:test
[INFO]    +- org.slf4j:slf4j-simple:jar:1.7.32:test
[INFO]    \- commons-logging:commons-logging:jar:1.2:test

 

Dependency 제외

해결 방법은 충돌하고 있는 두 Dependency중 한 부분에서 제외를 시켜주는 것이다. 나의 경우에는 Embedded-redis에서 제외를 시켜 주었다. 

변경된 pom.xml - <exclusions> 블럭이 추가되었다. 

<dependency>
    <groupId>it.ozimov</groupId>
    <artifactId>embedded-redis</artifactId>
    <version>${embedded.redis.version}</version>
    <scope>test</scope>
    <exclusions>
        <exclusion>
            <groupId>org.slf4j</groupId>
            <artifactId>slf4j-simple</artifactId>
        </exclusion>
    </exclusions>
</dependency>

 

간단하게 해결!

728x90
반응형

+ Recent posts