Handler.cs 2.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960
  1. using Application.Abstractions.Data;
  2. using Application.Abstractions.Messaging;
  3. using Microsoft.EntityFrameworkCore;
  4. namespace Application.Features.Admin.Forum.Trash.Comment.PermanentDelete;
  5. public sealed class Handler(IAppDbContext db) : ICommandHandler<Command>
  6. {
  7. public async Task Handle(Command request, CancellationToken ct)
  8. {
  9. if (request.IDs is null || request.IDs.Length == 0)
  10. {
  11. return;
  12. }
  13. var comments = await db.Comment.Where(c => request.IDs.Contains(c.ID) && c.IsDeleted).ToListAsync(ct);
  14. if (comments.Count == 0)
  15. {
  16. return;
  17. }
  18. // 자식 댓글(대댓글) 재귀 조회 → 삭제 대상에 포함
  19. var allIDs = comments.Select(c => c.ID).ToHashSet();
  20. var parentIDs = allIDs.ToList();
  21. while (parentIDs.Count > 0)
  22. {
  23. var childComments = await db.Comment.Where(c => c.ParentID != null && parentIDs.Contains(c.ParentID.Value) && !allIDs.Contains(c.ID)).ToListAsync(ct);
  24. if (childComments.Count == 0)
  25. {
  26. break;
  27. }
  28. parentIDs = [.. childComments.Select(c => c.ID)];
  29. allIDs.UnionWith(parentIDs);
  30. comments.AddRange(childComments);
  31. }
  32. var allIDList = allIDs.ToList();
  33. // 로그 테이블 선삭제 (Restrict FK 충돌 방지)
  34. var fileDownLogs = await db.CommentFileDownLog.Where(c => allIDList.Contains(c.CommentID)).ToListAsync(ct);
  35. if (fileDownLogs.Count > 0)
  36. {
  37. db.CommentFileDownLog.RemoveRange(fileDownLogs);
  38. }
  39. var linkClickLogs = await db.CommentLinkClickLog.Where(c => allIDList.Contains(c.CommentID)).ToListAsync(ct);
  40. if (linkClickLogs.Count > 0)
  41. {
  42. db.CommentLinkClickLog.RemoveRange(linkClickLogs);
  43. }
  44. // 자식 먼저 삭제되도록 Depth 역순 정렬
  45. var orderedComments = comments.OrderByDescending(c => c.Depth).ToList();
  46. db.Comment.RemoveRange(orderedComments);
  47. await db.SaveChangesAsync(ct);
  48. }
  49. }