워밍업 클럽 4기 백엔드 - Day 18 미션
Practical Testing: 실용적인 테스트 가이드
1. @Mock, @MockBean, @Spy, @SpyBean, @InjectMocks 의 차이를 한번 정리해 봅시다.
@Mock, @MockBean은 순수 Mock 객체 생성 시에 사용
실제 객체 없이 동작을 모의하는 가짜 객체
@Spy, @SpyBean은 Spy 객체 생성 시에 사용
실제 객체의 동작을 그대로 실행하면서 일부 기능을 Stubbing(실제 객체를 감싸는 Spy 객체 생성)
Mock과 달리 Stubbing 할 메소드를 별도로 지정해주어야 함
@MockBean, @SpyBean은 Spring Bean 형태로 객체 생성
생성한 Mock, Spy 객체를 Spring Context에 등록하여 기존의 Bean을 대체
Spring Boot 서버를 띄우는 테스트에 사용 가능(@SpringBootTest)
@Mock, @Spy는 Spring Boot 컨텍스트 없이 객체 생성
@InjectMocks는 해당 어노테이션이 달린 객체를 생성하면서, @Mock으로 생성한 Mock 객체를 주입해주는 역할
@Mock으로 생성한 객체에 의존하는 객체 생성 + Mock 객체 의존성 주입
@MockBean의 경우, @WebMvcTest(*.class)와 함께 사용하며 @MockBean으로 생성한 Bean이 Spring Context에 등록되어 의존성 주입이 이루어지므로 @InjectMocks와 함께 사용하지 않음
상황별 선택 방법
Spring Context 없이 POJO로 순수 로직을 검증할 때는
@Mock & @InjectMocks, @Spy 사용
Spring 환경에서 테스트 시
@MockBean, @SpyBean 사용
실제 객체를 사용하고, 일부만 스텁 시
@Spy, @SpyBean 사용
실제 객체를 사용하지 않는 경우
@Mock(+@InjectMocks), @MockBean 사용
2. 아래 3개의 테스트가 있습니다.
내용을 살펴보고, 각 항목을 @BeforeEach, given절, when절에 배치한다면 어떻게 배치하고 싶으신가요?
(@BeforeEach에 올라간 내용은 공통 항목으로 합칠 수 있습니다. ex. 1-1과 2-1을 하나로 합쳐서 @BeforeEach에 배치)
@BeforeEach
void setUp() {
// 각 테스트에 공통적으로 필요한 내용 준비
0-1. 사용자 생성에 필요한 내용 준비
0-2. 사용자 생성
0-3. 게시물 생성에 필요한 내용 준비
0-4. 게시물 생성
}
@DisplayName("사용자가 댓글을 작성할 수 있다.")
@Test
void writeComment() {
// given
1-1. 댓글 생성에 필요한 내용 준비
// when
1-2. 댓글 생성
// then
검증
}
@DisplayName("사용자가 댓글을 수정할 수 있다.")
@Test
void updateComment() {
// given
2-1. 댓글 생성에 필요한 내용 준비
2-2. 댓글 생성
// when
2-3. 댓글 수정
// then
검증
}
@DisplayName("자신이 작성한 댓글이 아니면 수정할 수 없다.")
@Test
void cannotUpdateCommentWhenUserIsNotWriter() {
// given
3-1. 사용자2 생성에 필요한 내용 준비
3-2. 사용자2 생성
3-3. 사용자1의 댓글 생성에 필요한 내용 준비
3-4. 사용자1의 댓글 생성
// when
3-5. 사용자2가 사용자1의 댓글 수정 시도
// then
검증
}
판단 기준
when 절에는 검증하고자 하는 핵심 동작을 작성
given 절에서는 when 절의 동작을 위해 필요한 데이터를 준비
@BeforeEach에서는 모든 테스트에서 공통적으로 필요한 데이터만 준비하고, 일부 테스트에서만 사용되는 데이터(사용자 1 댓글, 사용자 2 생성 등)는 해당 테스트 내에서 독립적으로 준비
댓글을 작성해보세요.