Framework/Spring
Spring boot starter test 와 Junit5를 이용한 테스트
코딩하는 오징어
2018. 8. 24. 23:51
반응형
Junit 5 & Spring Test을 이용한 TDD 환경 세팅
기본 세팅
@ExtendWith(SpringExtension.class) @SpringBootTest(classes = KkApplication.class) @ActiveProfiles("test") public abstract class SpringTestSupport { }
- TestContext를 사용하려면 위의 SpringTestSupport를 상속받아 테스트 코드를 개발한다.
- @ExtendWith는 Junit4의 RunWith(SpringRunner.class)와 비슷하다고 생각하면 된다.
- @SpringBootTest는 @SpringBootApplication이 붙은 애너테이션을 찾아 context를 찾는다.
- @SpringBootApplication을 사용하지 않고 다음과 같이 애너테이션을 풀어 개발했다면 위와 같이 classes필드를 통해 지정해야한다.
@Configuration @EnableAutoConfiguration(exclude = { DataSourceAutoConfiguration.class, HibernateJpaAutoConfiguration.class }) @ComponentScan( basePackageClasses = KkApplication.class, excludeFilters = { @ComponentScan.Filter(type = FilterType.CUSTOM, classes = TypeExcludeFilter.class), @ComponentScan.Filter(type = FilterType.CUSTOM, classes = AutoConfigurationExcludeFilter.class) }) public class KkApplication { public static void main(String[] args) { SpringApplication.run(KkApplication.class, args); } }
MockMvc 테스트 세팅
@AutoConfigureMockMvc public abstract class SpringMockMvcTestSupport extends SpringTestSupport { @Autowired protected MockMvc mockMvc; }
- @AutoConfigureMockMvc를 통해 MockMvc를 Builder없이 주입 받을 수 있다. 스프링부트의 매력인 Auto Config의 장점이다. 이 애너테이션을 상속받아 controller 테스트들을 진행한다. 다음은 샘플 테스트 코드이다.
class HealthCheckerControllerTest extends SpringMockMvcTestSupport { @Test void healthCheckTest() throws Exception { this.mockMvc.perform(get("/v1/health-checker")) .andDo(print()) .andExpect(status().is(HttpStatus.OK.value())) .andExpect(content().contentType(MediaType.APPLICATION_JSON_UTF8_VALUE)) .andExpect(jsonPath("$.deployDate").exists()) .andExpect(jsonPath("$.deployVersion").exists()) .andExpect(jsonPath("$.distributor").exists()); } }
Mock 테스트 세팅
public class MockitoExtension implements TestInstancePostProcessor, ParameterResolver { @Override public void postProcessTestInstance(Object testInstance, ExtensionContext context) { MockitoAnnotations.initMocks(testInstance); } @Override public boolean supportsParameter(ParameterContext parameterContext, ExtensionContext extensionContext) { return parameterContext.getParameter().isAnnotationPresent(Mock.class); } @Override public Object resolveParameter(ParameterContext parameterContext, ExtensionContext extensionContext) { return getMock(parameterContext.getParameter(), extensionContext); } private Object getMock(Parameter parameter, ExtensionContext extensionContext) { Class<?> mockType = parameter.getType(); Store mocks = extensionContext.getStore(Namespace.create(MockitoExtension.class, mockType)); String mockName = getMockName(parameter); if (mockName != null) { return mocks.getOrComputeIfAbsent(mockName, key -> mock(mockType, mockName)); } else { return mocks.getOrComputeIfAbsent(mockType.getCanonicalName(), key -> mock(mockType)); } } private String getMockName(Parameter parameter) { String explicitMockName = parameter.getAnnotation(Mock.class).name().trim(); if (!explicitMockName.isEmpty()) { return explicitMockName; } else if (parameter.isNamePresent()) { return parameter.getName(); } return null; } }
- 위의 MockitoExtension 클래스를 @ExtendWith의 value 필드에 넣음으로써 Mock테스트에 필요한 곳에 애너테이션을 달면 된다. 다음은 샘플 코드이다.
import static org.mockito.Mockito.when; @ExtendWith(MockitoExtension.class) class HealthCheckerControllerTest extends SpringMockMvcTestSupport { @Mock private BookRepository bookRepository; @InjectMocks private BookService bookService; private givenBookRepository() { when(bookRepository.findByXXX(..)).thenReturn(...); } @Test void healthCheckTest() throws Exception { this.mockMvc.perform(get("/v1/health-checker")) .andDo(print()) .andExpect(status().is(HttpStatus.OK.value())) .andExpect(content().contentType(MediaType.APPLICATION_JSON_UTF8_VALUE)) .andExpect(jsonPath("$.deployDate").exists()) .andExpect(jsonPath("$.deployVersion").exists()) .andExpect(jsonPath("$.distributor").exists()); } }
- 여기서 @Test메서드에 @Mock 파라미터로 바로 줄 수도 있다. @Test메서드에 파라미터를 줄 수 있는 것은 Junit4와 달라진 Junit5의 장점인 것 같다.
주의
- org.junit.jupiter.api.Test를 import하여 @Test를 이용하여야 @ExtendWith(SpringExtension.class)를 통해 컨텍스트를 끌어 올 수 있다. null pointer exception이 발생한다면 org.junit.test가 import 되어있는지 확인해보자.
- build.gradle에 다음과 같이 설정을 해야 @Test가 달린 메서드들을 테스트한다. gradle 버전이 4.6이상이어야 함을 참고하자.
build.gradle
dependencies { ... testCompile('org.springframework.boot:spring-boot-starter-test') testCompile("org.junit.platform:junit-platform-launcher:1.1.1") testCompile("org.junit.platform:junit-platform-suite-api:1.1.1") testCompile("org.junit.jupiter:junit-jupiter-engine:5.1.1") testCompile("org.junit.vintage:junit-vintage-engine:5.1.1") ... } test { useJUnitPlatform() //Gradle 4.6이상에서 사용할 수 있는 설정 systemProperty 'spring.profiles.active', 'test' }
/gradle/wrapper/gradle-wrapper.properties
...
distributionUrl=https\://services.gradle.org/distributions/gradle-4.8.1-all.zip
//필자는 Junit5로 테스트를 짰지만 gradle test task로 테스트들이 실행되지 않음을 발견하였다.
//4.5.1 -> 4.8.1로 올려서 task를 실행시켜주었더니 필자가 겪은 에러가 해결되었다.
반응형