해당 코드는 Github 를 확인해주세요.
소개
스프링은 다양한 테스트 전략을 제공하고 있습니다. 대표적으로 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 처리가 안되는 테스트 진행을 하기 어려움
Code
IntegrationTest
1 2 3 4 5 6 7 8 9 10 11 @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
어노테이션을 추가합니다.
Test Code
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 public class MemberApiTest extends IntegrationTest { @Autowired private MemberSetup memberSetup; @Test public void 회원가입_성공() throws Exception { final Member member = MemberBuilder.build(); final Email email = member.getEmail(); final Name name = member.getName(); final SignUpRequest dto = SignUpRequestBuilder.build(email, name); final ResultActions resultActions = requestSignUp(dto); 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 { final Member member = memberSetup.save(); final ResultActions resultActions = requestGetMember(member.getId()); 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 라이브러리에 대한 학습 비용이 발생합니다.
Code
MockTest
1 2 3 4 5 6 @RunWith (MockitoJUnitRunner.class)@ActiveProfiles (TestProfile.TEST)@Ignore public class MockTest {}
주로 Service 영역을 테스트 합니다.
MockitoJUnitRunner
을 통해서 Mock 테스트를 진행합니다.
Test Code
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 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 회원가입_성공() { 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); final Member signUpMember = memberSignUpService.doSignUp(dto); 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 회원가입_이메일중복_경우() { final Email email = member.getEmail(); final Name name = member.getName(); final SignUpRequest dto = SignUpRequestBuilder.build(email, name); given(memberRepository.existsByEmail(any())).willReturn(true ); memberSignUpService.doSignUp(dto); } }
MockTest
객체를 상속받아 테스트의 일관성을 갖습니다.
회원가입_성공
테스트는 오직 회원 가입에대한 단위 테스트만 진행합니다.
existsByEmail
을 모킹해서 해당 이메일이 중복되지 않았다는 가정을 합니다.
then
에서는 회원 객체가 해당 비즈니스 요구사항에 맞게 생성됬는지를 검사합니다.
실제 데이터베이스에 Insert 됬는지 여부는 해당 테스트의 관심사가 아닙니다.
회원가입_이메일중복_경우
테스트는 회원가입시 이메일이 중복됬는지 여부를 확인합니다.
existsByEmail
을 모킹해서 이메일이 중복됬다는 가정을 합니다.
expected
으로 이메일이 중복되었을 경우 EmailDuplicateException
예외가 발생하는지 확인합니다.
해당 이메일이 데이터베이스에 실제로 있어서 예외가 발생하는지는 관심사가 아닙니다. 작성한 코드가 제대로 동작 여부만이 해당 테스트의 관심사 입니다.
오직 테스트의 관심사만 테스트를 진행하기 때문에 예외 발생시 디버깅 작업도 명확해집니다.
외부 의존도가 낮기 때문에 테스트 하고자하는 부분만 명확하게 테스트가 가능합니다.
이것은 단점이기도 합니다. 해당 테스트만 진행하지 외부 의존을 갖는 코드까지 테스트하지 않으니 실제 환경에서 제대로 동작하지 않을 가능성이 있습니다. 외부 의존에대한 테스트는 통합 테스트에서 진행합니다.
Mock API 테스트
장점
Mock 테스트와 장점은 거의 같습니다.
WebApplication
관련된 Bean들만 등록하기 때문에 통합 테스트보다 빠르게 테스트할 수 있습니다.
통합 테스트를 진행하기 어려운 테스트를 진행합니다.
외부 API 같은 Rollback 처리가 힘들거나 불가능한 테스트를 주로 사용합니다.
예를 들어 외부 결제 모듈 API를 콜하면 안 되는 케이스에서 주로 사용 할 수 있습니다.
이런 문제는 통합 테스트에서 해당 객체를 Mock 객체로 변경해서 테스트를 변경해서 테스트할 수도 있습니다.
단점
Mcok 테스트와 다점은 거의 같습니다.
요청부터 응답까지 모든 테스트를 Mock 기반으로 테스트하기 때문에 실제 환경에서는 제대로 동작하지 않을 가능성이 매우 큽니다.
Code
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 @WebMvcTest (MemberApi.class)public class MemberMockApiTest extends MockApiTest { @MockBean private MemberSignUpService memberSignUpService; @MockBean private MemberHelperService memberHelperService; ... @Test public void 회원가입_유효하지않은_입력값() throws Exception { 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); final ResultActions resultActions = requestSignUp(dto); resultActions .andExpect(status().isBadRequest()) ; }
@WebMvcTest(MemberApi.class)
어노테이션을 통해서 하고자하는 MemberApi
의 테스트를 진행합니다.
@MockBean
으로 객체를 주입받아 Mocking 작업을 진행합니다.
테스트의 관심사는 오직 Request와 그에 따른 Response 입니다.
Repository Test
장점
Repository
관련된 Bean들만 등록하기 때문에 통합 테스트에 비해서 빠릅니다.
Repository
에 대한 관심사만 갖기 때문에 테스트 범위가 작습니다.
단점
테스트 범위가 작기 때문에 실제 환경과 차이가 발생합니다.
Code
RepositoryTest
1 2 3 4 5 6 7 @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등의 커스텀하게 추가된 메서드를 테스트합니다.
Test Code
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 public class MemberRepositoryTest extends RepositoryTest { @Autowired private MemberRepository memberRepository; private Member saveMember; private Email email; @Before public void setUp () throws Exception { final String value = "cheese10yun@gmail.com" ; 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("ehdgoanfrhkqortntksdls@asd.com" )); assertThat(existsByEmail).isFalse(); } }
setUp()
메서드를 통해서 Member를 데이터베이스에 insert 합니다.
setUp()
메서드는 메번 테스트 코드가 실행되기전에 실행됩니다. 즉 테스트 코드 실핼 할 때마다 insert -> rollback이 자동으로 이루어집니다.
추가 작성한 쿼리 메서드 existsByEmail
을 테스트 진행합니다.
실제로 작성된 쿼리가 어떻게 출력되는지 show-sql
옵션을 통해서 확인 합니다. ORM은 SQL을 직접 장성하지 않으니 실제 쿼리가 어떻게 출력되는지 확인하는 습관을 반드시 가져야합니다.
POJO 테스트
설명
각 엔티티(Embeddable, Entity, 일반 POJO, 모든 객체) 객체들의 기능이 풍부해야 합니다. 객체 본인의 책임을 충분히 다하지 않고 있으면 다른 영역으로 그 객체의 책임이 넘어 가게됩니다. 예를 들어 Name
객체가 getFullName()
메서드를 제공해주지 않는다면 getFullName()
메서드를 만족시키는 메서드들이 다른 계층에서 구현하게 되고 어느 계층에서 어떻게 사용되고 있는지 모르기 때문에 누군가는 중복코드를 만들게 됩니다.
객체지향에서 본인의 책임(기능)은 본인 스스로가 제공해야 합니다. 특히 엔티티 객체들은 가장 핵심 객체이고 이 객체를 사용하는 계층들이 다양하게 분포되기 때문에 반드시 테스트 코드를 작성해야합니다.
장점
POJO 객체이므로 테스트하기 편합니다. 외부에서 주입 받을 의존성도 없고 Mocking할 대상도 없습니다.
엔티티 객체는 사용하는 계층이 많으므로 테스트의 효율성이 높습니다.
단점
단점은 없다고 생각합니다. POJO를 테스트 하므로 테스트 속도 및 난도가 낮지만 높은 안전성을 갖게 됩니다.
Code
Embeddable
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 @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
의 이름을 적절하게 조합해서 제공해줍니다.
Test Code
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 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라고 말하는 게 어렵습니다. 그래도 테스트 코드의 중요성은 이미 많은 개발자가 공감하고 있는 만큼 보다 효율적인 테스트 코드 환경을 구축하려는 노력이 많이 선행되어야 한다고 생각합니다.