Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Documentation of why Kotlin JDSL returns a nullable list in the Spring Support module. #685

Merged
merged 3 commits into from
Apr 29, 2024
Merged
Show file tree
Hide file tree
Changes from 2 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
# Why does Kotlin JDSL allow nullable return types?

Kotlin JDSL helps you write queries in a type-safe way.
You may wonder if it also allows non-null data to be null when returning a bunch of data (e.g. `Slice<T?>` in `findSlice`, `Page<T?>` in `findPage`).
This article explains why the Kotlin JDSL allows nullable return types.

## Why do we need nullable return types?

The reason why Kotlin JDSL supports nullable return types is that depending on how the JPQL query is written, some of the returned list items may contain null values.
The simplest example is when you look up a column or entity without using DTO Projection.

The table below shows some of the cases where nullable return types can occur when using the Kotlin JDSL.

| Item | Nullable or not | Reason |
|----------------|-----------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------|
| DTO Projection | X | Calling the constructor for all ROWs did not result in the object being created to allow nulls.<br/>Fields in DTO can be null, but DTO object cannot be null. |
| Column | O | Exists when looking up a field that is null. |
| Entity | O | Exists if the Entity being joined on Left Join is null. |

As another example, the code below shows a situation where the Author entity exists, but the BookAuthor entity, the target of the left join, may be null.

```kotlin
val query = jpql {
select(
path(BookAuthor),
cj848 marked this conversation as resolved.
Show resolved Hide resolved
).from(
entity(Author::class),
leftJoin(BookAuthor::class).on(path(Author::authorId).equal(path(BookAuthor::authorId)))
)
}
```

## Background on the design decision

In the early development of Kotlin JDSL 3.0, we tried nullable inference.
This was an attempt to automatically infer whether a query result is nullable through the type system.
However, during the development of queries that use JOIN, we encountered the problem that perfect nullable inference was not possible.

Starting with Kotlin JDSL 3.0, we're settling on a path that utilizes the user's query writing knowledge as much as possible, with a small learning curve.
This avoids confusion due to different interfaces for different users, and allows them to build on their existing JPQL knowledge without having to learn Kotlin JDSL syntax separately.
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
# 왜 Kotlin JDSL은 Nullable한 반환 타입을 허용하나요?

Kotlin JDSL을 사용하면 Type Safe하게 쿼리를 작성할 수 있게 도와주지만
다건의 데이터를 반환할 때 non-null한 데이터 또한 null을 허용하는지 궁금증을 가질 수 있습니다. (예를 들어 `findSlice`의 `Slice<T?>`, `findPage`의 `Page<T?>`)
이 글은 Kotlin JDSL에서 nullable한 반환 타입을 허용하는 이유를 설명합니다.

## Nullable 반환 타입의 필요성

Kotlin JDSL이 nullable한 반환 타입을 지원하는 이유는 JPQL 쿼리의 작성 방법에 따라 반환되는 리스트 항목 중 null 값이 포함될 수 있기 때문입니다.
가장 간단한 예로는 DTO Projection을 사용하지 않고, column이나 entity를 조회할 때를 들 수 있습니다.

아래 표는 Kotlin JDSL을 사용할 때 nullable한 반환 타입이 발생할 수 있는 몇 가지 경우를 나타냅니다.

| 항목 | null 여부 | 이유 |
|----------------|---------|-------------------------------------------------------------------------------------------------------|
| DTO Projection | X | 모든 ROW에 대한 생성자 호출하기 때문에 객체가 생성되어 null을 허용하는 결과가 나오지 않음<br/>DTO의 필드가 null일 수는 있으나 DTO 객체가 null일 수는 없음 |
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
| DTO Projection | X | 모든 ROW에 대한 생성자 호출하기 때문에 객체가 생성되어 null을 허용하는 결과가 나오지 않음<br/>DTO의 필드가 null일 수는 있으나 DTO 객체가 null일 수는 없음 |
| DTO Projection | X | 모든 ROW에 대한 생성자를 호출하기 때문에 객체가 생성되어 null을 허용하는 결과가 나오지 않음<br/>DTO의 필드가 null일 수는 있으나 DTO 객체가 null일 수는 없음 |

i think this is correct.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thank you for the detailed review! I will edit it. ☺️

| Column | O | null인 필드를 조회하는 경우 존재 |
| Entity | O | Left Join시 조인의 대상이 되는 Entity가 null인 경우 존재 |

또 다른 예로 아래 코드는 Author 엔티티가 있지만, left join의 대상인 BookAuthor 엔티티는 null일 수 있는 상황을 보여줍니다.

```kotlin
val query = jpql {
select(
path(BookAuthor),
cj848 marked this conversation as resolved.
Show resolved Hide resolved
).from(
entity(Author::class),
leftJoin(BookAuthor::class).on(path(Author::authorId).equal(path(BookAuthor::authorId))),
)
}
```

## 설계 결정의 배경

초기 Kotlin JDSL 3.0 개발 시에는 nullable 추론을 시도했습니다.
이는 쿼리 결과의 nullable 여부를 타입 시스템을 통해 자동으로 추론하려는 시도였습니다.
하지만, join을 사용하는 쿼리의 개발 과정에서 완벽한 nullable 추론이 불가능하다는 문제에 직면했습니다.

Kotlin JDSL 3.0부터는 되도록 사용자가 알고 있는 쿼리 작성 지식만을 활용해서 적은 러닝 커브로 사용할 수 있게 하는 노선으로 정착합니다.
이는 사용자가 바라보는 인터페이스가 각자 달라 혼란을 방지하고, Kotlin JDSL 문법을 별도로 학습할 필요 없이 기존의 JPQL 지식을 기반으로 활용할 수 있게 합니다.
Loading