강의

멘토링

커뮤니티

인프런 커뮤니티 질문&답변

손준수님의 프로필 이미지
손준수

작성한 질문수

실전! 스프링 부트와 JPA 활용1 - 웹 애플리케이션 개발

단위 테스트 작성 시 트랜젝션, 영속성 컨덱스트 문제로 질문이 있습니다.

해결된 질문

작성

·

145

0

안녕하세요.  강의 수강 후 실무 작성 중 JUNIT TEST 작성 중 애로사항이 생겨서 질문 남깁니다.

상황은 이렇습니다.

API 통신이며 JUNIT TEST에서 API 호출 시는 mock로 호출하고 있습니다.

A객체와 C객체는 N:M 관계며 강의에 있는대로 A객체와 B객체가 1:N, C객체와 B객체와 1:N 식으로 연관관계가 주어져있습니다.

A객체 저장 API 호출 시 A-C 연결을 담은 B객체도 같이 저장되며 B객체만 독립적으로 저장하는 API는 없습니다. 

A객체 아이디와 C객체 아이디로 B객체를 찾아내서 B객체의 특정 정보를 업데이트하는 API의 JUNIT TEST를 작성하고 있습니다.

해당 API를 직접 호출하면 문제없이 작동합니다.

하지만 JUNIT TEST에서 A객체 저장 API를 호출해서 A객체와 B객체를 저장하고 테스트할 API를 호출하면 백앤드 로직에서 A객체 아이디와 C객체 아이디로 조회해도 B객체가 조회되지 않습니다.  개인적인 추측으로 트랜젝션이 종료되지 않은 상황이라  B객체를 JPQL로 조회하기 떄문에 영속성 컨덱스트가 아니라 아직 반영되지 않은 DB를 조회해서 그런듯 싶은데 em.flush()로 플리싱을 해도 특별히 변화가 없네요. 

이런 경우는 어떻게 해야할까요?

소스 스크린샷 첨부하며 addCall이 A객체 저장 api며 마지막에 실행하는게 테스트할 api입니다. 

감사합니다.

답변 1

0

김영한님의 프로필 이미지
김영한
지식공유자

안녕하세요. 준수님

아마도 JUnit에 설정한 트랜잭션이 mock.perform으로 호출하는 부분까지 영향을 주지는 못할꺼에요.

따라서 JUnit에서 사용하는 트랜잭션과 mock.perform으로 스프링을 호출해서 사용하는 트랜잭션이 서로 다릅니다.

이렇게 되면 트랜잭션 자체가 다르기 때문에, 첫번째 트랜잭션이 데이터를 저장하고 커밋하기 전까지, 두번째 트랜잭션은 해당 데이터를 읽을 수 없습니다.

따라서 이 테스트는 다음과 같이 해결할 수 있습니다.

1. 데이터를 저장하는 요청도 mock.perform으로 수행한다. 그리고 해당 수행이 끝나면 조회한다.

2. 테스트 케이스에 @Tx를 사용하지 않고, 데이터를 저장하고 커밋까지 한 다음에, mock.perform을 통해서 조회한다.

물론 이 경우 롤백은 데이터를 직접 삭제하는 코드를 넣어주셔야 합니다.

감사합니다.

손준수님의 프로필 이미지
손준수
질문자

아;;; 문제 해결되었습니다. 굉장히 단순한 문제였고 트랜젝션이나 영속성 컨덱스트 같은것과는 관계가 없습니다.  답변 감사합니다. 

손준수님의 프로필 이미지
손준수

작성한 질문수

질문하기