LocalFileStorage.cs 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120
  1. using Microsoft.AspNetCore.Hosting;
  2. using Microsoft.AspNetCore.Http;
  3. using Microsoft.Extensions.Logging;
  4. using SharedKernel.Storage;
  5. namespace Infrastructure.Storage
  6. {
  7. public sealed class LocalFileStorage : IFileStorage
  8. {
  9. private readonly IWebHostEnvironment _env;
  10. private readonly ILogger<LocalFileStorage> _log;
  11. public LocalFileStorage(IWebHostEnvironment env, ILogger<LocalFileStorage> log)
  12. {
  13. _env = env;
  14. _log = log;
  15. }
  16. public async Task<FileUploadResult?> SaveFileAsync(IFormFile? file, FileStoragePath path, string[] allowedExtensions, CancellationToken ct = default)
  17. {
  18. if (file is null || file.Length <= 0)
  19. {
  20. return null;
  21. }
  22. var ext = Path.GetExtension(file.FileName).ToLowerInvariant();
  23. if (!allowedExtensions.Contains(ext))
  24. {
  25. throw new ArgumentException("허용되지 않는 파일 확장자입니다.");
  26. }
  27. var rel = path.ToRelativePath();
  28. var dir = CombineUnderWebRoot(rel);
  29. var fileName = $"{Guid.NewGuid():N}{ext}";
  30. var fullPath = Path.Combine(dir, fileName);
  31. await using var fs = new FileStream(fullPath, FileMode.CreateNew, FileAccess.Write, FileShare.None);
  32. await file.CopyToAsync(fs, ct);
  33. var url = "/" + $"{rel}/{fileName}".Replace("\\", "/");
  34. return new FileUploadResult(url, fileName, file.Length, ext);
  35. }
  36. public async Task<FileUploadResult?> SaveBytesAsync(ReadOnlyMemory<byte> bytes, string extension, FileStoragePath path, CancellationToken ct = default)
  37. {
  38. if (bytes.Length <= 0)
  39. {
  40. return null;
  41. }
  42. var ext = FileUtils.NormalizeExtension(extension);
  43. if (ext is not (".jpg" or ".png" or ".gif" or ".bmp" or ".webp"))
  44. {
  45. throw new ArgumentException("허용되지 않는 파일 확장자입니다.");
  46. }
  47. var rel = path.ToRelativePath();
  48. var dir = CombineUnderWebRoot(rel);
  49. Directory.CreateDirectory(dir); // 경로 생성(없으면)
  50. var fileName = $"{Guid.NewGuid():N}{ext}";
  51. var fullPath = Path.Combine(dir, fileName);
  52. await File.WriteAllBytesAsync(fullPath, bytes.ToArray(), ct);
  53. var url = "/" + $"{rel}/{fileName}".Replace("\\", "/");
  54. return new FileUploadResult(url, fileName, bytes.Length, ext);
  55. }
  56. public void DeleteByUrl(string? url)
  57. {
  58. if (string.IsNullOrWhiteSpace(url))
  59. {
  60. return;
  61. }
  62. if (!FileUtils.IsLocalPublicUrl(url))
  63. {
  64. _log.LogWarning("삭제가 불가한 주소입니다. {0}", url);
  65. return;
  66. }
  67. var fullPath = CombineUnderWebRoot(url.TrimStart('/'));
  68. if (!File.Exists(fullPath))
  69. {
  70. _log.LogWarning("파일을 찾을 수 없습니다. {0}", fullPath);
  71. return;
  72. }
  73. try
  74. {
  75. File.Delete(fullPath);
  76. _log.LogInformation("파일이 삭제됨 : {0}", fullPath);
  77. }
  78. catch (Exception e)
  79. {
  80. _log.LogError(e, "파일 삭제 중 오류가 발생했습니다. {0}", fullPath);
  81. }
  82. }
  83. private string CombineUnderWebRoot(string relative)
  84. {
  85. var webRoot = Path.GetFullPath(_env.WebRootPath);
  86. var fullPath = Path.GetFullPath(Path.Combine(_env.WebRootPath, relative));
  87. if (!fullPath.StartsWith(webRoot, StringComparison.OrdinalIgnoreCase))
  88. {
  89. throw new UnauthorizedAccessException("경로를 찾을 수 없습니다.");
  90. }
  91. return fullPath;
  92. }
  93. }
  94. }