-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathAdminController.java
More file actions
255 lines (208 loc) · 14.2 KB
/
AdminController.java
File metadata and controls
255 lines (208 loc) · 14.2 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
package com.linglevel.api.admin.controller;
import com.linglevel.api.admin.dto.ArticleReleaseNotificationRequest;
import com.linglevel.api.admin.dto.ArticleReleaseNotificationResponse;
import com.linglevel.api.admin.dto.GrantTicketRequest;
import com.linglevel.api.admin.dto.GrantTicketResponse;
import com.linglevel.api.admin.dto.NotificationBroadcastRequest;
import com.linglevel.api.admin.dto.NotificationBroadcastResponse;
import com.linglevel.api.admin.dto.NotificationSendResponse;
import com.linglevel.api.admin.dto.NotificationSendRequest;
import com.linglevel.api.admin.dto.RecoverStreakRequest;
import com.linglevel.api.admin.dto.RecoverStreakResponse;
import com.linglevel.api.admin.dto.ResetTodayStreakRequest;
import com.linglevel.api.fcm.dto.FcmMessageRequest;
import com.linglevel.api.admin.dto.UpdateChunkRequest;
import com.linglevel.api.admin.service.AdminService;
import com.linglevel.api.admin.service.NotificationService;
import com.linglevel.api.common.dto.ExceptionResponse;
import com.linglevel.api.common.dto.MessageResponse;
import com.linglevel.api.common.exception.CommonErrorCode;
import com.linglevel.api.common.exception.CommonException;
import com.linglevel.api.user.ticket.exception.TicketException;
import com.linglevel.api.common.dto.PageResponse;
import com.linglevel.api.content.book.dto.ChunkResponse;
import com.linglevel.api.content.article.dto.ArticleChunkResponse;
import com.linglevel.api.content.article.dto.ArticleOriginResponse;
import com.linglevel.api.content.article.dto.GetArticleOriginsRequest;
import com.linglevel.api.content.article.service.ArticleService;
import com.linglevel.api.streak.entity.UserStudyReport;
import com.linglevel.api.streak.service.StreakService;
import com.linglevel.api.user.entity.User;
import com.linglevel.api.user.repository.UserRepository;
import com.linglevel.api.user.ticket.service.TicketService;
import com.linglevel.api.version.dto.VersionUpdateRequest;
import com.linglevel.api.version.dto.VersionUpdateResponse;
import com.linglevel.api.version.service.VersionService;
import io.swagger.v3.oas.annotations.Operation;
import io.swagger.v3.oas.annotations.Parameter;
import io.swagger.v3.oas.annotations.security.SecurityRequirement;
import io.swagger.v3.oas.annotations.tags.Tag;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.springdoc.core.annotations.ParameterObject;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.*;
import jakarta.validation.Valid;
import java.util.List;
@RestController
@RequestMapping("/api/v1/admin")
@RequiredArgsConstructor
@Slf4j
@Tag(name = "Admin", description = "어드민 관리 API")
@SecurityRequirement(name = "adminApiKey")
public class AdminController {
private final AdminService adminService;
private final VersionService versionService;
private final NotificationService notificationService;
private final TicketService ticketService;
private final UserRepository userRepository;
private final ArticleService articleService;
private final StreakService streakService;
@Operation(summary = "책 청크 수정", description = "어드민 권한으로 특정 책의 청크 내용을 수정합니다.")
@PutMapping("/books/{bookId}/chapters/{chapterId}/chunks/{chunkId}")
public ResponseEntity<ChunkResponse> updateBookChunk(
@Parameter(description = "책 ID", required = true) @PathVariable String bookId,
@Parameter(description = "챕터 ID", required = true) @PathVariable String chapterId,
@Parameter(description = "청크 ID", required = true) @PathVariable String chunkId,
@Parameter(description = "청크 수정 요청", required = true) @Valid @RequestBody UpdateChunkRequest request) {
log.info("Admin updating book chunk - bookId: {}, chapterId: {}, chunkId: {}", bookId, chapterId, chunkId);
ChunkResponse response = adminService.updateBookChunk(bookId, chapterId, chunkId, request);
return ResponseEntity.ok(response);
}
@Operation(summary = "기사 청크 수정", description = "어드민 권한으로 특정 기사의 청크 내용을 수정합니다.")
@PutMapping("/articles/{articleId}/chunks/{chunkId}")
public ResponseEntity<ArticleChunkResponse> updateArticleChunk(
@Parameter(description = "기사 ID", required = true) @PathVariable String articleId,
@Parameter(description = "청크 ID", required = true) @PathVariable String chunkId,
@Parameter(description = "청크 수정 요청", required = true) @Valid @RequestBody UpdateChunkRequest request) {
log.info("Admin updating article chunk - articleId: {}, chunkId: {}", articleId, chunkId);
ArticleChunkResponse response = adminService.updateArticleChunk(articleId, chunkId, request);
return ResponseEntity.ok(response);
}
@Operation(summary = "책 삭제", description = "어드민 권한으로 특정 책과 관련된 모든 데이터(챕터, 청크, 진도, S3 파일 등)를 삭제합니다.")
@DeleteMapping("/books/{bookId}")
public ResponseEntity<MessageResponse> deleteBook(
@Parameter(description = "책 ID", required = true) @PathVariable String bookId) {
log.info("Admin deleting book - bookId: {}", bookId);
adminService.deleteBook(bookId);
return ResponseEntity.ok(new MessageResponse("Book and all related data deleted successfully."));
}
@Operation(summary = "기사 삭제", description = "어드민 권한으로 특정 기사와 관련된 모든 데이터(청크, S3 파일 등)를 삭제합니다.")
@DeleteMapping("/articles/{articleId}")
public ResponseEntity<MessageResponse> deleteArticle(
@Parameter(description = "기사 ID", required = true) @PathVariable String articleId) {
log.info("Admin deleting article - articleId: {}", articleId);
adminService.deleteArticle(articleId);
return ResponseEntity.ok(new MessageResponse("Article and all related data deleted successfully."));
}
@Operation(summary = "앱 버전 업데이트", description = "어드민 권한으로 앱의 최신 버전 및 최소 요구 버전을 부분 업데이트합니다.")
@PatchMapping("/version")
public ResponseEntity<VersionUpdateResponse> updateVersion(
@Parameter(description = "버전 업데이트 요청", required = true) @Valid @RequestBody VersionUpdateRequest request) {
log.info("Admin updating app version - latestVersion: {}, minimumVersion: {}",
request.getLatestVersion(), request.getMinimumVersion());
VersionUpdateResponse response = versionService.updateVersion(request);
return ResponseEntity.ok(response);
}
@Operation(summary = "브로드캐스트 알림 전송", description = "어드민 권한으로 전체 사용자에게 FCM 푸시 알림을 브로드캐스트합니다.")
@PostMapping("/notifications/broadcast")
public ResponseEntity<NotificationBroadcastResponse> broadcastNotification(
@Parameter(description = "브로드캐스트 알림 전송 요청", required = true) @Valid @RequestBody NotificationBroadcastRequest request) {
NotificationBroadcastResponse response = notificationService.sendBroadcastNotification(request);
return ResponseEntity.ok(response);
}
@Operation(summary = "푸시 알림 전송", description = "어드민 권한으로 특정 사용자들에게 FCM 푸시 알림을 전송합니다.")
@PostMapping("/notifications/send")
public ResponseEntity<NotificationSendResponse> sendNotification(
@Parameter(description = "알림 전송 요청", required = true) @Valid @RequestBody NotificationSendRequest request) {
NotificationSendResponse response = notificationService.sendNotificationFromRequest(request);
return ResponseEntity.ok(response);
}
@Operation(summary = "아티클 출시 알림 전송", description = "AI 서버가 새 아티클 출시 시 국가와 카테고리 기반으로 타겟 사용자에게 푸시 알림을 전송합니다.")
@PostMapping("/notifications/article-release")
public ResponseEntity<ArticleReleaseNotificationResponse> sendArticleReleaseNotification(
@Parameter(description = "아티클 출시 알림 전송 요청", required = true) @Valid @RequestBody ArticleReleaseNotificationRequest request) {
ArticleReleaseNotificationResponse response = notificationService.sendArticleReleaseNotification(request);
return ResponseEntity.ok(response);
}
@Operation(summary = "티켓 지급", description = "어드민 권한으로 사용자에게 티켓을 지급합니다.")
@PostMapping("/tickets/grant")
public ResponseEntity<GrantTicketResponse> grantTicket(
@Parameter(description = "티켓 지급 요청", required = true) @Valid @RequestBody GrantTicketRequest request) {
log.info("Admin granting tickets - userId: {}, amount: {}, reason: {}",
request.getUserId(), request.getAmount(), request.getReason());
// 사용자 존재 여부 확인
User user = userRepository.findById(request.getUserId())
.orElseThrow(() -> new CommonException(CommonErrorCode.RESOURCE_NOT_FOUND, "User not found."));
String reason = request.getReason() != null ? request.getReason() : "관리자 지급";
int newBalance = ticketService.grantTicket(user.getId(), request.getAmount(), reason);
GrantTicketResponse response = GrantTicketResponse.builder()
.message("Tickets granted successfully.")
.userId(request.getUserId())
.amount(request.getAmount())
.newBalance(newBalance)
.build();
return ResponseEntity.ok(response);
}
@Operation(summary = "아티클 원본 URL 목록 조회", description = "어드민 권한으로 아티클의 원본 URL 목록을 필터링하여 조회합니다.")
@GetMapping("/articles/origins")
public ResponseEntity<PageResponse<ArticleOriginResponse>> getArticleOrigins(
@ParameterObject @ModelAttribute GetArticleOriginsRequest request) {
log.info("Admin fetching article origins - tags: {}, targetLanguageCode: {}",
request.getTags(), request.getTargetLanguageCode());
PageResponse<ArticleOriginResponse> response = articleService.getArticleOrigins(request);
return ResponseEntity.ok(response);
}
@Operation(summary = "아티클 targetLanguageCode 마이그레이션", description = "targetLanguageCode가 null이거나 없는 모든 아티클에 [KO, EN, JA]를 설정합니다.")
@PostMapping("/articles/migrate/target-language-code")
public ResponseEntity<MessageResponse> migrateArticleTargetLanguageCode() {
log.info("Admin migrating article targetLanguageCode");
long updatedCount = articleService.migrateTargetLanguageCode();
String message = String.format("Successfully updated %d articles with default target language codes [KO, EN, JA]", updatedCount);
return ResponseEntity.ok(new MessageResponse(message));
}
@Operation(summary = "[Debug] 오늘의 스트릭 학습 상태 리셋", description = "디버깅용 API. 특정 사용자의 오늘 학습 완료 기록을 삭제하여, 스트릭을 달성하지 않은 상태로 되돌립니다.")
@PostMapping("/streaks/reset-today")
public ResponseEntity<MessageResponse> resetTodayStreak(@Valid @RequestBody ResetTodayStreakRequest request) {
adminService.resetTodayStreak(request.getUserId());
return ResponseEntity.ok(new MessageResponse("User " + request.getUserId() + "'s streak status for today has been reset."));
}
@Operation(summary = "스트릭 복구", description = "어드민 권한으로 특정 사용자의 누락된 스트릭을 복구합니다. MISSED 날짜는 COMPLETED로 변경하고, FREEZE_USED는 COMPLETED로 변경하며 프리즈를 보상합니다. 복구 범위 이후 날짜들도 프리즈를 사용하여 최대한 연결합니다.")
@PostMapping("/streaks/recover")
public ResponseEntity<RecoverStreakResponse> recoverStreak(
@Parameter(description = "스트릭 복구 요청", required = true) @Valid @RequestBody RecoverStreakRequest request) {
log.info("Admin recovering streak - userId: {}, startDate: {}, endDate: {}",
request.getUserId(), request.getStartDate(), request.getEndDate());
// 사용자 존재 여부 확인
User user = userRepository.findById(request.getUserId())
.orElseThrow(() -> new CommonException(CommonErrorCode.RESOURCE_NOT_FOUND, "User not found."));
// 스트릭 복구 실행
streakService.recoverStreak(request.getUserId(), request.getStartDate(), request.getEndDate());
// 복구 후 UserStudyReport 조회
UserStudyReport updatedReport = streakService.recalculateUserStudyReport(request.getUserId());
RecoverStreakResponse response = RecoverStreakResponse.builder()
.message("Streak recovered successfully.")
.userId(request.getUserId())
.startDate(request.getStartDate())
.endDate(request.getEndDate())
.currentStreak(updatedReport.getCurrentStreak())
.longestStreak(updatedReport.getLongestStreak())
.lastCompletionDate(updatedReport.getLastCompletionDate())
.build();
log.info("Streak recovery completed for user {} - currentStreak: {}, longestStreak: {}",
request.getUserId(), updatedReport.getCurrentStreak(), updatedReport.getLongestStreak());
return ResponseEntity.ok(response);
}
@ExceptionHandler(TicketException.class)
public ResponseEntity<ExceptionResponse> handleTicketException(TicketException e) {
log.error("Admin Ticket Exception: {}", e.getMessage());
return ResponseEntity.status(e.getStatus())
.body(new ExceptionResponse(e));
}
@ExceptionHandler(CommonException.class)
public ResponseEntity<ExceptionResponse> handleCommonException(CommonException e) {
log.error("Admin Common Exception: {}", e.getMessage());
return ResponseEntity.status(e.getStatus())
.body(new ExceptionResponse(e));
}
}