반응형
Notice
Recent Posts
Recent Comments
Link
«   2025/01   »
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
Archives
Today
Total
01-22 13:27
관리 메뉴

ImJay

[Java Spring] 1-12. 회원 서비스 테스트 본문

Java Spring/스프링 입문

[Java Spring] 1-12. 회원 서비스 테스트

ImJay 2023. 1. 30. 13:05
반응형

[Java Spring] 1-12. 회원 서비스 테스트


1. 로직을 개발했으면 잘 작동하는지 이제는 당연히 테스트해봐야겠다.

 

2. 기존에는 테스트 전용 패키지를 직접 생성해주었다면, 훨씬 간단한 방법이 존재한다.

 

3. 개발한 로직 클래스(MemberService) 에 Ctrl + Shift + T 를 누른다.

 

4. 라이브러리는 JUnit5 를 선택하고, 테스트 메서드는 전부 선택한다.

 

5.  테스트 케이스 틀이 자동으로 생성되는 모습이다.

 

6. 회원가입(join) 테스트케이스 코드 추가

package hello.hellospring.service;

import hello.hellospring.domain.Member;
import org.assertj.core.api.Assertions;
import org.junit.jupiter.api.Test;

import static org.assertj.core.api.Assertions.assertThat;
import static org.junit.jupiter.api.Assertions.*;

class MemberServiceTest {

    MemberService memberService = new MemberService();

    @Test
    void 회원가입() {
        //given
        Member member = new Member();
        member.setName("hello");

        //when
        Long saveId = memberService.join(member);

        //then
        Member findMember = memberService.findOne(saveId).get();
        assertThat(member.getName()).isEqualTo(findMember.getName());
    }

    @Test
    void findMembers() {
    }

    @Test
    void findOne() {
    }
}

- 외국계 회사와 일하는게 아니라면, 테스트 메서드는 직관적으로 한글로 작성해도 된다. 유일하게 한글을 사용해도 되는 네이밍이다.

- 테스트 케이스 작성 기법 : given, when, then

  • given : 무언가 주어졌을 때
  • when : 이것을 실행했을 때
  • then : 결과가 이게 나와야된다!

위 틀을 가지고 하면서 상황에 따라 점점 변형해가면 좋다.

 

7. 그러나 사실 join 에서의 핵심은 중복 회원 가입 로직의 예외처리가 정상적으로 작동하는지 확인하는 것이다.

따라서, 중복 회원 가입 코드 테스트 케이스를 추가한다.

    @Test
    public void 중복_회원_예외() {
        //given
        Member member1 = new Member();
        member1.setName("spring");

        Member member2 = new Member();
        member2.setName("spring");

        //when
        memberService.join(member1);
        try {
            memberService.join(member2);
            fail();
        } catch (IllegalStateException e) {
            assertThat(e.getMessage()).isEqualTo("이미 존재하는 회원입니다.");
        }

        //then
    }

- fail() 은 실패를 알리는 메서드이다.

 

8. 근데 이런 걸로(?) try catch 를 사용하기엔 애매하다. 그래서 좋은 문법을 제공한다.

    @Test
    public void 중복_회원_예외() {
        //given
        Member member1 = new Member();
        member1.setName("spring");

        Member member2 = new Member();
        member2.setName("spring");

        //when
        memberService.join(member1);
        assertThrows(IllegalStateException.class, () -> memberService.join(member2));

//        try {
//            memberService.join(member2);
//            fail();
//        } catch (IllegalStateException e) {
//            assertThat(e.getMessage()).isEqualTo("이미 존재하는 회원입니다.");
//        }

        //then
    }

- assertThrows 문법을 통해 예외처리를 확인할 수 있다.

 

9. 그렇다면 메세지는 어떻게 검증할까요? 

9-1. assertThrows 에 커서를 대고 Ctrl + Alt + V

    @Test
    public void 중복_회원_예외() {
        //given
        Member member1 = new Member();
        member1.setName("spring");

        Member member2 = new Member();
        member2.setName("spring");

        //when
        memberService.join(member1);
        IllegalStateException e = assertThrows(IllegalStateException.class, () -> memberService.join(member2));

        assertThat(e.getMessage()).isEqualTo("이미 존재하는 회원입니다.");

//        try {
//            memberService.join(member2);
//            fail();
//        } catch (IllegalStateException e) {
//            assertThat(e.getMessage()).isEqualTo("이미 존재하는 회원입니다.");
//        }

        //then
    }

 

10. 또 DB에 중복저장을 피하기 위해 초기화를 진행해주어야 한다.

    MemberService memberService = new MemberService();
    MemoryMemberRepository memberRepository = new MemoryMemberRepository();

    @AfterEach
    public void afterEach() {
        memberRepository.clearStore();
    }

- 기존에 테스트케이스에 사용했던 방식을 그대로 사용해준다!

- Ctrl + R 을 누르면 바로 이전에 실행했던 빌드를 그대로 실행해준다. ( 매우 유용 )

 

11. 근데 여기서 문제가 되는 부분은 다음과 같다.

- MemberService 클래스에 존재하는 MemoryMemberRepository 와

MemberServiceTest 클래스에 존재하는 MemoryMemberRepository 는 서로 다른 인스턴스이다.

- 지금 당장은 MemoryMemberRepository 의 멤버변수들이 static 이기 때문에 문제상황이 발생하지 않지만,

만약 멤버변수가 static 이 아니라면?

- 바로 다른 DB가 돼버리면서 문제가 발생한다.

- 그뿐만 아니라 결론적으로 지금 같은 레포지토리로 테스트가 되어야하는데 서로 다른 페로지토리로 테스트가 진행되고 있다. 

- 해결 방법(같은 레포지토리를 사용하게 하는 방법)은 다음과 같다.

public class MemberService {

    private final MemberRepository memberRepository;

    public MemberService(MemberRepository memberRepository) {
        this.memberRepository = memberRepository;
    }

- MemberService 클래스를 위와 같이 수정한다. ( 생성자 단축키 : ALT + INSERT )

 

class MemberServiceTest {

    MemberService memberService;
    MemoryMemberRepository memberRepository;

    @BeforeEach
    public void beforeEach() {
        memberRepository = new MemoryMemberRepository();
        memberService = new MemberService(memberRepository);
    }

- MemberServiceTest 클래스도 변수 초기화를 수정하고, beforeEach 메서드를 추가한다.

 

- MemberService 입장에서 보면, 직접 유효하지 않고 리포지토리를 외부에서 직접 넣어준다.

이러한 방식을 Dependency Injection 이라고 한다. 

반응형
Comments