스프링 DB 1편(데이터 접근 핵심 원리)

Ch03. 트랜잭션 이해 - 트랜잭션(적용 전)

webmaster 2022. 4. 25. 16:07
728x90

MemberServiceV1

@RequiredArgsConstructor
public class MemberServiceV1 {
    private final MemberRepositoryV1 memberRepository;

    public void accountTransfer(String fromId, String toId, int money) throws SQLException {
        Member fromMember = memberRepository.findById(fromId);
        Member toMember = memberRepository.findById(toId);

        memberRepository.update(fromId, fromMember.getMoney() - money);
        validation(toMember);
        memberRepository.update(toId, toMember.getMoney() + money);
    }

    private void validation(Member toMember) {
        if( toMember.getMemberId().equals("ex")){
            throw new IllegalStateException("이체중 예외 발생");
        }
    }
}
  • formId 의 회원을 조회해서 toId 의 회원에게 money 만큼의 돈을 계좌이체 하는 로직이다.
    • fromId 회원의 돈을 money 만큼 감소한다. -> UPDATE SQL 실행 
    • toId 회원의 돈을 money 만큼 증가한다. -> UPDATE SQL 실행
  • 예외 상황을 테스트해보기 위해 toId 가 "ex" 인 경우 예외를 발생한다.

MemberServiceV1Test

/**
 * 기본 동작, 트랜잭션이 없어서 문제 발생
 */
class MemberServiceV1Test {

    public static final String MEMBER_A = "memberA";
    public static final String MEMBER_B = "memberB";
    public static final String MEMBER_EX = "ex";

    private MemberRepositoryV1 memberRepository;
    private MemberServiceV1 memberService;

    @BeforeEach
    public void before() {
        DriverManagerDataSource dataSource = new DriverManagerDataSource(URL, USERNAME, PASSWORD);
        memberRepository = new MemberRepositoryV1(dataSource);
        memberService = new MemberServiceV1(memberRepository);
    }

    @AfterEach
    public void after() throws SQLException {
        memberRepository.delete(MEMBER_A);
        memberRepository.delete(MEMBER_B);
        memberRepository.delete(MEMBER_EX);
    }

    @Test
    @DisplayName("정상 이체")
    public void accountTransfer() throws SQLException {
        //given
        Member memberA = new Member(MEMBER_A, 10000);
        Member memberB = new Member(MEMBER_B, 10000);
        memberRepository.save(memberA);
        memberRepository.save(memberB);

        //when
        memberService.accountTransfer(memberA.getMemberId(), memberB.getMemberId(), 2000);

        //then
        Member findMemberA = memberRepository.findById(memberA.getMemberId());
        Member findMemberB = memberRepository.findById(memberB.getMemberId());
        assertThat(findMemberA.getMoney()).isEqualTo(8000);
        assertThat(findMemberB.getMoney()).isEqualTo(12000);
    }

    @Test
    @DisplayName("이체중 예외 발생")
    public void accountTransferEx() throws SQLException {
        //given
        Member memberA = new Member(MEMBER_A, 10000);
        Member memberB = new Member(MEMBER_EX, 10000);
        memberRepository.save(memberA);
        memberRepository.save(memberB);

        //when
        assertThatThrownBy(
            () -> memberService.accountTransfer(memberA.getMemberId(), memberB.getMemberId(), 2000))
            .isInstanceOf(IllegalStateException.class);

        //then
        Member findMemberA = memberRepository.findById(memberA.getMemberId());
        Member findMemberB = memberRepository.findById(memberB.getMemberId());
        assertThat(findMemberA.getMoney()).isEqualTo(8000);
        assertThat(findMemberB.getMoney()).isEqualTo(10000);
    }
}

정상 이체

  • given: 다음 데이터를 저장해서 테스트를 준비한다.
    • memberA 10000원
    • memberB 10000원
  • when: 계좌이체 로직을 실행한다.
    • memberService.accountTransfer() 를 실행한다.
    • memberA memberB 로 2000원 계좌이체 한다.
      • memberA 의 금액이 2000원 감소한다.
      • memberB 의 금액이 2000원 증가한다.
  • then: 계좌이체가 정상 수행되었는지 검증한다.
    • memberA 8000원 = 2000원 감소
    • memberB 12000원 = 2000원 증가

테스트 데이터 제거

  • @BeforeEach : 각각의 테스트가 수행되기 전에 실행
  • @afterEach : 각각의 테스트가 실행되고 난 이후에 실행
  • 테스트 데이터를 제거하는 과정이 불편하지만, 다음 테스트에 영향을 주지 않으려면 테스트에서 사용한 데이터를 모두 제거해야 한다. 그렇지 않으면 이번 테스트에서 사용한 데이터 때문에 다음 테스트에서 데이터 중복으로 오류가 발생할 수 있다.
  • 테스트에서 사용한 데이터를 제거하는 더 나은 방법으로는 트랜잭션을 활용하면 된다. 테스트 전에 트랜잭션을 시작하고, 테스트 이후에 트랜잭션을 롤백해버리면 데이터가 처음 상태로 돌아온다. 이 방법은 이후에 설명하겠다.

이체 중 예외 발생

  • given: 다음 데이터를 저장해서 테스트를 준비한다.
    • memberA 10000원
    • memberEx 10000원
  • when: 계좌이체 로직을 실행한다.
    • memberService.accountTransfer() 를 실행한다.
    • memberA -> memberEx 로 2000원 계좌이체 한다.
      • memberA 의 금액이 2000원 감소한다.
      • memberEx 회원의 ID는 ex 이므로 중간에 예외가 발생한다. 
  • then: 계좌이체는 실패한다. memberA 의 돈만 2000원 줄어든다.
    • memberA 8000원 - 2000원 감소
    • memberB 10000원 - 중간에 실패로 로직이 수행되지 않았다. 따라서 그대로 10000원으로 남아있게 된다

 

728x90