스프링은 다양한 테스트 전략을 제공하고 있습니다. 대표적으로 Slice Test 라는 것으로 특정 레이어에 대해서 Bean을 최소한으로 등록시켜 테스트 하고자 하는 부분에 최대한 단위 테스트를 지원합니다. 다양하게 지원해주는 만큼 테스트 코드를 통일성 있게 관리하는 것이 중요합니다. 더 안전하고 통일성 있게 테스트를 진행하는 방법에 대해서 제 나름의 노하우를 정리해보았습니다.
어노테이션 | 설명 | 부모 클래스 | Bean |
---|---|---|---|
@SpringBootTest | 통합 테스트, 전체 | IntegrationTest | Bean 전체 |
@WebMvcTest | 단위 테스트, Mvc 테스트 | MockApiTest | MVC 관련된 Bean |
@DataJpaTest | 단위 테스트, Jpa 테스트 | RepositoryTest | JPA 관련 Bean |
None | 단위 테스트, Service 테스트 | MockTest | None |
None | POJO, 도메인 테스트 | None | None |
- 모든 Bean을 올리고 테스트를 진행하기 때문에 쉽게 테스트 진행 가능
- 모든 Bean을 올리고 테스트를 진행하기 때문에 운영환경과 가장 유사하게 테스트 가능
- API를 테스트할 경우 요청부터 응답까지 전체적인 테스트 진행 가능
- 모든 Bean을 올리고 테스트를 진행하기 때문에 테스트 시간이 오래 걸림
- 테스트의 단위가 크기 때문에 테스트 실패시 디버깅이 어려움
- 외부 API 콜같은 Rollback 처리가 안되는 테스트 진행을 하기 어려움
@RunWith(SpringRunner.class)
@SpringBootTest(classes = ApiApp.class)
@AutoConfigureMockMvc
@ActiveProfiles(TestProfile.TEST)
@Transactional
@Ignore
public class IntegrationTest {
@Autowired protected MockMvc mvc;
@Autowired protected ObjectMapper objectMapper;
...
}
- 통합 테스트의 Base 클래스입니다. Base 클래스를 통해서 테스트 전략을 통일성 있게 가져갈 수 있습니다.
- 통합 테스트는 주로 컨트롤러 테스트를 주로 하며 요청부터 응답까지의 전체 플로우를 테스트합니다.
@ActiveProfiles(TestProfile.TEST)
설정으로 테스트에 profile을 지정합니다. 환경별로 properties 파일을 관리하듯이 test도 반드시 별도의 properties 파일로 관리하는 것이 바람직합니다.- 인터페이스나 enum 클래스를 통해서 profile을 관리합니다. 오타 실수를 줄일 수 있으며 전체적인 프로필이 몇 개 있는지 한 번에 확인할 수 있습니다.
@Transactional
트랜잭션 어노테이션을 추가하면 테스트코드의 데이터베이스 정보가 자동으로 Rollback 됩니다. 베이스 클래스에 이 속성을 추가 해야지 실수 없이 진행할 수 있습니다.@Transactional
을 추가하면 자연스럽게 데이터베이스 상태 의존적인 테스트를 자연스럽게 하지 않을 수 있게 됩니다.- 통합 테스트 시 필요한 기능들을
protected
로 제공해줄 수 있습니다. API 테스트를 주로 하게 되니 ObjectMapper 등을 제공해줄 수 있습니다. 유틸성 메서드들도protected
로 제공해주면 중복 코드 및 테스트 코드의 편의성이 높아집니다. - 실제로 동작할 필요가 없으니
@Ignore
어노테이션을 추가합니다.
public class MemberApiTest extends IntegrationTest {
@Autowired
private MemberSetup memberSetup;
@Test
public void 회원가입_성공() throws Exception {
//given
final Member member = MemberBuilder.build();
final Email email = member.getEmail();
final Name name = member.getName();
final SignUpRequest dto = SignUpRequestBuilder.build(email, name);
//when
final ResultActions resultActions = requestSignUp(dto);
//then
resultActions
.andExpect(status().isOk())
.andExpect(jsonPath("email.value").value(email.getValue()))
.andExpect(jsonPath("email.host").value(email.getHost()))
.andExpect(jsonPath("email.id").value(email.getId()))
.andExpect(jsonPath("name.first").value(name.getFirst()))
.andExpect(jsonPath("name.middle").value(name.getMiddle()))
.andExpect(jsonPath("name.last").value(name.getLast()))
.andExpect(jsonPath("name.fullName").value(name.getFullName()))
;
}
@Test
public void 회원조회() throws Exception {
//given
final Member member = memberSetup.save();
//when
final ResultActions resultActions = requestGetMember(member.getId());
//then
resultActions
.andExpect(status().isOk())
.andExpect(jsonPath("email.value").value(member.getEmail().getValue()))
.andExpect(jsonPath("email.host").value(member.getEmail().getHost())
.andExpect(jsonPath("email.id").value(member.getEmail().getId()))
.andExpect(jsonPath("name.first").value(member.getName().getFirst()))
.andExpect(jsonPath("name.middle").value(member.getName().getMiddle()))
.andExpect(jsonPath("name.last").value(member.getName().getLast()))
.andExpect(jsonPath("name.fullName").value(member.getName().getFullName()))
;
}
private ResultActions requestSignUp(SignUpRequest dto) throws Exception {
return mvc.perform(post("/members")
.contentType(MediaType.APPLICATION_JSON_UTF8)
.content(objectMapper.writeValueAsString(dto)))
.andDo(print());
}
...
}
IntegrationTest
클래스를 상속받습니다. 이 상속을 통해서 MemberApiTest에서 테스트를 위한 어노테이션이 생략되며 어떤 통합 테스트라도 항상 통일성을 가질 수 있습니다.given
,when
,then
키워드로 테스트 흐름을 알려줍니다. 다른 사람의 테스트 코드의 가독성이 높아지기 때문에 해당 키워드로 적절하게 표시하는 것을 권장합니다.- 요청에 대한 메서드를
requestSignUp(...)
으로 분리해서 재사용성을 높입니다. 해당 메서드로 valdate 실패하는 케이스도 작성합니다andDo(print())
메서드를 추가해서 해당 요청에 대한 출력을 확인합니다. 디버깅에 매우 유용합니다. - 모든 response에 대한
andExpect
를 작성합니다. 간혹.andExpect(content().string(containsString("")))
이런 테스트를 진행하는데 특정 문자열이 들어 있는지 없는지 확인하는 것보다 모- response에 하나라도 빠지거나 변경되면 API 변경이 이루어진 것이고 그 변경에 맞게 테스트 코드도 변경되어야 합니다.
회원 조회
테스트 같은 경우memberSetup.save();
메서드로 테스트전에 데이터베이스에 insert 합니다.- 데이터베이스에 미리 있는 값을 검증하는 것은 데이터베이스 상태에 의존한 코드가 되며 누군가가 회원 정보를 변경하게 되면 테스트 코드가 실패하게 됩니다.
- 테스트 전에 데이터를 insert하지 않는다면 테스트 코드 구동 전에
.sql
으로 미리 데이터베이스를 준비시킵니다 ApplicationRunner를 이용해서 데이터베이스를 준비시키 방법도 있습니다. - 중요한 것은 데이터베이스 상태에 너무 의존적인 테스트는 향후 로직의 문제가 없더라도 테스트가 실패하는 상황이 자주 만나게 됩니다.
- 진행하고자 하는 테스트에만 집중할 수 있습니다.
- 테스트 진행시 중요 관점이 아닌 것들은 Mocking 처리해서 외부 의존성들을 줄일 수 있습니다.
- 예를 들어 주문 할인 로직이 제대로 동작하는지에 대한 테스트만 진행하지 이게 실제로 데이터베이스에 insert되는지는 해당 테스트의 관심사가 아닙니다.
- 테스트 속도가 빠릅니다.
- 의존성 있는 객체를 Mocking 하기 때문에 문제가 완결된 테스트는 아닙니다.
- Mocking 하기가 귀찮습니다.
- Mocking 라이브러리에 대한 학습 비용이 발생합니다.
@RunWith(MockitoJUnitRunner.class)
@ActiveProfiles(TestProfile.TEST)
@Ignore
public class MockTest {
}
- 주로 Service 영역을 테스트 합니다.
MockitoJUnitRunner
을 통해서 Mock 테스트를 진행합니다.
public class MemberSignUpServiceTest extends MockTest {
@InjectMocks
private MemberSignUpService memberSignUpService;
@Mock
private MemberRepository memberRepository;
private Member member;
@Before
public void setUp() throws Exception {
member = MemberBuilder.build();
}
@Test
public void 회원가입_성공() {
//given
final Email email = member.getEmail();
final Name name = member.getName();
final SignUpRequest dto = SignUpRequestBuilder.build(email, name);
given(memberRepository.existsByEmail(any())).willReturn(false);
given(memberRepository.save(any())).willReturn(member);
//when
final Member signUpMember = memberSignUpService.doSignUp(dto);
//then
assertThat(signUpMember).isNotNull();
assertThat(signUpMember.getEmail().getValue()).isEqualTo(member.getEmail().getValue());
assertThat(signUpMember.getName().getFullName()).isEqualTo(member.getName().getFullName());
}
@Test(expected = EmailDuplicateException.class)
public void 회원가입_이메일중복_경우() {
//given
final Email email = member.getEmail();
final Name name = member.getName();
final SignUpRequest dto = SignUpRequestBuilder.build(email, name);
given(memberRepository.existsByEmail(any())).willReturn(true);
//when
memberSignUpService.doSignUp(dto);
}
}
MockTest
객체를 상속받아 테스트의 일관성을 갖습니다.회원가입_성공
테스트는 오직 회원 가입에대한 단위 테스트만 진행합니다.existsByEmail
을 모킹해서 해당 이메일이 중복되지 않았다는 가정을 합니다.then
에서는 회원 객체가 해당 비지니스 요구사항에 맞게 생성됬는지를 검사합니다.- 실제 데이터베이스에 Insert 됐는지 여부는 해당 테스트의 관심사가 아닙니다.
회원가입_이메일중복_경우
테스트는 회원가입시 이메일이 중복됬는지 여부를 확인합니다.existsByEmail
을 모킹해서 이메일이 중복됬다는 가정을 합니다.expected
으로 이메일이 중복되었을 경우EmailDuplicateException
예외가 발생하는지 확인합니다.- 해당 이메일이 데이터베이스에 실제로 있어서 예외가 발생하는지는 관심사가 아닙니다. 작성한 코드가 제대로 동작 여부만이 해당 테스트의 관심사 입니다.
- 오직 테스트의 관심사만 테스트를 진행하기 때문에 예외 발생시 디버깅 작업도 명확해집니다.
- 외부 의존도가 낮기 때문에 테스트 하고자하는 부분만 명확하게 테스트가 가능합니다.
- 이것은 단점이기도 합니다. 해당 테스트만 진행하지 외부 의존을 갖는 코드까지 테스트하지 않으니 실제 환경에서 제대로 동작하지 않을 가능성이 있습니다. 외부 의존에 대한 테스트는 통합 테스트에서 진행합니다.
- Mock 테스트와 장점은 거의 같습니다.
WebApplication
관련된 Bean들만 등록하기 때문에 통합 테스트보다 빠르게 테스트할 수 있습니다.- 통합 테스트를 진행하기 어려운 테스트를 진행합니다.
- 외부 API 같은 Rollback 처리가 힘들거나 불가능한 테스트를 주로 사용합니다.
- 예를 들어 외부 결제 모듈 API를 콜하면 안 되는 케이스에서 주로 사용 할 수 있습니다.
- 이런 문제는 통합 테스트에서 해당 객체를 Mock 객체로 변경해서 테스트를 변경해서 테스트할 수도 있습니다.
- Mock 테스트와 단점은 거의 같습니다.
- 요청부터 응답까지 모든 테스트를 Mock기반으로 테스트하기 때문에 실제 환경에서는 제대로 동작하지 않을 가능성이 매우 큽니다.
@WebMvcTest(MemberApi.class)
public class MemberMockApiTest extends MockApiTest {
@MockBean private MemberSignUpService memberSignUpService;
@MockBean private MemberHelperService memberHelperService;
...
@Test
public void 회원가입_유효하지않은_입력값() throws Exception {
//given
final Email email = Email.of("asdasd@d"); // 이메일 형식이 유효하지 않음
final Name name = Name.builder().build();
final SignUpRequest dto = SignUpRequestBuilder.build(email, name);
final Member member = MemberBuilder.build();
given(memberSignUpService.doSignUp(any())).willReturn(member);
//when
final ResultActions resultActions = requestSignUp(dto);
//then
resultActions
.andExpect(status().isBadRequest())
;
}
@WebMvcTest(MemberApi.class)
어노테이션을 통해서 하고자 하는MemberApi
의 테스트를 진행합니다.@MockBean
으로 객체를 주입받아 Mocking 작업을 진행합니다.- 테스트의 관심사는 오직 Request와 그에 따른 Response 입니다.
Repository
관련된 Bean들만 등록하기 때문에 통합 테스트에 비해서 빠릅니다.Repository
에 대한 관심사만 갖기 때문에 테스트 범위가 작습니다.
- 테스트 범위가 작기 때문에 실제 환경과 차이가 발생합니다.
@RunWith(SpringRunner.class)
@DataJpaTest
@ActiveProfiles(TestProfile.TEST)
@AutoConfigureTestDatabase(replace = AutoConfigureTestDatabase.Replace.NONE)
@Ignore
public class RepositoryTest {
}
@DataJpaTest
어노테이션을 통해서Repository
에 대한 Bean만 등록합니다.@DataJpaTest
는 기본적으로 메모리 데이터베이스에 대한 테스트를 진행합니다.@AutoConfigureTestDatabase
어노테이션을 통해서 profile에 등록된 데이터베이스 정보로 대체할 수 있습니다.JpaRepository
에서 기본적으로 기본적으로 재공해주는findById
,findByAll
,deleteById
등은 테스트를 하지 않습니다.- 기본적으로
save()
null 제약 조건 등의 테스트는 진행해도 좋다고 생각합니다. - 주로 커스텀하게 작성한 쿼리 메서드,
@Query
으로 작성된 JPQL등의 커스텀하게 추가된 메서드를 테스트합니다.
- 기본적으로
public class MemberRepositoryTest extends RepositoryTest {
@Autowired
private MemberRepository memberRepository;
private Member saveMember;
private Email email;
@Before
public void setUp() throws Exception {
final String value = "[email protected]";
email = EmailBuilder.build(value);
final Name name = NameBuilder.build();
saveMember = memberRepository.save(MemberBuilder.build(email, name));
}
...
@Test
public void existsByEmail_존재하는경우_true() {
final boolean existsByEmail = memberRepository.existsByEmail(email);
assertThat(existsByEmail).isTrue();
}
@Test
public void existsByEmail_존재하지않은_경우_false() {
final boolean existsByEmail = memberRepository.existsByEmail(Email.of("[email protected]"));
assertThat(existsByEmail).isFalse();
}
}
setUp()
메서드를 통해서 Member를 데이터베이스에 insert 합니다.setUp()
메서드는 메번 테스트 코드가 실행되기전에 실행됩니다. 즉 테스트 코드를 실행할 때마다 insert -> rollback이 자동으로 이루어집니다.
- 추가 작성한 쿼리메서드
existsByEmail
을 테스트 진행합니다.- 실제로 작성된 쿼리가 어떻게 출력되는지
show-sql
옵션을 통해서 확인 합니다. ORM은 SQL을 직접 작성하지 않으니 실제 쿼리가 어떻게 출력되는지 확인하는 습관을 반드시 가져야합니다.
- 실제로 작성된 쿼리가 어떻게 출력되는지
각 엔티티(Embeddable, Entity, 일반 POJO, 모든 객체) 객체들의 기능이 풍부해야 합니다. 객체 본인의 책임을 충분히 다하지 않고 있으면 다른 영역으로 그 객체의 책임이 넘어 가게됩니다. 예를 들어 Name
객체가 getFullName()
메서드를 제공해주지 않는다면 getFullName()
메서드를 만족시키는 메서드들이 다른 계층에서 구현하게 되고 어느 계층에서 어떻게 사용되고 있는지 모르기 때문에 누군가는 중복코드를 만들게 됩니다.
객체지향에서 본인의 책임(기능)은 본인 스스로가 제공해야 합니다. 특히 엔티티 객체들은 가장 핵심 객체이고 이 객체를 사용하는 계층들이 다양하게 분포되기 때문에 반드시 테스트 코드를 작성해야합니다.
- POJO 객체이므로 테스트하기 편합니다. 외부에서 주입 받을 의존성도 없고 Mocking할 대상도 없습니다.
- 엔티티 객체는 사용하는 계층이 많으므로 테스트의 효율성이 높습니다.
- 단점은 없다고 생각합니다. POJO를 테스트 하므로 테스트 속도 및 난도가 낮지만 높은 안전성을 갖게 됩니다.
@Embeddable
@Getter
@NoArgsConstructor(access = AccessLevel.PROTECTED)
@ToString(of = {"first", "middle", "last"})
public class Name {
@NotEmpty
@Column(name = "first_name", length = 50)
private String first;
@Column(name = "middle_name", length = 50)
private String middle;
@NotEmpty
@Column(name = "last_name", length = 50)
private String last;
@Builder
public Name(final String first, final String middle, final String last) {
this.first = first;
this.middle = StringUtils.isEmpty(middle) ? null : middle;
this.last = last;
}
public String getFullName() {
if (this.middle == null) {
return String.format("%s %s", this.first, this.last);
}
return String.format("%s %s %s", this.first, this.middle, this.last);
}
}
Name
객체는Member
객체에서 사용하고 있습니다. 이처럼 Name 이라는 객체를Embeddable
으로 별도로 가지고 있으면 데이터의 응집력 재사용성이 높아집니다.- 예를 들어 주문 시 주문자 정보를 받아야 된다면
Order
라는 객체에도 동일하게Name
객체를 사용하면 재사용성이 높아집니다.
- 예를 들어 주문 시 주문자 정보를 받아야 된다면
Embeddable
객체에서도 다른 객체와 마찬가지로Name
관련된 기능을 충분히 제공해야 합니다.getFullName()
메서드 처럼first
,last
,middle
의 이름을 적절하게 조합해서 제공해줍니다.
public class NameTest {
@Test
public void getFullName_isFullName_ReturnFullName() {
final Name name = Name.builder()
.first("first")
.middle("middle")
.last("last")
.build();
final String fullName = name.getFullName();
assertThat(fullName, is("first middle last"));
}
@Test
public void getFullName_WithoutMiddle_ReturnMiddleNameIsNull() {
final Name name = Name.builder()
.first("first")
.middle("")
.last("last")
.build();
final String fullName = name.getFullName();
assertThat(fullName, is("first last"));
assertThat(name.getMiddle(), is(nullValue()));
}
@Test
public void getFullName_MiddleNameIsNull_ReturnMiddleNameIsNull() {
final Name name = Name.builder()
.first("first")
.middle("")
.last("last")
.build();
final String fullName = name.getFullName();
assertThat(fullName, is("first last"));
assertThat(name.getMiddle(), is(nullValue()));
}
}
entity
,Embeddable
객체 등의 객체들도 반드시 테스트 코드를 작성해야합니다.- middle값이 비어있을 경우 null로 잘 들어가는지,
getFullName()
메서드가 잘 동작하는지 테스트합니다.
각자의 프로젝트 환경이 다르기 때문에 어느 한 방법이 Best Practice라고 말하는 게 어렵습니다. 그래도 테스트 코드의 중요성은 이미 많은 개발자가 공감하고 있는 만큼 보다 효율적인 테스트 코드 환경을 구축하려는 노력이 많이 선행되어야 한다고 생각합니다.