JPA + Querydsl group_concat 사용법

해당 코드는 Github에서 확인할 수 있습니다.

JPA + Querydsl group_concat 사용법

Querydsl 기반으로 작업하다 보면 sql 함수가 필요한 경우가 있습니다. 대표적으로 sum(), max() 등이 있고 해당 함수는 Querydsl 자체에서 지원해 주고 있습니다. 하지만 group_concat과 같은 함수를 사용하기 위해서는 별도의 설정이 필요합니다. 본 포스팅의 내용은 Mysql 환경에서 JPA + Querydsl 조합에서 group_concat을 사용하는 방법을 정리한 것입니다.

Mysql 환경에서 group_concat을 사용하기 위해서 별다른 설정 없이 Expressions.stringTemplate() 함수를 이용해서 group_concat사용하는 경우 아래와 같은 에러가 발생합니다.

1
java.lang.IllegalArgumentException: No pattern found for GROUP_CONCAT

SQLExpressions.groupConcat을 사용해서 group_concat을 사용하는 방법도 있지만 해당 설정은 의존성과, 설정이 조금 복잡하기 때문에 Expressions.stringTemplate 기반으로 group_concat을 사용하겠습니다.

설정 방법

1
2
3
4
5
6
7
8
9
10
package com.example.querydsl.config
    ...
    class MysqlCustomDialect : MySQL57Dialect() {
        init {
            registerFunction(
                "GROUP_CONCAT",
                StandardSQLFunction("group_concat", StandardBasicTypes.STRING)
            )
        }
    }

MySQL57Dialect을 상속받는 커스텀한 Dialect클래스를 작성하고 registerFunction()메서드를 통해서 group_concat SQL Function을 등록합니다.

1
2
3
4
5
6
spring:
      jpa:
        ...
        show-sql: true
        database-platform: com.example.querydsl.config.MysqlCustomDialect
        ...

application.myl or application.properties 설정에서 위에서 작성한 MysqlCustomDialect 패키지 경로를 입력합니다.

테스트

  • 위와 같은 관계에서 테스트를 진행하겠습니다.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
data class MemberGroupConcat @QueryProjection constructor(
        val usernameGroupConcat: String,
        val ageGroupConcat: String
    )
    @ActiveProfiles("mysql")
    @Transactional
    internal class GroupConcatTest(
        private val em: EntityManager
    ) : SpringBootTestSupport() {
        @Test
        internal fun `group concat test`() {
            //given
            val teamA = Team("teamA")
            em.persist(teamA)
            (1..20).map {
                em.persist(Member(username = "member-$it", age = it, team = teamA))
            }
            //when
            val members =
                query
                    .select(QMemberGroupConcat(
                        Expressions.stringTemplate("group_concat({0})", qMember.username),
                        Expressions.stringTemplate("group_concat({0})", qMember.age)
                    ))
                    .from(qMember)
                    .groupBy(qMember.team)
                    .fetch()
            for (member in members) {
                println(member) // MemberGroupConcat(usernameGroupConcat=member-1,member-2,member-3,member-4,member-5,member-6,member-7,member-8,member-9,member-10,member-11,member-12,member-13,member-14,member-15,member-16,member-17,member-18,member-19,member-20, ageGroupConcat=1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20)
            }
        }
    }

teamA를 하나 생성하고 member-1 ~ member-20까지 member를 생성하고 teamA에 연결합니다. 리턴 받는 객체는 MemberGroupConcat으로 받게 설정합니다. Projection을 하는 여러 가지 방법이 있지만 @QueryProjection을 이용한 방법을 선호합니다. 해당 포스팅은 Querydsl Projection 방법 소개 및 선호하는 패턴 정리에서 정리한 적 있습니다.

의도한 것처럼 group_concat이 잘 동작하는 것을 확인할 수 있습니다.

주의점

group_concat사용할 때 max length를 주의해서 사용해야 합니다. group_concat max length size가 있고 그것을 넘어가면 문자열을 더 가져오지 않습니다.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
@Test
    internal fun `group concat max length size`() {
        //given
        val teamA = Team("teamA")
        em.persist(teamA)
        (1..1000).map {
            em.persist(Member(username = "member-$it", age = it, team = teamA))
        }
        //when
        val members =
            query
                .select(QMemberGroupConcat(
                    Expressions.stringTemplate("group_concat({0})", qMember.username),
                    Expressions.stringTemplate("group_concat({0})", qMember.age)
                ))
                .from(qMember)
                .groupBy(qMember.team)
                .fetch()
        for (member in members) {
            println(member) // MemberGroupConcat(usernameGroupConcat=member-1,member-2,member-3,...member-101,member-102,member-103, ageGroupConcat=1,2,3,,281,282,283,)
        }
    }

member를 1000개 저장하고 group_concat을 사용해서 print()을 하면 전체 결과가 나오지 않고 일정 length에서 잘리는 것을 알 수 있습니다.

1
show variables like 'group_concat_max_len';

위 명령어로 group_concat max length를 확인할 수 있습니다. 물론 해당 max length를 늘려서 사용할 수 있습니다.

정리

group_concat을 사용하는 방법을 정리하긴 했지만 가능하면 사용하는 것을 권장하지 않습니다. JPA 연관관계를 통해서 가져오는 것이 더 선호하며 group_concat 정도는 괜찮지만 SQL에 함수에 너무 의존적인 코드를 작성하는 것은 지양하는 것이 좋다고 생각합니다.


Popit은 페이스북 댓글만 사용하고 있습니다. 페이스북 로그인 후 글을 보시면 댓글이 나타납니다.