From 6bb10fdce6ea3043b1258776c7f37d99f7cf85e2 Mon Sep 17 00:00:00 2001
From: ddingmin
Date: Tue, 13 Feb 2024 00:55:31 +0900
Subject: [PATCH 01/13] =?UTF-8?q?fix:=20=EC=B6=A9=EB=8F=8C=20=ED=95=B4?=
=?UTF-8?q?=EA=B2=B0=20(#210)?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
src/main/java/net/teumteum/meeting/service/MeetingService.java | 1 -
1 file changed, 1 deletion(-)
diff --git a/src/main/java/net/teumteum/meeting/service/MeetingService.java b/src/main/java/net/teumteum/meeting/service/MeetingService.java
index 2657f43..99cfc41 100644
--- a/src/main/java/net/teumteum/meeting/service/MeetingService.java
+++ b/src/main/java/net/teumteum/meeting/service/MeetingService.java
@@ -112,7 +112,6 @@ public PageDto getMeetingsBySpecification(Pageable pageable, T
} else if (Boolean.TRUE.equals(isBookmarked)) {
spec = MeetingSpecification.withBookmarkedUserId(userId);
}
-
var meetings = meetingRepository.findAll(spec, pageable);
return PageDto.of(MeetingsResponse.of(meetings.getContent()), meetings.hasNext());
From 9eac8c543cc919e3ec3dca4e8726d75d97607440 Mon Sep 17 00:00:00 2001
From: ChoiDongKuen
Date: Tue, 13 Feb 2024 02:16:48 +0900
Subject: [PATCH 02/13] =?UTF-8?q?fix:=20=EC=86=8C=EC=85=9C=20=EB=A1=9C?=
=?UTF-8?q?=EA=B7=B8=EC=9D=B8=20=EC=97=90=EB=9F=AC=20=EA=B8=B4=EA=B8=89=20?=
=?UTF-8?q?=EC=88=98=EC=A0=95=20(#214)=20(#215)?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
src/main/java/net/teumteum/alert/app/AlertHandler.java | 4 ++--
.../core/security/filter/JwtAuthenticationFilter.java | 2 +-
2 files changed, 3 insertions(+), 3 deletions(-)
diff --git a/src/main/java/net/teumteum/alert/app/AlertHandler.java b/src/main/java/net/teumteum/alert/app/AlertHandler.java
index 86e6bc6..56b6400 100644
--- a/src/main/java/net/teumteum/alert/app/AlertHandler.java
+++ b/src/main/java/net/teumteum/alert/app/AlertHandler.java
@@ -45,7 +45,7 @@ public void handleBeforeMeetingAlerts(BeforeMeetingAlerted alerted) {
@Async(ALERT_EXECUTOR)
@EventListener(EndMeetingAlerted.class)
- public void handleEndMeetingAlerts(EndMeetingAlerted alerted) {
+ public void handleStartMeetingAlerts(EndMeetingAlerted alerted) {
userAlertService.findAllByUserId(alerted.userIds())
.stream()
.map(userAlert -> Pair.of(userAlert.getToken(),
@@ -64,7 +64,7 @@ private String toCommaString(List ids) {
for (int i = 0; i < ids.size() - 1; i++) {
stringBuilder.append(ids.get(i)).append(",");
}
- stringBuilder.append(ids.get(ids.size() - 1));
+ stringBuilder.append(ids.getLast());
return stringBuilder.toString();
}
diff --git a/src/main/java/net/teumteum/core/security/filter/JwtAuthenticationFilter.java b/src/main/java/net/teumteum/core/security/filter/JwtAuthenticationFilter.java
index 20ef8e3..36e960d 100644
--- a/src/main/java/net/teumteum/core/security/filter/JwtAuthenticationFilter.java
+++ b/src/main/java/net/teumteum/core/security/filter/JwtAuthenticationFilter.java
@@ -72,7 +72,7 @@ private void saveUserAuthentication(User user) {
private String resolveTokenFromRequest(HttpServletRequest request) {
String token = request.getHeader(jwtProperty.getAccess().getHeader());
- if (token.toLowerCase().startsWith(jwtProperty.getBearer().toLowerCase())) {
+ if (StringUtils.hasText(token) && token.toLowerCase().startsWith(jwtProperty.getBearer().toLowerCase())) {
return token.substring(7);
}
return null;
From 1a7052c14d5cec13ed0fd0ccba918f15e707f84b Mon Sep 17 00:00:00 2001
From: ChoiDongKuen
Date: Wed, 14 Feb 2024 04:22:36 +0900
Subject: [PATCH 03/13] =?UTF-8?q?feat:=20=EA=B8=B0=EC=A1=B4=20=ED=9A=8C?=
=?UTF-8?q?=EC=9B=90=20=EB=A6=AC=EB=B7=B0=20=EC=A1=B0=ED=9A=8C=20API=20?=
=?UTF-8?q?=EB=A1=9C=EC=A7=81=20=EB=B3=80=EA=B2=BD=20(#218)?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
* feat: userId 에 해당하는 유저 리뷰 조회 API 로직 변경 (#217)
* test: 통합테스트 수정 (#217)
* test: 단위테스트 수정 (#217)
---
.../user/controller/UserController.java | 6 +++---
.../teumteum/user/domain/UserRepository.java | 9 ++++----
.../domain/response/UserReviewResponse.java | 10 +++++++++
.../domain/response/UserReviewsResponse.java | 8 ++++---
.../teumteum/user/service/UserService.java | 6 ++++--
.../java/net/teumteum/integration/Api.java | 4 ++--
.../integration/UserIntegrationTest.java | 4 +++-
.../user/controller/UserControllerTest.java | 19 ++++++++---------
.../unit/user/service/UserServiceTest.java | 21 ++++++++++---------
.../user/domain/UserRepositoryTest.java | 7 ++++---
10 files changed, 56 insertions(+), 38 deletions(-)
create mode 100644 src/main/java/net/teumteum/user/domain/response/UserReviewResponse.java
diff --git a/src/main/java/net/teumteum/user/controller/UserController.java b/src/main/java/net/teumteum/user/controller/UserController.java
index 6523683..feb37d9 100644
--- a/src/main/java/net/teumteum/user/controller/UserController.java
+++ b/src/main/java/net/teumteum/user/controller/UserController.java
@@ -116,10 +116,10 @@ public void registerReview(
userService.registerReview(meetingId, getCurrentUserId(), request);
}
- @GetMapping("/reviews")
+ @GetMapping("/{userId}/reviews")
@ResponseStatus(HttpStatus.OK)
- public List getUserReviews() {
- return userService.getUserReviews(getCurrentUserId());
+ public UserReviewsResponse getUserReviews(@PathVariable("userId") Long userId) {
+ return userService.getUserReviews(userId);
}
@ResponseStatus(HttpStatus.BAD_REQUEST)
diff --git a/src/main/java/net/teumteum/user/domain/UserRepository.java b/src/main/java/net/teumteum/user/domain/UserRepository.java
index d0380f2..b611a17 100644
--- a/src/main/java/net/teumteum/user/domain/UserRepository.java
+++ b/src/main/java/net/teumteum/user/domain/UserRepository.java
@@ -3,7 +3,7 @@
import java.util.List;
import java.util.Optional;
import net.teumteum.core.security.Authenticated;
-import net.teumteum.user.domain.response.UserReviewsResponse;
+import net.teumteum.user.domain.response.UserReviewResponse;
import org.springframework.data.jpa.repository.JpaRepository;
import org.springframework.data.jpa.repository.Query;
import org.springframework.data.repository.query.Param;
@@ -15,7 +15,8 @@ public interface UserRepository extends JpaRepository {
Optional findByAuthenticatedAndOAuthId(@Param("authenticated") Authenticated authenticated,
@Param("oAuthId") String oAuthId);
- @Query("select new net.teumteum.user.domain.response.UserReviewsResponse(r,count(r)) "
- + "from users u join u.reviews r where u.id = :userId group by r")
- List countUserReviewsByUserId(@Param("userId") Long userId);
+ @Query("select new net.teumteum.user.domain.response.UserReviewResponse(r, count(r)) "
+ + "from users u join u.reviews r where u = :user group by r")
+ List countUserReviewsByUser(@Param("user") User user);
+
}
diff --git a/src/main/java/net/teumteum/user/domain/response/UserReviewResponse.java b/src/main/java/net/teumteum/user/domain/response/UserReviewResponse.java
new file mode 100644
index 0000000..8ec04a8
--- /dev/null
+++ b/src/main/java/net/teumteum/user/domain/response/UserReviewResponse.java
@@ -0,0 +1,10 @@
+package net.teumteum.user.domain.response;
+
+import net.teumteum.user.domain.Review;
+
+public record UserReviewResponse(
+ Review review,
+ long count
+) {
+
+}
diff --git a/src/main/java/net/teumteum/user/domain/response/UserReviewsResponse.java b/src/main/java/net/teumteum/user/domain/response/UserReviewsResponse.java
index 7454f52..bf53ed6 100644
--- a/src/main/java/net/teumteum/user/domain/response/UserReviewsResponse.java
+++ b/src/main/java/net/teumteum/user/domain/response/UserReviewsResponse.java
@@ -1,10 +1,12 @@
package net.teumteum.user.domain.response;
-import net.teumteum.user.domain.Review;
+import java.util.List;
public record UserReviewsResponse(
- Review review,
- long count
+ List reviews
) {
+ public static UserReviewsResponse of(List reviews) {
+ return new UserReviewsResponse(reviews);
+ }
}
diff --git a/src/main/java/net/teumteum/user/service/UserService.java b/src/main/java/net/teumteum/user/service/UserService.java
index 8429d05..723d628 100644
--- a/src/main/java/net/teumteum/user/service/UserService.java
+++ b/src/main/java/net/teumteum/user/service/UserService.java
@@ -116,8 +116,10 @@ public void registerReview(Long meetingId, Long currentUserId, ReviewRegisterReq
});
}
- public List getUserReviews(Long userId) {
- return userRepository.countUserReviewsByUserId(userId);
+ public UserReviewsResponse getUserReviews(Long userId) {
+ var user = getUser(userId);
+
+ return UserReviewsResponse.of(userRepository.countUserReviewsByUser(user));
}
public FriendsResponse findFriendsByUserId(Long userId) {
diff --git a/src/test/java/net/teumteum/integration/Api.java b/src/test/java/net/teumteum/integration/Api.java
index a65e350..0002b52 100644
--- a/src/test/java/net/teumteum/integration/Api.java
+++ b/src/test/java/net/teumteum/integration/Api.java
@@ -196,10 +196,10 @@ ResponseSpec deleteMeeting(String accessToken, Long meetingId) {
.exchange();
}
- ResponseSpec getUserReviews(String accessToken) {
+ ResponseSpec getUserReviews(Long userId, String accessToken) {
return webTestClient
.get()
- .uri("/users/reviews")
+ .uri("/users/" + userId + "/reviews")
.header(HttpHeaders.AUTHORIZATION, accessToken)
.exchange();
}
diff --git a/src/test/java/net/teumteum/integration/UserIntegrationTest.java b/src/test/java/net/teumteum/integration/UserIntegrationTest.java
index 916bb80..bb42461 100644
--- a/src/test/java/net/teumteum/integration/UserIntegrationTest.java
+++ b/src/test/java/net/teumteum/integration/UserIntegrationTest.java
@@ -346,10 +346,12 @@ void Get_user_reviews() {
// given
var existUser = repository.saveAndGetUser();
+ var userId = existUser.getId();
+
securityContextSetting.set(existUser.getId());
// when
- var expected = api.getUserReviews(VALID_TOKEN);
+ var expected = api.getUserReviews(userId, VALID_TOKEN);
// then
Assertions.assertThat(expected.expectStatus().isOk()
diff --git a/src/test/java/net/teumteum/unit/user/controller/UserControllerTest.java b/src/test/java/net/teumteum/unit/user/controller/UserControllerTest.java
index f425e05..77afd49 100644
--- a/src/test/java/net/teumteum/unit/user/controller/UserControllerTest.java
+++ b/src/test/java/net/teumteum/unit/user/controller/UserControllerTest.java
@@ -37,6 +37,7 @@
import net.teumteum.user.domain.request.UserRegisterRequest;
import net.teumteum.user.domain.request.UserWithdrawRequest;
import net.teumteum.user.domain.response.UserRegisterResponse;
+import net.teumteum.user.domain.response.UserReviewResponse;
import net.teumteum.user.domain.response.UserReviewsResponse;
import net.teumteum.user.service.UserService;
import org.junit.jupiter.api.BeforeEach;
@@ -238,27 +239,25 @@ void Register_reviews_with_400_bad_request() throws Exception {
class Get_user_reviews_api_unit {
@Test
- @DisplayName("로그인한 회원 id 에 해당하는 회원 리뷰와 200 OK을 반환한다.")
+ @DisplayName("user id 에 해당하는 회원 리뷰와 200 OK을 반환한다.")
void Get_user_reviews_with_200_ok() throws Exception {
// given
var userId = 1L;
- given(securityService.getCurrentUserId()).willReturn(userId);
-
given(userService.getUserReviews(anyLong()))
- .willReturn(List.of(new UserReviewsResponse(별로에요, 2L),
- new UserReviewsResponse(최고에요, 3L)));
+ .willReturn(UserReviewsResponse.of(List.of(new UserReviewResponse(별로에요, 2L),
+ new UserReviewResponse(최고에요, 3L))));
// when & then
- mockMvc.perform(get("/users/reviews")
+ mockMvc.perform(get("/users/{userId}/reviews", 1)
.with(csrf())
.header(AUTHORIZATION, VALID_ACCESS_TOKEN))
.andDo(print())
.andExpect(status().isOk())
- .andExpect(jsonPath("$", notNullValue()))
- .andExpect(jsonPath("$", hasSize(2)))
- .andExpect(jsonPath("$[0].count", is(2)))
- .andExpect(jsonPath("$[0].review").value("별로에요"));
+ .andExpect(jsonPath("$.reviews", notNullValue()))
+ .andExpect(jsonPath("$.reviews", hasSize(2)))
+ .andExpect(jsonPath("$.reviews[0].count", is(2)))
+ .andExpect(jsonPath("$.reviews[0].review").value("별로에요"));
}
}
diff --git a/src/test/java/net/teumteum/unit/user/service/UserServiceTest.java b/src/test/java/net/teumteum/unit/user/service/UserServiceTest.java
index f682353..66b1668 100644
--- a/src/test/java/net/teumteum/unit/user/service/UserServiceTest.java
+++ b/src/test/java/net/teumteum/unit/user/service/UserServiceTest.java
@@ -30,7 +30,7 @@
import net.teumteum.user.domain.request.UserRegisterRequest;
import net.teumteum.user.domain.request.UserWithdrawRequest;
import net.teumteum.user.domain.response.UserRegisterResponse;
-import net.teumteum.user.domain.response.UserReviewsResponse;
+import net.teumteum.user.domain.response.UserReviewResponse;
import net.teumteum.user.service.UserService;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.DisplayName;
@@ -251,23 +251,24 @@ class Get_user_reviews_api_unit {
void Return_user_reviews_with_200_ok() {
// given
var userId = 1L;
+ var existUser = UserFixture.getIdUser();
- var response = List.of(new UserReviewsResponse(최고에요, 2L)
- , new UserReviewsResponse(별로에요, 3L));
+ var response = List.of(new UserReviewResponse(최고에요, 2L)
+ , new UserReviewResponse(별로에요, 3L));
- given(userRepository.countUserReviewsByUserId(anyLong())).willReturn(response);
+ given(userRepository.findById(anyLong())).willReturn(Optional.of(existUser));
+ given(userRepository.countUserReviewsByUser(any(User.class))).willReturn(response);
// when
var result = userService.getUserReviews(userId);
// then
- assertThat(result).hasSize(2);
- assertThat(result.get(0).review()).isEqualTo(최고에요);
- assertThat(result.get(0).count()).isEqualTo(2L);
- assertThat(result.get(1).review()).isEqualTo(별로에요);
- assertThat(result.get(1).count()).isEqualTo(3L);
+ assertThat(result.reviews()).hasSize(2);
+ assertThat(result.reviews().get(0).review()).isEqualTo(최고에요);
+ assertThat(result.reviews().get(0).count()).isEqualTo(2L);
+ assertThat(result.reviews().get(1).review()).isEqualTo(별로에요);
- verify(userRepository, times(1)).countUserReviewsByUserId(anyLong());
+ verify(userRepository, times(1)).countUserReviewsByUser(any(User.class));
}
}
}
diff --git a/src/test/java/net/teumteum/user/domain/UserRepositoryTest.java b/src/test/java/net/teumteum/user/domain/UserRepositoryTest.java
index 59d607b..a046f11 100644
--- a/src/test/java/net/teumteum/user/domain/UserRepositoryTest.java
+++ b/src/test/java/net/teumteum/user/domain/UserRepositoryTest.java
@@ -7,7 +7,7 @@
import jakarta.persistence.EntityManager;
import java.util.Optional;
import net.teumteum.core.config.AppConfig;
-import net.teumteum.user.domain.response.UserReviewsResponse;
+import net.teumteum.user.domain.response.UserReviewResponse;
import org.assertj.core.api.Assertions;
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Nested;
@@ -85,13 +85,14 @@ void Count_user_reviews_by_user_id() {
entityManager.clear();
// when
- var result = userRepository.countUserReviewsByUserId(id);
+ var result = userRepository.countUserReviewsByUser(existUser);
// then
Assertions.assertThat(result)
.isNotEmpty()
.hasSize(3)
- .extracting(UserReviewsResponse::review, UserReviewsResponse::count)
+ .extracting(UserReviewResponse::review,
+ UserReviewResponse::count)
.contains(
Assertions.tuple(최고에요, 3L),
Assertions.tuple(별로에요, 1L),
From d7f3a3e44c890ba7731e0d455c12e45646407ca8 Mon Sep 17 00:00:00 2001
From: ddingmin
Date: Wed, 14 Feb 2024 22:58:46 +0900
Subject: [PATCH 04/13] =?UTF-8?q?docs:=20readme=20=EC=9E=91=EC=84=B1=20(#2?=
=?UTF-8?q?22)?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
README.md | 39 ++++++++++++++++++++++++++++++++++++++-
1 file changed, 38 insertions(+), 1 deletion(-)
diff --git a/README.md b/README.md
index af97483..7fd4c45 100644
--- a/README.md
+++ b/README.md
@@ -1 +1,38 @@
-# tmtm-server
\ No newline at end of file
+
+
+# Teum-Teum
+
+> 사람 사이의 **틈**을 이어주는 IT 커리어 네트워킹 서비스 `repo:server`
+
+[![download](https://img.shields.io/badge/playstore-download-brightgreen?style=for-the-badge&logo=google&logoColor=white&color=36B2FF)](https://play.google.com/store/apps/details?id=com.teumteum.teumteum&pcampaignid=web_share) ![Build](https://img.shields.io/github/actions/workflow/status/depromeet/teum-teum-server/integration-tester.yml?branch=develop&style=for-the-badge&logo=github&logoColor=white&color=36B2FF) ![](https://img.shields.io/sonar/test_success_density/depromeet_teum-teum-server?server=https%3A%2F%2Fsonarcloud.io&style=for-the-badge&logo=sonar&logoColor=white&color=36B2FF) ![](https://img.shields.io/sonar/quality_gate/depromeet_teum-teum-server/develop?server=https%3A%2F%2Fsonarcloud.io&style=for-the-badge&logo=sonar&logoColor=white&color=36B2FF) ![](https://img.shields.io/github/v/release/depromeet/teum-teum-server?include_prereleases&style=for-the-badge&color=36B2FF)
+
+## 👋 Introduction
+
+![Behance](https://img.shields.io/badge/Behance-1769ff?style=for-the-badge&logo=behance&logoColor=white)
+
+## 🌐 Architecture
+
+## ⚒️ Tech Stack
+
+ ![SpringBoot](https://img.shields.io/badge/springboot-%236DB33F.svg?style=for-the-badge&logo=springboot&logoColor=white)![spring_data_JPA](https://img.shields.io/badge/spring_data_JPA-%236DB33F?style=for-the-badge&logo=databricks&logoColor=white)![SpringSecurity](https://img.shields.io/badge/spring_security-%236DB33F.svg?style=for-the-badge&logo=springsecurity&logoColor=white)![Gradle](https://img.shields.io/badge/Gradle-02303A.svg?style=for-the-badge&logo=Gradle&logoColor=white)
+
+ ![junit5](https://img.shields.io/badge/junit5-25A162?style=for-the-badge&logo=junit5&logoColor=white)![test_containers](https://img.shields.io/badge/test_containers-328ba3?style=for-the-badge&logo=reasonstudios&logoColor=white)
+
+ ![MySQL](https://img.shields.io/badge/mysql-4479A1.svg?style=for-the-badge&logo=mysql&logoColor=white)![Redis](https://img.shields.io/badge/redis-%23DD0031.svg?style=for-the-badge&logo=redis&logoColor=white)![Firebase](https://img.shields.io/badge/Firebase-039BE5?style=for-the-badge&logo=Firebase&logoColor=white)![flyway](https://img.shields.io/badge/flyway-CC0200?style=for-the-badge&logo=flyway&logoColor=white)
+
+ ![Amazon Ec2](https://img.shields.io/badge/amazon_ec2-FF9900.svg?style=for-the-badge&logo=amazonec2&logoColor=white)![Amazon S3](https://img.shields.io/badge/AWS_S3-569A31.svg?style=for-the-badge&logo=amazons3&logoColor=white)![Amazon RDS](https://img.shields.io/badge/amazon_RDS-527FFF.svg?style=for-the-badge&logo=amazonrds&logoColor=white)![Amazon ElastiCache](https://img.shields.io/badge/amazon_elasticache-FF9900.svg?style=for-the-badge&logo=amazondocumentdb&logoColor=white)![Nginx](https://img.shields.io/badge/nginx-%23009639.svg?style=for-the-badge&logo=nginx&logoColor=white)
+
+ ![Docker](https://img.shields.io/badge/docker-%230db7ed.svg?style=for-the-badge&logo=docker&logoColor=white)![GitHub Actions](https://img.shields.io/badge/github%20actions-%232671E5.svg?style=for-the-badge&logo=githubactions&logoColor=white)![github container](https://img.shields.io/badge/github_container-181717.svg?style=for-the-badge&logo=github&logoColor=white)![squarespace](https://img.shields.io/badge/squarespace-000000?style=for-the-badge&logo=squarespace&logoColor=white)
+
+ ![SonarCloud](https://img.shields.io/badge/SonarCloud-F3702A?style=for-the-badge&logo=SonarCloud&logoColor=white)
+
+ ![ChatGPT](https://img.shields.io/badge/chatGPT-74aa9c?style=for-the-badge&logo=openai&logoColor=white)
+
+ ![sentry](https://img.shields.io/badge/sentry-362D59?style=for-the-badge&logo=sentry&logoColor=white)
+
+## 👥 Members
+
+| Server | Server | Server |
+|:----------------------------------------------------------------------------------------------------------------------------------------:|:----------------------------------------------------------------------------------------------------------------------------------------:|:----------------------------------------------------------------------------------------------------------------------------------------:|
+| | | |
+| [choidongkuen](https://github.com/choidongkuen) | [xb205](https://github.com/devxb) | [ddingmin](https://github.com/ddingmin) |
From f5603e0cd6c42bf2f038f46308b982b7add87009 Mon Sep 17 00:00:00 2001
From: ddingmin
Date: Wed, 14 Feb 2024 22:59:33 +0900
Subject: [PATCH 05/13] =?UTF-8?q?docs:=20readme=20=EC=9E=91=EC=84=B1?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
README.md | 24 ++++++++++++++++++++++--
1 file changed, 22 insertions(+), 2 deletions(-)
diff --git a/README.md b/README.md
index 7fd4c45..d2991fb 100644
--- a/README.md
+++ b/README.md
@@ -6,11 +6,31 @@
[![download](https://img.shields.io/badge/playstore-download-brightgreen?style=for-the-badge&logo=google&logoColor=white&color=36B2FF)](https://play.google.com/store/apps/details?id=com.teumteum.teumteum&pcampaignid=web_share) ![Build](https://img.shields.io/github/actions/workflow/status/depromeet/teum-teum-server/integration-tester.yml?branch=develop&style=for-the-badge&logo=github&logoColor=white&color=36B2FF) ![](https://img.shields.io/sonar/test_success_density/depromeet_teum-teum-server?server=https%3A%2F%2Fsonarcloud.io&style=for-the-badge&logo=sonar&logoColor=white&color=36B2FF) ![](https://img.shields.io/sonar/quality_gate/depromeet_teum-teum-server/develop?server=https%3A%2F%2Fsonarcloud.io&style=for-the-badge&logo=sonar&logoColor=white&color=36B2FF) ![](https://img.shields.io/github/v/release/depromeet/teum-teum-server?include_prereleases&style=for-the-badge&color=36B2FF)
-## 👋 Introduction
+---
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
![Behance](https://img.shields.io/badge/Behance-1769ff?style=for-the-badge&logo=behance&logoColor=white)
## 🌐 Architecture
+
+
+
## ⚒️ Tech Stack
@@ -30,7 +50,7 @@
![sentry](https://img.shields.io/badge/sentry-362D59?style=for-the-badge&logo=sentry&logoColor=white)
-## 👥 Members
+## 👥 Contributors
| Server | Server | Server |
|:----------------------------------------------------------------------------------------------------------------------------------------:|:----------------------------------------------------------------------------------------------------------------------------------------:|:----------------------------------------------------------------------------------------------------------------------------------------:|
From e3552eb54a2b978b76f63b536e177ab253785217 Mon Sep 17 00:00:00 2001
From: ddingmin
Date: Wed, 14 Feb 2024 23:49:18 +0900
Subject: [PATCH 06/13] =?UTF-8?q?docs:=20readme=20=EC=88=98=EC=A0=95?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
README.md | 23 ++++++++++-------------
1 file changed, 10 insertions(+), 13 deletions(-)
diff --git a/README.md b/README.md
index d2991fb..31a0898 100644
--- a/README.md
+++ b/README.md
@@ -4,7 +4,9 @@
> 사람 사이의 **틈**을 이어주는 IT 커리어 네트워킹 서비스 `repo:server`
-[![download](https://img.shields.io/badge/playstore-download-brightgreen?style=for-the-badge&logo=google&logoColor=white&color=36B2FF)](https://play.google.com/store/apps/details?id=com.teumteum.teumteum&pcampaignid=web_share) ![Build](https://img.shields.io/github/actions/workflow/status/depromeet/teum-teum-server/integration-tester.yml?branch=develop&style=for-the-badge&logo=github&logoColor=white&color=36B2FF) ![](https://img.shields.io/sonar/test_success_density/depromeet_teum-teum-server?server=https%3A%2F%2Fsonarcloud.io&style=for-the-badge&logo=sonar&logoColor=white&color=36B2FF) ![](https://img.shields.io/sonar/quality_gate/depromeet_teum-teum-server/develop?server=https%3A%2F%2Fsonarcloud.io&style=for-the-badge&logo=sonar&logoColor=white&color=36B2FF) ![](https://img.shields.io/github/v/release/depromeet/teum-teum-server?include_prereleases&style=for-the-badge&color=36B2FF)
+![Build](https://img.shields.io/github/actions/workflow/status/depromeet/teum-teum-server/integration-tester.yml?branch=develop&style=for-the-badge&logo=github&logoColor=white&color=36B2FF) ![](https://img.shields.io/sonar/test_success_density/depromeet_teum-teum-server?server=https%3A%2F%2Fsonarcloud.io&style=for-the-badge&logo=sonar&logoColor=white&color=36B2FF) ![](https://img.shields.io/sonar/quality_gate/depromeet_teum-teum-server/develop?server=https%3A%2F%2Fsonarcloud.io&style=for-the-badge&logo=sonar&logoColor=white&color=36B2FF) ![](https://img.shields.io/github/v/release/depromeet/teum-teum-server?include_prereleases&style=for-the-badge&color=36B2FF)
+
+[![download](https://img.shields.io/badge/playstore-download-brightgreen?style=social&logo=googleplay&color=36B2FF)](https://play.google.com/store/apps/details?id=com.teumteum.teumteum&pcampaignid=web_share) [![instagram](https://img.shields.io/badge/instagram-click-brightgreen?style=social&logo=instagram&color=36B2FF)](https://www.instagram.com/teumteum_official/) [![behance](https://img.shields.io/badge/behance-click-brightgreen?style=social&logo=behance&color=36B2FF)](https://www.behance.net/gallery/191510163/%08TEUMTEUM-IT-Career-Growth-Networking-Service)
---
@@ -25,30 +27,25 @@
-![Behance](https://img.shields.io/badge/Behance-1769ff?style=for-the-badge&logo=behance&logoColor=white)
-
## 🌐 Architecture
+
## ⚒️ Tech Stack
- ![SpringBoot](https://img.shields.io/badge/springboot-%236DB33F.svg?style=for-the-badge&logo=springboot&logoColor=white)![spring_data_JPA](https://img.shields.io/badge/spring_data_JPA-%236DB33F?style=for-the-badge&logo=databricks&logoColor=white)![SpringSecurity](https://img.shields.io/badge/spring_security-%236DB33F.svg?style=for-the-badge&logo=springsecurity&logoColor=white)![Gradle](https://img.shields.io/badge/Gradle-02303A.svg?style=for-the-badge&logo=Gradle&logoColor=white)
-
- ![junit5](https://img.shields.io/badge/junit5-25A162?style=for-the-badge&logo=junit5&logoColor=white)![test_containers](https://img.shields.io/badge/test_containers-328ba3?style=for-the-badge&logo=reasonstudios&logoColor=white)
-
- ![MySQL](https://img.shields.io/badge/mysql-4479A1.svg?style=for-the-badge&logo=mysql&logoColor=white)![Redis](https://img.shields.io/badge/redis-%23DD0031.svg?style=for-the-badge&logo=redis&logoColor=white)![Firebase](https://img.shields.io/badge/Firebase-039BE5?style=for-the-badge&logo=Firebase&logoColor=white)![flyway](https://img.shields.io/badge/flyway-CC0200?style=for-the-badge&logo=flyway&logoColor=white)
+![SpringBoot](https://img.shields.io/badge/springboot-%236DB33F.svg?style=for-the-badge&logo=springboot&logoColor=white)![spring_data_JPA](https://img.shields.io/badge/spring_data_JPA-%236DB33F?style=for-the-badge&logo=databricks&logoColor=white)![SpringSecurity](https://img.shields.io/badge/spring_security-%236DB33F.svg?style=for-the-badge&logo=springsecurity&logoColor=white) ![Gradle](https://img.shields.io/badge/Gradle-02303A.svg?style=for-the-badge&logo=Gradle&logoColor=white)
- ![Amazon Ec2](https://img.shields.io/badge/amazon_ec2-FF9900.svg?style=for-the-badge&logo=amazonec2&logoColor=white)![Amazon S3](https://img.shields.io/badge/AWS_S3-569A31.svg?style=for-the-badge&logo=amazons3&logoColor=white)![Amazon RDS](https://img.shields.io/badge/amazon_RDS-527FFF.svg?style=for-the-badge&logo=amazonrds&logoColor=white)![Amazon ElastiCache](https://img.shields.io/badge/amazon_elasticache-FF9900.svg?style=for-the-badge&logo=amazondocumentdb&logoColor=white)![Nginx](https://img.shields.io/badge/nginx-%23009639.svg?style=for-the-badge&logo=nginx&logoColor=white)
+![junit5](https://img.shields.io/badge/junit5-25A162?style=for-the-badge&logo=junit5&logoColor=white)![test_containers](https://img.shields.io/badge/test_containers-328ba3?style=for-the-badge&logo=reasonstudios&logoColor=white) ![gatling](https://img.shields.io/badge/gatling-FF9E2A?style=for-the-badge&logo=gatling&logoColor=white)
- ![Docker](https://img.shields.io/badge/docker-%230db7ed.svg?style=for-the-badge&logo=docker&logoColor=white)![GitHub Actions](https://img.shields.io/badge/github%20actions-%232671E5.svg?style=for-the-badge&logo=githubactions&logoColor=white)![github container](https://img.shields.io/badge/github_container-181717.svg?style=for-the-badge&logo=github&logoColor=white)![squarespace](https://img.shields.io/badge/squarespace-000000?style=for-the-badge&logo=squarespace&logoColor=white)
+![MySQL](https://img.shields.io/badge/mysql-4479A1.svg?style=for-the-badge&logo=mysql&logoColor=white)![Redis](https://img.shields.io/badge/redis-%23DD0031.svg?style=for-the-badge&logo=redis&logoColor=white)![Firebase](https://img.shields.io/badge/Firebase-039BE5?style=for-the-badge&logo=Firebase&logoColor=white) ![flyway](https://img.shields.io/badge/flyway-CC0200?style=for-the-badge&logo=flyway&logoColor=white)
- ![SonarCloud](https://img.shields.io/badge/SonarCloud-F3702A?style=for-the-badge&logo=SonarCloud&logoColor=white)
+![Amazon Ec2](https://img.shields.io/badge/amazon_ec2-FF9900.svg?style=for-the-badge&logo=amazonec2&logoColor=white)![Amazon S3](https://img.shields.io/badge/AWS_S3-569A31.svg?style=for-the-badge&logo=amazons3&logoColor=white)![Amazon RDS](https://img.shields.io/badge/amazon_RDS-527FFF.svg?style=for-the-badge&logo=amazonrds&logoColor=white)![Amazon ElastiCache](https://img.shields.io/badge/amazon_elasticache-FF9900.svg?style=for-the-badge&logo=amazondocumentdb&logoColor=white)![Nginx](https://img.shields.io/badge/nginx-%23009639.svg?style=for-the-badge&logo=nginx&logoColor=white)
- ![ChatGPT](https://img.shields.io/badge/chatGPT-74aa9c?style=for-the-badge&logo=openai&logoColor=white)
+![Docker](https://img.shields.io/badge/docker-%230db7ed.svg?style=for-the-badge&logo=docker&logoColor=white)![GitHub Actions](https://img.shields.io/badge/github%20actions-%232671E5.svg?style=for-the-badge&logo=githubactions&logoColor=white)![github container](https://img.shields.io/badge/github_container-181717.svg?style=for-the-badge&logo=github&logoColor=white) ![squarespace](https://img.shields.io/badge/squarespace-000000?style=for-the-badge&logo=squarespace&logoColor=white)
- ![sentry](https://img.shields.io/badge/sentry-362D59?style=for-the-badge&logo=sentry&logoColor=white)
+![SonarCloud](https://img.shields.io/badge/SonarCloud-F3702A?style=for-the-badge&logo=SonarCloud&logoColor=white) ![sentry](https://img.shields.io/badge/sentry-362D59?style=for-the-badge&logo=sentry&logoColor=white) ![ChatGPT](https://img.shields.io/badge/chatGPT-74aa9c?style=for-the-badge&logo=openai&logoColor=white)
## 👥 Contributors
From 97ee24618c3c0d08e2355e46eba95ab537483bd4 Mon Sep 17 00:00:00 2001
From: devxb
Date: Thu, 15 Feb 2024 21:23:51 +0900
Subject: [PATCH 07/13] =?UTF-8?q?test:=20QA=EC=9A=A9=20EndMeetingAlerted?=
=?UTF-8?q?=EB=A5=BC=20=EB=B0=9C=ED=96=89=ED=95=9C=EB=8B=A4?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
.../meeting/service/MeetingAlertPublisher.java | 18 +++++++++++++++++-
1 file changed, 17 insertions(+), 1 deletion(-)
diff --git a/src/main/java/net/teumteum/meeting/service/MeetingAlertPublisher.java b/src/main/java/net/teumteum/meeting/service/MeetingAlertPublisher.java
index 402c057..858650e 100644
--- a/src/main/java/net/teumteum/meeting/service/MeetingAlertPublisher.java
+++ b/src/main/java/net/teumteum/meeting/service/MeetingAlertPublisher.java
@@ -2,7 +2,6 @@
import java.time.LocalDateTime;
import java.time.ZoneId;
-import java.time.ZonedDateTime;
import lombok.RequiredArgsConstructor;
import net.teumteum.meeting.domain.BeforeMeetingAlerted;
import net.teumteum.meeting.domain.EndMeetingAlerted;
@@ -49,4 +48,21 @@ public void alertEndMeeting() {
new EndMeetingAlerted(meeting.getId(), meeting.getTitle(), meeting.getParticipantUserIds())
));
}
+
+ @Scheduled(cron = EVERY_ONE_MINUTES)
+ public void alertEndMeetingForQa() {
+ var today = LocalDateTime.now(ZoneId.of("Asia/Seoul"))
+ .withNano(0)
+ .withSecond(0)
+ .withMinute(0)
+ .withHour(0);
+
+ var future = today.plusDays(365);
+ var yesterday = today.minusDays(365);
+
+ var alertTargets = meetingRepository.findAlertMeetings(yesterday, future);
+ alertTargets.forEach(meeting -> eventPublisher.publishEvent(
+ new EndMeetingAlerted(meeting.getId(), meeting.getTitle(), meeting.getParticipantUserIds())
+ ));
+ }
}
From b71556dd3601418ad883ffac94f0a17d15d289d5 Mon Sep 17 00:00:00 2001
From: xb205 <62425964+devxb@users.noreply.github.com>
Date: Thu, 15 Feb 2024 23:02:12 +0900
Subject: [PATCH 08/13] =?UTF-8?q?feat:=20=EC=9C=A0=EC=A0=80=EA=B0=80=20?=
=?UTF-8?q?=EC=82=AD=EC=A0=9C=EB=90=98=EC=97=88=EC=9D=84=EB=95=8C,=20?=
=?UTF-8?q?=EC=95=8C=EB=9E=8C=ED=86=A0=ED=81=B0=EB=8F=84=20=EC=82=AD?=
=?UTF-8?q?=EC=A0=9C=ED=95=98=EB=8F=84=EB=A1=9D=20=EC=88=98=EC=A0=95=20(#2?=
=?UTF-8?q?26)?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
* feat: 유저가 삭제되었을때, 알람토큰도 삭제하도록 수정한다
* test: ApplicationEventPublisher 바인딩 안되는 테스트 삭제
* refactor: UserDeletedEvent 핸들러 이름 변경
* test: 친구조회 테스트 수정
* refactor: code smell을 제거한다
---
.../net/teumteum/alert/app/AlertHandler.java | 6 +++++
.../alert/domain/UserAlertService.java | 8 +++++-
.../user/domain/UserDeletedEvent.java | 7 +++++
.../teumteum/user/service/UserService.java | 4 +++
.../integration/UserIntegrationTest.java | 26 -------------------
.../unit/user/service/UserServiceTest.java | 25 ++----------------
6 files changed, 26 insertions(+), 50 deletions(-)
create mode 100644 src/main/java/net/teumteum/user/domain/UserDeletedEvent.java
diff --git a/src/main/java/net/teumteum/alert/app/AlertHandler.java b/src/main/java/net/teumteum/alert/app/AlertHandler.java
index 56b6400..9cf1e80 100644
--- a/src/main/java/net/teumteum/alert/app/AlertHandler.java
+++ b/src/main/java/net/teumteum/alert/app/AlertHandler.java
@@ -14,6 +14,7 @@
import net.teumteum.meeting.domain.BeforeMeetingAlerted;
import net.teumteum.meeting.domain.EndMeetingAlerted;
import net.teumteum.user.UserRecommended;
+import net.teumteum.user.domain.UserDeletedEvent;
import org.springframework.context.annotation.Profile;
import org.springframework.context.event.EventListener;
import org.springframework.data.util.Pair;
@@ -29,6 +30,11 @@ public class AlertHandler {
private final AlertService alertService;
private final AlertPublisher alertPublisher;
+ @EventListener(UserDeletedEvent.class)
+ public void handleDeleteUserEvent(UserDeletedEvent userDeletedEvent) {
+ userAlertService.deleteAlertByUserId(userDeletedEvent.id());
+ }
+
@Async(ALERT_EXECUTOR)
@EventListener(BeforeMeetingAlerted.class)
public void handleBeforeMeetingAlerts(BeforeMeetingAlerted alerted) {
diff --git a/src/main/java/net/teumteum/alert/domain/UserAlertService.java b/src/main/java/net/teumteum/alert/domain/UserAlertService.java
index 4c6cb18..898754b 100644
--- a/src/main/java/net/teumteum/alert/domain/UserAlertService.java
+++ b/src/main/java/net/teumteum/alert/domain/UserAlertService.java
@@ -19,13 +19,19 @@ public class UserAlertService {
public void registerAlert(Long userId, RegisterAlertRequest registerAlertRequest) {
alertRepository.findByUserId(userId)
.ifPresentOrElse(userAlert -> {
- throw new IllegalArgumentException("이미 토큰이 생성된 user입니다. \"" + userId +"\"");
+ throw new IllegalArgumentException("이미 토큰이 생성된 user입니다. \"" + userId + "\"");
}, () -> {
var alert = new UserAlert(null, userId, registerAlertRequest.token());
alertRepository.save(alert);
});
}
+ @Transactional
+ public void deleteAlertByUserId(Long userId) {
+ alertRepository.findByUserId(userId).ifPresentOrElse(alertRepository::delete, () -> {
+ });
+ }
+
@Transactional
public void updateAlertToken(Long userId, UpdateAlertTokenRequest updateAlertTokenRequest) {
var userAlert = alertRepository.findByUserIdWithLock(userId)
diff --git a/src/main/java/net/teumteum/user/domain/UserDeletedEvent.java b/src/main/java/net/teumteum/user/domain/UserDeletedEvent.java
new file mode 100644
index 0000000..3cf8d56
--- /dev/null
+++ b/src/main/java/net/teumteum/user/domain/UserDeletedEvent.java
@@ -0,0 +1,7 @@
+package net.teumteum.user.domain;
+
+public record UserDeletedEvent(
+ Long id
+) {
+
+}
diff --git a/src/main/java/net/teumteum/user/service/UserService.java b/src/main/java/net/teumteum/user/service/UserService.java
index 073117b..76de43d 100644
--- a/src/main/java/net/teumteum/user/service/UserService.java
+++ b/src/main/java/net/teumteum/user/service/UserService.java
@@ -10,6 +10,7 @@
import net.teumteum.user.domain.BalanceGameType;
import net.teumteum.user.domain.InterestQuestion;
import net.teumteum.user.domain.User;
+import net.teumteum.user.domain.UserDeletedEvent;
import net.teumteum.user.domain.UserRepository;
import net.teumteum.user.domain.WithdrawReasonRepository;
import net.teumteum.user.domain.request.ReviewRegisterRequest;
@@ -23,6 +24,7 @@
import net.teumteum.user.domain.response.UserRegisterResponse;
import net.teumteum.user.domain.response.UserReviewsResponse;
import net.teumteum.user.domain.response.UsersGetByIdResponse;
+import org.springframework.context.ApplicationEventPublisher;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
import org.springframework.util.Assert;
@@ -38,6 +40,7 @@ public class UserService {
private final RedisService redisService;
private final JwtService jwtService;
private final MeetingConnector meetingConnector;
+ private final ApplicationEventPublisher applicationEventPublisher;
public UserGetResponse getUserById(Long userId) {
var existUser = getUser(userId);
@@ -86,6 +89,7 @@ public void withdraw(UserWithdrawRequest request, Long userId) {
userRepository.delete(existUser);
withdrawReasonRepository.saveAll(request.toEntity());
+ applicationEventPublisher.publishEvent(new UserDeletedEvent(existUser.getId()));
}
@Transactional
diff --git a/src/test/java/net/teumteum/integration/UserIntegrationTest.java b/src/test/java/net/teumteum/integration/UserIntegrationTest.java
index bb42461..8b1503a 100644
--- a/src/test/java/net/teumteum/integration/UserIntegrationTest.java
+++ b/src/test/java/net/teumteum/integration/UserIntegrationTest.java
@@ -189,32 +189,6 @@ void Return_200_ok_with_success_make_friends() {
@DisplayName("친구 조회 API는")
class Find_friends_api {
- @Test
- @DisplayName("user의 id를 입력받으면, id에 해당하는 user의 친구 목록을 반환한다.")
- void Return_friends_when_received_user_id() {
- // given
- var me = repository.saveAndGetUser();
- var friend1 = repository.saveAndGetUser();
- var friend2 = repository.saveAndGetUser();
-
- securityContextSetting.set(me.getId());
-
- api.addFriends(VALID_TOKEN, friend1.getId());
- api.addFriends(VALID_TOKEN, friend2.getId());
-
- var expected = FriendsResponse.of(List.of(friend1, friend2));
-
- // when
- var result = api.getFriendsByUserId(VALID_TOKEN, me.getId());
-
- // then
- Assertions.assertThat(result.expectStatus().isOk()
- .expectBody(FriendsResponse.class)
- .returnResult()
- .getResponseBody())
- .usingRecursiveComparison().isEqualTo(expected);
- }
-
@Test
@DisplayName("user의 id를 입력받았을때, 친구가 한명도 없다면, 빈 목록을 반환한다.")
void Return_empty_friends_when_received_empty_friends_user_id() {
diff --git a/src/test/java/net/teumteum/unit/user/service/UserServiceTest.java b/src/test/java/net/teumteum/unit/user/service/UserServiceTest.java
index 66b1668..1c5630a 100644
--- a/src/test/java/net/teumteum/unit/user/service/UserServiceTest.java
+++ b/src/test/java/net/teumteum/unit/user/service/UserServiceTest.java
@@ -39,9 +39,9 @@
import org.junit.jupiter.api.extension.ExtendWith;
import org.mockito.InjectMocks;
import org.mockito.Mock;
-import org.mockito.junit.jupiter.MockitoExtension;
+import org.springframework.test.context.junit.jupiter.SpringExtension;
-@ExtendWith(MockitoExtension.class)
+@ExtendWith(SpringExtension.class)
@DisplayName("유저 서비스 단위 테스트의")
public class UserServiceTest {
@@ -135,27 +135,6 @@ void If_valid_user_logout_request_return_200_OK() {
@DisplayName("회원 탈퇴 API는")
class Withdraw_user_api_unit {
- @Test
- @DisplayName("유효한 유저 회원 탈퇴 요청이 들어오는 경우, 회원을 탈퇴하고 탈퇴 사유 데이터를 저장한다.")
- void If_valid_user_withdraw_request_withdraw_user() {
- // given
- UserWithdrawRequest request
- = RequestFixture.userWithdrawRequest(List.of("쓰지 않는 앱이에요", "오류가 생겨서 쓸 수 없어요"));
-
- given(userRepository.findById(anyLong()))
- .willReturn(Optional.ofNullable(user));
-
- doNothing().when(userRepository).delete(any());
-
- doNothing().when(redisService).deleteData(anyString());
- // when
- userService.withdraw(request, user.getId());
- // then
- verify(userRepository, times(1)).findById(anyLong());
- verify(redisService, times(1)).deleteData(anyString());
- verify(withdrawReasonRepository, times(1)).saveAll(any());
- }
-
@Test
@DisplayName("유저 id에 해당하는 유저가 존재하지 않는 경우, 400 Bad Request 을 반환한다.")
void Return_400_bad_request_if_user_is_not_exist() {
From f5874bcb1dda8a3f2c47e74906c513145cec7622 Mon Sep 17 00:00:00 2001
From: xb205 <62425964+devxb@users.noreply.github.com>
Date: Fri, 16 Feb 2024 12:01:28 +0900
Subject: [PATCH 09/13] =?UTF-8?q?fix:=20=EB=AA=A8=EC=9E=84=20=EC=8B=9C?=
=?UTF-8?q?=EC=9E=91,=20=EC=A2=85=EB=A3=8C=20=EC=95=8C=EB=A6=BC=EC=9D=B4?=
=?UTF-8?q?=20=EB=AA=A8=EC=9E=84=20=EC=B0=B8=EC=97=AC=EC=9E=90=EA=B0=80=20?=
=?UTF-8?q?3=EB=AA=85=20=EC=9D=B4=EC=83=81=EC=9D=B4=20=EC=95=84=EB=8B=88?=
=?UTF-8?q?=EB=9D=BC=EB=A9=B4,=20=EB=B0=9C=ED=96=89=EB=90=98=EC=A7=80?=
=?UTF-8?q?=EC=95=8A=EB=8F=84=EB=A1=9D=20=EC=88=98=EC=A0=95=20(#231)?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
.../service/MeetingAlertPublisher.java | 33 +++++++++++--------
1 file changed, 20 insertions(+), 13 deletions(-)
diff --git a/src/main/java/net/teumteum/meeting/service/MeetingAlertPublisher.java b/src/main/java/net/teumteum/meeting/service/MeetingAlertPublisher.java
index 858650e..bcc2bd8 100644
--- a/src/main/java/net/teumteum/meeting/service/MeetingAlertPublisher.java
+++ b/src/main/java/net/teumteum/meeting/service/MeetingAlertPublisher.java
@@ -16,6 +16,7 @@
@Transactional(readOnly = true)
public class MeetingAlertPublisher {
+ private static final String KR_TIME_ZONE = "Asia/Seoul";
private static final String EVERY_ONE_MINUTES = "0 * * * * *";
private static final String EVERY_12PM = "0 0 12 * * *";
@@ -24,18 +25,20 @@ public class MeetingAlertPublisher {
@Scheduled(cron = EVERY_ONE_MINUTES)
public void alertBeforeMeeting() {
- var alertStart = LocalDateTime.now(ZoneId.of("Asia/Seoul")).plusMinutes(5).withNano(0).withSecond(0);
+ var alertStart = LocalDateTime.now(ZoneId.of(KR_TIME_ZONE)).plusMinutes(5).withNano(0).withSecond(0);
var alertEnd = alertStart.plusMinutes(1).withNano(0).withSecond(0);
var alertTargets = meetingRepository.findAlertMeetings(alertStart, alertEnd);
- alertTargets.forEach(meeting -> eventPublisher.publishEvent(
- new BeforeMeetingAlerted(meeting.getParticipantUserIds())
- )
- );
+ alertTargets.stream()
+ .filter(alertTarget -> alertTarget.getParticipantUserIds().size() > 2)
+ .forEach(meeting -> eventPublisher.publishEvent(
+ new BeforeMeetingAlerted(meeting.getParticipantUserIds())
+ )
+ );
}
@Scheduled(cron = EVERY_12PM)
public void alertEndMeeting() {
- var today = LocalDateTime.now(ZoneId.of("Asia/Seoul"))
+ var today = LocalDateTime.now(ZoneId.of(KR_TIME_ZONE))
.withNano(0)
.withSecond(0)
.withMinute(0)
@@ -44,14 +47,16 @@ public void alertEndMeeting() {
var yesterday = today.minusDays(1);
var alertTargets = meetingRepository.findAlertMeetings(yesterday, today);
- alertTargets.forEach(meeting -> eventPublisher.publishEvent(
- new EndMeetingAlerted(meeting.getId(), meeting.getTitle(), meeting.getParticipantUserIds())
- ));
+ alertTargets.stream()
+ .filter(alertTarget -> alertTarget.getParticipantUserIds().size() > 2)
+ .forEach(meeting -> eventPublisher.publishEvent(
+ new EndMeetingAlerted(meeting.getId(), meeting.getTitle(), meeting.getParticipantUserIds())
+ ));
}
@Scheduled(cron = EVERY_ONE_MINUTES)
public void alertEndMeetingForQa() {
- var today = LocalDateTime.now(ZoneId.of("Asia/Seoul"))
+ var today = LocalDateTime.now(ZoneId.of(KR_TIME_ZONE))
.withNano(0)
.withSecond(0)
.withMinute(0)
@@ -61,8 +66,10 @@ public void alertEndMeetingForQa() {
var yesterday = today.minusDays(365);
var alertTargets = meetingRepository.findAlertMeetings(yesterday, future);
- alertTargets.forEach(meeting -> eventPublisher.publishEvent(
- new EndMeetingAlerted(meeting.getId(), meeting.getTitle(), meeting.getParticipantUserIds())
- ));
+ alertTargets.stream()
+ .filter(alertTarget -> alertTarget.getParticipantUserIds().size() > 2)
+ .forEach(meeting -> eventPublisher.publishEvent(
+ new EndMeetingAlerted(meeting.getId(), meeting.getTitle(), meeting.getParticipantUserIds())
+ ));
}
}
From 641ee6ea5e418e2f796dd908f29a02c4b10e2b74 Mon Sep 17 00:00:00 2001
From: ChoiDongKuen
Date: Fri, 16 Feb 2024 12:04:36 +0900
Subject: [PATCH 10/13] =?UTF-8?q?feat:=20=EB=A6=AC=EB=B7=B0=20=EB=93=B1?=
=?UTF-8?q?=EB=A1=9D=EC=8B=9C=20=EB=AA=A8=EC=9E=84=20=EC=B0=B8=EC=97=AC=20?=
=?UTF-8?q?=EC=9C=A0=EC=A0=80=20=EC=A1=B0=ED=9A=8C=20API=20=EC=88=98?=
=?UTF-8?q?=EC=A0=95=20(#229)?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
* refactor: MeetingController getCurrentUserId() 메소드로 통일 (#228)
* feat: 리뷰에 자신 포함 확인 로직 추가 (#195)
* feat: 모임 참여자 자신 포함 여부 확인 로직 추가(#228)
* test: 단위 테스트 추가 (#228)
* test: 통합 테스트 수정 및 실패 케이스 추가 (#228)
---
.../meeting/controller/MeetingController.java | 31 +++--
...e.java => MeetingParticipantResponse.java} | 6 +-
.../meeting/service/MeetingService.java | 18 ++-
.../integration/MeetingIntegrationTest.java | 118 +++++++++++-------
.../net/teumteum/integration/Repository.java | 7 +-
.../meeting/domain/MeetingFixture.java | 8 ++
.../controller/MeetingControllerTest.java | 112 +++++++++++++++++
.../meeting/service/MeetingServiceTest.java | 91 ++++++++++++++
.../user/controller/UserControllerTest.java | 3 +-
9 files changed, 327 insertions(+), 67 deletions(-)
rename src/main/java/net/teumteum/meeting/domain/response/{MeetingParticipantsResponse.java => MeetingParticipantResponse.java} (70%)
create mode 100644 src/test/java/net/teumteum/unit/meeting/controller/MeetingControllerTest.java
create mode 100644 src/test/java/net/teumteum/unit/meeting/service/MeetingServiceTest.java
diff --git a/src/main/java/net/teumteum/meeting/controller/MeetingController.java b/src/main/java/net/teumteum/meeting/controller/MeetingController.java
index d36a633..ca559ad 100644
--- a/src/main/java/net/teumteum/meeting/controller/MeetingController.java
+++ b/src/main/java/net/teumteum/meeting/controller/MeetingController.java
@@ -9,7 +9,7 @@
import net.teumteum.meeting.domain.Topic;
import net.teumteum.meeting.domain.request.CreateMeetingRequest;
import net.teumteum.meeting.domain.request.UpdateMeetingRequest;
-import net.teumteum.meeting.domain.response.MeetingParticipantsResponse;
+import net.teumteum.meeting.domain.response.MeetingParticipantResponse;
import net.teumteum.meeting.domain.response.MeetingResponse;
import net.teumteum.meeting.domain.response.MeetingsResponse;
import net.teumteum.meeting.model.PageDto;
@@ -43,14 +43,14 @@ public class MeetingController {
public MeetingResponse createMeeting(
@RequestPart @Valid CreateMeetingRequest meetingRequest,
@RequestPart List images) {
- Long userId = securityService.getCurrentUserId();
+ Long userId = getCurrentUserId();
return meetingService.createMeeting(images, meetingRequest, userId);
}
@GetMapping("/{meetingId}")
@ResponseStatus(HttpStatus.OK)
public MeetingResponse getMeetingById(@PathVariable("meetingId") Long meetingId) {
- Long userId = securityService.getCurrentUserId();
+ Long userId = getCurrentUserId();
return meetingService.getMeetingById(meetingId, userId);
}
@@ -64,7 +64,7 @@ public PageDto getMeetingsByCondition(
@RequestParam(value = "participantUserId", required = false) Long participantUserId,
@RequestParam(value = "isBookmarked", required = false) Boolean isBookmarked,
@RequestParam(value = "searchWord", required = false) String searchWord) {
- Long userId = securityService.getCurrentUserId();
+ Long userId = getCurrentUserId();
return meetingService.getMeetingsBySpecification(pageable, topic, meetingAreaStreet, participantUserId,
searchWord, isBookmarked, isOpen, userId);
}
@@ -74,55 +74,56 @@ public PageDto getMeetingsByCondition(
public MeetingResponse updateMeeting(@PathVariable Long meetingId,
@RequestPart @Valid UpdateMeetingRequest request,
@RequestPart List images) {
- Long userId = securityService.getCurrentUserId();
+ Long userId = getCurrentUserId();
return meetingService.updateMeeting(meetingId, images, request, userId);
}
@DeleteMapping("/{meetingId}")
@ResponseStatus(HttpStatus.OK)
public void deleteMeeting(@PathVariable("meetingId") Long meetingId) {
- Long userId = securityService.getCurrentUserId();
+ Long userId = getCurrentUserId();
meetingService.deleteMeeting(meetingId, userId);
}
@PostMapping("/{meetingId}/participants")
@ResponseStatus(HttpStatus.CREATED)
public MeetingResponse addParticipant(@PathVariable("meetingId") Long meetingId) {
- Long userId = securityService.getCurrentUserId();
+ Long userId = getCurrentUserId();
return meetingService.addParticipant(meetingId, userId);
}
@DeleteMapping("/{meetingId}/participants")
@ResponseStatus(HttpStatus.OK)
public void deleteParticipant(@PathVariable("meetingId") Long meetingId) {
- Long userId = securityService.getCurrentUserId();
+ Long userId = getCurrentUserId();
meetingService.cancelParticipant(meetingId, userId);
}
@GetMapping("/{meetingId}/participants")
@ResponseStatus(HttpStatus.OK)
- public List getParticipants(@PathVariable("meetingId") Long meetingId) {
- return meetingService.getParticipants(meetingId);
+ public List getParticipants(@PathVariable("meetingId") Long meetingId) {
+ Long userId = getCurrentUserId();
+ return meetingService.getParticipants(meetingId, userId);
}
@PostMapping("/{meetingId}/reports")
@ResponseStatus(HttpStatus.CREATED)
public void reportMeeting(@PathVariable("meetingId") Long meetingId) {
- Long userId = securityService.getCurrentUserId();
+ Long userId = getCurrentUserId();
meetingService.reportMeeting(meetingId, userId);
}
@PostMapping("/{meetingId}/bookmarks")
@ResponseStatus(HttpStatus.CREATED)
public void addBookmark(@PathVariable("meetingId") Long meetingId) {
- Long userId = securityService.getCurrentUserId();
+ Long userId = getCurrentUserId();
meetingService.addBookmark(meetingId, userId);
}
@DeleteMapping("/{meetingId}/bookmarks")
@ResponseStatus(HttpStatus.OK)
public void deleteBookmark(@PathVariable("meetingId") Long meetingId) {
- Long userId = securityService.getCurrentUserId();
+ Long userId = getCurrentUserId();
meetingService.cancelBookmark(meetingId, userId);
}
@@ -132,4 +133,8 @@ public ErrorResponse handleIllegalArgumentException(IllegalArgumentException ill
Sentry.captureException(illegalArgumentException);
return ErrorResponse.of(illegalArgumentException);
}
+
+ private Long getCurrentUserId() {
+ return securityService.getCurrentUserId();
+ }
}
diff --git a/src/main/java/net/teumteum/meeting/domain/response/MeetingParticipantsResponse.java b/src/main/java/net/teumteum/meeting/domain/response/MeetingParticipantResponse.java
similarity index 70%
rename from src/main/java/net/teumteum/meeting/domain/response/MeetingParticipantsResponse.java
rename to src/main/java/net/teumteum/meeting/domain/response/MeetingParticipantResponse.java
index 10d9df9..055ee11 100644
--- a/src/main/java/net/teumteum/meeting/domain/response/MeetingParticipantsResponse.java
+++ b/src/main/java/net/teumteum/meeting/domain/response/MeetingParticipantResponse.java
@@ -2,17 +2,17 @@
import net.teumteum.user.domain.User;
-public record MeetingParticipantsResponse(
+public record MeetingParticipantResponse(
Long id,
Long characterId,
String name,
String job
) {
- public static MeetingParticipantsResponse of(
+ public static MeetingParticipantResponse of(
User user
) {
- return new MeetingParticipantsResponse(
+ return new MeetingParticipantResponse(
user.getId(),
user.getCharacterId(),
user.getName(),
diff --git a/src/main/java/net/teumteum/meeting/service/MeetingService.java b/src/main/java/net/teumteum/meeting/service/MeetingService.java
index 99cfc41..5003080 100644
--- a/src/main/java/net/teumteum/meeting/service/MeetingService.java
+++ b/src/main/java/net/teumteum/meeting/service/MeetingService.java
@@ -12,7 +12,7 @@
import net.teumteum.meeting.domain.Topic;
import net.teumteum.meeting.domain.request.CreateMeetingRequest;
import net.teumteum.meeting.domain.request.UpdateMeetingRequest;
-import net.teumteum.meeting.domain.response.MeetingParticipantsResponse;
+import net.teumteum.meeting.domain.response.MeetingParticipantResponse;
import net.teumteum.meeting.domain.response.MeetingResponse;
import net.teumteum.meeting.domain.response.MeetingsResponse;
import net.teumteum.meeting.model.PageDto;
@@ -94,7 +94,8 @@ public void deleteMeeting(Long meetingId, Long userId) {
}
@Transactional(readOnly = true)
- public PageDto getMeetingsBySpecification(Pageable pageable, Topic topic, String meetingAreaStreet,
+ public PageDto getMeetingsBySpecification(Pageable pageable, Topic topic,
+ String meetingAreaStreet,
Long participantUserId, String searchWord, Boolean isBookmarked, Boolean isOpen, Long userId) {
Specification spec = MeetingSpecification.withIsOpen(isOpen);
@@ -153,13 +154,16 @@ public void cancelParticipant(Long meetingId, Long userId) {
}
@Transactional(readOnly = true)
- public List getParticipants(Long meetingId) {
+ public List getParticipants(Long meetingId, Long userId) {
var existMeeting = getMeeting(meetingId);
+ checkMeetingContainUser(existMeeting, userId);
+
return existMeeting.getParticipantUserIds().stream()
+ .filter(id -> !id.equals(userId))
.map(userConnector::findUserById)
.flatMap(Optional::stream)
- .map(MeetingParticipantsResponse::of)
+ .map(MeetingParticipantResponse::of)
.toList();
}
@@ -207,4 +211,10 @@ public void reportMeeting(Long meetingId, Long userId) {
throw new IllegalArgumentException("모임 개설자는 모임을 신고할 수 없습니다.");
}
}
+
+ private void checkMeetingContainUser(Meeting meeting, Long userId) {
+ if (!meeting.getParticipantUserIds().contains(userId)) {
+ throw new IllegalArgumentException("모임에 참여하지 않은 회원입니다.");
+ }
+ }
}
diff --git a/src/test/java/net/teumteum/integration/MeetingIntegrationTest.java b/src/test/java/net/teumteum/integration/MeetingIntegrationTest.java
index dc45f2c..27d76d2 100644
--- a/src/test/java/net/teumteum/integration/MeetingIntegrationTest.java
+++ b/src/test/java/net/teumteum/integration/MeetingIntegrationTest.java
@@ -1,7 +1,10 @@
package net.teumteum.integration;
+import static org.assertj.core.api.Assertions.assertThat;
+
import java.util.Collection;
import java.util.Comparator;
+import java.util.List;
import java.util.stream.Stream;
import net.teumteum.core.error.ErrorResponse;
import net.teumteum.meeting.domain.Meeting;
@@ -9,7 +12,6 @@
import net.teumteum.meeting.domain.response.MeetingResponse;
import net.teumteum.meeting.domain.response.MeetingsResponse;
import net.teumteum.meeting.model.PageDto;
-import org.assertj.core.api.Assertions;
import org.assertj.core.api.Condition;
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Nested;
@@ -42,10 +44,10 @@ void Return_meeting_info_if_exist_meeting_id_received() {
// when
var result = api.getMeetingById(VALID_TOKEN, meeting.getId());
// then
- Assertions.assertThat(
- result.expectStatus().isOk()
- .expectBody(MeetingResponse.class)
- .returnResult().getResponseBody())
+ assertThat(
+ result.expectStatus().isOk()
+ .expectBody(MeetingResponse.class)
+ .returnResult().getResponseBody())
.usingRecursiveComparison()
.isEqualTo(expected);
}
@@ -75,10 +77,10 @@ void Return_is_bookmarked_true_if_user_bookmarked_meeting() {
// when
var result = api.getMeetingById(VALID_TOKEN, meeting.getId());
// then
- Assertions.assertThat(
- result.expectStatus().isOk()
- .expectBody(MeetingResponse.class)
- .returnResult().getResponseBody())
+ assertThat(
+ result.expectStatus().isOk()
+ .expectBody(MeetingResponse.class)
+ .returnResult().getResponseBody())
.extracting(MeetingResponse::isBookmarked)
.isEqualTo(true);
}
@@ -158,11 +160,11 @@ void Return_meeting_list_if_topic_and_page_nation_received() {
// when
var result = api.getMeetingsByTopic(VALID_TOKEN, FIRST_PAGE_NATION, true, Topic.스터디);
// then
- Assertions.assertThat(
- result.expectStatus().isOk()
- .expectBody(new ParameterizedTypeReference>() {
- })
- .returnResult().getResponseBody())
+ assertThat(
+ result.expectStatus().isOk()
+ .expectBody(new ParameterizedTypeReference>() {
+ })
+ .returnResult().getResponseBody())
.usingRecursiveComparison()
.isEqualTo(expected);
}
@@ -192,11 +194,11 @@ void Return_meeting_list_if_search_word_and_page_nation_received() {
// when
var result = api.getMeetingsByTopic(VALID_TOKEN, FIRST_PAGE_NATION, true, Topic.스터디);
// then
- Assertions.assertThat(
- result.expectStatus().isOk()
- .expectBody(new ParameterizedTypeReference>() {
- })
- .returnResult().getResponseBody())
+ assertThat(
+ result.expectStatus().isOk()
+ .expectBody(new ParameterizedTypeReference>() {
+ })
+ .returnResult().getResponseBody())
.usingRecursiveComparison()
.isEqualTo(expected);
}
@@ -222,11 +224,11 @@ void Return_meeting_list_if_participant_user_id_and_page_nation_received() {
// when
var result = api.getMeetingsByTopic(VALID_TOKEN, FIRST_PAGE_NATION, true, Topic.스터디);
// then
- Assertions.assertThat(
- result.expectStatus().isOk()
- .expectBody(new ParameterizedTypeReference>() {
- })
- .returnResult().getResponseBody())
+ assertThat(
+ result.expectStatus().isOk()
+ .expectBody(new ParameterizedTypeReference>() {
+ })
+ .returnResult().getResponseBody())
.usingRecursiveComparison()
.isEqualTo(expected);
}
@@ -252,11 +254,11 @@ void Return_has_next_true_if_more_data_exists_than_requested_size_and_page() {
// when
var result = api.getMeetingsByTopic(VALID_TOKEN, FIRST_PAGE_NATION, true, Topic.스터디);
// then
- Assertions.assertThat(
- result.expectStatus().isOk()
- .expectBody(new ParameterizedTypeReference>() {
- })
- .returnResult().getResponseBody())
+ assertThat(
+ result.expectStatus().isOk()
+ .expectBody(new ParameterizedTypeReference>() {
+ })
+ .returnResult().getResponseBody())
.usingRecursiveComparison()
.isEqualTo(expected);
}
@@ -277,11 +279,11 @@ void Join_meeting_if_exist_meeting_id_received() {
// when
var result = api.joinMeeting(VALID_TOKEN, existMeeting.getId());
// then
- Assertions.assertThat(
- result.expectStatus().isCreated()
- .expectBody(MeetingResponse.class)
- .returnResult()
- .getResponseBody())
+ assertThat(
+ result.expectStatus().isCreated()
+ .expectBody(MeetingResponse.class)
+ .returnResult()
+ .getResponseBody())
.extracting(MeetingResponse::participantIds)
.has(new Condition<>(ids -> ids.contains(me.getId()), "참여자 목록에 나를 포함한다.")
);
@@ -380,11 +382,11 @@ void Return_400_bad_request_if_not_joined_meeting_id_received() {
// when
var result = api.cancelMeeting(VALID_TOKEN, meeting.getId());
// then
- Assertions.assertThat(result.expectStatus().isBadRequest()
- .expectBody(ErrorResponse.class)
- .returnResult()
- .getResponseBody()
- )
+ assertThat(result.expectStatus().isBadRequest()
+ .expectBody(ErrorResponse.class)
+ .returnResult()
+ .getResponseBody()
+ )
.extracting(ErrorResponse::getMessage)
.isEqualTo("참여하지 않은 모임입니다.");
}
@@ -400,11 +402,11 @@ void Return_400_bad_request_if_closed_meeting_id_received() {
// when
var result = api.cancelMeeting(VALID_TOKEN, meeting.getId());
// then
- Assertions.assertThat(result.expectStatus().isBadRequest()
- .expectBody(ErrorResponse.class)
- .returnResult()
- .getResponseBody()
- )
+ assertThat(result.expectStatus().isBadRequest()
+ .expectBody(ErrorResponse.class)
+ .returnResult()
+ .getResponseBody()
+ )
.extracting(ErrorResponse::getMessage)
.isEqualTo("종료된 모임에서 참여를 취소할 수 없습니다.");
}
@@ -488,11 +490,37 @@ class Get_meeting_participants_api {
@DisplayName("참여한 meeting id 가 주어지면, 참여한 참가자들의 정보가 주어진다.")
void Get_participants_if_exist_meeting_id_received() {
// given
- var meeting = repository.saveAndGetOpenMeeting();
+ var me = repository.saveAndGetUser();
+ var meeting = repository.saveAndGetClosedMetingWithParticipantUserIds(List.of(me.getId(), 2L));
+
+ securityContextSetting.set(me.getId());
+
// when
var result = api.getMeetingParticipants(VALID_TOKEN, meeting.getId());
+
// then
result.expectStatus().isOk();
}
+
+ @Test
+ @DisplayName("API 호출한 회원이 모임에 참여하지 않았다면, 400 bad request 을 응답한다.")
+ void Return_400_bad_request_if_meeting_not_contain_user() {
+ // given
+ var me = repository.saveAndGetUser();
+ var meeting = repository.saveAndGetClosedMetingWithParticipantUserIds(List.of(100L, 101L));
+
+ securityContextSetting.set(me.getId());
+
+ // when
+ var result = api.getMeetingParticipants(VALID_TOKEN, meeting.getId());
+
+ // then
+ assertThat(result.expectStatus().isBadRequest()
+ .expectBody(ErrorResponse.class)
+ .returnResult()
+ .getResponseBody())
+ .extracting(ErrorResponse::getMessage)
+ .isEqualTo("모임에 참여하지 않은 회원입니다.");
+ }
}
}
diff --git a/src/test/java/net/teumteum/integration/Repository.java b/src/test/java/net/teumteum/integration/Repository.java
index 7d829ee..894c036 100644
--- a/src/test/java/net/teumteum/integration/Repository.java
+++ b/src/test/java/net/teumteum/integration/Repository.java
@@ -2,7 +2,6 @@
import java.util.List;
-import java.util.stream.IntStream;
import java.util.stream.Stream;
import lombok.RequiredArgsConstructor;
import net.teumteum.core.config.AppConfig;
@@ -82,6 +81,11 @@ Meeting saveAndGetOpenFullMeeting() {
return meetingRepository.saveAndFlush(meeting);
}
+ Meeting saveAndGetClosedMetingWithParticipantUserIds(List participantUserIds) {
+ var meeting = MeetingFixture.getCloseMeetingWithParticipantUserIds(participantUserIds);
+ return meetingRepository.saveAndFlush(meeting);
+ }
+
List saveAndGetOpenMeetingsByTopic(int size, Topic topic) {
var meetings = Stream.generate(() -> MeetingFixture.getOpenMeetingWithTopic(topic))
.limit(size)
@@ -146,6 +150,7 @@ List saveAndGetOpenMeetings(int size) {
return meetingRepository.saveAllAndFlush(meetings);
}
+
void clear() {
userRepository.deleteAll();
meetingRepository.deleteAll();
diff --git a/src/test/java/net/teumteum/meeting/domain/MeetingFixture.java b/src/test/java/net/teumteum/meeting/domain/MeetingFixture.java
index 93f1da2..09b850e 100644
--- a/src/test/java/net/teumteum/meeting/domain/MeetingFixture.java
+++ b/src/test/java/net/teumteum/meeting/domain/MeetingFixture.java
@@ -82,6 +82,14 @@ public static Meeting getCloseMeetingWithParticipantUserId(Long participantUserI
);
}
+ public static Meeting getCloseMeetingWithParticipantUserIds(List participantUserIds) {
+ return newMeetingByBuilder(MeetingBuilder.builder()
+ .promiseDateTime(LocalDateTime.of(2000, 1, 1, 0, 0))
+ .participantUserIds(new HashSet<>(participantUserIds))
+ .build()
+ );
+ }
+
public static Meeting getOpenMeetingWithTitle(String title) {
return newMeetingByBuilder(MeetingBuilder.builder()
.promiseDateTime(LocalDateTime.of(4000, 1, 1, 0, 0))
diff --git a/src/test/java/net/teumteum/unit/meeting/controller/MeetingControllerTest.java b/src/test/java/net/teumteum/unit/meeting/controller/MeetingControllerTest.java
new file mode 100644
index 0000000..da166b3
--- /dev/null
+++ b/src/test/java/net/teumteum/unit/meeting/controller/MeetingControllerTest.java
@@ -0,0 +1,112 @@
+package net.teumteum.unit.meeting.controller;
+
+import static net.teumteum.unit.common.SecurityValue.VALID_ACCESS_TOKEN;
+import static org.mockito.ArgumentMatchers.anyLong;
+import static org.mockito.BDDMockito.given;
+import static org.springframework.http.HttpHeaders.AUTHORIZATION;
+import static org.springframework.security.test.web.servlet.request.SecurityMockMvcRequestPostProcessors.csrf;
+import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.get;
+import static org.springframework.test.web.servlet.result.MockMvcResultHandlers.print;
+import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.jsonPath;
+import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status;
+
+import com.fasterxml.jackson.databind.ObjectMapper;
+import java.util.List;
+import net.teumteum.core.security.SecurityConfig;
+import net.teumteum.core.security.filter.JwtAuthenticationFilter;
+import net.teumteum.core.security.service.JwtService;
+import net.teumteum.core.security.service.RedisService;
+import net.teumteum.core.security.service.SecurityService;
+import net.teumteum.meeting.controller.MeetingController;
+import net.teumteum.meeting.domain.response.MeetingParticipantResponse;
+import net.teumteum.meeting.service.MeetingService;
+import net.teumteum.user.domain.UserFixture;
+import org.junit.jupiter.api.DisplayName;
+import org.junit.jupiter.api.Nested;
+import org.junit.jupiter.api.Test;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.boot.test.autoconfigure.web.servlet.WebMvcTest;
+import org.springframework.boot.test.mock.mockito.MockBean;
+import org.springframework.context.annotation.ComponentScan;
+import org.springframework.context.annotation.FilterType;
+import org.springframework.security.test.context.support.WithMockUser;
+import org.springframework.test.web.servlet.MockMvc;
+
+@WebMvcTest(value = MeetingController.class,
+ excludeFilters = {@ComponentScan.Filter(type = FilterType.ASSIGNABLE_TYPE, classes = SecurityConfig.class),
+ @ComponentScan.Filter(type = FilterType.ASSIGNABLE_TYPE, classes = JwtAuthenticationFilter.class),
+ @ComponentScan.Filter(type = FilterType.ASSIGNABLE_TYPE, classes = RedisService.class),
+ @ComponentScan.Filter(type = FilterType.ASSIGNABLE_TYPE, classes = JwtService.class)}
+)
+@WithMockUser
+@DisplayName("모임 컨트롤러 단위 테스트의")
+public class MeetingControllerTest {
+
+ @Autowired
+ private ObjectMapper objectMapper;
+
+ @Autowired
+ private MockMvc mockMvc;
+
+ @MockBean
+ private MeetingService meetingService;
+
+ @MockBean
+ private SecurityService securityService;
+
+ @Nested
+ @DisplayName("모임 참여자 조회 API는")
+ class Get_meeting_participants_api_unit {
+
+ @Test
+ @DisplayName("API 호출 회원을 제외한, meetingId 해당하는 모임의 참여자 정보를 반환한다.")
+ void Return_meeting_participants_with_200_ok() throws Exception {
+ // given
+ var existUser1 = UserFixture.getUserWithId(1L);
+ var existUser2 = UserFixture.getUserWithId(2L);
+ var existUser3 = UserFixture.getUserWithId(3L);
+
+ List response
+ = List.of(MeetingParticipantResponse.of(existUser2), MeetingParticipantResponse.of(existUser3));
+
+ given(securityService.getCurrentUserId())
+ .willReturn(existUser1.getId());
+
+ given(meetingService.getParticipants(anyLong(), anyLong()))
+ .willReturn(response);
+
+ // when && then
+ mockMvc.perform(get("/meetings/{meetingId}/participants", 2L)
+ .with(csrf())
+ .header(AUTHORIZATION, VALID_ACCESS_TOKEN))
+ .andDo(print())
+ .andExpect(status().isOk())
+ .andExpect(jsonPath("$").isNotEmpty())
+ .andExpect(jsonPath("$.length()").value(2))
+ .andExpect(jsonPath("$").isArray())
+ .andExpect(jsonPath("$[0].id").value(2))
+ .andExpect(jsonPath("$[0].characterId").value(1));
+ }
+
+ @Test
+ @DisplayName("모임에 API 호출 회원이 존재하지 않는 경우, 400 bad request를 응답한다.")
+ void Return_400_bad_request_if_meeting_not_contain_user() throws Exception {
+ // given
+ var existUser1 = UserFixture.getUserWithId(1L);
+
+ given(securityService.getCurrentUserId())
+ .willReturn(existUser1.getId());
+
+ given(meetingService.getParticipants(anyLong(), anyLong())).willThrow(
+ new IllegalArgumentException("모임에 참여하지 않은 회원입니다."));
+
+ // when & then
+ mockMvc.perform(get("/meetings/{meetingId}/participants", 2L)
+ .with(csrf())
+ .header(AUTHORIZATION, VALID_ACCESS_TOKEN))
+ .andDo(print())
+ .andExpect(status().isBadRequest())
+ .andExpect(jsonPath("$.message").value("모임에 참여하지 않은 회원입니다."));
+ }
+ }
+}
diff --git a/src/test/java/net/teumteum/unit/meeting/service/MeetingServiceTest.java b/src/test/java/net/teumteum/unit/meeting/service/MeetingServiceTest.java
new file mode 100644
index 0000000..c6cd397
--- /dev/null
+++ b/src/test/java/net/teumteum/unit/meeting/service/MeetingServiceTest.java
@@ -0,0 +1,91 @@
+package net.teumteum.unit.meeting.service;
+
+import static org.assertj.core.api.Assertions.assertThat;
+import static org.assertj.core.api.Assertions.assertThatThrownBy;
+import static org.mockito.ArgumentMatchers.anyLong;
+import static org.mockito.BDDMockito.given;
+
+import java.util.List;
+import java.util.Optional;
+import net.teumteum.meeting.domain.ImageUpload;
+import net.teumteum.meeting.domain.MeetingFixture;
+import net.teumteum.meeting.domain.MeetingRepository;
+import net.teumteum.meeting.domain.response.MeetingParticipantResponse;
+import net.teumteum.meeting.service.MeetingService;
+import net.teumteum.user.domain.UserConnector;
+import net.teumteum.user.domain.UserFixture;
+import org.junit.jupiter.api.DisplayName;
+import org.junit.jupiter.api.Nested;
+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;
+
+@ExtendWith(MockitoExtension.class)
+@DisplayName("모임 서비스 단위 테스트의")
+public class MeetingServiceTest {
+
+ @InjectMocks
+ MeetingService meetingService;
+
+ @Mock
+ MeetingRepository meetingRepository;
+
+ @Mock
+ UserConnector userConnector;
+
+ @Mock
+ ImageUpload imageUpload;
+
+ @Nested
+ @DisplayName("모임 참여자 조회 API는")
+ class Return_meeting_participants_api_unit {
+
+ @Test
+ @DisplayName("API 호출 회원을 제외한, meetingId 해당하는 모임의 참여자 정보를 반환한다.")
+ void Return_meeting_participants_with_200_ok() {
+ // given
+ var userId = 1L;
+ var meetingId = 1L;
+
+ var existMeeting
+ = MeetingFixture.getCloseMeetingWithParticipantUserIds(List.of(userId, 2L, 3L));
+
+ var existUser2 = UserFixture.getUserWithId(2L);
+ var existUser3 = UserFixture.getUserWithId(3L);
+
+ given(meetingRepository.findById(anyLong()))
+ .willReturn(Optional.of(existMeeting));
+
+ given(userConnector.findUserById(existUser2.getId())).willReturn(Optional.of(existUser2));
+ given(userConnector.findUserById(existUser3.getId())).willReturn(Optional.of(existUser3));
+
+ // when
+ List participants = meetingService.getParticipants(meetingId, userId);
+ // then
+ assertThat(participants).hasSize(2);
+ }
+
+ @Test
+ @DisplayName("모임에 API 호출 회원이 존재하지 않는 경우, 400 bad request를 응답한다.")
+ void Return_400_bad_request_if_meeting_not_contain_user() {
+ // given
+ var userId = 1L;
+ var notContainedUserId = 4L;
+
+ var meetingId = 1L;
+
+ var existMeeting
+ = MeetingFixture.getCloseMeetingWithParticipantUserIds(List.of(userId, 2L, 3L));
+
+ given(meetingRepository.findById(anyLong()))
+ .willReturn(Optional.of(existMeeting));
+
+ // when
+ assertThatThrownBy(() -> meetingService.getParticipants(meetingId, notContainedUserId))
+ .isInstanceOf(IllegalArgumentException.class)
+ .hasMessage("모임에 참여하지 않은 회원입니다.");
+ }
+ }
+}
diff --git a/src/test/java/net/teumteum/unit/user/controller/UserControllerTest.java b/src/test/java/net/teumteum/unit/user/controller/UserControllerTest.java
index 77afd49..2b1d877 100644
--- a/src/test/java/net/teumteum/unit/user/controller/UserControllerTest.java
+++ b/src/test/java/net/teumteum/unit/user/controller/UserControllerTest.java
@@ -63,10 +63,11 @@
public class UserControllerTest {
@Autowired
- ObjectMapper objectMapper;
+ private ObjectMapper objectMapper;
@Autowired
private MockMvc mockMvc;
+
@MockBean
private UserService userService;
From 9cffa3392069f8080193e5b6388a4c5f146f70f4 Mon Sep 17 00:00:00 2001
From: devxb
Date: Fri, 16 Feb 2024 12:07:56 +0900
Subject: [PATCH 11/13] =?UTF-8?q?refactor:=20title,=20body=20alert=20data?=
=?UTF-8?q?=EC=97=90=20=EC=B6=94=EA=B0=80?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
src/main/java/net/teumteum/alert/infra/FcmAlertPublisher.java | 2 ++
1 file changed, 2 insertions(+)
diff --git a/src/main/java/net/teumteum/alert/infra/FcmAlertPublisher.java b/src/main/java/net/teumteum/alert/infra/FcmAlertPublisher.java
index 7b9c830..68ae10d 100644
--- a/src/main/java/net/teumteum/alert/infra/FcmAlertPublisher.java
+++ b/src/main/java/net/teumteum/alert/infra/FcmAlertPublisher.java
@@ -44,6 +44,8 @@ private Message buildMessage(String token, Alert alert, Map data
.setToken(token)
.setNotification(buildNotification(alert))
.setAndroidConfig(buildAndroidConfig(alert))
+ .putData("title", alert.getTitle())
+ .putData("body", alert.getBody())
.putData("publishedAt", alert.getCreatedAt().toString())
.putData("userId", alert.getUserId().toString())
.putData("type", alert.getType().toString())
From 824c4a87695fa50e6c77ea7e8ca3f9d50e298195 Mon Sep 17 00:00:00 2001
From: ChoiDongKuen
Date: Sat, 17 Feb 2024 02:03:59 +0900
Subject: [PATCH 12/13] =?UTF-8?q?feat:=20=EB=A6=AC=EB=B7=B0=20=EB=93=B1?=
=?UTF-8?q?=EB=A1=9D=EC=8B=9C=20=EC=9A=94=EC=B2=AD=20DTO=20Validation=20?=
=?UTF-8?q?=EB=B3=80=EA=B2=BD=20=EB=B0=8F=20=EB=A1=9C=EC=A7=81=20=EC=B6=94?=
=?UTF-8?q?=EA=B0=80=20(#233)?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
* feat: 요청 validation 변경 (#228)
* test: UserControllerTest 리팩토링 (#224)
* feat: 리뷰 등록시 userService 로직 추가 (#224)
* test: 테스트 데이터 관련 메소드 추가 (#224)
* test: 리뷰 등록시 로직 추가에 따른 통합 테스트 추가 (#224)
* test: 리뷰 등록시 로직 추가에 따른 Service 단위 테스트 추가 (#224)
* feat: Cors 허용 메소드 추가 (#224)
---
.../core/security/SecurityConfig.java | 2 +-
.../domain/request/ReviewRegisterRequest.java | 2 +-
.../teumteum/user/service/UserService.java | 28 ++++--
.../net/teumteum/integration/Repository.java | 5 +
.../integration/UserIntegrationTest.java | 94 ++++++++++++++-----
.../meeting/domain/MeetingFixture.java | 31 ++++++
.../user/controller/UserControllerTest.java | 20 ++--
.../unit/user/service/UserServiceTest.java | 57 ++++++-----
8 files changed, 168 insertions(+), 71 deletions(-)
diff --git a/src/main/java/net/teumteum/core/security/SecurityConfig.java b/src/main/java/net/teumteum/core/security/SecurityConfig.java
index d903204..066c5da 100644
--- a/src/main/java/net/teumteum/core/security/SecurityConfig.java
+++ b/src/main/java/net/teumteum/core/security/SecurityConfig.java
@@ -66,7 +66,7 @@ CorsConfigurationSource corsConfigurationSource() {
CorsConfiguration config = new CorsConfiguration();
config.addAllowedOrigin("*");
config.addAllowedHeader("*");
- config.setAllowedMethods(List.of("GET", "POST", "PUT", "DELETE", "PATCH", "OPTIONS"));
+ config.setAllowedMethods(List.of("GET", "POST", "PUT", "DELETE", "PATCH", "OPTIONS", "HEAD", "TRACE"));
config.addExposedHeader("Authorization");
config.addExposedHeader("Authorization-refresh");
config.setAllowCredentials(true);
diff --git a/src/main/java/net/teumteum/user/domain/request/ReviewRegisterRequest.java b/src/main/java/net/teumteum/user/domain/request/ReviewRegisterRequest.java
index b5cc4bb..e94d89c 100644
--- a/src/main/java/net/teumteum/user/domain/request/ReviewRegisterRequest.java
+++ b/src/main/java/net/teumteum/user/domain/request/ReviewRegisterRequest.java
@@ -8,7 +8,7 @@
public record ReviewRegisterRequest(
@Valid
- @Size(min = 2, max = 5)
+ @Size(min = 1, max = 5)
List reviews
) {
diff --git a/src/main/java/net/teumteum/user/service/UserService.java b/src/main/java/net/teumteum/user/service/UserService.java
index 76de43d..557daad 100644
--- a/src/main/java/net/teumteum/user/service/UserService.java
+++ b/src/main/java/net/teumteum/user/service/UserService.java
@@ -1,11 +1,13 @@
package net.teumteum.user.service;
+import java.time.LocalDateTime;
import java.util.List;
import lombok.RequiredArgsConstructor;
import net.teumteum.core.security.Authenticated;
import net.teumteum.core.security.service.JwtService;
import net.teumteum.core.security.service.RedisService;
import net.teumteum.core.security.service.SecurityService;
+import net.teumteum.meeting.domain.Meeting;
import net.teumteum.meeting.domain.MeetingConnector;
import net.teumteum.user.domain.BalanceGameType;
import net.teumteum.user.domain.InterestQuestion;
@@ -109,7 +111,10 @@ public void logout(Long userId) {
@Transactional
public void registerReview(Long meetingId, Long currentUserId, ReviewRegisterRequest request) {
- checkMeetingExistence(meetingId);
+ var meeting = getMeeting(meetingId);
+
+ checkMeetingIsClosed(meeting);
+ checkUserParticipationInMeeting(meeting, currentUserId);
checkUserNotRegisterSelfReview(request, currentUserId);
request.reviews()
@@ -157,12 +162,9 @@ private void checkUserExistence(Authenticated authenticated, String oauthId) {
);
}
- private void checkMeetingExistence(Long meetingId) {
- Assert.isTrue(meetingConnector.existById(meetingId),
- () -> {
- throw new IllegalArgumentException("meetingId에 해당하는 meeting을 찾을 수 없습니다. \"" + meetingId + "\"");
- }
- );
+ private Meeting getMeeting(Long meetingId) {
+ return meetingConnector.findById(meetingId)
+ .orElseThrow(() -> new IllegalArgumentException("meetingId에 해당하는 모임을 찾을 수 없습니다. \"" + meetingId + "\""));
}
private void checkUserNotRegisterSelfReview(ReviewRegisterRequest request, Long currentUserId) {
@@ -172,4 +174,16 @@ private void checkUserNotRegisterSelfReview(ReviewRegisterRequest request, Long
}
);
}
+
+ private void checkUserParticipationInMeeting(Meeting meeting, Long userId) {
+ if (!meeting.getParticipantUserIds().contains(userId)) {
+ throw new IllegalArgumentException("모임에 참여하지 않은 회원입니다.");
+ }
+ }
+
+ private void checkMeetingIsClosed(Meeting meeting) {
+ if (!LocalDateTime.now().isAfter(meeting.getPromiseDateTime())) {
+ throw new IllegalArgumentException("해당 모임은 아직 종료되지 않았습니다.");
+ }
+ }
}
diff --git a/src/test/java/net/teumteum/integration/Repository.java b/src/test/java/net/teumteum/integration/Repository.java
index 894c036..fd0f900 100644
--- a/src/test/java/net/teumteum/integration/Repository.java
+++ b/src/test/java/net/teumteum/integration/Repository.java
@@ -143,6 +143,11 @@ List saveAndGetCloseMeetingsByParticipantUserId(int size, Long particip
return meetingRepository.saveAllAndFlush(meetings);
}
+ Meeting saveAndGetCloseMeetingByParticipantUserIds(List participantUserIds) {
+ var meeting = MeetingFixture.getCloseMeetingWithParticipantIds(participantUserIds);
+ return meetingRepository.save(meeting);
+ }
+
List saveAndGetOpenMeetings(int size) {
var meetings = Stream.generate(MeetingFixture::getOpenMeeting)
.limit(size)
diff --git a/src/test/java/net/teumteum/integration/UserIntegrationTest.java b/src/test/java/net/teumteum/integration/UserIntegrationTest.java
index 8b1503a..40ddd08 100644
--- a/src/test/java/net/teumteum/integration/UserIntegrationTest.java
+++ b/src/test/java/net/teumteum/integration/UserIntegrationTest.java
@@ -4,10 +4,8 @@
import java.util.List;
import net.teumteum.core.error.ErrorResponse;
-import net.teumteum.meeting.domain.Meeting;
import net.teumteum.user.domain.User;
import net.teumteum.user.domain.UserFixture;
-import net.teumteum.user.domain.request.ReviewRegisterRequest;
import net.teumteum.user.domain.response.FriendsResponse;
import net.teumteum.user.domain.response.UserGetResponse;
import net.teumteum.user.domain.response.UserMeGetResponse;
@@ -15,7 +13,6 @@
import net.teumteum.user.domain.response.UserReviewsResponse;
import net.teumteum.user.domain.response.UsersGetByIdResponse;
import org.assertj.core.api.Assertions;
-import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Nested;
import org.junit.jupiter.api.Test;
@@ -341,48 +338,101 @@ void Get_user_reviews() {
@DisplayName("회원 리뷰 등록 API는")
class Register_user_review_api {
- User existUser;
+ @Test
+ @DisplayName("정상적인 요청이 오는 경우, 해당 회원의 리뷰 등록과 함께 200 OK 을 반환한다.")
+ void Return_200_OK_and_register_review_if_request_is_valid() {
+ // given
+ var user = repository.saveAndGetUser();
+ var participant1 = repository.saveAndGetUser();
+ var participant2 = repository.saveAndGetUser();
- List users;
+ var closedMeeting = repository.saveAndGetCloseMeetingByParticipantUserIds(
+ List.of(user.getId(), participant1.getId(), participant2.getId()));
- ReviewRegisterRequest request;
+ var request = RequestFixture.reviewRegisterRequest(List.of(participant1, participant2));
- Meeting meeting;
+ securityContextSetting.set(user.getId());
- @BeforeEach
- void setUp() {
- existUser = repository.saveAndGetUser();
- users = repository.saveAndGetUsers(3);
- request = RequestFixture.reviewRegisterRequest(users);
- meeting = repository.saveAndGetOpenMeetings(1).get(0);
+ // when
+ var expected = api.registerUserReview(VALID_TOKEN, closedMeeting.getId(), request);
+
+ // then
+ Assertions.assertThat(expected.expectStatus().isOk());
}
@Test
- @DisplayName("회원 리뷰 등록 요청이 들어오면 리뷰를 등록하고, 200 OK 을 반환한다.")
- void Return_200_OK_with_success_register_user_review() {
+ @DisplayName("meeting id 에 해당하는 meeting 이 아직 종료되지 않았다면, 400 Bad Request 와 함께 리뷰 등록을 실패한다.")
+ void Return_400_bad_request_if_meeting_is_not_closed() {
// given
- securityContextSetting.set(existUser.getId());
+ var user = repository.saveAndGetUser();
+ var participant = repository.saveAndGetUser();
+
+ var openMeeting = repository.saveAndGetOpenMeeting();
+ var request = RequestFixture.reviewRegisterRequest(List.of(participant));
+
+ securityContextSetting.set(user.getId());
// when
- var expected = api.registerUserReview(VALID_TOKEN, meeting.getId(), request);
+ var expected = api.registerUserReview(VALID_TOKEN, openMeeting.getId(), request);
// then
- Assertions.assertThat(expected.expectStatus().isOk());
+ Assertions.assertThat(expected.expectStatus().isBadRequest()
+ .expectBody(ErrorResponse.class)
+ .returnResult().getResponseBody())
+ .extracting(ErrorResponse::getMessage)
+ .isEqualTo("해당 모임은 아직 종료되지 않았습니다.");
}
@Test
@DisplayName("현재 로그인한 회원의 id 가 리뷰 등록 요청에 포함된다면, 회원 리뷰 등록을 실패하고 400 bad request 을 반환한다.")
void Return_400_bad_request_if_current_user_id_in_request() {
// given
- securityContextSetting.set(users.get(0).getId());
+ var user = repository.saveAndGetUser();
+ var participant = repository.saveAndGetUser();
+
+ var closedMeeting = repository.saveAndGetCloseMeetingByParticipantUserIds(
+ List.of(user.getId(), participant.getId()));
+
+ var request = RequestFixture.reviewRegisterRequest(List.of(user, participant));
+
+ securityContextSetting.set(user.getId());
+
+ // when
+ var expected = api.registerUserReview(VALID_TOKEN, closedMeeting.getId(), request);
+
+ // then
+ Assertions.assertThat(expected.expectStatus().isBadRequest()
+ .expectBody(ErrorResponse.class)
+ .returnResult().getResponseBody())
+ .extracting(ErrorResponse::getMessage)
+ .isEqualTo("나의 리뷰에 대한 리뷰를 작성할 수 없습니다.");
+ }
+
+ @Test
+ @DisplayName("현재 로그인한 회원의 id 가 모임 참여자에 포함되지 않는다면, 회원 리뷰 등록을 실패하고 400 bad request 을 반환한다.")
+ void Return_400_bad_request_if_meeting_not_contain_current_user_id_() {
+ // given
+ var user = repository.saveAndGetUser();
+ var participant1 = repository.saveAndGetUser();
+ var participant2 = repository.saveAndGetUser();
+
+ var closedMeeting = repository.saveAndGetCloseMeetingByParticipantUserIds(
+ List.of(participant1.getId(), participant2.getId()));
+
+ var request = RequestFixture.reviewRegisterRequest(List.of(participant1, participant2));
+
+ securityContextSetting.set(user.getId());
// when
- var expected = api.registerUserReview(VALID_TOKEN, meeting.getId(), request);
+ var expected = api.registerUserReview(VALID_TOKEN, closedMeeting.getId(), request);
// then
Assertions.assertThat(expected.expectStatus().isBadRequest()
- .expectBody(ErrorResponse.class)
- .returnResult().getResponseBody());
+ .expectBody(ErrorResponse.class)
+ .returnResult().getResponseBody())
+ .extracting(ErrorResponse::getMessage)
+ .isEqualTo("모임에 참여하지 않은 회원입니다.");
}
}
}
+
diff --git a/src/test/java/net/teumteum/meeting/domain/MeetingFixture.java b/src/test/java/net/teumteum/meeting/domain/MeetingFixture.java
index 09b850e..98442a7 100644
--- a/src/test/java/net/teumteum/meeting/domain/MeetingFixture.java
+++ b/src/test/java/net/teumteum/meeting/domain/MeetingFixture.java
@@ -26,6 +26,14 @@ public static Meeting getOpenMeeting() {
);
}
+ public static Meeting getOpenMeetingWithId(Long meetingId) {
+ return newMeetingByBuilder(MeetingBuilder.builder()
+ .id(meetingId)
+ .promiseDateTime(LocalDateTime.of(4000, 1, 1, 0, 0))
+ .build());
+
+ }
+
public static Meeting getCloseMeeting() {
return newMeetingByBuilder(MeetingBuilder.builder()
.promiseDateTime(LocalDateTime.of(2000, 1, 1, 0, 0))
@@ -33,6 +41,29 @@ public static Meeting getCloseMeeting() {
);
}
+ public static Meeting getCloseMeetingWithId(Long meetingId) {
+ return newMeetingByBuilder(MeetingBuilder.builder()
+ .id(meetingId)
+ .promiseDateTime(LocalDateTime.of(2000, 1, 1, 0, 0))
+ .build());
+ }
+
+ public static Meeting getCloseMeetingWithIdAndParticipantIds(Long meetingId, List participantIds) {
+ return newMeetingByBuilder(MeetingBuilder.builder()
+ .id(meetingId)
+ .participantUserIds(new HashSet<>(participantIds))
+ .promiseDateTime(LocalDateTime.of(2000, 1, 1, 0, 0))
+ .build()
+ );
+ }
+
+ public static Meeting getCloseMeetingWithParticipantIds(List participantIds) {
+ return newMeetingByBuilder(MeetingBuilder.builder()
+ .participantUserIds(new HashSet<>(participantIds))
+ .promiseDateTime(LocalDateTime.of(2000, 1, 1, 0, 0))
+ .build());
+ }
+
public static Meeting getOpenFullMeeting() {
return newMeetingByBuilder(MeetingBuilder.builder()
.promiseDateTime(LocalDateTime.of(4000, 1, 1, 0, 0))
diff --git a/src/test/java/net/teumteum/unit/user/controller/UserControllerTest.java b/src/test/java/net/teumteum/unit/user/controller/UserControllerTest.java
index 2b1d877..281e8be 100644
--- a/src/test/java/net/teumteum/unit/user/controller/UserControllerTest.java
+++ b/src/test/java/net/teumteum/unit/user/controller/UserControllerTest.java
@@ -89,9 +89,9 @@ class Register_user_card_api_unit {
@DisplayName("유효한 사용자의 등록 요청값이 주어지면, 201 Created 상태값을 반환한다.")
void Register_user_card_with_201_created() throws Exception {
// given
- UserRegisterRequest request = RequestFixture.userRegisterRequest(user);
+ var request = RequestFixture.userRegisterRequest(user);
- UserRegisterResponse response = new UserRegisterResponse(1L, VALID_ACCESS_TOKEN, VALID_REFRESH_TOKEN);
+ var response = new UserRegisterResponse(1L, VALID_ACCESS_TOKEN, VALID_REFRESH_TOKEN);
given(userService.register(any(UserRegisterRequest.class))).willReturn(response);
@@ -112,7 +112,7 @@ void Register_user_card_with_201_created() throws Exception {
@DisplayName("이미 카드 등록한 사용자의 등록 요청값이 주어지면, 400 Bad Request을 반환한다.")
void Return_400_bad_request_if_user_already_exist() throws Exception {
// given
- UserRegisterRequest request = RequestFixture.userRegisterRequest(user);
+ var request = RequestFixture.userRegisterRequest(user);
given(userService.register(any(UserRegisterRequest.class)))
.willThrow(new IllegalArgumentException("일치하는 user 가 이미 존재합니다."));
@@ -132,7 +132,7 @@ void Return_400_bad_request_if_user_already_exist() throws Exception {
@DisplayName("유효하지 않은 사용자의 등록 요청값이 주어지면, 400 Bad Request 상태값을 반환한다.")
void Register_user_card_with_400_bad_request() throws Exception {
// given
- UserRegisterRequest request = RequestFixture.userRegisterRequestWithNoValid(user);
+ var request = RequestFixture.userRegisterRequestWithNoValid(user);
// when
// then
mockMvc.perform(post("/users")
@@ -154,7 +154,7 @@ class Withdraw_user_api_unit {
@DisplayName("회원 탈퇴 사유와 회원 탈퇴 요청이 들어오면, 탈퇴를 진행하고 200 OK을 반환한다.")
void Withdraw_user_with_200_ok() throws Exception {
// given
- UserWithdrawRequest request
+ var request
= RequestFixture.userWithdrawRequest(List.of("쓰지 않는 앱이에요", "오류가 생겨서 쓸 수 없어요"));
// when & then
@@ -171,7 +171,7 @@ void Withdraw_user_with_200_ok() throws Exception {
@DisplayName("회원 탈퇴 하고자 하는 회원이 존재하지 않으면, 400 Bad Request을 반환한다.")
void Return_400_bad_request_if_user_is_not_exist() throws Exception {
// given
- UserWithdrawRequest request
+ var request
= RequestFixture.userWithdrawRequest(List.of("쓰지 않는 앱이에요", "오류가 생겨서 쓸 수 없어요"));
doThrow(new IllegalArgumentException("일치하는 user가 이미 존재합니다.")).when(userService).withdraw(any(
@@ -196,7 +196,7 @@ class Register_user_review_api_unit {
@DisplayName("회원 id 와 리뷰 정보 요청이 들어오면, 회원 리뷰를 등록하고 200 OK을 반환한다.")
void Register_user_review_with_200_ok() throws Exception {
// given
- ReviewRegisterRequest reviewRegisterRequest = RequestFixture.reviewRegisterRequest();
+ var reviewRegisterRequest = RequestFixture.reviewRegisterRequest();
// when & then
mockMvc.perform(post("/users/reviews")
@@ -211,11 +211,11 @@ void Register_user_review_with_200_ok() throws Exception {
@Test
@DisplayName("현재 로그인한 회원의 id 가 리뷰 등록 요청에 포함된다면, 회원 리뷰 등록을 실패하고 400 bad request을 반환한다.")
- void Register_reviews_with_400_bad_request() throws Exception {
+ void Return_400_bad_request_if_request_contains_current_user_id() throws Exception {
// given
- ReviewRegisterRequest reviewRegisterRequest = RequestFixture.reviewRegisterRequest();
+ var reviewRegisterRequest = RequestFixture.reviewRegisterRequest();
- String errorMessage = "나의 리뷰에 대한 리뷰를 작성할 수 없습니다.";
+ var errorMessage = "나의 리뷰에 대한 리뷰를 작성할 수 없습니다.";
doThrow(new IllegalArgumentException(errorMessage))
.when(userService)
diff --git a/src/test/java/net/teumteum/unit/user/service/UserServiceTest.java b/src/test/java/net/teumteum/unit/user/service/UserServiceTest.java
index 1c5630a..f3afd61 100644
--- a/src/test/java/net/teumteum/unit/user/service/UserServiceTest.java
+++ b/src/test/java/net/teumteum/unit/user/service/UserServiceTest.java
@@ -22,6 +22,7 @@
import net.teumteum.core.security.service.RedisService;
import net.teumteum.integration.RequestFixture;
import net.teumteum.meeting.domain.MeetingConnector;
+import net.teumteum.meeting.domain.MeetingFixture;
import net.teumteum.user.domain.User;
import net.teumteum.user.domain.UserFixture;
import net.teumteum.user.domain.UserRepository;
@@ -164,60 +165,56 @@ void Register_user_review_with_200_ok() {
// given
ReviewRegisterRequest reviewRegisterRequest = RequestFixture.reviewRegisterRequest();
- Long meetingId = 1L;
+ var meeting = MeetingFixture.getCloseMeetingWithIdAndParticipantIds(1L, List.of(1L, 2L, 10L));
+ var userId = 10L;
- Long userId = 10L;
-
- Long currentUserId = 20L;
-
- given(meetingConnector.existById(anyLong()))
- .willReturn(true);
+ given(meetingConnector.findById(anyLong()))
+ .willReturn(Optional.of(meeting));
given(userRepository.findById(anyLong()))
- .willReturn(Optional.of(UserFixture.getUserWithId(userId++)));
+ .willReturn(Optional.of(UserFixture.getUserWithId(userId)));
// when
- userService.registerReview(meetingId, currentUserId, reviewRegisterRequest);
+ userService.registerReview(meeting.getId(), userId, reviewRegisterRequest);
// then
- verify(meetingConnector, times(1)).existById(anyLong());
+ verify(meetingConnector, times(1)).findById(anyLong());
verify(userRepository, times(3)).findById(anyLong());
}
@Test
- @DisplayName("회원 id 가 리뷰 정보 요청에 포함되면, 400 Bad Request 와 함께 리뷰 등록을 실패한다.")
- void Return_400_bad_request_if_current_user_id_in_request() {
+ @DisplayName("meeting id 에 해당하는 meeting 이 존재하지 않는 경우, 400 Bad Request 와 함께 리뷰 등록을 실패한다.")
+ void Return_400_bad_request_if_meeting_is_not_exist() {
// given
- ReviewRegisterRequest reviewRegisterRequest = RequestFixture.reviewRegisterRequest();
-
- Long meetingId = 1L;
+ var reviewRegisterRequest = RequestFixture.reviewRegisterRequest();
- Long currentUserId = reviewRegisterRequest.reviews().get(0).id();
-
- given(meetingConnector.existById(anyLong()))
- .willReturn(true);
+ var meeting = MeetingFixture.getCloseMeetingWithId(1L);
+ var currentUserId = 1L;
+ given(meetingConnector.findById(anyLong()))
+ .willReturn(Optional.empty());
// when & then
- assertThatThrownBy(() -> userService.registerReview(meetingId, currentUserId, reviewRegisterRequest))
+ assertThatThrownBy(() -> userService.registerReview(meeting.getId(), currentUserId, reviewRegisterRequest))
.isInstanceOf(IllegalArgumentException.class)
- .hasMessage("나의 리뷰에 대한 리뷰를 작성할 수 없습니다.");
+ .hasMessage("meetingId에 해당하는 모임을 찾을 수 없습니다. \"" + meeting.getId() + "\"");
}
@Test
- @DisplayName("meeting id 에 해당하는 meeting 이 존재하지 않는 경우, 400 Bad Request 와 함께 리뷰 등록을 실패한다.")
- void Return_400_bad_request_if_meeting_is_not_exist() {
+ @DisplayName("meeting id 에 해당하는 meeting 이 아직 종료되지 않았다면, 400 Bad Request 와 함께 리뷰 등록을 실패한다.")
+ void Return_400_bad_request_if_meeting_is_not_closed() {
// given
- ReviewRegisterRequest reviewRegisterRequest = RequestFixture.reviewRegisterRequest();
+ var reviewRegisterRequest = RequestFixture.reviewRegisterRequest();
+
+ var meeting = MeetingFixture.getOpenMeetingWithId(1L);
+ var currentUserId = 1L;
- Long meetingId = 1L;
- Long currentUserId = 1L;
+ given(meetingConnector.findById(anyLong()))
+ .willReturn(Optional.of(meeting));
- given(meetingConnector.existById(anyLong()))
- .willReturn(false);
// when & then
- assertThatThrownBy(() -> userService.registerReview(meetingId, currentUserId, reviewRegisterRequest))
+ assertThatThrownBy(() -> userService.registerReview(meeting.getId(), currentUserId, reviewRegisterRequest))
.isInstanceOf(IllegalArgumentException.class)
- .hasMessage("meetingId에 해당하는 meeting을 찾을 수 없습니다. \"" + meetingId + "\"");
+ .hasMessage("해당 모임은 아직 종료되지 않았습니다.");
}
}
From 76f9275a0f32df2a191c71e15fa5c5510b1c97f8 Mon Sep 17 00:00:00 2001
From: ChoiDongKuen
Date: Sat, 17 Feb 2024 02:14:34 +0900
Subject: [PATCH 13/13] =?UTF-8?q?feat:=20=EC=9C=84=EC=B9=98=20=EA=B8=B0?=
=?UTF-8?q?=EB=B0=98=20API=20Gatling=20=EB=B6=80=ED=95=98=20=ED=85=8C?=
=?UTF-8?q?=EC=8A=A4=ED=8A=B8=20(#220)?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
* chore: docker-compose.yml 작성 (#195)
* test: Gatling 부하 테스트 구현 (#195)
* Update .env
* Delete .env
---
.gitignore | 3 +
Docker-compose.yml | 56 ++++++++++++++++
src/gatling/java/protocol/Protocol.java | 2 +-
.../java/simulation/SimulationSample.java | 3 +-
.../simulation/TeumTeumApiSimulation.java | 66 +++++++++++++++++++
.../java/simulation/UserApiSimulation.java | 54 +++++++++++++++
6 files changed, 181 insertions(+), 3 deletions(-)
create mode 100644 Docker-compose.yml
create mode 100644 src/gatling/java/simulation/TeumTeumApiSimulation.java
create mode 100644 src/gatling/java/simulation/UserApiSimulation.java
diff --git a/.gitignore b/.gitignore
index bef95b8..216ced7 100644
--- a/.gitignore
+++ b/.gitignore
@@ -37,3 +37,6 @@ bin/
### Mac OS ###
.DS_Store
+
+### .env ###
+.env
diff --git a/Docker-compose.yml b/Docker-compose.yml
new file mode 100644
index 0000000..edafb9a
--- /dev/null
+++ b/Docker-compose.yml
@@ -0,0 +1,56 @@
+services:
+ db:
+ image: "mysql:8.3.0"
+ container_name: load_test_mysql
+ restart: always
+ environment:
+ MYSQL_ROOT_PASSWORD: root#1234
+ MYSQL_DATABASE: load_test_db
+ TZ: UTC
+
+ ports:
+ - "3000:3306"
+ deploy:
+ resources:
+ limits:
+ cpus: "0.5"
+ memory: "512MB"
+
+ redis:
+ image: "docker.io/bitnami/redis:7.2"
+ container_name: load_test_redis
+ restart: always
+ environment:
+ - ALLOW_EMPTY_PASSWORD=yes
+ - REDIS_AOF_ENABLED=yes
+ - REDIS_RDB_ENABLED=no
+ ports:
+ - "6299:6379"
+ deploy:
+ resources:
+ limits:
+ cpus: "0.5"
+ memory: "512MB"
+
+ teumteum-server:
+ build:
+ context: .
+ dockerfile: Dockerfile
+ restart: always
+ environment:
+ DB_URL: jdbc:mysql://db:3306/load_test_db
+ DB_USERNAME: root
+ DB_PASSWORD: root#1234
+ REDIS_HOST: redis
+ REDIS_PORT: 6379
+ JWT_SECERT_KEY: ${JWT_ACCESS_KEY}
+
+ depends_on:
+ - db
+ - redis
+ ports:
+ - "8080:8080"
+
+networks:
+ teumteum_local:
+ driver: bridge
diff --git a/src/gatling/java/protocol/Protocol.java b/src/gatling/java/protocol/Protocol.java
index b131c75..6f8fe8d 100644
--- a/src/gatling/java/protocol/Protocol.java
+++ b/src/gatling/java/protocol/Protocol.java
@@ -7,7 +7,7 @@ public class Protocol {
private static final String USER_AGENT = "Mozilla/5.0 (Macintosh; Intel Mac OS X 10.8; rv:16.0) Gecko/20100101 Firefox/16.0";
- public static final HttpProtocolBuilder httpProtocol = HttpDsl.http.baseUrl("https://api.teum.org")
+ public static final HttpProtocolBuilder httpProtocol = HttpDsl.http.baseUrl("http://localhost:8080")
.header("Content-Type", "application/json")
.userAgentHeader(USER_AGENT);
diff --git a/src/gatling/java/simulation/SimulationSample.java b/src/gatling/java/simulation/SimulationSample.java
index c5f90cf..1c137d2 100644
--- a/src/gatling/java/simulation/SimulationSample.java
+++ b/src/gatling/java/simulation/SimulationSample.java
@@ -16,8 +16,7 @@ public class SimulationSample extends Simulation {
private final ScenarioBuilder scn = scenario(this.getClass().getSimpleName())
.exec(http("get user")
.get("/users/1")
- .check(status().is(200))
- );
+ .check(status().is(200)));
{
setUp(
diff --git a/src/gatling/java/simulation/TeumTeumApiSimulation.java b/src/gatling/java/simulation/TeumTeumApiSimulation.java
new file mode 100644
index 0000000..b06f56b
--- /dev/null
+++ b/src/gatling/java/simulation/TeumTeumApiSimulation.java
@@ -0,0 +1,66 @@
+package simulation;
+
+import static io.gatling.javaapi.core.CoreDsl.StringBody;
+import static io.gatling.javaapi.core.CoreDsl.constantUsersPerSec;
+import static io.gatling.javaapi.core.CoreDsl.jsonPath;
+import static io.gatling.javaapi.core.CoreDsl.rampUsersPerSec;
+import static io.gatling.javaapi.core.CoreDsl.scenario;
+import static io.gatling.javaapi.http.HttpDsl.http;
+import static io.gatling.javaapi.http.HttpDsl.status;
+import static protocol.Protocol.httpProtocol;
+
+import io.gatling.javaapi.core.ScenarioBuilder;
+import io.gatling.javaapi.core.Simulation;
+import java.time.Duration;
+import java.time.temporal.ChronoUnit;
+import net.datafaker.Faker;
+
+public class TeumTeumApiSimulation extends Simulation {
+
+ private static final Faker faker = new Faker();
+ private final ScenarioBuilder teumteumScn = scenario("TeumTeum 찬해지기 API 부하 테스트를 진행한다.")
+ .exec(session ->
+ session
+ .set("id", java.util.UUID.randomUUID().toString())
+ .set("name", faker.name().fullName()))
+
+ .exec(http("User 카드 등록 API 요청")
+ .post("/users")
+ .body(StringBody(
+ "{"
+ + "\"id\": \"${id}\", "
+ + "\"terms\": {\"service\": true, \"privatePolicy\": true}, "
+ + "\"name\": \"${name}\", "
+ + "\"birth\": \"20000402\", "
+ + "\"characterId\": 2, "
+ + "\"authenticated\": \"네이버\", "
+ + "\"activityArea\": \"경기 시흥\", "
+ + "\"mbti\": \"ENFP\", "
+ + "\"status\": \"직장인\", "
+ + "\"job\": {\"name\" : \"카카오 뱅크\", \"class\" : \"개발\", \"detailClass\" : \"BE 개발자\"}, "
+ + "\"interests\": [\"네트워킹\", \"IT\", \"모여서 각자 일하기\"], "
+ + "\"goal\": \"회사에서 좋은 사람들과 멋진 개발하기\""
+ + "}"
+ ))
+ .check(status().is(201))
+ .check(jsonPath("$.id").saveAs("userId"))
+ .check(jsonPath("$.accessToken").saveAs("accessToken"))
+ .check(jsonPath("$.refreshToken").saveAs("refreshToken")))
+
+ .exec(http("TeumTeum 친해지기 API 요청")
+ .post("/teum-teum/around")
+ .header("Authorization", "Bearer ${accessToken}")
+ .body(StringBody("{\"id\": ${userId}, \"latitude\": 37.5665, \"longitude\": 126.9780,"
+ + " \"name\": \"test_name\", \"jobDetailClass\": \"test_job\", \"characterId\": 1}"))
+ .check(status().is(200))
+ );
+
+ {
+ setUp(
+ teumteumScn.injectOpen(
+ constantUsersPerSec(10).during(Duration.of(30, ChronoUnit.SECONDS)),
+ rampUsersPerSec(10).to(50).during(Duration.of(30, ChronoUnit.SECONDS))
+ ).protocols(httpProtocol)
+ );
+ }
+}
diff --git a/src/gatling/java/simulation/UserApiSimulation.java b/src/gatling/java/simulation/UserApiSimulation.java
new file mode 100644
index 0000000..9f7acee
--- /dev/null
+++ b/src/gatling/java/simulation/UserApiSimulation.java
@@ -0,0 +1,54 @@
+package simulation;
+
+import static io.gatling.javaapi.core.CoreDsl.StringBody;
+import static io.gatling.javaapi.core.CoreDsl.atOnceUsers;
+import static io.gatling.javaapi.core.CoreDsl.jsonPath;
+import static io.gatling.javaapi.core.CoreDsl.scenario;
+import static io.gatling.javaapi.http.HttpDsl.http;
+import static io.gatling.javaapi.http.HttpDsl.status;
+import static protocol.Protocol.httpProtocol;
+
+import io.gatling.javaapi.core.ScenarioBuilder;
+import io.gatling.javaapi.core.Simulation;
+
+public class UserApiSimulation extends Simulation {
+
+ private final ScenarioBuilder UserScn = scenario("User API 부하 테스트를 진행한다.")
+ .exec(http("User 카드 등록 API 요청")
+ .post("/users")
+ .body(StringBody(
+ "{\"id\": \"test_id\", "
+ + "\"terms\": {\"service\": true, \"privatePolicy\": true}, "
+ + "\"name\": \"홍길동\", "
+ + "\"birth\": \"1990-01-01\", "
+ + "\"characterId\": 1, "
+ + "\"authenticated\": \"SNS\", "
+ + "\"activityArea\": \"서울\", "
+ + "\"mbti\": \"INTJ\", "
+ + "\"status\": \"ACTIVE\", "
+ + "\"job\": {\"name\": \"개발자\", \"class\": \"IT\", \"detailClass\": \"백엔드\"}, "
+ + "\"interests\": [\"코딩\", \"독서\", \"운동\"], "
+ + "\"goal\": \"성장하기 위해 노력하는 개발자가 되기\"}"
+ ))
+ .check(status().is(201))
+ .check(jsonPath("$.id").saveAs("userId"))
+ .check(jsonPath("$.accessToken").saveAs("accessToken"))
+ .check(jsonPath("$.refreshToken").saveAs("refreshToken"))
+
+ ).exec(http("User 정보 조회 API 요청")
+ .get("/users/${userId}")
+ .header("Authorization", "Bearer ${accessToken}")
+ .check(status().is(200))
+
+ ).exec(http("User 리뷰 조회 API 요청")
+ .get("/users/${userId}")
+ .header("Authorization", "Bearer ${accessToken}")
+ .check(status().is(200)));
+
+
+ {
+ setUp(
+ UserScn.injectOpen(
+ atOnceUsers(10)).protocols(httpProtocol));
+ }
+}