-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathProjectService.java
More file actions
396 lines (315 loc) · 16.4 KB
/
ProjectService.java
File metadata and controls
396 lines (315 loc) · 16.4 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
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
package com.kwcapstone.Service;
import com.kwcapstone.Domain.Dto.Request.EmailInviteRequestDto;
import com.kwcapstone.Domain.Dto.Request.ProjectDeleteRequestDto;
import com.kwcapstone.Domain.Dto.Request.ProjectNameEditRequestDto;
import com.kwcapstone.Domain.Dto.Response.*;
import com.kwcapstone.Domain.Entity.*;
import com.kwcapstone.Repository.*;
import com.kwcapstone.Security.PrincipalDetails;
import lombok.RequiredArgsConstructor;
import org.bson.types.ObjectId;
import org.springframework.http.HttpStatus;
import org.springframework.security.core.annotation.AuthenticationPrincipal;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;
import org.springframework.web.server.ResponseStatusException;
import java.time.LocalDateTime;
import java.util.ArrayList;
import java.util.List;
import java.util.Optional;
import java.util.UUID;
import java.util.stream.Collectors;
@Service
@Transactional
@RequiredArgsConstructor
public class ProjectService {
private final ProjectRepository projectRepository;
private final InviteRepository inviteRepository;
private final EmailService emailService;
private final MemberRepository memberRepository;
private final MemberToProjectRepository memberToProjectRepository;
private final NoticeRepository noticeRepository;
// 이메일로 프로젝트에 사용자 추가하기
public InviteEmailResponseDto addByEmailUser(PrincipalDetails principalDetails,
String projectId, EmailInviteRequestDto emailInviteRequestDto) {
ObjectId memberId = principalDetails.getId();
Optional<Member> member = memberRepository.findById(memberId);
if(!member.isPresent()) {
throw new ResponseStatusException(HttpStatus.NOT_FOUND, "잘못된 ObjectId 형식입니다.");
}
//사용자 추가할 때 projectId에 memberId가 회원가입 처리가 되어있어야 함.
Project project = getProject(projectId);
// 1. 초대 코드 생성 (UUID 또는 토큰)
String inviteCode = UUID.randomUUID().toString();
// 2. 이메일 전송 - 이거 올릴땐 다른 주소로 해야함. www.moaba.site로
String inviteLink = "https://www.moaba.site/main/project/" + projectId + "/accept?code=" + inviteCode;
String inviterName = principalDetails.getUsername();
String projectName = project.getProjectName();
Optional<Member> isMember = memberRepository.findByEmail(emailInviteRequestDto.getEmail());
if (!isMember.isPresent()) {
throw new ResponseStatusException(HttpStatus.NOT_FOUND, "회원가입된 사용자가 아닙니다.");
}
emailService.sendProjectInviteMessage(
emailInviteRequestDto.getEmail(),
inviteLink,
inviterName,
projectName
);
saveInviteCode(inviteCode, projectId, emailInviteRequestDto.getEmail(), memberId);
// 코드 보내고 모아바 웹에 알림 보내기
String noticeTitle = inviterName + "님이 " + projectName + " 프로젝트에 초대하였습니다.";
Notice notice = Notice.builder()
.title(noticeTitle)
.url(inviteLink)
.createAt(LocalDateTime.now())
.isRead(false)
.official(false)
.userId(isMember.get().getMemberId())
.senderId(memberId)
.build();
noticeRepository.save(notice);
return new InviteEmailResponseDto(
projectId,
emailInviteRequestDto.getEmail(),
inviteCode
);
}
// 프로젝트 찾기
private Project getProject(String projectId) {
ObjectId objProjectId = new ObjectId(projectId);
Optional<Project> project = projectRepository.findByProjectId(objProjectId);
if(!project.isPresent()) {
throw new ResponseStatusException(HttpStatus.NOT_FOUND, "프로젝트를 찾을 수 없습니다.");
}
return project.get();
}
private void saveInviteCode(String inviteCode, String projectId, String email, ObjectId userId) {
Invite invite = Invite.builder()
.inviteCode(inviteCode)
.projectId(new ObjectId(projectId))
.email(email)
.userId(userId)
.createdAt(LocalDateTime.now())
.expiredAt(LocalDateTime.now().plusHours(72)) // 유효시간 : 3일
.build();
inviteRepository.save(invite);
}
// 초대 수락
public void acceptInvite(PrincipalDetails principalDetails, String projectId, String code) {
ObjectId memberId = principalDetails.getId();
Optional<Member> isMember = memberRepository.findById(memberId);
if(!isMember.isPresent()) {
throw new ResponseStatusException(HttpStatus.NOT_FOUND, "초대 수락 전 로그인을 먼저 하세요.");
}
// 초대 코드 검증
Invite invite = validateInviteCode(code, projectId);
Member invitedMember = memberRepository.findByEmail(invite.getEmail())
.orElseThrow(() -> new ResponseStatusException(HttpStatus.NOT_FOUND, "가입된 사용자가 아닙니다."));
// principalDetails에서 ID를 가져와서 초대된 사용자의 ID와 비교
if (!invitedMember.getMemberId().equals(principalDetails.getId())) {
throw new ResponseStatusException(HttpStatus.FORBIDDEN, "초대된 사용자가 아닙니다.");
}
//project가 있는지를 확인하기
ObjectId projectObjectId = new ObjectId(projectId);
Optional<Project> project = projectRepository.findByProjectId(projectObjectId);
if(!project.isPresent()) {
throw new ResponseStatusException(HttpStatus.NOT_FOUND, "프로젝트를 찾을 수 없습니다.");
}
//member가 있는지 확인하기
Member member = memberRepository.findByMemberId(principalDetails.getId())
.orElseThrow(() -> new ResponseStatusException(HttpStatus.NOT_FOUND, "사용자를 찾을 수 없습니다."));
// 이미 추가된 프로젝트의 경우 예외처리
if (memberToProjectRepository.existsByProjectIdAndMemberId(projectObjectId, memberId)) {
throw new ResponseStatusException(HttpStatus.CONFLICT, "이미 추가된 프로젝트입니다.");
}
//관계 추가하기
MemberToProject memberToProject = MemberToProject.builder()
.projectId(new ObjectId(projectId))
.memberId(member.getMemberId())
.build();
memberToProjectRepository.save(memberToProject);
//member에도 관계성 추가
if (member.getProjectIds() == null) {
member.setProjectIds(new ArrayList<>());
}
member.getProjectIds().add(new ObjectId(projectId));
memberRepository.save(member);
}
//유효성 검사
private Invite validateInviteCode(String code, String projectId) {
Invite invite = inviteRepository.findByInviteCode(code)
.orElseThrow(() -> new ResponseStatusException(HttpStatus.NOT_FOUND, "유효하지 않은 초대 코드입니다."));
if (!invite.getProjectId().toString().equals(projectId)) {
throw new ResponseStatusException(HttpStatus.NOT_FOUND, "초대 코드가 잘못되었습니다.");
}
if(invite.getExpiredAt().isBefore(LocalDateTime.now())){
throw new ResponseStatusException(HttpStatus.GONE, "초대 코드가 만료되었습니다.");
}
return invite;
}
// 프로젝트 삭제 기능
public void deleteProject(PrincipalDetails principalDetails, List<ProjectDeleteRequestDto> deleteRequestList) {
ObjectId memberId = principalDetails.getId();
for (ProjectDeleteRequestDto dto : deleteRequestList) {
ObjectId projectId = dto.getProjectId();
String type = dto.getType().toLowerCase();
Optional<Project> project = projectRepository.findByProjectId(projectId);
if(!project.isPresent()){
throw new ResponseStatusException(HttpStatus.NOT_FOUND, "프로젝트를 찾을 수 없습니다.");
}
if (!project.get().getCreator().equals(memberId)) {
throw new ResponseStatusException(HttpStatus.FORBIDDEN, "프로젝트 삭제 권한이 없습니다.");
}
if (type.equals("project")) {
projectRepository.deleteByProjectId(projectId);
memberToProjectRepository.deleteByProjectId(projectId);
} else if (type.equals("record")) {
project.get().setZipFile(null);
project.get().setScript(null);
projectRepository.save(project.get());
} else if (type.equals("summary")) {
project.get().setSummary(null);
projectRepository.save(project.get());
} else {
throw new ResponseStatusException(HttpStatus.BAD_REQUEST, "유효하지 않은 삭제 타입입니다.");
}
}
}
//프로젝트 이름 수정
public ProjectNameEditResponseDto editProjectName(String projectId,
ProjectNameEditRequestDto projectNameEditRequestDto){
ObjectId ObjprojectId = new ObjectId(projectId);
Optional<Project> project = projectRepository.findByProjectId(ObjprojectId);
if(!project.isPresent()){
throw new ResponseStatusException(HttpStatus.NOT_FOUND, "프로젝트를 찾을 수 없습니다.");
}
project.get().editName(projectNameEditRequestDto.getProjectName());
projectRepository.save(project.get());
return new ProjectNameEditResponseDto(
projectId,
project.get().getProjectName());
}
//프로젝트 공유 모달 띄우기
public GetProjectShareModalResponseDto getProjectShareModal(String projectId,
PrincipalDetails principalDetails) {
ObjectId memberId = principalDetails.getId();
ObjectId ObjprojectId = new ObjectId(projectId);
Optional<Project> project = projectRepository.findByProjectId(ObjprojectId);
if(!project.isPresent()){
throw new ResponseStatusException(HttpStatus.NOT_FOUND, "프로젝트를 찾을 수 없습니다.");
}
//inviteLink
String inviteUrl = "https://www.moaba.site/project/" + projectId;
//참여자 목록 가져오기
List<MemberToProject> connections = memberToProjectRepository.findByProjectId(ObjprojectId);
if(connections == null){
throw new ResponseStatusException(HttpStatus.INTERNAL_SERVER_ERROR, "프로젝트 참여자 목록에 null 값입니다.");
}
List<MemberInfoDto> sharedMembers = connections.stream()
.map(conn -> {
Member member = memberRepository.findByMemberId(conn.getMemberId())
.orElseThrow(() -> new ResponseStatusException(HttpStatus.NOT_FOUND, "참여자 목록에서 회원 정보를 찾을 수 없는 참가자가 존재합니다."));
//회의 생성자일때
if((project.get().getCreator()).equals(conn.getMemberId())) {
return new MemberInfoDto(member.getName(), "회의 생성자");
}else{
return new MemberInfoDto(member.getName(), "참석자");
}
}).sorted((a, b) -> {
// "회의 생성자"가 먼저 오도록 정렬
if (a.getRole().equals("회의 생성자")) return -1;
if (b.getRole().equals("회의 생성자")) return 1;
return 0;
}).collect(Collectors.toList());
return new GetProjectShareModalResponseDto(inviteUrl, sharedMembers);
}
//프로젝트 공유링크로 들어왔을 때 사용자 추가
public boolean addByLink(PrincipalDetails principalDetails,
String projectId){
ObjectId memberId = principalDetails.getId();
if(memberId == null){
throw new ResponseStatusException(HttpStatus.INTERNAL_SERVER_ERROR, "토큰에서 넘겨진 memberId 가 null 입니다.");
}
ObjectId objProjectId = new ObjectId(projectId);
Optional<Project> project = projectRepository.findByProjectId(objProjectId);
if(!project.isPresent()){
throw new ResponseStatusException(HttpStatus.NOT_FOUND, "프로젝트를 찾을 수 없습니다.");
}
//이미 참여중인가?
boolean alreadyJoined = memberToProjectRepository.existsByProjectIdAndMemberId(objProjectId, memberId);
if(alreadyJoined){
return true;
}
//참가자 등록
MemberToProject mapping = MemberToProject.builder()
.projectId(objProjectId)
.memberId(memberId)
.build();
memberToProjectRepository.save(mapping);
//사용자 객체에도 pojectid
Optional<Member> member = memberRepository.findByMemberId(memberId);
if(!member.isPresent()){
throw new ResponseStatusException(HttpStatus.NOT_FOUND, "존재하지 않는 MemberId 입니다.");
}
if(member.get().getProjectIds() == null){
member.get().setProjectIds(new ArrayList<>());
}
member.get().getProjectIds().add(objProjectId);
memberRepository.save(member.get());
return false;
}
//프로젝트 status 띄우기
public ProjectStatusResponseDto getProjectStatus(PrincipalDetails principalDetails,
String projectId){
ObjectId memberId = principalDetails.getId();
if(memberId == null){
throw new ResponseStatusException(HttpStatus.INTERNAL_SERVER_ERROR, "토큰에서 넘겨진 memberId 가 null 입니다.");
}
ObjectId objProjectId = new ObjectId(projectId);
Optional<Project> project = projectRepository.findByProjectId(objProjectId);
if(!project.isPresent()){
throw new ResponseStatusException(HttpStatus.NOT_FOUND, "프로젝트를 찾을 수 없습니다.");
}
ObjectId creatorId = project.get().getCreator();
String creatorName = memberRepository.findByMemberId(creatorId)
.map(Member::getName)
.orElse("알 수 없음");
String projectStatus = project.get().getStatus();
String projectName = project.get().getProjectName();
LocalDateTime updatedAt = project.get().getUpdatedAt();
return new ProjectStatusResponseDto(projectId, creatorName, projectStatus, projectName, updatedAt);
}
//프로젝트 추출
public ExportProjectResponseDto exportProject(PrincipalDetails principalDetails,
String projectId,
ExportKind kind){
ObjectId memberId = principalDetails.getId();
Optional<Member> member = memberRepository.findByMemberId(memberId);
if(!member.isPresent()){
throw new ResponseStatusException(HttpStatus.UNAUTHORIZED, "토큰에 있는 memberId 정보가 db에 존재하지 않습니다.");
}
ObjectId ObjProject = new ObjectId(projectId);
Optional<Project> project = projectRepository.findByProjectId(ObjProject);
if(!project.isPresent()){
throw new ResponseStatusException(HttpStatus.NOT_FOUND, "프로젝트를 찾을 수 없습니다.");
}
String projectUrl = "";
//kind에 따라서 프로젝트 추출하기
switch (kind) {
case MINDMAP -> {
// MindMap 추출 로직
projectUrl = project.get().getProjectImage();
}
case SUMMARY -> {
// Summary 추출 로직
projectUrl = project.get().getSummary().getSummaryUrl();
}
case RECORDING -> {
// Recording 추출 로직
projectUrl = project.get().getZipFile().getZipUrl();
}
default -> throw new IllegalArgumentException("Unsupported export kind: " + kind);
}
return new ExportProjectResponseDto(projectId, projectUrl);
}
}