인프런 영문 브랜드 로고
인프런 영문 브랜드 로고

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

김기현님의 프로필 이미지

작성한 질문수

DGS: NETFLIX가 스프링 부트로 대용량 트래픽을 처리하는 방법

Query DataFetcher

LazyInitializationException JPA 오류 발생

작성

·

38

0

안녕하세요. Burger님 Netflix-DGS 강의 잘 보고 있습니다. GraphQL 사용 시, Lazy 로딩 오류가 발생하는데요. AllOpen을 사용해서 Hibernate가 Kotlin에서 프록시객체를 생성할 수 있게 열어뒀음에도 불구하고 LazyInitializationException이 발생하고 있습니다.
원인을 모르겠어서 질문 남깁니다.
FetchType을 EAGER로 바꾸면 사실 단일 조회는 가능하나. 단순히 EAGER로 바꿔서 정상적으로 조회를 한다해도.. 이 방법이면 성능상 문제가 될 수 있는 가능성이 있을 것 같아서 LAZY 로딩으로 해결하고 싶습니다.

답변 2

0

Burger님의 프로필 이미지
Burger
지식공유자

안녕하세요 김기현님!

 

강의 수강해주셔서 감사합니다.

혹시 어떤 쿼리, 어떤 필드를 요청했을 때 났는지 알 수 있을까요?

그리고 엔티티 설정 상태나 데이터페처 코드, 스프링 부트 버전도 알 수 있으면 좋을것 같습니다!

김기현님의 프로필 이미지
김기현
질문자

깃허브를 드리는게 나을 듯 하여 깃허브로 공유드려요!
깃허브 링크
DgsQuery 조회 시 발생되며, 저는 h2 Database를 사용하지 않고
MySQL를 사용하였습니다.

@Entity

class Movie {

@Id @GeneratedValue(strategy = GenerationType.IDENTITY)

val id: Long? = null

@Column(nullable = false)

val title: String? = null

@Column(nullable = false)

val releaseDate: LocalDate? = null

@ManyToOne(fetch = FetchType.LAZY, optional = false)

@JoinColumn(name = "director_id")

val director: Director? = null

@OneToMany(mappedBy = "movie")

val reviews: List<Review> = emptyList()

}

 

@Entity

@Table(name = "director")

class Director {

@Id @GeneratedValue(strategy = GenerationType.IDENTITY)

val id: Long? = null

@Column(nullable = false)

val name: String? = null

@OneToMany(mappedBy = "director")

val movies: List<Movie> = emptyList()

}

 

@DgsComponent

class MovieDataFetcher(

private val movieRepository: MovieRepository

) {

@DgsQuery(field = "movies")

fun getMovies(): MutableList<Movie> {

return movieRepository.findAll()

}

}

 

위 처럼 입력하였으며, 선생님의 수업내용을 따라하였습니다.
Spring Boot 버전은 3.4.3 버전입니다.

김기현님의 프로필 이미지
김기현
질문자

추가로 조회한 쿼리는
{
movies{
id,
title,
releaseDate,
director{
id,
name
}
}
}

입니다!

김기현님의 프로필 이미지
김기현
질문자

해결하였습니다~!


application.yml 설정에서
spring.jpa.open-in-view 설정을 false에서 true로 설정하여 해결하였습니다.

OSIV(Open Session I View)
스프링의 경우 보통 비즈니스 로직(Service Layer)에서 트랜잭션을 설정하고(@Transactional) 이 트랜잭션이 유지가 되는 영속성 컨텍스트의 범위는 뷰 렌더링까지는 되지는 않는다는 것을 알았고, Controller에서 Entity를 조회할 때 Lazy loading이 되어있어서 LazyInitializationException의 예외가 발생하게 되었습니다.

다음 게시물을 참고해서 해결하였습니다. https://zzang9ha.tistory.com/406

Burger님의 프로필 이미지
Burger
지식공유자

안녕하세요 기현님!

 

오전에 정신이 없어서 이제 확인했는데, 잘 해결되셔서 다행입니다 ㅎㅎ

도움 일찍 못드려 죄송하고, 혹시 진행하시면서 추가로 문제가 발생하면 언제든 문의 남겨주세요.

 

감사합니다!

0

인프런 AI 인턴님의 프로필 이미지

안녕하세요, 인프런 AI 인턴입니다. 김기현님.

LazyInitializationException은 주로 JPA 엔티티를 정보가 필요한 시점에서 ‘지연 로딩’ 방식으로 가져올 때, 해당 엔티티가 더 이상 활성화된 세션이나 트랜잭션 내에 존재하지 않는 경우 발생하는 오류입니다. 이 문제는 특히 GraphQL 리졸버나 컨트롤러 메소드에서 발생할 가능성이 큽니다.

이 문제를 해결하기 위한 몇 가지 접근 방법을 제안드리겠습니다:

  1. Transaction 관리: 엔티티를 접근해야 하는 메소드나 리졸버에 @Transactional 어노테이션을 추가해보세요. 이는 Hibernate 세션이 열린 상태에서 엔티티를 로딩하고 접근할 수 있게 해 줍니다.

  2. DTO 사용: 필요한 데이터만 담고 있는 DTO(Data Transfer Object) 클래스를 사용하는 방법도 있습니다. DTO에 데이터를 추출해서 서비스 계층에서 각 필드로 설정해주면 프록시 객체를 초기화할 필요 없이 안전하게 사용할 수 있습니다.

  3. Join Fetch 사용: JPQL(Query Language)을 사용하여 Fetch Join을 통해 필요한 관계 엔티티를 즉시 로딩하도록 설정할 수 있습니다. 하지만 이 방법은 특수한 경우에만 사용하는 것이 좋습니다.

   @Query("SELECT u FROM User u JOIN FETCH u.roles WHERE u.id = :id")
   Optional<User> findUserWithRoles(Long id);

위의 접근 방법을 통해 Lazy 로딩과 관련된 문제를 해결할 수 있기를 바랍니다. 추가로 아래의 유사한 질문과 답변에서도 해결 방법을 참조하실 수 있습니다:

저는 질문자님의 이해를 돕고자 지식공유자님의 기존 답변을 참고하여 유사한 답변 링크를 추천해드리고 있습니다.
현재 베타 기능으로 답변이 만족스럽지 않을 수 있는 점 양해 부탁드립니다. 🙏
추가적으로 궁금한 점이 있으시면, 이어서 질문해 주세요. 곧 지식공유자께서 답변해 주실 것입니다.