diff --git a/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/api/query/MemberGroupQueryApi.java b/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/api/query/MemberGroupQueryApi.java index 9d4efaebc..f65919c37 100644 --- a/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/api/query/MemberGroupQueryApi.java +++ b/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/api/query/MemberGroupQueryApi.java @@ -10,7 +10,7 @@ import org.springframework.http.ResponseEntity; import site.timecapsulearchive.core.domain.group.data.response.GroupMemberInfosResponse; import site.timecapsulearchive.core.domain.member_group.data.response.GroupReceivingInvitesSliceResponse; -import site.timecapsulearchive.core.domain.member_group.data.response.GroupSendingInvitesResponse; +import site.timecapsulearchive.core.domain.member_group.data.response.GroupSendingInvitesSliceResponse; import site.timecapsulearchive.core.global.common.response.ApiSpec; public interface MemberGroupQueryApi { @@ -69,10 +69,16 @@ ResponseEntity> findGroupMemberInfos( description = "ok" ) }) - ResponseEntity> findGroupSendingInvites( + ResponseEntity> findGroupSendingInvites( Long memberId, @Parameter(in = ParameterIn.PATH, description = "그룹 아이디", required = true) - Long groupId + Long groupId, + + @Parameter(in = ParameterIn.QUERY, description = "마지막 그룹 초대 아이디") + Long groupInviteId, + + @Parameter(in = ParameterIn.QUERY, description = "페이지 크기", required = true) + int size ); } diff --git a/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/api/query/MemberGroupQueryApiController.java b/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/api/query/MemberGroupQueryApiController.java index ba7eb7371..b036638c7 100644 --- a/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/api/query/MemberGroupQueryApiController.java +++ b/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/api/query/MemberGroupQueryApiController.java @@ -15,8 +15,9 @@ import site.timecapsulearchive.core.domain.group.data.response.GroupMemberInfosResponse; import site.timecapsulearchive.core.domain.member_group.data.dto.GroupInviteSummaryDto; import site.timecapsulearchive.core.domain.member_group.data.dto.GroupSendingInviteMemberDto; +import site.timecapsulearchive.core.domain.member_group.data.dto.GroupSendingInvitesSliceRequestDto; import site.timecapsulearchive.core.domain.member_group.data.response.GroupReceivingInvitesSliceResponse; -import site.timecapsulearchive.core.domain.member_group.data.response.GroupSendingInvitesResponse; +import site.timecapsulearchive.core.domain.member_group.data.response.GroupSendingInvitesSliceResponse; import site.timecapsulearchive.core.domain.member_group.service.MemberGroupQueryService; import site.timecapsulearchive.core.global.common.response.ApiSpec; import site.timecapsulearchive.core.global.common.response.SuccessCode; @@ -84,17 +85,25 @@ public ResponseEntity> findGroupMemberInfos( produces = {"application/json"} ) @Override - public ResponseEntity> findGroupSendingInvites( + public ResponseEntity> findGroupSendingInvites( @AuthenticationPrincipal final Long memberId, - @PathVariable(value = "group_id") final Long groupId + @PathVariable(value = "group_id") final Long groupId, + @RequestParam(value = "group_invite_id", required = false) final Long groupInviteId, + @RequestParam(value = "size") final int size ) { - List groupSendingInvites = memberGroupQueryService.findGroupSendingInvites( - memberId, groupId); + GroupSendingInvitesSliceRequestDto dto = GroupSendingInvitesSliceRequestDto.create( + memberId, groupId, groupInviteId, size); + + Slice groupSendingInvitesSlice = memberGroupQueryService.findGroupSendingInvites( + dto); return ResponseEntity.ok( ApiSpec.success( SuccessCode.SUCCESS, - GroupSendingInvitesResponse.createOf(groupSendingInvites) + GroupSendingInvitesSliceResponse.createOf( + groupSendingInvitesSlice.getContent(), + groupSendingInvitesSlice.hasNext() + ) ) ); } diff --git a/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/data/dto/GroupSendingInvitesSliceRequestDto.java b/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/data/dto/GroupSendingInvitesSliceRequestDto.java new file mode 100644 index 000000000..823e958b5 --- /dev/null +++ b/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/data/dto/GroupSendingInvitesSliceRequestDto.java @@ -0,0 +1,18 @@ +package site.timecapsulearchive.core.domain.member_group.data.dto; + +public record GroupSendingInvitesSliceRequestDto( + Long memberId, + Long groupId, + Long groupInviteId, + int size +) { + + public static GroupSendingInvitesSliceRequestDto create( + final Long memberId, + final Long groupId, + final Long groupInviteId, + final int size + ) { + return new GroupSendingInvitesSliceRequestDto(memberId, groupId, groupInviteId, size); + } +} diff --git a/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/data/response/GroupSendingInvitesResponse.java b/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/data/response/GroupSendingInvitesSliceResponse.java similarity index 62% rename from backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/data/response/GroupSendingInvitesResponse.java rename to backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/data/response/GroupSendingInvitesSliceResponse.java index 4be590970..24b231bbe 100644 --- a/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/data/response/GroupSendingInvitesResponse.java +++ b/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/data/response/GroupSendingInvitesSliceResponse.java @@ -5,19 +5,23 @@ import site.timecapsulearchive.core.domain.member_group.data.dto.GroupSendingInviteMemberDto; @Schema(description = "그룹 초대 보낸 목록") -public record GroupSendingInvitesResponse( +public record GroupSendingInvitesSliceResponse( @Schema(description = "초대 보낸 그룹원 정보 리스트") - List responses + List groupSendingInviteMembers, + + @Schema(description = "다음 페이지 유무") + boolean hasNext ) { - public static GroupSendingInvitesResponse createOf( - final List groupSendingInviteMemberDtos + public static GroupSendingInvitesSliceResponse createOf( + final List groupSendingInviteMemberDtos, + final boolean hasNext ) { List groupSendingInviteMemberResponses = groupSendingInviteMemberDtos.stream() .map(GroupSendingInviteMemberDto::toResponse) .toList(); - return new GroupSendingInvitesResponse(groupSendingInviteMemberResponses); + return new GroupSendingInvitesSliceResponse(groupSendingInviteMemberResponses, hasNext); } } diff --git a/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/repository/group_invite_repository/GroupInviteQueryRepository.java b/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/repository/group_invite_repository/GroupInviteQueryRepository.java index c9c5c3166..64e878abc 100644 --- a/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/repository/group_invite_repository/GroupInviteQueryRepository.java +++ b/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/repository/group_invite_repository/GroupInviteQueryRepository.java @@ -5,6 +5,7 @@ import org.springframework.data.domain.Slice; import site.timecapsulearchive.core.domain.member_group.data.dto.GroupInviteSummaryDto; import site.timecapsulearchive.core.domain.member_group.data.dto.GroupSendingInviteMemberDto; +import site.timecapsulearchive.core.domain.member_group.data.dto.GroupSendingInvitesSliceRequestDto; public interface GroupInviteQueryRepository { @@ -18,8 +19,7 @@ Slice findGroupReceivingInvitesSlice( final ZonedDateTime createdAt ); - List findGroupSendingInvites( - final Long memberId, - final Long groupId + Slice findGroupSendingInvites( + final GroupSendingInvitesSliceRequestDto dto ); } diff --git a/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/repository/group_invite_repository/GroupInviteQueryRepositoryImpl.java b/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/repository/group_invite_repository/GroupInviteQueryRepositoryImpl.java index 7f2b33ec6..831b4e8e3 100644 --- a/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/repository/group_invite_repository/GroupInviteQueryRepositoryImpl.java +++ b/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/repository/group_invite_repository/GroupInviteQueryRepositoryImpl.java @@ -6,6 +6,7 @@ import static site.timecapsulearchive.core.domain.member_group.entity.QGroupInvite.groupInvite; import com.querydsl.core.types.Projections; +import com.querydsl.core.types.dsl.BooleanExpression; import com.querydsl.jpa.impl.JPAQueryFactory; import java.sql.PreparedStatement; import java.sql.SQLException; @@ -14,14 +15,13 @@ import java.time.ZonedDateTime; import java.util.List; import lombok.RequiredArgsConstructor; -import org.springframework.data.domain.Pageable; import org.springframework.data.domain.Slice; -import org.springframework.data.domain.SliceImpl; import org.springframework.jdbc.core.BatchPreparedStatementSetter; import org.springframework.jdbc.core.JdbcTemplate; import org.springframework.stereotype.Repository; import site.timecapsulearchive.core.domain.member_group.data.dto.GroupInviteSummaryDto; import site.timecapsulearchive.core.domain.member_group.data.dto.GroupSendingInviteMemberDto; +import site.timecapsulearchive.core.domain.member_group.data.dto.GroupSendingInvitesSliceRequestDto; import site.timecapsulearchive.core.global.util.SliceUtil; @Repository @@ -101,11 +101,10 @@ public Slice findGroupReceivingInvitesSlice( return SliceUtil.makeSlice(size, groupInviteSummaryDtos); } - public List findGroupSendingInvites( - final Long memberId, - final Long groupId + public Slice findGroupSendingInvites( + final GroupSendingInvitesSliceRequestDto dto ) { - return jpaQueryFactory + List groupSendingInviteMemberDtos = jpaQueryFactory .select( Projections.constructor( GroupSendingInviteMemberDto.class, @@ -117,8 +116,24 @@ public List findGroupSendingInvites( ) .from(groupInvite) .join(groupInvite.groupMember, member) - .where(groupInvite.group.id.eq(groupId) - .and(groupInvite.groupOwner.id.eq(memberId))) + .where( + groupInviteIdPagingCursorCondition(dto), + groupInvite.group.id.eq(dto.groupId()) + .and(groupInvite.groupOwner.id.eq(dto.memberId())) + ) + .orderBy(groupInvite.id.desc()) + .limit(dto.size() + 1) .fetch(); + + return SliceUtil.makeSlice(dto.size(), groupSendingInviteMemberDtos); + } + + private BooleanExpression groupInviteIdPagingCursorCondition( + GroupSendingInvitesSliceRequestDto dto) { + if (dto.groupInviteId() == null) { + return null; + } + + return groupInvite.id.lt(dto.groupInviteId()); } } diff --git a/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/service/MemberGroupQueryService.java b/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/service/MemberGroupQueryService.java index dd8141997..bd9f1c113 100644 --- a/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/service/MemberGroupQueryService.java +++ b/backend/core/src/main/java/site/timecapsulearchive/core/domain/member_group/service/MemberGroupQueryService.java @@ -10,6 +10,7 @@ import site.timecapsulearchive.core.domain.group.exception.GroupNotFoundException; import site.timecapsulearchive.core.domain.member_group.data.dto.GroupInviteSummaryDto; import site.timecapsulearchive.core.domain.member_group.data.dto.GroupSendingInviteMemberDto; +import site.timecapsulearchive.core.domain.member_group.data.dto.GroupSendingInvitesSliceRequestDto; import site.timecapsulearchive.core.domain.member_group.repository.group_invite_repository.GroupInviteRepository; import site.timecapsulearchive.core.domain.member_group.repository.member_group_repository.MemberGroupRepository; @@ -41,10 +42,8 @@ public List findGroupMemberInfos( return memberGroupRepository.findGroupMemberInfos(memberId, groupId); } - public List findGroupSendingInvites( - final Long memberId, - final Long groupId - ) { - return groupInviteRepository.findGroupSendingInvites(memberId, groupId); + public Slice findGroupSendingInvites(final + GroupSendingInvitesSliceRequestDto dto) { + return groupInviteRepository.findGroupSendingInvites(dto); } } diff --git a/backend/core/src/test/java/site/timecapsulearchive/core/domain/member_group/repository/GroupInviteQueryRepositoryTest.java b/backend/core/src/test/java/site/timecapsulearchive/core/domain/member_group/repository/GroupInviteQueryRepositoryTest.java index 5e8522e18..03c45d9fc 100644 --- a/backend/core/src/test/java/site/timecapsulearchive/core/domain/member_group/repository/GroupInviteQueryRepositoryTest.java +++ b/backend/core/src/test/java/site/timecapsulearchive/core/domain/member_group/repository/GroupInviteQueryRepositoryTest.java @@ -23,6 +23,7 @@ import site.timecapsulearchive.core.domain.member.entity.Member; import site.timecapsulearchive.core.domain.member_group.data.dto.GroupInviteSummaryDto; import site.timecapsulearchive.core.domain.member_group.data.dto.GroupSendingInviteMemberDto; +import site.timecapsulearchive.core.domain.member_group.data.dto.GroupSendingInvitesSliceRequestDto; import site.timecapsulearchive.core.domain.member_group.entity.GroupInvite; import site.timecapsulearchive.core.domain.member_group.repository.group_invite_repository.GroupInviteQueryRepository; import site.timecapsulearchive.core.domain.member_group.repository.group_invite_repository.GroupInviteQueryRepositoryImpl; @@ -37,6 +38,7 @@ class GroupInviteQueryRepositoryTest extends RepositoryTest { private Long groupId; private Long groupOwnerId; private Long groupMemberId; + private Long firstGroupInviteStartId; GroupInviteQueryRepositoryTest( JdbcTemplate jdbcTemplate, @@ -69,6 +71,10 @@ void setUp(@Autowired EntityManager entityManager) { GroupInvite groupInvite = GroupInvite.createOf(groups.get(i), groupOwners.get(i), groupMember); entityManager.persist(groupInvite); + + if (i == 0) { + firstGroupInviteStartId = groupInvite.getId(); + } } } @@ -129,19 +135,22 @@ void setUp(@Autowired EntityManager entityManager) { @Test void 그룹장은_자신이_보낸_그룹_초대_목록을_조회하면_그룹_초대목록이_나온다() { //given + GroupSendingInvitesSliceRequestDto requestDto = new GroupSendingInvitesSliceRequestDto( + groupOwnerId, groupId, null, 20 + ); + //when - List groupSendingInvites = groupInviteRepository.findGroupSendingInvites( - groupOwnerId, groupId); + Slice groupSendingInvites = groupInviteRepository.findGroupSendingInvites(requestDto); //then SoftAssertions.assertSoftly(softly -> { - softly.assertThat(groupSendingInvites).isNotEmpty(); - softly.assertThat(groupSendingInvites).allMatch(dto -> dto.id() != null); - softly.assertThat(groupSendingInvites) + softly.assertThat(groupSendingInvites.hasContent()).isTrue(); + softly.assertThat(groupSendingInvites.getContent()).allMatch(dto -> dto.id() != null); + softly.assertThat(groupSendingInvites.getContent()) .allMatch(dto -> dto.nickname() != null && !dto.nickname().isBlank()); - softly.assertThat(groupSendingInvites) + softly.assertThat(groupSendingInvites.getContent()) .allMatch(dto -> dto.profileUrl() != null && !dto.profileUrl().isBlank()); - softly.assertThat(groupSendingInvites) + softly.assertThat(groupSendingInvites.getContent()) .allMatch(dto -> dto.sendingInvitesCreatedAt() != null); }); } @@ -149,11 +158,16 @@ void setUp(@Autowired EntityManager entityManager) { @Test void 그룹원은_자신이_보낸_그룹_초대_목록을_조회하면_빈_리스트가_나온다() { //given + GroupSendingInvitesSliceRequestDto requestDto = new GroupSendingInvitesSliceRequestDto( + groupOwnerId, groupId, firstGroupInviteStartId - 1, 20 + ); + //when - List groupSendingInvites = groupInviteRepository.findGroupSendingInvites( - groupMemberId, groupId); + Slice groupSendingInvites = groupInviteRepository.findGroupSendingInvites( + requestDto + ); //then - assertThat(groupSendingInvites).isEmpty(); + assertThat(groupSendingInvites.isEmpty()).isTrue(); } }