Handler.cs 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186
  1. using Application.Abstractions.Messaging;
  2. using Application.Abstractions.Data;
  3. using SharedKernel.Storage;
  4. using Microsoft.EntityFrameworkCore;
  5. using System.Text.RegularExpressions;
  6. namespace Application.Features.Admin.Forum.Post.Update;
  7. public sealed class Handler(IAppDbContext db, IFileStorage fileStorage) : ICommandHandler<Command>
  8. {
  9. private static readonly string[] AllowedImageExtensions = [".jpg", ".jpeg", ".png", ".gif", ".webp", ".bmp"];
  10. 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"];
  11. private static readonly Regex ImgBase64Regex = new(@"<img\b[^>]*?\bsrc\s*=\s*""(?<src>data:image/(?<ext>[a-zA-Z0-9.+-]+);base64,(?<data>[a-zA-Z0-9+/=\s]+))""", RegexOptions.Compiled | RegexOptions.IgnoreCase);
  12. public async Task Handle(Command request, CancellationToken ct)
  13. {
  14. var post = await db.Post.FirstOrDefaultAsync(x => x.ID == request.ID, ct);
  15. if (post is null)
  16. {
  17. throw new KeyNotFoundException("게시글을 찾을 수 없습니다.");
  18. }
  19. post.BoardPrefixID = request.BoardPrefixID;
  20. post.Subject = request.Subject;
  21. post.Content = request.Content ?? string.Empty;
  22. post.IsNotice = request.IsNotice;
  23. post.IsSecret = request.IsSecret;
  24. post.IsAnonymous = request.IsAnonymous;
  25. post.UpdatedAt = DateTime.UtcNow;
  26. var uploadPath = new FileStoragePath(UploadTarget.Upload, UploadFolder.Post, post.ID);
  27. // 에디터 이미지 처리 (base64 → 파일 + PostImage 레코드)
  28. var content = post.Content;
  29. var matches = ImgBase64Regex.Matches(content);
  30. if (matches.Count > 0)
  31. {
  32. byte newImageCount = 0;
  33. var firstImageUrl = (string?)null;
  34. foreach (Match match in matches)
  35. {
  36. var ext = FileUtils.NormalizeExtension(match.Groups["ext"].Value);
  37. var bytes = Convert.FromBase64String(match.Groups["data"].Value);
  38. var result = await fileStorage.SaveBytesAsync(bytes, ext, uploadPath, ct);
  39. if (result is not null)
  40. {
  41. content = content.Replace(match.Groups["src"].Value, result.Url);
  42. firstImageUrl ??= result.Url;
  43. await db.PostImage.AddAsync(new Domain.Entities.Forum.Posts.PostImage
  44. {
  45. BoardID = post.BoardID,
  46. PostID = post.ID,
  47. FileName = result.FileName,
  48. HashedName = result.FileName,
  49. Path = uploadPath.ToRelativePath(),
  50. Url = result.Url,
  51. Extension = ext,
  52. ContentType = $"image/{match.Groups["ext"].Value}",
  53. Size = result.Size,
  54. Width = result.Width,
  55. Height = result.Height
  56. }, ct);
  57. newImageCount++;
  58. }
  59. }
  60. post.Content = content;
  61. var existingImageCount = await db.PostImage.CountAsync(x => x.PostID == post.ID && !x.IsDisabled, ct);
  62. post.Images = (byte)Math.Min(existingImageCount + newImageCount, byte.MaxValue);
  63. if (string.IsNullOrEmpty(post.Thumbnail) && firstImageUrl is not null)
  64. {
  65. post.Thumbnail = firstImageUrl;
  66. }
  67. }
  68. // 썸네일 처리
  69. if (request.ThumbnailFile is not null)
  70. {
  71. if (!string.IsNullOrEmpty(post.Thumbnail))
  72. {
  73. fileStorage.DeleteByUrl(post.Thumbnail);
  74. }
  75. var result = await fileStorage.SaveFileAsync(request.ThumbnailFile, uploadPath, AllowedImageExtensions, ct);
  76. post.Thumbnail = result?.Url;
  77. }
  78. // 파일 처리 (새 파일 추가)
  79. if (request.Files is { Count: > 0 })
  80. {
  81. foreach (var file in request.Files)
  82. {
  83. var result = await fileStorage.SaveFileAsync(file, uploadPath, AllowedFileExtensions, ct);
  84. if (result is not null)
  85. {
  86. var ext = Path.GetExtension(file.FileName).ToLowerInvariant();
  87. await db.PostFile.AddAsync(new Domain.Entities.Forum.Posts.PostFile
  88. {
  89. BoardID = post.BoardID,
  90. PostID = post.ID,
  91. UUID = Guid.NewGuid(),
  92. FileName = file.FileName,
  93. HashedName = result.FileName,
  94. Path = uploadPath.ToRelativePath(),
  95. Url = result.Url,
  96. Extension = ext,
  97. ContentType = file.ContentType,
  98. Size = result.Size
  99. }, ct);
  100. }
  101. }
  102. var fileCount = await db.PostFile.CountAsync(x => x.PostID == post.ID && !x.IsDisabled, ct);
  103. post.Files = (byte)Math.Min(fileCount + request.Files.Count, byte.MaxValue);
  104. }
  105. // 태그 처리 (diff 기반)
  106. if (request.Tags is not null)
  107. {
  108. var existingPostTags = await db.PostTag.Include(x => x.Tag).Where(x => x.PostID == post.ID).ToListAsync(ct);
  109. var newTagSlugs = request.Tags.Where(t => !string.IsNullOrWhiteSpace(t)).Select(t => t.Trim().ToLowerInvariant().Replace(' ', '-')).ToHashSet();
  110. var existingSlugs = existingPostTags.Select(x => x.Tag.Slug).ToHashSet();
  111. // 삭제할 태그: 기존에 있지만 새 목록에 없는 것
  112. var toRemove = existingPostTags.Where(x => !newTagSlugs.Contains(x.Tag.Slug)).ToList();
  113. foreach (var postTag in toRemove)
  114. {
  115. db.PostTag.Remove(postTag);
  116. if (postTag.Tag.UsageCount > 0)
  117. {
  118. postTag.Tag.UsageCount--;
  119. postTag.Tag.UpdatedAt = DateTime.UtcNow;
  120. }
  121. }
  122. // 추가할 태그: 새 목록에 있지만 기존에 없는 것
  123. var toAdd = request.Tags
  124. .Where(t => !string.IsNullOrWhiteSpace(t))
  125. .Select(t => t.Trim())
  126. .Where(t => !existingSlugs.Contains(t.ToLowerInvariant().Replace(' ', '-')))
  127. .ToList();
  128. foreach (var tagName in toAdd)
  129. {
  130. var slug = tagName.ToLowerInvariant().Replace(' ', '-');
  131. var tag = await db.Tag.FirstOrDefaultAsync(x => x.Slug == slug, ct);
  132. if (tag is null)
  133. {
  134. tag = new Domain.Entities.Forum.Posts.Tag
  135. {
  136. Name = tagName,
  137. Slug = slug
  138. };
  139. await db.Tag.AddAsync(tag, ct);
  140. await db.SaveChangesAsync(ct);
  141. }
  142. tag.UsageCount++;
  143. tag.UpdatedAt = DateTime.UtcNow;
  144. await db.PostTag.AddAsync(new Domain.Entities.Forum.Posts.PostTag
  145. {
  146. BoardID = post.BoardID,
  147. PostID = post.ID,
  148. TagID = tag.ID
  149. }, ct);
  150. }
  151. post.Tags = (byte)Math.Min(newTagSlugs.Count, byte.MaxValue);
  152. }
  153. await db.SaveChangesAsync(ct);
  154. }
  155. }