너무 궁금해서 미칠것 같은 내용을 공부후 정리한 내용입니다. 영혼을 갈아서 정리해 보았습니다!
항상 모든 인터페이스의 구현체를 만들어야만 테스트가 가능할까요?
컨트롤러를 테스트 하고 싶은데, 서비스의 구현체 없이 인터페이스 만을 활용하여 테스트 할수는 없는것 일까요?
이에 대한 궁금증에서 시작한 공부 내용을 정리해 봅니다.
▶ 목차
1. Mock의 의미
2. Mock 객체 만들기
3. CGLIB 프록시를 활용한 Mock의 원리 이해하기
Mock? Mocking? MockUp?
- Mock의 의미는 "테스트를 위해 만든 모형" 을 의미합니다.
- Mocking의 의미는 테스트를 위해 실제 객체와 비슷한 모의 객체를 만든는 것을 말합니다.
- MockUp의 의미는 모킹한 객체를 메모리에서 얻어내는 과정을 말합니다.
Mock 객체 만들기
Spring을 사용할때 Controller에서는 서비스의 인터페이스를 사용하여 비지니스 컴포넌트(ServiceImpl)의 메소드를 호출하게 됩니다. 우선 다음 그림을 살펴보시죠!
일반적인 경우 Service 인터페이스를 만들고, 이를 구현하는 ServiceImpl 클래스를 만들게 됩니다.
그리고 이러한 구현체를 사용하기 위해 클라이언트에게는 Service 인터페이스를 제공하면 되죠!
하지만 항상 모든 구현체를 만들고 테스트를 작성해야 할까요?
비지니스 컴포넌트(서비스 구현체)를 생성하는데 많은 자원과 시간이 필요하거나, 아직 비지니스 컴포넌트가 완성되지 않아 인터페이스만 제공되고 있는 경우도 있을수 있습니다.
이런 상황에서 비지니스 컴포넌트까지 다 필요한 테스트는 문제가 될수 있겠죠?
이러한 문제를 해결하기위해 Mockito를 활용하여 컴포넌트 자체를 모킹하여 테스트하는 방법을 제공합니다!
즉, 인터페이스가 구현되어 있다는 가정하에 구성한 Service 로직을 실행하고자 할 때 Mock 객체를 만들어 테스트 해야한다!
우선 인터페이스는 필요하겠죠? 간단하게 하나 만들어 봅시다!
▶ Service Interface
public interface BoardService {
String hello(String name);
BoardVO getBoard();
List<BoardVO> getBoardList();
}
이제 이 interface를 통하여 테스트를 진행해볼까요? 우선 코드보고 뒤에 설명을 할께요~
▶ @MockBean을 활용한 Mock객체 생성하기
import static org.mockito.Mockito.when;
@SpringBootTest(webEnvironment = WebEnvironment.MOCK)
@AutoConfigureMockMvc
public class BoardControllerTest {
@Autowired
private MockMvc mockMvc;
@MockBean
private BoardService boardService;
@Test
public void testHello() throws Exception {
//given (stub 생성)
Mockito.when(boardService.hello("zbqmgldjfh")).thenReturn("Hello : zbqmgldjfh");
// when
ResultActions result = mockMvc.perform(get("/hello").param("name", "zbqmgldjfh"));
// then
result.andExpect(status().isOk())
.andExpect(content().string("Hello : zbqmgldjfh"))
.andDo(print());
}
}
- @SpringBootTest(webEnvironment = WebEnvironment.MOCK)
위 에노테이션을 통하여 서블릿 컨테이너를 모킹하는 실습을 진행할수 있습니다.
webEnvironment값으로 MOCK을 주었는데, 이는 테스트를 진행할때 톰켓서버를 실행하지 않으며 직접적으로 서블릿 컨테이너를 구동하지 않는다는 의미이다.
이를 통해 서블릿 컨테이너의 Mock을 생성하게 되는 것 입니다!
또한 이렇게 모킹한 객체를 DI 받으려면 @AutoConfigureMockMvc 이 필요해서 추가해주었습니다.
- mockMvc 자동 주입
@Autowired private MockMvc mockMvc;
위에서 @AutoConfigureMockMvc를 지정해주었기 때문에 MockMvc를 모킹한 객체를 주입받을 수 있습니다.
- stub 생성
//given (stub 생성)
Mockito.when(boardService.hello("zbqmgldjfh")).thenReturn("Hello : zbqmgldjfh");
기대 행위를 작성하는것을 Stubbing이라고 부릅니다.
생각해보면 테스트 할 대상인 boardService는 구현체를 만든적이 없습니다.
따라서 인터페이스를 통해 메서드를 호출해도 처리될 logic 부분이 없죠!
하지만 내부처리 logic의 구현은 나중에 생각하고, 우선적으로 input과 output만 생각하여 stub를 구현하면 해당 메서드를 호출시 결과는 받을수가 있겠죠? 이를 활용하여 테스트를 진행하는것 입니다!
- 테스트 실행시 조건부
// when
ResultActions result = mockMvc.perform(get("/hello").param("name", "zbqmgldjfh"));
어떠한 요청을 전달할지 설정하는 부분입니다. 테스트를 진행하려면 요청부분을 설정해야 겠죠?
mockMvc의 perform메서드를 통하여 이를 지정할수 있습니다. 반환값으로는 ResultActions 을 반환하게 됩니다.
- 결과 확인
// then
result.andExpect(status().isOk()).andExpect(content().string("Hello : zbqmgldjfh")).andDo(print());
최종적으로 원하는 결과가 나오는지 확인할 수 있습니다!
테스트 케이스가 성공적으로 작동하였죠!
이번 테스트를 통하여 인터페이스만을 통해 테스트를 진행해 보았습니다.
구현체 없이도 테스트를 진행한 놀라운 경험을 저 또한 오늘 해보게 되었습니다!
▶ @MockBean
Mock 과 비슷한 @MockBean이라는 어노테이션이 있습니다. 이름이 비슷한것처럼 실제로도 비슷하게 동작합니다.
하지만 @Mockbean은 경로(org.springframework.boot.test.mock.mockito.MockBean)를 봐도 알 수 있듯이 @Mock과는 다르게 spring 영역에 있는 어노테이션이라는 것을 알 수 있습니다.
@MockBean은 스프링 컨텍스트(스프링 빈을 관리하는 컨테이너)에 mock객체를 등록하게 되고 스프링 컨텍스트에 의해 @Autowired가 동작할 때 등록된 mock객체를 사용할 수 있도록 동작합니다.
따라서 @Mock이 사용하는 @InjectMocks 과 같은 의존성 주입을 같이 사용하면 오류가 발생하게 됩니다.
(자세한 @Mock, @MockBean의 차이는 나중에 정리하기로...)
좀더 deep dive 해볼까요?
여기서 끝내는건 저의 성격상 참을수가 없습니다. 아직도 궁금한게 산더미.... 가장 궁금한게 하나 있는데...
Stub을 만드는 when()은 어떻게 작동할까요? 더 나아가 Mock의 원리는 뭘까요?
출처 입력
대충 프록시를 이용할것 같다는 느낌은 듭니다만... 궁금하면 공부하고 정리해야겠죠?
when(boardService.hello("zbqmgldjfh")).thenReturn("Hello : zbqmgldjfh")
일단 프록시가 만들어지는 과정부터 대략적으로 알아야 합니다.
▶ 간단한 Mock프레임 워크 만들어보기 (중요)
우리는 프록시 객체를 반환하는 정적인 mock메서드를 만들것 입니다. 다음 코드를 살펴보시죠!
public class Mock {
public static <T> T mock(Class<T> clazz) {
MockInvocationHandler invocationHandler = new MockInvocationHandler();
T proxy = (T) Proxy.newProxyInstance(Mock.class.getClassLoader(), new Class[]{clazz}, invocationHandler);
return proxy;
}
}
이렇게 하면 clazz에 대한 프록시 객체를 생성하기 위해서 내부적으로 MockInvocationHandler를 호출하게 됩니다.
코드 중간 부분에 프록시를 만드는 부분을 확인할 수 있습니다. 다음과 같이 말이죠
public static Object newProxyInstance(ClassLoaser loader, Class<?>[] interfaces, InvocationHandler handler)
- loader : 프록시 클래스를 정의하는 클래스 로더
- interfaces : 프록시 클래스가 구현하는 인터페이스 리스트
- handler : 메서드 호출을 처리하는 핸들러 (클라이언트에서 타겟의 메소드를 호출할 때 프록시에서 내부적으로 호출하는 InvocationHandler의 구현체 입니다.)
newProxyInstance()로 생성된 프록시 객체는 전달한 Interfaces들의 메서드를 포함하고 있습니다.
MockInvocationHandler는 다음과 같습니다.
private static class MockInvocationHandler implements InvocationHandler {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
return null;
}
}
invoke 메서드를 보면 3개의 파라미터를 갖고 있습니다.
method 파라미터는 메서드 이름, 파라미터 타임, 리턴 타입 등의 정보를 갖고있습니다.
args 파라미터는 proxy 매서드 호출에 전달한 파라미터를 포함합니다.
최종적으로 프록시 객체의 메서드에 대한 호출은 null을 반환합니다.
이 단계에서는 메서드를 호출할 수 있는 프록시 객체가 메모리 상에 존재합니다.
▶ when() 메서드를 통해 실질적으로 얻는것은?
위 코드의 경우 null 객체만 얻을 수 있습니다. 프록시의 코드를 통해 확인해 봤죠!
첫 번째 단계에서 우리는 항상 null을 반환하는 프록시인 boardService객체를 만들었습니다.
그럼 boardService.hello니까 "when() 메서드가 hello() 메서드를 호출하면서 "zbqmgldjfh" 라는 인자를 넘기고" 그러는 거
아닙니다!
우린 인터페이스만 알고 있어요!! 구현체는 전혀 모르자나요!
프록시에 대한 호출로 null을 반환하고 null 값을 when() 메서드에 전달합니다. 이게 전부에요!
▶ thenReturn() 메서드는 뭘까?
별거 없습니다. 작은 속임수 정도를 부리는 녀석이죠!
해결책은 간단합니다. 상태가 저장되는 static변수를 사용하는거죠!
1) MockInvocationHandler 의 구현체인 invocationHandler 안에는 우리가 이전에 프록시를 만들때 사용한 함수와 인자들이 저장되어 있습니다.
2) Mock class 안에는 마지막으로 불려진 MockInvocationHandler에 대한 참조를 invocationHandler 로 저장하고 있습니다.
위 2 단계는 프록시 객체가 내부적으로 invoke를 호출하기 때문에 발생합니다.
when() 메서드는 어떠한 logic도 없습니다.
하지만 thenReturn()을 호출하는 순간 MockInvocationHandler을 반환하는데, 이 안에 이전에 저장된 인자와 함수가 포함되어 있습니다.
출처
'BackEnd > TDD' 카테고리의 다른 글
[TDD] JUnit 만들기 - 2 (2) | 2022.07.15 |
---|---|
[TDD] JUnit 만들기 - 1 (2) | 2022.07.15 |
[junit5] MockMvc에서 NestedServletException 통과시키기 (0) | 2022.03.11 |
[TDD] 테스트 코드 작성 팁 (2/2) (0) | 2022.01.30 |
[TDD] 테스트 코드 작성 팁 (1/2) (0) | 2022.01.28 |
댓글