본문 바로가기
spring

4.회원 서비스 개발

by Thumper 2021. 3. 26.
  • 'MemberService'를 테스트 해보는 시간으로 'MemberServiceTest'를 만들어 봅시다.

 

 'Go to Test'로 단축키로 'create Test'를 할 수 있습니다.

다음과 같이 체크(v)하여 만듭니다.

 

package hello.hellospring.Service;

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

//import org.assertj.core.api.Assertions.assertThat;


class MemberServiceTest {
   
    @Test
    void join() {
        }
        
    @Test
    void findMembers() {
    }

    @Test
    void findOne() {
    }
}

위의 코드처럼 '껍데기'와 같은 틀이 생성됩니다.

 

 

 


 

Q. 회원가입() 테스트 코드 만들기

위에서 만든 '껍데기'에 이제 코드를 만들어 봅시다.

 

 @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());  //assertj static import
    }

Q.한글로 함수명 적어도 되는지?

  • 테스트 코드는 한글로 적어도 된다. 
  • 영어권이 아니라면, 직관적으로 파악할 수 있어 좋다.
  • 빌드될 때, 실제 코드로 포함되지 않기 때문에 문제는 없다.

 

 

구현 패턴규칙

  • given : 무언가 주어졌을 때("~한 데이터가 기반인지?")
  • when : 이것을 실행하면 ("검증대상")
  • then : 이렇게 결과가 나와야 한다. ("결과")

join()함수_회원가입

hello 이름의 사람이 가입하고 나서, 가입확인을 검증합니다.

 

Assertions.assertThat(member.getName()).isEqualTo(findMember.getName());

assertj버전의 라이브러리로 가입확인 검증을 합니다.

static import옵셥을 통해 줄여 놓습니다.

 

 

 

 

 

 

1. 예외처리: try() catch()문

Q. 테스트는 예외 플로우가 훨씬 중요하다.

  • 테스트는 정상 플로우도 중요하지만 예외 플로우가 훨씬 중요합니다.
  • 정상검증을 잘 따져서 예외처리하는 게 중요합니다.

 

 

Q1. try() catch()문으로 처리한다.

 

package hello.hellospring.Service;

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

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

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
    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("error");
        } catch (IllegalStateException e) {
            assertThat(e.getMessage()).isEqualTo("이미 존재하는 회원입니다.");
        }
     
        //then
    }

    @Test
    void findMembers() {
    }

    @Test
    void findOne() {
    }
}
  • given : "spring" 이름의  member1, member2이 존재하고,
  • when : member1, member2가 가입할 때, 중복가입의 경우에는 IllegalStateException 예외가 터지게 만든다.

▶결과: 

정상동작

정상동작되어 에러가 안 터집니다.

 

 

 

 

 


 

 Q. 만약 에러가 터지면?

MemberService의 예외처리

검증할 때, 

코드가 MemberService의 예외처리의 구문과 다른 구문을 try() catch()구문에 넣었다면 당연히 에러가 터집니다.

 

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

 

 

에러 화면

위의 사진처럼,  IllegalStateException 예외구문이 기대 대상과 실제구문이  다르다고 알려줍니다.


 

 

 

 

 

 

 

 

 

2. 예외처리: assertThrows()문

try() catch' 방법은 "애매하다"

그래서 더 좋은 방법인 assertThrows()로 해봅시다.

 

 

 

Q. aseertThrows: 예외 유형의 유효성을 검사합니다

 

▶IllegalStateException경우

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

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

    //when
    memberService.join(member1);
    //junit.jupiter.api.Assertions.assertThrows; 버전일 때,
    assertThrows(IllegalStateException.class, () -> memberService.join(member2));
 }

member1가입 후, member1 이름과 같은 member2가 회원가입(join)을 할 때 예외가 터지는지 확인합니다.

결과 확인

▶NullPointerException경우

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

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

    //when
    memberService.join(member1);
    //junit.jupiter.api.Assertions.assertThrows; 버전일 때,
    assertThrows(NullPointerException.class, () -> memberService.join(member2));
 }

 


Q. 메시지는 어떻게 검증하는지?

@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("이미 존재하는 회원입니다.");
 }

 

assertThat()으로 메시지를 반환하여 받을 수 있다.

결과 확인

 

 

 

 

 

2. 'MemberServiceTest' 코드를 모두 실행한다면?

Q. 만약에 '회원가입'과 '중복_회원_예외' 테스트코드에 동일한 이름의 회원이 있다면?

package hello.hellospring.Service;

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

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

class MemberServiceTest {

    MemberService memberService = new MemberService();

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

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

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

    @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("이미 존재하는 회원입니다.");

        //then
    }

    @Test
    void findMembers() {
    }

    @Test
    void findOne() {
    }
}

 

 

에러가 터져버립니다...

 

테스트코드 때 했던 것처럼, '중복_회원_예외'함수에서 등록된 'spring'이름의 회원이 있기 때문에 예외가 터져버립니다.

 

 

 

 

 

Q. 그렇다면 또 clear()하면 될까요?

네. 그렇지만 MemberService밖에 없기 때문에 MemoryMemberRepository를 추가해주고 clear()해주면 됩니다.

class MemberServiceTest {

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

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

이렇게 하면 테스트코드 돌 때마다 메모리초기화 해주기 때문에 잘 동작합니다.

성공!

 

 

 

3. 여기서 애매한 점

MemberService, MemberServiceTest에 각각 만들어 놓은 MemberRepository객체가 있다. (서로 다른 인스턴스인 상황)

public class MemberService {
    private final MemberRepository memberRepository = new MemoryMemberRepository();

 

class MemberServiceTest {

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

▶이런 경우에는 하나로 만들어 같이 쓰는게 좋습니다.

 

 

MemoryMemberRepository를 보면 물론 static으로 되어 있어서 문제가 없다.

(static은 인스턴스와 상관없이 클래스에 붙기 때문에 지금은 크게 상관없다...)

public class MemoryMemberRepository implements MemberRepository {

    private static Map<Long, Member> store = new HashMap<>();
    private static long sequence = 0L;

 

(그런데...문제는)

'new MemoryMemberRepository;'처럼 new로 다른 객체(repository)가 생성이 되면 

다른 인스턴스이기 때문에 혹시라도 내용물이 달라질 수 있기 때문에 걱정인거죠...

 

 

Q.그래서 어떻게 바꾸나요?

>1. 먼저 Memberservice클래스 수정

 private final MemberRepository memberRepository;

    public MemberService(MemberRepository memberRepository) { //3
        this.memberRepository = memberRepository;
    }
  • new 생성자를 지우고 constructor를 만듭니다.
  • 변화 :기존에는 회원 서비스가 메모리 회원 리포지토리를 직접 생성하게 했지만, 외부에서 넣어줄 수 있게 바꿉니다.

>2. 먼저 MemberserviceTest클래스 수정

class MemberServiceTest {

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

 

>3. (동작원리)

각각의 테스트코드가 시작하기 전에  beforeEach()가 실행되게 합니다. (1~3)

  • MemoryMemberRepository()를 만들어  memberRepository에 넣어주고,
  • memberRepository를 member MemberService()에 넣어줍니다.
  • 넣어주면, MemberService클래스에 있는 constructor에 담겨집니다.

▶그렇게하면 같은 (두 개->하나로) memoryMemberRepository를 사용하게 됩니다.

 

 

 

 

 

 

 

 

 

4. DI방식: Dependecy Injection

 

스프링의 핵심 기능중 하나인 DI(Dependecy Injection)

DI는 말 그대로 "의존성을 주입"시켜준다-입니다. 객체를 직접 생성하는 게 아니라 외부에서 생성한 후 주입을 시켜주는 방식인데요.

예) MemberService입장에서는  (MemberServiceTest)외부에서 주입받았다



 

 

마무리

다음 시간에는 DI에 대해 집중적으로 알아봅시다.

 

 

참고자료

1. Junit - assertThrows, beforeEach

gmlwjd9405.github.io/2019/11/26/junit5-guide-basic.html

2. 스프링 DI

private.tistory.com/39

댓글