Handler.cs 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398
  1. using Application.Abstractions.Messaging;
  2. using Application.Abstractions.Messaging.Email;
  3. using Application.Abstractions.Data;
  4. using Application.Abstractions.Forum;
  5. using Domain.Entities.Forum.Comments;
  6. using Domain.Entities.Forum.ValueObject;
  7. using SharedKernel.Results;
  8. using SharedKernel.Storage;
  9. using Microsoft.EntityFrameworkCore;
  10. using System.Text.RegularExpressions;
  11. namespace Application.Features.Api.Forum.Comment.Create;
  12. public sealed class Handler(IAppDbContext db, IFileStorage fileStorage, IBoardPermissionService permissionService, IMailService mailService) : ICommandHandler<Command, Result<int>>
  13. {
  14. private static readonly string[] AllowedImageExtensions = [".jpg", ".jpeg", ".png", ".gif", ".webp", ".bmp"];
  15. private static readonly string[] AllowedFileExtensions = [".jpg", ".jpeg", ".png", ".gif", ".webp", ".bmp", ".pdf", ".doc", ".docx", ".xls", ".xlsx", ".ppt", ".pptx", ".txt", ".zip", ".rar", ".7z", ".hwp", ".hwpx", ".csv"];
  16. public async Task<Result<int>> Handle(Command request, CancellationToken ct)
  17. {
  18. if (string.IsNullOrWhiteSpace(request.Content))
  19. {
  20. return Result.Failure<int>(Error.Problem("Comment.ContentRequired", "내용을 입력해주세요."));
  21. }
  22. var post = await db.Post.FirstOrDefaultAsync(x => x.ID == request.PostID, ct);
  23. if (post is null)
  24. {
  25. return Result.Failure<int>(Error.NotFound("Comment.PostNotFound", "게시글을 찾을 수 없습니다."));
  26. }
  27. var member = await db.Member.AsNoTracking().FirstOrDefaultAsync(x => x.ID == request.MemberID, ct);
  28. if (member is null)
  29. {
  30. return Result.Failure<int>(Error.NotFound("Comment.MemberNotFound", "회원 정보를 찾을 수 없습니다."));
  31. }
  32. // 댓글 권한 확인
  33. var boardMeta = await db.BoardMeta.AsNoTracking().FirstOrDefaultAsync(x => x.BoardID == post.BoardID, ct);
  34. var isQnABoard = boardMeta?.List.Layout == BoardLayout.QnA;
  35. if (boardMeta is not null)
  36. {
  37. // 댓글 내용 길이 검증
  38. var contentLength = request.Content.Trim().Length;
  39. if (boardMeta.Comment.MinContentLength > 0 && contentLength < boardMeta.Comment.MinContentLength)
  40. {
  41. return Result.Failure<int>(Error.Problem("Comment.MinContentLength", $"내용은 {boardMeta.Comment.MinContentLength}자 이상 작성해주세요."));
  42. }
  43. if (boardMeta.Comment.MaxContentLength > 0 && contentLength > boardMeta.Comment.MaxContentLength)
  44. {
  45. return Result.Failure<int>(Error.Problem("Comment.MaxContentLength", $"내용은 {boardMeta.Comment.MaxContentLength}자 이내로 작성해주세요."));
  46. }
  47. // 1:1 문의 게시판: 최고관리자·매니저만 댓글/대댓글 작성 가능
  48. if (isQnABoard)
  49. {
  50. if (!member.IsAdmin)
  51. {
  52. var mgr = await permissionService.GetBoardManagerAsync(post.BoardID, member.ID, ct);
  53. if (mgr is null)
  54. {
  55. return Result.Failure<int>(Error.Forbidden("Comment.QnAPermissionDenied", "1:1 문의 게시판은 관리자만 답변할 수 있습니다."));
  56. }
  57. }
  58. }
  59. else
  60. {
  61. var requiredPermission = request.ParentID.HasValue ? boardMeta.Permission.ReplyWrite : boardMeta.Permission.CommentWrite;
  62. var requiredPermissionName = request.ParentID.HasValue ? "답글 작성 권한" : "댓글 작성 권한";
  63. if (!await permissionService.HasPermissionAsync(member, post.BoardID, requiredPermission, ct))
  64. {
  65. return Result.Failure<int>(Error.Forbidden("Comment.PermissionDenied", $"{requiredPermissionName}이 없습니다."));
  66. }
  67. }
  68. // 파일 업로드 권한 확인
  69. if ((request.Images is { Count: > 0 } || request.Files is { Count: > 0 }) &&
  70. !await permissionService.HasPermissionAsync(member, post.BoardID, boardMeta.Permission.FileUpload, ct)
  71. ) {
  72. return Result.Failure<int>(Error.Forbidden("Comment.FileUploadDenied", "파일 업로드 권한이 없습니다."));
  73. }
  74. }
  75. sbyte depth = 0;
  76. var isReply = false;
  77. // 부모 댓글이 있을 경우 답글로 처리
  78. if (request.ParentID.HasValue)
  79. {
  80. var parent = await db.Comment.AsNoTracking().FirstOrDefaultAsync(x => x.ID == request.ParentID.Value, ct);
  81. if (parent is null)
  82. {
  83. return Result.Failure<int>(Error.NotFound("Comment.ParentNotFound", "상위 댓글을 찾을 수 없습니다."));
  84. }
  85. depth = (sbyte)(parent.Depth + 1);
  86. isReply = true;
  87. // 부모 댓글 답글 수 증가
  88. var parentEntity = await db.Comment.FirstOrDefaultAsync(x => x.ID == request.ParentID.Value, ct);
  89. if (parentEntity is not null)
  90. {
  91. parentEntity.Replies++;
  92. }
  93. }
  94. var comment = new Domain.Entities.Forum.Comments.Comment
  95. {
  96. BoardID = post.BoardID,
  97. PostID = request.PostID,
  98. MemberID = request.MemberID,
  99. ParentID = request.ParentID,
  100. Depth = depth,
  101. Content = request.Content.Trim(),
  102. IsReply = isReply,
  103. IsSecret = request.IsSecret,
  104. Name = member.Name,
  105. SID = member.SID,
  106. Email = member.Email
  107. };
  108. // Mention 처리
  109. if (!string.IsNullOrWhiteSpace(request.Mention))
  110. {
  111. var rawHandle = request.Mention.Trim();
  112. var handleValue = rawHandle.TrimStart('@').Trim();
  113. var mentionMember = await db.Member.AsNoTracking().FirstOrDefaultAsync(x => x.Name == handleValue || x.SID == handleValue, ct);
  114. if (mentionMember is not null)
  115. {
  116. comment.MentionMemberID = mentionMember.ID;
  117. comment.CommentMention = new CommentMention
  118. {
  119. BoardID = post.BoardID,
  120. PostID = request.PostID,
  121. MemberID = mentionMember.ID,
  122. RawHandle = rawHandle
  123. };
  124. }
  125. }
  126. await db.Comment.AddAsync(comment, ct);
  127. // Post 댓글 카운트 증가
  128. post.Comments++;
  129. post.LastCommentUpdatedAt = DateTime.UtcNow;
  130. // 1:1 문의 게시판: 댓글 작성 시 답변 완료 표시
  131. if (isQnABoard && !request.ParentID.HasValue)
  132. {
  133. post.IsReply = true;
  134. }
  135. // Board 댓글 카운트 증가
  136. var board = await db.Board.FirstOrDefaultAsync(x => x.ID == post.BoardID, ct);
  137. if (board is not null)
  138. {
  139. board.Comments++;
  140. board.UpdatedAt = DateTime.UtcNow;
  141. }
  142. // MemberStats 댓글 수 증가
  143. var memberStats = await db.MemberStats.FirstOrDefaultAsync(x => x.MemberID == request.MemberID, ct);
  144. if (memberStats is not null)
  145. {
  146. memberStats.CommentCount++;
  147. }
  148. await db.SaveChangesAsync(ct);
  149. // 파일 처리 (comment.ID 확보 후)
  150. var uploadPath = new FileStoragePath(UploadTarget.Upload, UploadFolder.Comment, comment.ID);
  151. // 이미지 처리
  152. if (request.Images is { Count: > 0 })
  153. {
  154. byte imageCount = 0;
  155. var savedImageUrls = new List<string>();
  156. foreach (var image in request.Images)
  157. {
  158. var result = await fileStorage.SaveFileAsync(image, uploadPath, AllowedImageExtensions, ct);
  159. if (result is not null)
  160. {
  161. var ext = Path.GetExtension(image.FileName).ToLowerInvariant();
  162. await db.CommentImage.AddAsync(new CommentImage
  163. {
  164. BoardID = post.BoardID,
  165. PostID = request.PostID,
  166. CommentID = comment.ID,
  167. FileName = image.FileName,
  168. HashedName = result.FileName,
  169. Path = uploadPath.ToRelativePath(),
  170. Url = result.Url,
  171. Extension = ext,
  172. ContentType = image.ContentType,
  173. Size = result.Size,
  174. Width = result.Width,
  175. Height = result.Height
  176. }, ct);
  177. savedImageUrls.Add(result.Url);
  178. imageCount++;
  179. }
  180. }
  181. // content의 data:image/ 플레이스홀더를 실제 이미지 경로로 순서대로 치환
  182. var content = comment.Content;
  183. foreach (var url in savedImageUrls)
  184. {
  185. var idx = content.IndexOf("data:image/", StringComparison.Ordinal);
  186. if (idx >= 0)
  187. {
  188. var endIdx = content.IndexOfAny(['"', '\''], idx);
  189. if (endIdx > idx)
  190. {
  191. content = string.Concat(content.AsSpan(0, idx), url, content.AsSpan(endIdx));
  192. }
  193. }
  194. }
  195. comment.Content = content;
  196. comment.Images = imageCount;
  197. }
  198. // 파일 처리
  199. if (request.Files is { Count: > 0 })
  200. {
  201. // content HTML에서 file-embed의 data-uuid와 data-name 매핑 추출
  202. var fileUuidMap = new List<(string Name, Guid Uuid)>();
  203. var uuidMatches = Regex.Matches(comment.Content, @"data-uuid=""([^""]+)""\s+data-name=""([^""]+)""");
  204. foreach (Match match in uuidMatches)
  205. {
  206. if (Guid.TryParse(match.Groups[1].Value, out var uuid))
  207. {
  208. fileUuidMap.Add((match.Groups[2].Value, uuid));
  209. }
  210. }
  211. byte fileCount = 0;
  212. foreach (var file in request.Files)
  213. {
  214. var result = await fileStorage.SaveFileAsync(file, uploadPath, AllowedFileExtensions, ct);
  215. if (result is not null)
  216. {
  217. var ext = Path.GetExtension(file.FileName).ToLowerInvariant();
  218. // content의 file-embed에서 매칭되는 UUID 사용, 없으면 새로 생성
  219. var mapIdx = fileUuidMap.FindIndex(x => x.Name == file.FileName);
  220. Guid fileUuid;
  221. if (mapIdx >= 0)
  222. {
  223. fileUuid = fileUuidMap[mapIdx].Uuid;
  224. fileUuidMap.RemoveAt(mapIdx);
  225. }
  226. else
  227. {
  228. fileUuid = Guid.NewGuid();
  229. }
  230. await db.CommentFile.AddAsync(new Domain.Entities.Forum.Comments.CommentFile
  231. {
  232. BoardID = post.BoardID,
  233. PostID = request.PostID,
  234. CommentID = comment.ID,
  235. UUID = fileUuid,
  236. FileName = file.FileName,
  237. HashedName = result.FileName,
  238. Path = uploadPath.ToRelativePath(),
  239. Url = result.Url,
  240. Extension = ext,
  241. ContentType = file.ContentType,
  242. Size = result.Size
  243. }, ct);
  244. fileCount++;
  245. }
  246. }
  247. comment.Files = fileCount;
  248. }
  249. // 미디어 처리
  250. if (request.Medias is { Count: > 0 })
  251. {
  252. byte mediaCount = 0;
  253. foreach (var mediaUrl in request.Medias)
  254. {
  255. if (!string.IsNullOrWhiteSpace(mediaUrl))
  256. {
  257. await db.CommentMedia.AddAsync(new CommentMedia
  258. {
  259. BoardID = post.BoardID,
  260. PostID = request.PostID,
  261. CommentID = comment.ID,
  262. Url = mediaUrl.Trim()
  263. }, ct);
  264. mediaCount++;
  265. }
  266. }
  267. comment.Medias = mediaCount;
  268. }
  269. await db.SaveChangesAsync(ct);
  270. // 이메일 알림 발송
  271. if (boardMeta?.NotifyTemplate is not null && boardMeta.Notify is not null)
  272. {
  273. try
  274. {
  275. var notify = boardMeta.Notify;
  276. var template = boardMeta.NotifyTemplate;
  277. var notifyFlags = request.ParentID.HasValue ? notify.ReplyWriteNotifyEnum : notify.CommentWriteNotifyEnum;
  278. var emailSubject = request.ParentID.HasValue ? template.ReplyWriteEmailNotifySubject : template.CommentWriteEmailNotifySubject;
  279. var emailContent = request.ParentID.HasValue ? template.ReplyWriteEmailNotifyContent : template.CommentWriteEmailNotifyContent;
  280. if (notifyFlags != 0 && !string.IsNullOrWhiteSpace(emailSubject) && !string.IsNullOrWhiteSpace(emailContent))
  281. {
  282. var recipients = new HashSet<string>(StringComparer.OrdinalIgnoreCase);
  283. // 게시글 작성자
  284. if (notifyFlags.HasFlag(BoardNotify.PostAuthor))
  285. {
  286. var postAuthor = await db.Member.AsNoTracking().FirstOrDefaultAsync(x => x.ID == post.MemberID, ct);
  287. if (postAuthor is not null && !string.IsNullOrWhiteSpace(postAuthor.Email))
  288. {
  289. recipients.Add(postAuthor.Email);
  290. }
  291. }
  292. // 부모 댓글 작성자 (대댓글일 때)
  293. if (notifyFlags.HasFlag(BoardNotify.CommentAuthor) && request.ParentID.HasValue)
  294. {
  295. var parentComment = await db.Comment.AsNoTracking().FirstOrDefaultAsync(x => x.ID == request.ParentID.Value, ct);
  296. if (parentComment is not null)
  297. {
  298. var commentAuthor = await db.Member.AsNoTracking().FirstOrDefaultAsync(x => x.ID == parentComment.MemberID, ct);
  299. if (commentAuthor is not null && !string.IsNullOrWhiteSpace(commentAuthor.Email))
  300. {
  301. recipients.Add(commentAuthor.Email);
  302. }
  303. }
  304. }
  305. // 최고관리자
  306. if (notifyFlags.HasFlag(BoardNotify.Admin))
  307. {
  308. var admins = await db.Member.AsNoTracking().Where(x => x.IsAdmin).ToListAsync(ct);
  309. foreach (var admin in admins)
  310. {
  311. if (!string.IsNullOrWhiteSpace(admin.Email))
  312. {
  313. recipients.Add(admin.Email);
  314. }
  315. }
  316. }
  317. // 게시판 매니저
  318. if (notifyFlags.HasFlag(BoardNotify.Manager))
  319. {
  320. var managers = await db.BoardManager.AsNoTracking().Where(x => x.BoardID == post.BoardID).ToListAsync(ct);
  321. foreach (var mgr in managers)
  322. {
  323. var mgrMember = await db.Member.AsNoTracking().FirstOrDefaultAsync(x => x.ID == mgr.MemberID, ct);
  324. if (mgrMember is not null && !string.IsNullOrWhiteSpace(mgrMember.Email))
  325. {
  326. recipients.Add(mgrMember.Email);
  327. }
  328. }
  329. }
  330. // 본인에게는 발송하지 않음
  331. recipients.Remove(member.Email);
  332. foreach (var email in recipients)
  333. {
  334. await mailService.SendAsync(new SendData(email, emailSubject, emailContent), ct);
  335. }
  336. }
  337. }
  338. catch
  339. {
  340. // 이메일 발송 실패 시 댓글 등록은 성공 처리
  341. }
  342. }
  343. return comment.ID;
  344. }
  345. }