From 027d91b30d03c1ec5595620e2dfd6045f65f8b48 Mon Sep 17 00:00:00 2001
From: sungHeeLee <70899677+hee9841@users.noreply.github.com>
Date: Thu, 26 Dec 2024 18:13:40 +0900
Subject: [PATCH] =?UTF-8?q?Feat:=20=EC=B1=8C=EB=A6=B0=EC=A7=80=20=EB=B0=8F?=
=?UTF-8?q?=20=EB=AA=A9=ED=91=9C=20=EC=9D=91=EB=8B=B5=20DTO=20=ED=86=B5?=
=?UTF-8?q?=ED=95=A9=20(#326)?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
* Feat: challenge, goal 달성 퍼센테이지 계산 구현
* Test: challenge, goal 달성 퍼센테이지 계산 테스트(러닝 데이터 추가 시)
* Feat: 챌린지, 목표 달성 결과 response dto 통합
* Feat: 챌린지, 목표 달성 결과 response dto 통합으로 RunningRecordResultResponseV2 수정
* Fix: 러닝 추가(목표 모드) 시 request goalValues에대한 validation 추가
* Fix: 러닝 기록 저장 시(챌린지 모드) 사용하지 않는 챌린지 일경우 예외처리 추가
* Fix: 챌린지, 목표 러닝 결과 response 통합으로 러닝 기록조회 서비스 함수 수정, 컨트롤러 response(v1) 수정
* Fix: 러닝 루트 리턴시 위도값이 경도 값으로 리턴되는 오류 수정
* Feat: 러닝 기록 조회(V2) 구현
* Docs: 러닝 기록 추가(V2) 문서 수정
---
.../running/RunningRecordService.java | 105 ++++-
.../running/dto/RunningResultDto.java | 15 +-
.../global/exception/type/ErrorType.java | 3 +
.../v1/running/RunningRecordController.java | 2 +-
.../v1/running/dto/ChallengeDto.java | 36 +-
.../v1/running/dto/GoalResultDto.java | 27 +-
.../response/RunningRecordQueryResponse.java | 47 +-
.../v2/running/RunningRecordControllerV2.java | 16 +-
.../v2/running/dto/AchievementResultDto.java | 23 +
.../v2/running/dto/RouteDtoV2.java | 2 +-
.../dto/request/RunningRecordRequestV2.java | 6 +-
.../RunningRecordResultResponseV2.java | 68 +--
.../running/RunningRecordServiceTest.java | 421 ++++++++++++++++--
.../RunningRecordControllerV2Test.java | 4 +-
14 files changed, 612 insertions(+), 163 deletions(-)
create mode 100644 src/main/java/com/dnd/runus/presentation/v2/running/dto/AchievementResultDto.java
diff --git a/src/main/java/com/dnd/runus/application/running/RunningRecordService.java b/src/main/java/com/dnd/runus/application/running/RunningRecordService.java
index d43b9537..d72a3094 100644
--- a/src/main/java/com/dnd/runus/application/running/RunningRecordService.java
+++ b/src/main/java/com/dnd/runus/application/running/RunningRecordService.java
@@ -3,7 +3,9 @@
import com.dnd.runus.application.running.dto.RunningResultDto;
import com.dnd.runus.application.running.event.RunningRecordAddedEvent;
import com.dnd.runus.domain.challenge.Challenge;
+import com.dnd.runus.domain.challenge.ChallengeCondition;
import com.dnd.runus.domain.challenge.ChallengeRepository;
+import com.dnd.runus.domain.challenge.ChallengeType;
import com.dnd.runus.domain.challenge.ChallengeWithCondition;
import com.dnd.runus.domain.challenge.GoalMetricType;
import com.dnd.runus.domain.challenge.achievement.ChallengeAchievement;
@@ -19,13 +21,15 @@
import com.dnd.runus.domain.running.DailyRunningRecordSummary;
import com.dnd.runus.domain.running.RunningRecord;
import com.dnd.runus.domain.running.RunningRecordRepository;
+import com.dnd.runus.global.exception.BusinessException;
import com.dnd.runus.global.exception.NotFoundException;
+import com.dnd.runus.global.exception.type.ErrorType;
import com.dnd.runus.presentation.v1.running.dto.WeeklyRunningRatingDto;
+import com.dnd.runus.presentation.v1.running.dto.request.RunningAchievementMode;
import com.dnd.runus.presentation.v1.running.dto.request.RunningRecordRequestV1;
import com.dnd.runus.presentation.v1.running.dto.request.RunningRecordWeeklySummaryType;
import com.dnd.runus.presentation.v1.running.dto.response.*;
import com.dnd.runus.presentation.v2.running.dto.request.RunningRecordRequestV2;
-import com.dnd.runus.presentation.v2.running.dto.request.RunningRecordRequestV2.ChallengeAchievedDto;
import com.dnd.runus.presentation.v2.running.dto.request.RunningRecordRequestV2.GoalAchievedDto;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.ApplicationEventPublisher;
@@ -79,7 +83,7 @@ public RunningRecordService(
}
@Transactional(readOnly = true)
- public RunningRecordQueryResponse getRunningRecord(long memberId, long runningRecordId) {
+ public RunningResultDto getRunningRecord(long memberId, long runningRecordId) {
RunningRecord runningRecord = runningRecordRepository
.findById(runningRecordId)
.filter(r -> r.member().memberId() == memberId)
@@ -97,7 +101,27 @@ public RunningRecordQueryResponse getRunningRecord(long memberId, long runningRe
.orElse(null)
: null;
- return RunningRecordQueryResponse.of(runningRecord, challengeAchievement, goalAchievement);
+ RunningAchievementMode runningAchievementMode = (challengeAchievement != null)
+ ? RunningAchievementMode.CHALLENGE
+ : (goalAchievement != null) ? RunningAchievementMode.GOAL : RunningAchievementMode.NORMAL;
+
+ switch (runningAchievementMode) {
+ case CHALLENGE -> {
+ return RunningResultDto.of(
+ runningRecord,
+ challengeAchievement,
+ calChallengeAchievementPercentage(memberId, challengeAchievement));
+ }
+ case GOAL -> {
+ int achievedGoalValue = goalAchievement.goalMetricType().getActualValue(runningRecord);
+ return RunningResultDto.of(
+ runningRecord,
+ goalAchievement,
+ calPercentage(achievedGoalValue, goalAchievement.achievementValue()));
+ }
+ }
+
+ return RunningResultDto.from(runningRecord);
}
@Transactional(readOnly = true)
@@ -223,16 +247,22 @@ public RunningResultDto addRunningRecordV2(long memberId, RunningRecordRequestV2
switch (request.achievementMode()) {
case CHALLENGE -> {
- ChallengeAchievedDto challengeAchievedForAdd = request.challengeValues();
Challenge challenge = challengeRepository
- .findById(challengeAchievedForAdd.challengeId())
- .orElseThrow(
- () -> new NotFoundException(Challenge.class, challengeAchievedForAdd.challengeId()));
+ .findById(request.challengeValues().challengeId())
+ .orElseThrow(() -> new NotFoundException(
+ Challenge.class, request.challengeValues().challengeId()));
+ if (!challenge.isActive()) {
+ throw new BusinessException(ErrorType.CHALLENGE_NOT_ACTIVE);
+ }
- ChallengeAchievement challengeAchievement = challengeAchievementRepository.save(
- new ChallengeAchievement(challenge, record, challengeAchievedForAdd.isSuccess()));
+ ChallengeAchievement challengeAchievement =
+ challengeAchievementRepository.save(new ChallengeAchievement(
+ challenge, record, request.challengeValues().isSuccess()));
- return RunningResultDto.of(record, challengeAchievement);
+ return RunningResultDto.of(
+ record,
+ challengeAchievement,
+ calChallengeAchievementPercentage(memberId, challengeAchievement));
}
case GOAL -> {
GoalAchievedDto goalAchievedForAdd = request.goalValues();
@@ -244,7 +274,10 @@ public RunningResultDto addRunningRecordV2(long memberId, RunningRecordRequestV2
: goalAchievedForAdd.goalTime(),
goalAchievedForAdd.isSuccess()));
- return RunningResultDto.of(record, goalAchievement);
+ int achievedGoalValue = goalAchievement.goalMetricType().getActualValue(record);
+
+ return RunningResultDto.of(
+ record, goalAchievement, calPercentage(achievedGoalValue, goalAchievement.achievementValue()));
}
}
return RunningResultDto.from(record);
@@ -337,4 +370,54 @@ private GoalAchievement handleGoalMode(RunningRecord runningRecord, Integer goal
GoalAchievement goalAchievement = new GoalAchievement(runningRecord, goalMetricType, goalValue, isAchieved);
return goalAchievementRepository.save(goalAchievement);
}
+
+ /**
+ * 챌린지 퍼센테이지 계산(V2 이후에서 사용)
+ * 어제 러닝 기록 이기기 관련 챌린지면 챌린지 목표값 재등록(기본 목표 값 + 어제 러닝 기록)한 후 계산
+ * @return Double(퍼센테이지 계산하기 힘든, 2 이전에 저장된 챌린지같은 경우 null로 리턴)
+ */
+ private Double calChallengeAchievementPercentage(long memberId, ChallengeAchievement challengeAchievement) {
+
+ Challenge challenge = challengeAchievement.challenge();
+ ChallengeWithCondition challengeWithCondition = challengeRepository
+ .findChallengeWithConditionsByChallengeId(challenge.challengeId())
+ .orElseThrow(() -> new NotFoundException(Challenge.class, challenge.challengeId()));
+
+ // DISTANCE_IN_TIME, PACE 챌린지인지 확인
+ // (v2이전에 퍼센테이지를 표현할 수 없는 애들일 경우, 퍼센테이지 null로 리턴함)
+ if (challenge.challengeType() == ChallengeType.DISTANCE_IN_TIME
+ || challengeWithCondition.conditions().stream()
+ .anyMatch(condition -> !condition.goalMetricType().hasPercentage())) {
+ return null;
+ }
+
+ ChallengeCondition condition = challengeWithCondition.conditions().getFirst();
+ RunningRecord runningRecord = challengeAchievement.runningRecord();
+
+ if (challenge.isDefeatYesterdayChallenge()) {
+ // 어제 러닝 기록 이기기 관련 챌린지면, 챌린지 목표값 재등록(어제 기록 + 목표값)
+ OffsetDateTime runningDate = runningRecord
+ .startAt()
+ .toLocalDate()
+ .atStartOfDay(runningRecord.startAt().getZone())
+ .toOffsetDateTime();
+
+ RunningRecord preRunningRecord =
+ runningRecordRepository
+ .findByMemberIdAndStartAtBetween(memberId, runningDate.minusDays(1), runningDate)
+ .stream()
+ .findFirst()
+ .orElseThrow(() -> new NotFoundException("이전 러닝 기록을 가져올 수 없습니다."));
+ condition.registerComparisonValue(condition.goalMetricType().getActualValue(preRunningRecord));
+ }
+
+ int achievedValue = condition.goalMetricType().getActualValue(runningRecord);
+ return calPercentage(achievedValue, condition.comparisonValue());
+ }
+
+ private double calPercentage(double part, double total) {
+ if (total <= 0) return 0;
+ double percentage = part / total;
+ return percentage > 1 ? 1 : percentage;
+ }
}
diff --git a/src/main/java/com/dnd/runus/application/running/dto/RunningResultDto.java b/src/main/java/com/dnd/runus/application/running/dto/RunningResultDto.java
index d410507b..c66c66b4 100644
--- a/src/main/java/com/dnd/runus/application/running/dto/RunningResultDto.java
+++ b/src/main/java/com/dnd/runus/application/running/dto/RunningResultDto.java
@@ -9,33 +9,38 @@ public record RunningResultDto(
RunningRecord runningRecord,
com.dnd.runus.presentation.v1.running.dto.request.RunningAchievementMode runningAchievementMode,
ChallengeAchievement challengeAchievement,
- GoalAchievement goalAchievement
+ GoalAchievement goalAchievement,
+ Double percentage
) {
public static RunningResultDto from(RunningRecord runningRecord) {
return new RunningResultDto(
runningRecord,
com.dnd.runus.presentation.v1.running.dto.request.RunningAchievementMode.NORMAL,
null,
+ null,
null
);
}
public static RunningResultDto of(RunningRecord runningRecord,
- ChallengeAchievement challengeAchievement) {
+ ChallengeAchievement challengeAchievement, Double percentage) {
return new RunningResultDto(
runningRecord,
com.dnd.runus.presentation.v1.running.dto.request.RunningAchievementMode.CHALLENGE,
challengeAchievement,
- null
+ null,
+ percentage
);
}
- public static RunningResultDto of(RunningRecord runningRecord, GoalAchievement goalAchievement) {
+ public static RunningResultDto of(RunningRecord runningRecord,
+ GoalAchievement goalAchievement, Double percentage) {
return new RunningResultDto(
runningRecord,
com.dnd.runus.presentation.v1.running.dto.request.RunningAchievementMode.GOAL,
null,
- goalAchievement
+ goalAchievement,
+ percentage
);
}
}
diff --git a/src/main/java/com/dnd/runus/global/exception/type/ErrorType.java b/src/main/java/com/dnd/runus/global/exception/type/ErrorType.java
index 310ebf4b..d896949a 100644
--- a/src/main/java/com/dnd/runus/global/exception/type/ErrorType.java
+++ b/src/main/java/com/dnd/runus/global/exception/type/ErrorType.java
@@ -50,6 +50,9 @@ public enum ErrorType {
GOAL_VALUES_REQUIRED_IN_GOAL_MODE(BAD_REQUEST, DEBUG, "RUNNING_005", "개인 목표 모드에서, 개인 목표 달성값은 필수 잆니다."),
CHALLENGE_VALUES_REQUIRED_IN_CHALLENGE_MODE(BAD_REQUEST, DEBUG, "RUNNING_006", "챌린지 모드에서, 챌린지 달성값은 필수 입니다."),
+ // ChallengeErrorType
+ CHALLENGE_NOT_ACTIVE(CONFLICT, DEBUG, "CHALLENGE_001", "해당 챌린지는 현재 활성화된 챌린지가 아닙니다."),
+
// WeatherErrorType
WEATHER_API_ERROR(SERVICE_UNAVAILABLE, WARN, "WEATHER_001", "날씨 API 호출 중 오류가 발생했습니다"),
;
diff --git a/src/main/java/com/dnd/runus/presentation/v1/running/RunningRecordController.java b/src/main/java/com/dnd/runus/presentation/v1/running/RunningRecordController.java
index a323050f..87a2cc87 100644
--- a/src/main/java/com/dnd/runus/presentation/v1/running/RunningRecordController.java
+++ b/src/main/java/com/dnd/runus/presentation/v1/running/RunningRecordController.java
@@ -31,7 +31,7 @@ public class RunningRecordController {
@GetMapping("/{runningRecordId}")
@Operation(summary = "러닝 기록 상세 조회", description = "RunngingRecord id로 러닝 상세 기록을 조회합니다.")
public RunningRecordQueryResponse getRunningRecord(@MemberId long memberId, @PathVariable long runningRecordId) {
- return runningRecordService.getRunningRecord(memberId, runningRecordId);
+ return RunningRecordQueryResponse.from(runningRecordService.getRunningRecord(memberId, runningRecordId));
}
@GetMapping("monthly-dates")
diff --git a/src/main/java/com/dnd/runus/presentation/v1/running/dto/ChallengeDto.java b/src/main/java/com/dnd/runus/presentation/v1/running/dto/ChallengeDto.java
index 055bd9a7..ccea6cd8 100644
--- a/src/main/java/com/dnd/runus/presentation/v1/running/dto/ChallengeDto.java
+++ b/src/main/java/com/dnd/runus/presentation/v1/running/dto/ChallengeDto.java
@@ -1,21 +1,31 @@
package com.dnd.runus.presentation.v1.running.dto;
+import com.dnd.runus.domain.challenge.achievement.ChallengeAchievement;
import io.swagger.v3.oas.annotations.media.Schema;
import jakarta.validation.constraints.NotBlank;
public record ChallengeDto(
- @Schema(description = "챌린지 ID")
- long challengeId,
- @NotBlank
- @Schema(description = "챌린지 이름", example = "오늘 30분 동안 뛰기")
- String title,
- @NotBlank
- @Schema(description = "챌린지 결과 문구", example = "성공했어요!")
- String subTitle,
- @NotBlank
- @Schema(description = "챌린지 아이콘 이미지 URL")
- String iconUrl,
- @Schema(description = "챌린지 성공 여부")
- boolean isSuccess
+ @Schema(description = "챌린지 ID")
+ long challengeId,
+ @NotBlank
+ @Schema(description = "챌린지 이름", example = "오늘 30분 동안 뛰기")
+ String title,
+ @NotBlank
+ @Schema(description = "챌린지 결과 문구", example = "성공했어요!")
+ String subTitle,
+ @NotBlank
+ @Schema(description = "챌린지 아이콘 이미지 URL")
+ String iconUrl,
+ @Schema(description = "챌린지 성공 여부")
+ boolean isSuccess
) {
+ public static ChallengeDto from(ChallengeAchievement achievement) {
+ return new ChallengeDto(
+ achievement.challenge().challengeId(),
+ achievement.challenge().name(),
+ achievement.description(),
+ achievement.challenge().imageUrl(),
+ achievement.isSuccess()
+ );
+ }
}
diff --git a/src/main/java/com/dnd/runus/presentation/v1/running/dto/GoalResultDto.java b/src/main/java/com/dnd/runus/presentation/v1/running/dto/GoalResultDto.java
index a72a052a..6d42a5da 100644
--- a/src/main/java/com/dnd/runus/presentation/v1/running/dto/GoalResultDto.java
+++ b/src/main/java/com/dnd/runus/presentation/v1/running/dto/GoalResultDto.java
@@ -1,17 +1,26 @@
package com.dnd.runus.presentation.v1.running.dto;
+import com.dnd.runus.domain.goalAchievement.GoalAchievement;
import io.swagger.v3.oas.annotations.media.Schema;
import jakarta.validation.constraints.NotBlank;
public record GoalResultDto(
- @Schema(description = "설정된 목표 제목", example = "2.5km 달성")
- String title,
- @Schema(description = "설정된 목표 결과 문구", example = "성공했어요!")
- String subTitle,
- @NotBlank
- @Schema(description = "챌린지 아이콘 이미지 URL")
- String iconUrl,
- @Schema(description = "설정된 목표 성공 여부")
- boolean isSuccess
+ @Schema(description = "설정된 목표 제목", example = "2.5km 달성")
+ String title,
+ @Schema(description = "설정된 목표 결과 문구", example = "성공했어요!")
+ String subTitle,
+ @NotBlank
+ @Schema(description = "챌린지 아이콘 이미지 URL")
+ String iconUrl,
+ @Schema(description = "설정된 목표 성공 여부")
+ boolean isSuccess
) {
+ public static GoalResultDto from(GoalAchievement achievement) {
+ return new GoalResultDto(
+ achievement.getTitle(),
+ achievement.getDescription(),
+ achievement.getIconUrl(),
+ achievement.isAchieved()
+ );
+ }
}
diff --git a/src/main/java/com/dnd/runus/presentation/v1/running/dto/response/RunningRecordQueryResponse.java b/src/main/java/com/dnd/runus/presentation/v1/running/dto/response/RunningRecordQueryResponse.java
index dbef30d5..091a28fd 100644
--- a/src/main/java/com/dnd/runus/presentation/v1/running/dto/response/RunningRecordQueryResponse.java
+++ b/src/main/java/com/dnd/runus/presentation/v1/running/dto/response/RunningRecordQueryResponse.java
@@ -1,7 +1,6 @@
package com.dnd.runus.presentation.v1.running.dto.response;
-import com.dnd.runus.domain.challenge.achievement.ChallengeAchievement;
-import com.dnd.runus.domain.goalAchievement.GoalAchievement;
+import com.dnd.runus.application.running.dto.RunningResultDto;
import com.dnd.runus.domain.running.RunningRecord;
import com.dnd.runus.global.constant.RunningEmoji;
import com.dnd.runus.presentation.v1.running.dto.ChallengeDto;
@@ -32,45 +31,18 @@ public record RunningRecordQueryResponse(
@NotNull
RunningRecordMetricsDto runningData
) {
- public static RunningRecordQueryResponse from(RunningRecord runningRecord) {
- return buildResponse(runningRecord, null, null, RunningAchievementMode.NORMAL);
- }
-
- public static RunningRecordQueryResponse of(RunningRecord runningRecord, ChallengeAchievement achievement) {
- return buildResponse(runningRecord,
- new ChallengeDto(
- achievement.challenge().challengeId(),
- achievement.challenge().name(),
- achievement.description(),
- achievement.challenge().imageUrl(),
- achievement.isSuccess()
- ),
- null,
- RunningAchievementMode.CHALLENGE
- );
- }
- public static RunningRecordQueryResponse of(RunningRecord runningRecord, GoalAchievement achievement) {
- return buildResponse(runningRecord,
- null,
- new GoalResultDto(
- achievement.getTitle(),
- achievement.getDescription(),
- achievement.getIconUrl(),
- achievement.isAchieved()
- ),
- RunningAchievementMode.GOAL
+ public static RunningRecordQueryResponse from(RunningResultDto runningResult) {
+ return buildResponse(
+ runningResult.runningRecord(),
+ runningResult.challengeAchievement() == null ? null
+ : ChallengeDto.from(runningResult.challengeAchievement()),
+ runningResult.goalAchievement() == null ? null
+ : GoalResultDto.from(runningResult.goalAchievement()),
+ runningResult.runningAchievementMode()
);
}
- public static RunningRecordQueryResponse of(RunningRecord runningRecord, ChallengeAchievement challengeAchievement, GoalAchievement goalAchievement) {
- if (challengeAchievement != null) {
- return of(runningRecord, challengeAchievement);
- } else if (goalAchievement != null) {
- return of(runningRecord, goalAchievement);
- }
- return from(runningRecord);
- }
private static RunningRecordQueryResponse buildResponse(RunningRecord runningRecord, ChallengeDto challenge, GoalResultDto goal, RunningAchievementMode achievementMode) {
return new RunningRecordQueryResponse(
@@ -89,4 +61,5 @@ private static RunningRecordQueryResponse buildResponse(RunningRecord runningRec
)
);
}
+
}
diff --git a/src/main/java/com/dnd/runus/presentation/v2/running/RunningRecordControllerV2.java b/src/main/java/com/dnd/runus/presentation/v2/running/RunningRecordControllerV2.java
index 5caf2f7f..db47200e 100644
--- a/src/main/java/com/dnd/runus/presentation/v2/running/RunningRecordControllerV2.java
+++ b/src/main/java/com/dnd/runus/presentation/v2/running/RunningRecordControllerV2.java
@@ -14,6 +14,7 @@
import lombok.RequiredArgsConstructor;
import org.springframework.http.HttpStatus;
import org.springframework.web.bind.annotation.GetMapping;
+import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
@@ -27,6 +28,12 @@ public class RunningRecordControllerV2 {
private final RunningRecordServiceV2 runningRecordService2;
private final RunningRecordService runningRecordService;
+ @GetMapping("/{runningRecordId}")
+ @Operation(summary = "러닝 기록 상세 조회", description = "RunngingRecord id로 러닝 상세 기록을 조회합니다.")
+ public RunningRecordResultResponseV2 getRunningRecord(@MemberId long memberId, @PathVariable long runningRecordId) {
+ return RunningRecordResultResponseV2.from(runningRecordService.getRunningRecord(memberId, runningRecordId));
+ }
+
@Operation(summary = "이번 달 러닝 기록 조회(홈화면) V2", description = """
홈화면의 이번 달 러닝 기록을 조회 합니다.
""")
@@ -47,9 +54,9 @@ public RunningRecordMonthlySummaryResponseV2 getMonthlyRunningSummary(@MemberId
"""
러닝 기록을 추가합니다.
러닝 기록은 시작 시간, 종료 시간, 러닝 평가(emotion), 러닝 데이터 등으로 구성됩니다.
- 챌린지 모드가 normal : challengeValues, goalValues 둘다 null
- 챌린지 모드가 challenge : challengeValues 필수 값
- 챌린지 모드가 goal : goalValues 필수 값
+ normal : challengeValues, goalValues 둘다 null
+ challenge : challengeValues 필수 값
+ goal : goalValues 필수 값
러닝 데이터는 위치, 거리, 시간, 칼로리, 평균 페이스, 러닝 경로로 구성됩니다.
러닝 기록 추가에 성공하면 러닝 기록 ID, 기록 정보를 반환합니다.
""")
@@ -58,7 +65,8 @@ public RunningRecordMonthlySummaryResponseV2 getMonthlyRunningSummary(@MemberId
ErrorType.CHALLENGE_VALUES_REQUIRED_IN_CHALLENGE_MODE,
ErrorType.GOAL_VALUES_REQUIRED_IN_GOAL_MODE,
ErrorType.GOAL_TIME_AND_DISTANCE_BOTH_EXIST,
- ErrorType.ROUTE_MUST_HAVE_AT_LEAST_TWO_COORDINATES
+ ErrorType.ROUTE_MUST_HAVE_AT_LEAST_TWO_COORDINATES,
+ ErrorType.CHALLENGE_NOT_ACTIVE
})
@PostMapping
@ResponseStatus(HttpStatus.OK)
diff --git a/src/main/java/com/dnd/runus/presentation/v2/running/dto/AchievementResultDto.java b/src/main/java/com/dnd/runus/presentation/v2/running/dto/AchievementResultDto.java
new file mode 100644
index 00000000..a2fb9c27
--- /dev/null
+++ b/src/main/java/com/dnd/runus/presentation/v2/running/dto/AchievementResultDto.java
@@ -0,0 +1,23 @@
+package com.dnd.runus.presentation.v2.running.dto;
+
+import io.swagger.v3.oas.annotations.media.Schema;
+import org.springframework.lang.Nullable;
+
+/**
+ * 챌린지, 목표 달성 결과 통합 DTO
+ */
+public record AchievementResultDto(
+ @Schema(description = "챌린지 이름 또는 설정된 목표 제목", examples = {"오늘 30분 동안 뛰기", "2.5km 달성"})
+ String title,
+ @Schema(description = "결과 문구", examples = {"정말 대단해요! 잘하셨어요", "아쉬워요. 내일 다시 도전해보세요!"})
+ String subTitle,
+ @Schema(description = "아이콘 이미지 URL")
+ String iconUrl,
+ @Schema(description = "성공 여부")
+ boolean isSuccess,
+ @Nullable
+ @Schema(description = "퍼센테이지 값, V2 이전에 퍼센테이지를 나타낼 수 없는 챌린지일 경우 null값을 리턴합니다.")
+ Double percentage
+) {
+
+}
diff --git a/src/main/java/com/dnd/runus/presentation/v2/running/dto/RouteDtoV2.java b/src/main/java/com/dnd/runus/presentation/v2/running/dto/RouteDtoV2.java
index c4ce3256..1ac8bf8f 100644
--- a/src/main/java/com/dnd/runus/presentation/v2/running/dto/RouteDtoV2.java
+++ b/src/main/java/com/dnd/runus/presentation/v2/running/dto/RouteDtoV2.java
@@ -14,7 +14,7 @@ public record RouteDtoV2(
) {
public record Point(double longitude, double latitude) {
public static Point from(CoordinatePoint point) {
- return new Point(point.longitude(), point.longitude());
+ return new Point(point.longitude(), point.latitude());
}
}
}
diff --git a/src/main/java/com/dnd/runus/presentation/v2/running/dto/request/RunningRecordRequestV2.java b/src/main/java/com/dnd/runus/presentation/v2/running/dto/request/RunningRecordRequestV2.java
index 7265748a..c6100820 100644
--- a/src/main/java/com/dnd/runus/presentation/v2/running/dto/request/RunningRecordRequestV2.java
+++ b/src/main/java/com/dnd/runus/presentation/v2/running/dto/request/RunningRecordRequestV2.java
@@ -31,7 +31,8 @@ public record RunningRecordRequestV2(
com.dnd.runus.presentation.v1.running.dto.request.RunningAchievementMode achievementMode,
@Schema(description = "챌린지 데이터, 챌린지를 하지 않은 경우 null이나 필드 없이 보내주세요")
ChallengeAchievedDto challengeValues,
- @Schema(description = "목표 데이터, 목표를 설정하지 않은 경우 null이나 필드 없이 보내주세요")
+ @Schema(description = "목표 데이터, 목표를 설정하지 않은 경우 null이나 필드 없이 보내주세요. "
+ + "goalDistance(거리) 또는 goalTime(시간)값 둘 중 하나는 null이어야 합니다.")
GoalAchievedDto goalValues,
@NotNull
RunningRecordMetrics runningData
@@ -52,7 +53,8 @@ public record RunningRecordRequestV2(
if(goalValues == null) {
throw new BusinessException(ErrorType.GOAL_VALUES_REQUIRED_IN_GOAL_MODE);
}
- if (goalValues.goalDistance() == null && goalValues.goalTime() == null) {
+ if ((goalValues.goalDistance() == null && goalValues.goalTime() == null)
+ || (goalValues.goalDistance() != null && goalValues.goalTime() != null)) {
throw new BusinessException(ErrorType.GOAL_TIME_AND_DISTANCE_BOTH_EXIST);
}
}
diff --git a/src/main/java/com/dnd/runus/presentation/v2/running/dto/response/RunningRecordResultResponseV2.java b/src/main/java/com/dnd/runus/presentation/v2/running/dto/response/RunningRecordResultResponseV2.java
index 57fd0841..6390b7ff 100644
--- a/src/main/java/com/dnd/runus/presentation/v2/running/dto/response/RunningRecordResultResponseV2.java
+++ b/src/main/java/com/dnd/runus/presentation/v2/running/dto/response/RunningRecordResultResponseV2.java
@@ -7,8 +7,8 @@
import com.dnd.runus.domain.goalAchievement.GoalAchievement;
import com.dnd.runus.domain.running.RunningRecord;
import com.dnd.runus.global.constant.RunningEmoji;
-import com.dnd.runus.presentation.v1.running.dto.ChallengeDto;
-import com.dnd.runus.presentation.v1.running.dto.GoalResultDto;
+import com.dnd.runus.presentation.v1.running.dto.request.RunningAchievementMode;
+import com.dnd.runus.presentation.v2.running.dto.AchievementResultDto;
import com.dnd.runus.presentation.v2.running.dto.RouteDtoV2;
import com.dnd.runus.presentation.v2.running.dto.RouteDtoV2.Point;
import io.swagger.v3.oas.annotations.media.Schema;
@@ -29,12 +29,9 @@ public record RunningRecordResultResponseV2(
RunningEmoji emotion,
@NotNull
@Schema(description = "달성 모드, normal: 일반(목표 설정 X), challenge: 챌린지, goal: 목표")
- com.dnd.runus.presentation.v1.running.dto.request.RunningAchievementMode achievementMode,
- //todo 챌린지, 및 goal 관련해서 다시 구성
- @Schema(description = "챌린지 정보, achievementMode가 challenge인 경우에만 값이 존재합니다.")
- ChallengeDto challenge,
- @Schema(description = "목표 결과 정보, achievementMode가 goal인 경우에만 값이 존재합니다.")
- GoalResultDto goal,
+ RunningAchievementMode achievementMode,
+ @Schema(description = "달성 값(챌린지 또는 목표), achievementMode가 challenge 또는 goal인 경우에만 값이 존재합니다.")
+ AchievementResultDto achievementResult,
@NotNull
RunningRecordMetrics runningData
) {
@@ -61,10 +58,7 @@ public static RunningRecordResultResponseV2 from(RunningResultDto runningRecordR
runningRecord.endAt().toLocalDateTime(),
runningRecord.emoji(),
runningRecordResult.runningAchievementMode(),
- runningRecordResult.runningAchievementMode() != com.dnd.runus.presentation.v1.running.dto.request.RunningAchievementMode.CHALLENGE ? null
- : buildChallengeDto(runningRecordResult.challengeAchievement()),
- runningRecordResult.runningAchievementMode() != com.dnd.runus.presentation.v1.running.dto.request.RunningAchievementMode.GOAL ? null
- : buildGoalResultDto(runningRecordResult.goalAchievement()),
+ buildAchievementResultOf(runningRecordResult),
new RunningRecordMetrics(
runningRecord.averagePace(),
runningRecord.duration(),
@@ -75,32 +69,38 @@ public static RunningRecordResultResponseV2 from(RunningResultDto runningRecordR
);
}
- private static ChallengeDto buildChallengeDto(ChallengeAchievement achievement) {
- if (achievement == null) {
- return null;
- }
- return new ChallengeDto(
- achievement.challenge().challengeId(),
- achievement.challenge().name(),
- achievement.description(),
- achievement.challenge().imageUrl(),
- achievement.isSuccess()
- );
- }
+ private static AchievementResultDto buildAchievementResultOf(RunningResultDto runningRecordResult) {
+ if (runningRecordResult.runningAchievementMode() == RunningAchievementMode.NORMAL) return null;
+ switch (runningRecordResult.runningAchievementMode()) {
+ case GOAL -> {
+ GoalAchievement goalAchievement = runningRecordResult.goalAchievement();
+ if (goalAchievement == null) return null;
+ return new AchievementResultDto(
+ goalAchievement.getTitle(),
+ goalAchievement.getDescription(),
+ goalAchievement.getIconUrl(),
+ goalAchievement.isAchieved(),
+ runningRecordResult.percentage()
+ );
+ }
+ case CHALLENGE -> {
+ ChallengeAchievement challengeAchievement = runningRecordResult.challengeAchievement();
+ if (challengeAchievement == null) return null;
+ return new AchievementResultDto(
+ challengeAchievement.challenge().name(),
+ challengeAchievement.description(),
+ challengeAchievement.challenge().imageUrl(),
+ challengeAchievement.isSuccess(),
+ runningRecordResult.percentage()
+ );
- private static GoalResultDto buildGoalResultDto(GoalAchievement achievement) {
- if (achievement == null) {
- return null;
+ }
+ default -> {
+ return null;
+ }
}
- return new GoalResultDto(
- achievement.getTitle(),
- achievement.getDescription(),
- achievement.getIconUrl(),
- achievement.isAchieved()
- );
}
-
private static List convertRouteDtoListFrom(
List runningRecordRoute) {
// route가 null, empty, 또는 경로데이터를 사용하지 않았을 버전의 데이터 값 인경우 null를 리턴
diff --git a/src/test/java/com/dnd/runus/application/running/RunningRecordServiceTest.java b/src/test/java/com/dnd/runus/application/running/RunningRecordServiceTest.java
index 1e11e21b..a57ed6a7 100644
--- a/src/test/java/com/dnd/runus/application/running/RunningRecordServiceTest.java
+++ b/src/test/java/com/dnd/runus/application/running/RunningRecordServiceTest.java
@@ -24,7 +24,6 @@
import com.dnd.runus.presentation.v1.running.dto.request.RunningRecordWeeklySummaryType;
import com.dnd.runus.presentation.v1.running.dto.response.RunningRecordAddResultResponseV1;
import com.dnd.runus.presentation.v1.running.dto.response.RunningRecordMonthlySummaryResponse;
-import com.dnd.runus.presentation.v1.running.dto.response.RunningRecordQueryResponse;
import com.dnd.runus.presentation.v1.running.dto.response.RunningRecordWeeklySummaryResponse;
import com.dnd.runus.presentation.v2.running.dto.RouteDtoV2;
import com.dnd.runus.presentation.v2.running.dto.request.RunningRecordRequestV2;
@@ -47,6 +46,7 @@
import static com.dnd.runus.global.constant.TimeConstant.SERVER_TIMEZONE;
import static java.time.temporal.ChronoField.DAY_OF_WEEK;
import static org.assertj.core.api.Assertions.assertThat;
+import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.*;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.ArgumentMatchers.eq;
@@ -117,11 +117,11 @@ void getRunningRecord() {
given(runningRecordRepository.findById(runningRecordId)).willReturn(Optional.of(runningRecord));
// when
- RunningRecordQueryResponse result = runningRecordService.getRunningRecord(memberId, runningRecordId);
+ RunningResultDto result = runningRecordService.getRunningRecord(memberId, runningRecordId);
// then
- assertEquals(runningRecordId, result.runningRecordId());
- assertEquals(runningRecord.emoji(), result.emotion());
+ assertEquals(runningRecordId, result.runningRecord().runningId());
+ assertEquals(runningRecord.emoji(), result.runningRecord().emoji());
}
@Test
@@ -156,6 +156,7 @@ void getRunningRecord_challenge() {
// given
long memberId = 1;
long runningRecordId = 1;
+ Challenge challenge = new Challenge(1L, "challenge", "image", true, ChallengeType.TODAY);
Member member = new Member(memberId, MemberRole.USER, "nickname1", OffsetDateTime.now(), OffsetDateTime.now());
RunningRecord runningRecord = new RunningRecord(
1L,
@@ -171,23 +172,126 @@ void getRunningRecord_challenge() {
"end location",
RunningEmoji.VERY_GOOD);
- ChallengeAchievement.Status challengeAchievementStatus = new ChallengeAchievement.Status(
- 1L, new Challenge(1L, "challenge", "image", true, ChallengeType.TODAY), true);
+ ChallengeAchievement.Status challengeAchievementStatus = new ChallengeAchievement.Status(1L, challenge, true);
given(runningRecordRepository.findById(runningRecordId)).willReturn(Optional.of(runningRecord));
given(challengeAchievementRepository.findByRunningRecordId(runningRecordId))
.willReturn(Optional.of(challengeAchievementStatus));
+ given(challengeRepository.findChallengeWithConditionsByChallengeId(challenge.challengeId()))
+ .willReturn(Optional.of(new ChallengeWithCondition(
+ challenge,
+ List.of(new ChallengeCondition(
+ GoalMetricType.DISTANCE, ComparisonType.GREATER_THAN_OR_EQUAL_TO, 1_000)))));
// when
- RunningRecordQueryResponse result = runningRecordService.getRunningRecord(memberId, runningRecordId);
+ RunningResultDto result = runningRecordService.getRunningRecord(memberId, runningRecordId);
// then
- assertEquals(runningRecordId, result.runningRecordId());
- assertEquals(runningRecord.emoji(), result.emotion());
+ assertEquals(runningRecordId, result.runningRecord().runningId());
+ assertEquals(runningRecord.emoji(), result.runningRecord().emoji());
+ assertNotNull(result.challengeAchievement());
+ assertEquals(RunningAchievementMode.CHALLENGE, result.runningAchievementMode());
assertEquals(
challengeAchievementStatus.challenge().name(),
- result.challenge().title());
- assertEquals(RunningAchievementMode.CHALLENGE, result.achievementMode());
+ result.challengeAchievement().challenge().name());
+ assertEquals(1, result.percentage());
+ }
+
+ @Test
+ @DisplayName("Challenge 모드의 러닝 기록 조회 : DISTANCE_IN_TIME 타입일 경우 퍼센테이지 null 리턴 ")
+ void getRunningRecord_challenge_DISTANCE_IN_TIME_Type() {
+ // given
+ long memberId = 1;
+ long runningRecordId = 1;
+ Challenge challenge = new Challenge(1L, "challenge", "image", true, ChallengeType.DISTANCE_IN_TIME);
+ Member member = new Member(memberId, MemberRole.USER, "nickname1", OffsetDateTime.now(), OffsetDateTime.now());
+ RunningRecord runningRecord = new RunningRecord(
+ 1L,
+ member,
+ 10_000,
+ Duration.ofSeconds(10_000),
+ 500,
+ new Pace(5, 30),
+ ZonedDateTime.now(),
+ ZonedDateTime.now(),
+ List.of(new CoordinatePoint(0, 0, 0), new CoordinatePoint(0, 0, 0)),
+ "start location",
+ "end location",
+ RunningEmoji.VERY_GOOD);
+
+ ChallengeAchievement.Status challengeAchievementStatus = new ChallengeAchievement.Status(1L, challenge, true);
+
+ given(runningRecordRepository.findById(runningRecordId)).willReturn(Optional.of(runningRecord));
+ given(challengeAchievementRepository.findByRunningRecordId(runningRecordId))
+ .willReturn(Optional.of(challengeAchievementStatus));
+ given(challengeRepository.findChallengeWithConditionsByChallengeId(challenge.challengeId()))
+ .willReturn(Optional.of(new ChallengeWithCondition(
+ challenge,
+ List.of(new ChallengeCondition(
+ GoalMetricType.DISTANCE, ComparisonType.GREATER_THAN_OR_EQUAL_TO, 1_000)))));
+
+ // when
+ RunningResultDto result = runningRecordService.getRunningRecord(memberId, runningRecordId);
+
+ // then
+ assertEquals(runningRecordId, result.runningRecord().runningId());
+ assertEquals(runningRecord.emoji(), result.runningRecord().emoji());
+ assertNotNull(result.challengeAchievement());
+ assertEquals(RunningAchievementMode.CHALLENGE, result.runningAchievementMode());
+ assertEquals(
+ challengeAchievementStatus.challenge().name(),
+ result.challengeAchievement().challenge().name());
+ assertNull(result.percentage());
+ }
+
+ @Test
+ @DisplayName("Challenge 모드의 러닝 기록 조회 : 퍼센테이지를 표시 할 수 없는 챌린지가 하나라도 있을 경우 퍼센테이지 null 리턴 ")
+ void getRunningRecord_challenge_hasNotPercentage() {
+ // given
+ long memberId = 1;
+ long runningRecordId = 1;
+ Challenge challenge = new Challenge(1L, "challenge", "image", true, ChallengeType.TODAY);
+ Member member = new Member(memberId, MemberRole.USER, "nickname1", OffsetDateTime.now(), OffsetDateTime.now());
+ RunningRecord runningRecord = new RunningRecord(
+ 1L,
+ member,
+ 10_000,
+ Duration.ofSeconds(10_000),
+ 500,
+ new Pace(5, 30),
+ ZonedDateTime.now(),
+ ZonedDateTime.now(),
+ List.of(new CoordinatePoint(0, 0, 0), new CoordinatePoint(0, 0, 0)),
+ "start location",
+ "end location",
+ RunningEmoji.VERY_GOOD);
+
+ ChallengeAchievement.Status challengeAchievementStatus = new ChallengeAchievement.Status(1L, challenge, true);
+
+ given(runningRecordRepository.findById(runningRecordId)).willReturn(Optional.of(runningRecord));
+ given(challengeAchievementRepository.findByRunningRecordId(runningRecordId))
+ .willReturn(Optional.of(challengeAchievementStatus));
+ given(challengeRepository.findChallengeWithConditionsByChallengeId(challenge.challengeId()))
+ .willReturn(Optional.of(new ChallengeWithCondition(
+ challenge,
+ List.of(
+ new ChallengeCondition(
+ GoalMetricType.DISTANCE, ComparisonType.GREATER_THAN_OR_EQUAL_TO, 1_000),
+ new ChallengeCondition(
+ GoalMetricType.PACE, ComparisonType.GREATER_THAN_OR_EQUAL_TO, 1_000)))));
+
+ // when
+ RunningResultDto result = runningRecordService.getRunningRecord(memberId, runningRecordId);
+
+ // then
+ assertEquals(runningRecordId, result.runningRecord().runningId());
+ assertEquals(runningRecord.emoji(), result.runningRecord().emoji());
+ assertNotNull(result.challengeAchievement());
+ assertEquals(RunningAchievementMode.CHALLENGE, result.runningAchievementMode());
+ assertEquals(
+ challengeAchievementStatus.challenge().name(),
+ result.challengeAchievement().challenge().name());
+ assertNull(result.percentage());
}
@Test
@@ -531,10 +635,25 @@ private RunningRecord createRunningRecord(RunningRecordRequestV1 request, Member
@DisplayName("러닝 결과 저장 V2")
class RunningRecordAddV2 {
+ private RunningRecordRequestV2.RunningRecordMetrics runningRecordMetrics;
+
+ @BeforeEach
+ void beforeEach() {
+ runningRecordMetrics = new RunningRecordRequestV2.RunningRecordMetrics(
+ Duration.ofSeconds(600),
+ 1000,
+ 500.0,
+ List.of(
+ new RouteDtoV2(new RouteDtoV2.Point(0, 0), new RouteDtoV2.Point(1, 1)),
+ new RouteDtoV2(new RouteDtoV2.Point(2, 2), new RouteDtoV2.Point(3, 3)),
+ new RouteDtoV2(new RouteDtoV2.Point(4, 4), new RouteDtoV2.Point(5, 5))));
+ }
+
@Test
@DisplayName("러닝 결과 저장 : 루트가 순서대로 들어갔는지 확인")
void addRunningRecordV2_CheckRoute() {
// given
+ Member member = new Member(MemberRole.USER, "nickname1");
RunningRecordRequestV2 request = new RunningRecordRequestV2(
LocalDateTime.of(2021, 1, 1, 12, 10, 30),
LocalDateTime.of(2021, 1, 1, 13, 12, 10),
@@ -544,16 +663,251 @@ void addRunningRecordV2_CheckRoute() {
com.dnd.runus.presentation.v1.running.dto.request.RunningAchievementMode.NORMAL,
null,
null,
- new RunningRecordRequestV2.RunningRecordMetrics(
- Duration.ofSeconds(10_100),
- 10_000,
- 500.0,
- List.of(
- new RouteDtoV2(new RouteDtoV2.Point(0, 0), new RouteDtoV2.Point(1, 1)),
- new RouteDtoV2(new RouteDtoV2.Point(2, 2), new RouteDtoV2.Point(3, 3)),
- new RouteDtoV2(new RouteDtoV2.Point(4, 4), new RouteDtoV2.Point(5, 5)))));
-
- List route = request.runningData().route().stream()
+ runningRecordMetrics);
+
+ RunningRecord expectedRecord = createRecordFrom(member, request);
+
+ given(memberRepository.findById(member.memberId())).willReturn(Optional.of(member));
+ given(runningRecordRepository.save(expectedRecord)).willReturn(expectedRecord);
+
+ // when
+ RunningResultDto response = runningRecordService.addRunningRecordV2(member.memberId(), request);
+
+ // then
+ assertEquals(
+ com.dnd.runus.presentation.v1.running.dto.request.RunningAchievementMode.NORMAL,
+ response.runningAchievementMode());
+ assertNull(response.challengeAchievement());
+ assertNull(response.goalAchievement());
+ assertNull(response.percentage());
+
+ RunningRecord resultRunning = response.runningRecord();
+ for (int i = 0; i < resultRunning.route().size(); i++) {
+ assertEquals(i, resultRunning.route().get(i).longitude());
+ assertEquals(i, resultRunning.route().get(i).latitude());
+ }
+ }
+
+ @Test
+ @DisplayName("러닝 모드가가 챌린지(어제보다 ~) : 기준이 되는 러닝 기록의 어제 러닝 기록이 없으면 NotFoundException를 발생한다.")
+ void addRunningRecordV2_ChallengeMode_DefeatYesterday_RunningRecordNotFoundError() {
+ // given
+ Member member = new Member(MemberRole.USER, "nickname1");
+ Challenge challenge =
+ new Challenge(1L, "어제보다 1km 더 달리기", 360, "image url", true, ChallengeType.DEFEAT_YESTERDAY);
+ RunningRecordRequestV2 request = new RunningRecordRequestV2(
+ LocalDateTime.of(2021, 1, 1, 12, 10, 30),
+ LocalDateTime.of(2021, 1, 1, 13, 12, 10),
+ "start location",
+ "end location",
+ RunningEmoji.VERY_GOOD,
+ com.dnd.runus.presentation.v1.running.dto.request.RunningAchievementMode.CHALLENGE,
+ new RunningRecordRequestV2.ChallengeAchievedDto(challenge.challengeId(), true),
+ null,
+ runningRecordMetrics);
+ RunningRecord expectedRecord = createRecordFrom(member, request);
+ ChallengeAchievement expectedChallengeAchievement = new ChallengeAchievement(
+ challenge, expectedRecord, request.challengeValues().isSuccess());
+ given(memberRepository.findById(member.memberId())).willReturn(Optional.of(member));
+ given(runningRecordRepository.save(expectedRecord)).willReturn(expectedRecord);
+ given(challengeRepository.findById(request.challengeValues().challengeId()))
+ .willReturn(Optional.of(challenge));
+ given(challengeAchievementRepository.save(expectedChallengeAchievement))
+ .willReturn(expectedChallengeAchievement);
+ given(challengeRepository.findChallengeWithConditionsByChallengeId(challenge.challengeId()))
+ .willReturn(Optional.of(new ChallengeWithCondition(
+ challenge,
+ List.of(new ChallengeCondition(
+ GoalMetricType.DISTANCE, ComparisonType.GREATER_THAN_OR_EQUAL_TO, 1000)))));
+ OffsetDateTime runningDate = expectedRecord
+ .startAt()
+ .toLocalDate()
+ .atStartOfDay(expectedRecord.startAt().getZone())
+ .toOffsetDateTime();
+ given(runningRecordRepository.findByMemberIdAndStartAtBetween(
+ member.memberId(), runningDate.minusDays(1), runningDate))
+ .willReturn(List.of());
+
+ // when, then
+ assertThrows(
+ NotFoundException.class, () -> runningRecordService.addRunningRecordV2(member.memberId(), request));
+ }
+
+ @Test
+ @DisplayName("러닝 모드가 챌린지(어제보다 ~km더 뛰기) : 챌린지 관련 값이 정상적으로 저장되었는지 확인한다.")
+ void addRunningRecordV2_ChallengeMode_DefeatYesterday_Distance() {
+ // given
+ Member member = new Member(MemberRole.USER, "nickname1");
+ Challenge challenge =
+ new Challenge(1L, "어제보다 1km 더 달리기", 360, "image url", true, ChallengeType.DEFEAT_YESTERDAY);
+ RunningRecordRequestV2 request = new RunningRecordRequestV2(
+ LocalDateTime.of(2021, 1, 1, 12, 10, 30),
+ LocalDateTime.of(2021, 1, 1, 13, 12, 10),
+ "start location",
+ "end location",
+ RunningEmoji.VERY_GOOD,
+ com.dnd.runus.presentation.v1.running.dto.request.RunningAchievementMode.CHALLENGE,
+ new RunningRecordRequestV2.ChallengeAchievedDto(challenge.challengeId(), false),
+ null,
+ runningRecordMetrics);
+ RunningRecord expectedRecord = createRecordFrom(member, request);
+ ChallengeAchievement expectedChallengeAchievement = new ChallengeAchievement(
+ challenge, expectedRecord, request.challengeValues().isSuccess());
+ OffsetDateTime runningDate = expectedRecord
+ .startAt()
+ .toLocalDate()
+ .atStartOfDay(expectedRecord.startAt().getZone())
+ .toOffsetDateTime();
+ RunningRecord yesterdayRunning = RunningRecord.builder()
+ .member(member)
+ .startAt(expectedRecord.startAt().minusDays(1))
+ .endAt(expectedRecord.endAt().minusDays(1))
+ .emoji(expectedRecord.emoji())
+ .startLocation(expectedRecord.startLocation())
+ .endLocation(expectedRecord.endLocation())
+ .distanceMeter(expectedRecord.distanceMeter())
+ .duration(expectedRecord.duration())
+ .calorie(expectedRecord.calorie())
+ .averagePace(expectedRecord.averagePace())
+ .route(expectedRecord.route())
+ .build();
+ given(memberRepository.findById(member.memberId())).willReturn(Optional.of(member));
+ given(runningRecordRepository.save(expectedRecord)).willReturn(expectedRecord);
+ given(challengeRepository.findById(request.challengeValues().challengeId()))
+ .willReturn(Optional.of(challenge));
+ given(challengeAchievementRepository.save(expectedChallengeAchievement))
+ .willReturn(expectedChallengeAchievement);
+ given(challengeRepository.findChallengeWithConditionsByChallengeId(challenge.challengeId()))
+ .willReturn(Optional.of(new ChallengeWithCondition(
+ challenge,
+ List.of(new ChallengeCondition(
+ GoalMetricType.DISTANCE, ComparisonType.GREATER_THAN_OR_EQUAL_TO, 1000)))));
+ given(runningRecordRepository.findByMemberIdAndStartAtBetween(
+ member.memberId(), runningDate.minusDays(1), runningDate))
+ .willReturn(List.of(yesterdayRunning));
+
+ // when
+ RunningResultDto result = runningRecordService.addRunningRecordV2(member.memberId(), request);
+
+ // then
+ assertEquals(
+ result.runningAchievementMode(),
+ com.dnd.runus.presentation.v1.running.dto.request.RunningAchievementMode.CHALLENGE);
+ assertNotNull(result.challengeAchievement());
+ assertNotNull(result.percentage());
+ assertEquals(0.5, result.percentage());
+ }
+
+ @Test
+ @DisplayName("러닝 모드가가 챌린지(어제보다 ~분 더 뛰기) : 챌린지 관련 값이 정상적으로 저장되었는지 확인한다.")
+ void addRunningRecordV2_ChallengeMode_DefeatYesterday_Time() {
+ // given
+ Member member = new Member(MemberRole.USER, "nickname1");
+ Challenge challenge =
+ new Challenge(1L, "어제보다 30분 더 달리기", 360, "image url", true, ChallengeType.DEFEAT_YESTERDAY);
+ RunningRecordRequestV2 request = new RunningRecordRequestV2(
+ LocalDateTime.of(2021, 1, 1, 12, 10, 30),
+ LocalDateTime.of(2021, 1, 1, 13, 12, 10),
+ "start location",
+ "end location",
+ RunningEmoji.VERY_GOOD,
+ com.dnd.runus.presentation.v1.running.dto.request.RunningAchievementMode.CHALLENGE,
+ new RunningRecordRequestV2.ChallengeAchievedDto(challenge.challengeId(), false),
+ null,
+ runningRecordMetrics);
+ RunningRecord expectedRecord = createRecordFrom(member, request);
+ ChallengeAchievement expectedChallengeAchievement = new ChallengeAchievement(
+ challenge, expectedRecord, request.challengeValues().isSuccess());
+ OffsetDateTime runningDate = expectedRecord
+ .startAt()
+ .toLocalDate()
+ .atStartOfDay(expectedRecord.startAt().getZone())
+ .toOffsetDateTime();
+ RunningRecord yesterdayRunning = RunningRecord.builder()
+ .member(member)
+ .startAt(expectedRecord.startAt().minusDays(1))
+ .endAt(expectedRecord.endAt().minusDays(1))
+ .emoji(expectedRecord.emoji())
+ .startLocation(expectedRecord.startLocation())
+ .endLocation(expectedRecord.endLocation())
+ .distanceMeter(expectedRecord.distanceMeter())
+ .duration(expectedRecord.duration())
+ .calorie(expectedRecord.calorie())
+ .averagePace(expectedRecord.averagePace())
+ .route(expectedRecord.route())
+ .build();
+ given(memberRepository.findById(member.memberId())).willReturn(Optional.of(member));
+ given(runningRecordRepository.save(expectedRecord)).willReturn(expectedRecord);
+ given(challengeRepository.findById(request.challengeValues().challengeId()))
+ .willReturn(Optional.of(challenge));
+ given(challengeAchievementRepository.save(expectedChallengeAchievement))
+ .willReturn(expectedChallengeAchievement);
+ given(challengeRepository.findChallengeWithConditionsByChallengeId(challenge.challengeId()))
+ .willReturn(Optional.of(new ChallengeWithCondition(
+ challenge,
+ List.of(new ChallengeCondition(
+ GoalMetricType.TIME, ComparisonType.GREATER_THAN_OR_EQUAL_TO, 1800)))));
+ given(runningRecordRepository.findByMemberIdAndStartAtBetween(
+ member.memberId(), runningDate.minusDays(1), runningDate))
+ .willReturn(List.of(yesterdayRunning));
+
+ // when
+ RunningResultDto result = runningRecordService.addRunningRecordV2(member.memberId(), request);
+
+ // then
+ assertEquals(
+ result.runningAchievementMode(),
+ com.dnd.runus.presentation.v1.running.dto.request.RunningAchievementMode.CHALLENGE);
+ assertNotNull(result.challengeAchievement());
+ assertNotNull(result.percentage());
+ assertEquals(0.25, result.percentage());
+ }
+
+ @Test
+ @DisplayName("러닝 모드가가 챌린지이고 챌린지가 성공일 경우 퍼센티이지 값은 1를 리턴한다.")
+ void addRunningRecordV2_ChallengeMode_CheckPercentage() {
+ // given
+ Member member = new Member(MemberRole.USER, "nickname1");
+ Challenge challenge = new Challenge(1L, "500m 달리기", 360, "image url", true, ChallengeType.TODAY);
+ RunningRecordRequestV2 request = new RunningRecordRequestV2(
+ LocalDateTime.of(2021, 1, 1, 12, 10, 30),
+ LocalDateTime.of(2021, 1, 1, 13, 12, 10),
+ "start location",
+ "end location",
+ RunningEmoji.VERY_GOOD,
+ com.dnd.runus.presentation.v1.running.dto.request.RunningAchievementMode.CHALLENGE,
+ new RunningRecordRequestV2.ChallengeAchievedDto(challenge.challengeId(), true),
+ null,
+ runningRecordMetrics);
+ RunningRecord expectedRecord = createRecordFrom(member, request);
+ ChallengeAchievement expectedChallengeAchievement = new ChallengeAchievement(
+ challenge, expectedRecord, request.challengeValues().isSuccess());
+ given(memberRepository.findById(member.memberId())).willReturn(Optional.of(member));
+ given(runningRecordRepository.save(expectedRecord)).willReturn(expectedRecord);
+ given(challengeRepository.findById(request.challengeValues().challengeId()))
+ .willReturn(Optional.of(challenge));
+ given(challengeAchievementRepository.save(expectedChallengeAchievement))
+ .willReturn(expectedChallengeAchievement);
+ given(challengeRepository.findChallengeWithConditionsByChallengeId(challenge.challengeId()))
+ .willReturn(Optional.of(new ChallengeWithCondition(
+ challenge,
+ List.of(new ChallengeCondition(
+ GoalMetricType.DISTANCE, ComparisonType.GREATER_THAN_OR_EQUAL_TO, 500)))));
+
+ // when
+ RunningResultDto result = runningRecordService.addRunningRecordV2(member.memberId(), request);
+
+ // then
+ assertEquals(
+ result.runningAchievementMode(),
+ com.dnd.runus.presentation.v1.running.dto.request.RunningAchievementMode.CHALLENGE);
+ assertNotNull(result.challengeAchievement());
+ assertNotNull(result.percentage());
+ assertEquals(1, result.percentage());
+ }
+
+ private RunningRecord createRecordFrom(Member member, RunningRecordRequestV2 request) {
+ List coordinatePoints = request.runningData().route().stream()
.flatMap(point -> Stream.of(
new CoordinatePoint(
point.start().longitude(), point.start().latitude()),
@@ -561,9 +915,7 @@ void addRunningRecordV2_CheckRoute() {
point.end().longitude(), point.end().latitude())))
.collect(Collectors.toList());
- Member member = new Member(MemberRole.USER, "nickname1");
-
- RunningRecord expectedRecord = RunningRecord.builder()
+ return RunningRecord.builder()
.member(member)
.startAt(request.startAt().atZone(defaultZoneOffset))
.endAt(request.endAt().atZone(defaultZoneOffset))
@@ -576,27 +928,8 @@ void addRunningRecordV2_CheckRoute() {
.averagePace(Pace.from(
request.runningData().distanceMeter(),
request.runningData().runningTime()))
- .route(route)
+ .route(coordinatePoints)
.build();
-
- given(memberRepository.findById(1L)).willReturn(Optional.of(member));
- given(runningRecordRepository.save(expectedRecord)).willReturn(expectedRecord);
-
- // when
- RunningResultDto response = runningRecordService.addRunningRecordV2(1L, request);
-
- // then
- assertEquals(
- com.dnd.runus.presentation.v1.running.dto.request.RunningAchievementMode.NORMAL,
- response.runningAchievementMode());
- assertNull(response.challengeAchievement());
- assertNull(response.goalAchievement());
-
- RunningRecord resultRunning = response.runningRecord();
- for (int i = 0; i < resultRunning.route().size(); i++) {
- assertEquals(i, resultRunning.route().get(i).longitude());
- assertEquals(i, resultRunning.route().get(i).latitude());
- }
}
}
}
diff --git a/src/test/java/com/dnd/runus/presentation/v2/running/RunningRecordControllerV2Test.java b/src/test/java/com/dnd/runus/presentation/v2/running/RunningRecordControllerV2Test.java
index 7dc00cd8..9bda3117 100644
--- a/src/test/java/com/dnd/runus/presentation/v2/running/RunningRecordControllerV2Test.java
+++ b/src/test/java/com/dnd/runus/presentation/v2/running/RunningRecordControllerV2Test.java
@@ -110,6 +110,7 @@ void addRunningRecord_Normal_CheckRunningPath() throws Exception {
createRunningRecordFrom(request),
com.dnd.runus.presentation.v1.running.dto.request.RunningAchievementMode.NORMAL,
null,
+ null,
null));
// when
@@ -121,8 +122,7 @@ void addRunningRecord_Normal_CheckRunningPath() throws Exception {
result.andExpect(status().isOk())
.andExpect(jsonPath("$.data.emotion").value("very-good"))
.andExpect(jsonPath("$.data.achievementMode").value("normal"))
- .andExpect(jsonPath("$.data.challenge").doesNotExist())
- .andExpect(jsonPath("$.data.goal").doesNotExist())
+ .andExpect(jsonPath("$.data.achievementResult").doesNotExist())
.andExpect(jsonPath("$.data.runningData.averagePace").value("5’30”"))
.andExpect(
jsonPath("$.data.runningData.route[0].start.longitude").value("1.0"))