- 'MemberService'를 테스트 해보는 시간으로 'MemberServiceTest'를 만들어 봅시다.
'Go to Test'로 단축키로 'create Test'를 할 수 있습니다.
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의 예외처리의 구문과 다른 구문을 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
'spring' 카테고리의 다른 글
6. DI_ 자바코드로 직접 스프링 빈 등록하기 (0) | 2021.03.27 |
---|---|
5. DI_ 컴포넌트 스캔과 자동 의존관계 설정 (0) | 2021.03.27 |
3. 서비스 개발 (0) | 2021.03.26 |
2. 테스트케이스 작성 (0) | 2021.03.26 |
1.회원관리예제_회원도메인, 리포지토리 만들기 (0) | 2021.03.24 |
댓글