Skip to content
Merged
Show file tree
Hide file tree
Changes from all 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
Expand Up @@ -12,6 +12,7 @@
import hanium.modic.backend.common.error.exception.LockException;
import hanium.modic.backend.common.property.property.VoteProperties;
import hanium.modic.backend.common.redis.distributedLock.LockManager;
import hanium.modic.backend.domain.user.service.UserVoteStreakService;
import hanium.modic.backend.domain.vote.entity.SimilarityVoteEntity;
import hanium.modic.backend.domain.vote.entity.SimilarityVoteResultEntity;
import hanium.modic.backend.domain.vote.entity.SimilarityVoteSummaryEntity;
Expand All @@ -24,6 +25,7 @@
import hanium.modic.backend.domain.post.enums.PostStatus;
import hanium.modic.backend.domain.post.repository.PostEntityRepository;
import hanium.modic.backend.web.vote.dto.response.VoteParticipationResponse;
import hanium.modic.backend.web.vote.dto.response.GetVoteStreakResponse;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;

Expand All @@ -45,6 +47,21 @@ public class VotingService {
private final VoteProperties voteProperties;
private final VoteRewardService voteRewardService;
private final VoteCompletionRewardService voteCompletionRewardService;
private final UserVoteStreakService userVoteStreakService;

/**
* 사용자의 투표 연속 정답 정보를 조회합니다.
*
* @param userId 조회 대상 사용자 ID
* @return 현재 연속 정답 수와 리워드 임계치를 담은 응답
*/
@Transactional(readOnly = true)
public GetVoteStreakResponse getVoteStreak(final Long userId) {
int streakCount = userVoteStreakService.getStreakCount(userId);
int rewardThreshold = voteProperties.getStreakRewardCount();
return GetVoteStreakResponse.of(streakCount, rewardThreshold);
}


/**
* 투표 참여 메서드
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@
import hanium.modic.backend.domain.vote.service.VotingService;
import hanium.modic.backend.web.vote.dto.request.VoteParticipationRequest;
import hanium.modic.backend.web.vote.dto.response.VoteParticipationResponse;
import hanium.modic.backend.web.vote.dto.response.GetVoteStreakResponse;
import hanium.modic.backend.web.vote.dto.response.VoteDetailResponse;
import hanium.modic.backend.web.vote.dto.response.VoteSummaryResponse;
import io.swagger.v3.oas.annotations.Operation;
Expand Down Expand Up @@ -55,6 +56,26 @@ public ResponseEntity<AppResponse<VoteSummaryResponse>> getVoteResults(
return ok(AppResponse.ok(response));
}

/**
* 현재 로그인한 사용자의 투표 연속 정답 수를 조회합니다.
*
* @param user 인증된 사용자
* @return 사용자의 현재 연속 정답 정보
*/
@GetMapping("/streak")
@Operation(
summary = "투표 연속 정답 수 조회",
description = "현재 로그인한 사용자의 투표 연속 정답 현황을 조회합니다.",
responses = {
@ApiResponse(responseCode = "200", description = "연속 정답 수 조회 성공"),
@ApiResponse(responseCode = "401", description = "인증이 필요합니다.[C-003]")
}
)
public ResponseEntity<AppResponse<GetVoteStreakResponse>> getVoteStreak(@CurrentUser UserEntity user) {
GetVoteStreakResponse response = votingService.getVoteStreak(user.getId());
return ok(AppResponse.ok(response));
}

@PostMapping("/{voteId}/decisions")
@Operation(summary = "투표 참여", description = "사용자가 특정 투표에 대해 자신의 결정을 제출합니다.")
@ApiResponse(responseCode = "200", description = "투표 참여 성공")
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
package hanium.modic.backend.web.vote.dto.response;

/**
* 사용자 투표 연속 정답 정보를 제공하는 응답 DTO.
*/
public record GetVoteStreakResponse(
int currentStreak,
int rewardThreshold
) {
/**
* 현재 연속 정답 수와 리워드 임계치를 기반으로 응답 객체를 생성합니다.
*
* @param currentStreak 사용자 연속 정답 수
* @param rewardThreshold 리워드 지급 임계치
* @return 연속 정답 정보를 담은 응답
*/
public static GetVoteStreakResponse of(int currentStreak, int rewardThreshold) {
return new GetVoteStreakResponse(currentStreak, rewardThreshold);
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,62 @@
package hanium.modic.backend.domain.vote.service;

import static org.assertj.core.api.Assertions.assertThat;
import static org.mockito.BDDMockito.given;

import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.extension.ExtendWith;
import org.mockito.InjectMocks;
import org.mockito.Mock;
import org.mockito.junit.jupiter.MockitoExtension;

import hanium.modic.backend.common.property.property.VoteProperties;
import hanium.modic.backend.common.redis.distributedLock.LockManager;
import hanium.modic.backend.domain.post.repository.PostEntityRepository;
import hanium.modic.backend.domain.user.service.UserVoteStreakService;
import hanium.modic.backend.domain.vote.repository.SimilarityVoteRepository;
import hanium.modic.backend.domain.vote.repository.SimilarityVoteResultRepository;
import hanium.modic.backend.domain.vote.repository.SimilarityVoteSummaryRepository;
import hanium.modic.backend.web.vote.dto.response.GetVoteStreakResponse;

@ExtendWith(MockitoExtension.class)
class VotingServiceTest {

@Mock
private SimilarityVoteRepository similarityVoteRepository;
@Mock
private SimilarityVoteResultRepository voteResultRepository;
@Mock
private SimilarityVoteSummaryRepository voteSummaryRepository;
@Mock
private PostEntityRepository postEntityRepository;
@Mock
private LockManager lockManager;
@Mock
private VoteProperties voteProperties;
@Mock
private VoteRewardService voteRewardService;
@Mock
private VoteCompletionRewardService voteCompletionRewardService;
@Mock
private UserVoteStreakService userVoteStreakService;

@InjectMocks
private VotingService votingService;

@Test
@DisplayName("투표 연속 정답 조회")
void getVoteStreak_underThreshold() {
// given
Long userId = 1L;
given(userVoteStreakService.getStreakCount(userId)).willReturn(2);
given(voteProperties.getStreakRewardCount()).willReturn(3);

// when
GetVoteStreakResponse response = votingService.getVoteStreak(userId);

// then
assertThat(response.currentStreak()).isEqualTo(2);
assertThat(response.rewardThreshold()).isEqualTo(3);
}
}