2. Mockito
Mockito 소개
- Mock : 진짜 객체와 비슷하게 동작하지만 프로그래머가 직접 그 객체의 행동을 관리하는 객체
- Mockito : Mock 객체를 쉽게 만들고 관리하고 검증할 수 있는 방법을 제공
Mockito 시작하기
- 스프링부트 2.2+ 프로젝트 생성시 spring-boot-starter-test에서 자동으로 Mockito 추가해 줌
ㆍmockito-core
ㆍmockito-junit-jupiter
- 다음 세 가지만 알면 Mock을 활용한 테스트를 쉽게 작성할 수 있다.
ㆍMock을 만드는 방법
ㆍMock이 어떻게 동작해야 하는지 관리하는 방법
ㆍMock의 행동을 검증하는 방법
Mock 객체 만들기
- Mockito.mock() 메소드로 만드는 방법
@Test
void createStudyService() {
MemberService memberService = Mockito.mock(MemberService.class);
StudyRepository studyRepository = Mockito.mock(StudyRepository.class);
StudyService studyService = new StudyService(memberService, studyRepository);
assertNotNull(studyService);
}
- @Mock 애노테이션으로 만드는 방법
ㆍJUnit 5 extension 으로 MockitoExtension을 사용해야 한다.
ㆍ필드
ㆍ메소드 매개변수
@ExtendWith(MockitoExtension.class)
class StudyServiceTest {
@Mock
MemberService memberService;
@Mock
StudyRepository studyRepository;
}
@ExtendWith(MockitoExtension.class)
class StudyServiceTest {
@Test
void createStudyService(@Mock MemberService memberService, @Mock StudyRepository studyRepository) {
StudyService studyService = new StudyService(memberService, studyRepository);
assertNotNull(studyService);
}
}
Mock 객체 Stubbing
- 모든 Mock 객체의 행동
ㆍNull을 리턴한다. (Optional 타입은 Optional.empty 리턴)
ㆍPrimitive 타입은 기본 Primitive 값
ㆍ콜렉션은 비어있는 콜렉션
ㆍVoid 메소드는 예외를 던지지 않고 아무런 일도 발생하지 않는다
- Mock 객체를 조작해서
ㆍ특정한 매개변수를 받은 결우 특정한 값을 리턴하거나 예외를 던지도록 만들 수 있다
Mockito.when(memberService.findById(Mockito.any())).thenReturn(Optional.of(member)); // memberService.findById() 호출되면 Optional.of(member) 리턴
ㆍVoid 메소드 특정 매개변수를 받거나 호출된 경우 예외를 발생 시킬 수 있다
Mockito.when(memberService.findById(1L)).thenThrow(new RuntimeException()); // 리턴타입이 있을 때 예외던지기
Mockito.doThrow(new IllegalArgumentException()).when(memberService).validate(1L); // 리턴타입이 void 일때
assertThrows(IllegalArgumentException.class, () -> {
memberService.validate(1L);
});
ㆍ메소드가 동일한 매개변수로 여러번 호출될 때 각기 다르게 행동하도록 조작할 수 있다.
Mockito.when(memberService.findById(Mockito.any()))
.thenReturn(Optional.of(member))
.thenThrow(new RuntimeException())
.thenReturn(Optional.empty());
assertEquals("phm@email.com",memberService.findById(1L).get().getEmail());
assertThrows(RuntimeException.class, () -> {
memberService.findById(2L);
});
assertEquals(Optional.empty(), memberService.findById(3L));
Mock 객체 확인
- Mock 객체가 어떻게 사용이 됐는지 확인할 수 있다
ㆍ특정 메소드가 특정 매개변수로 몇번 호출 되었는지, 최소 한번은 호출 됐는지, 전혀 호출되지 않았는지
Mockito.verify(memberService, Mockito.times(1)).notify(study); // memberService.notify(study)의 호출횟수 확인
Mockito.verify(memberService, Mockito.never()).validate(Mockito.any()); // memberService.validate() 은 호출 X
ㆍ어떤 순서대로 호출했는지
// 순서 확인
InOrder inOrder = Mockito.inOrder(memberService);
inOrder.verify(memberService).notify(study);
inOrder.verify(memberService).notify(member);
ㆍ특정 시간 이내에 호출했는지
ㆍ특정 시점 이후에 아무 일도 벌어지지 않았는지
// 액션이 없어야 한다는 것을 테스트
Mockito.verifyNoInteractions(memberService);
Mockito BDD 스타일 API
- BDD : 애플리케이션이 어떻게 "행동"해야 하는지에 대한 공통된 이해를 구성하는 방법으로, TDD에서 창안했다
- 행동에 대한 스펙
ㆍTitle
ㆍNarrative
º As a / I want / so that
ㆍAcceptance criteria
º Given / When Then
- Mockito는 BddMockito라는 클래스를 통해 BDD 스타일의 API를 제공한다
@Test
@DisplayName("BDD 스타일 Mockito API")
void BddMockitoAPI() {
// Given
StudyService studyService = new StudyService(memberService, studyRepository);
assertNotNull(studyService);
Member member = new Member();
member.setId(1L);
member.setEmail("phm@email.com");
Study study = new Study(10, "테스트");
// Mockito.when(memberService.findById(1L)).thenReturn(Optional.of(member));
// Mockito.when(studyRepository.save(study)).thenReturn(study);
BDDMockito.given(memberService.findById(1L)).willReturn(Optional.of(member));
BDDMockito.given(studyRepository.save(study)).willReturn(study);
// When
studyService.createNewStudy(1L, study);
// Then
assertEquals(member, study.getOwnerId());
// Mockito.verify(memberService, Mockito.times(1)).notify(study);
BDDMockito.then(memberService).should(Mockito.times(1)).notify(study);
// Mockito.verifyNoInteractions(memberService);
BDDMockito.then(memberService).shouldHaveNoMoreInteractions();
}