MultiOSSFileProvider.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257
  1. // Admin.NET 项目的版权、商标、专利和其他相关权利均受相应法律法规的保护。使用本项目应遵守相关法律法规和许可证的要求。
  2. //
  3. // 本项目主要遵循 MIT 许可证和 Apache 许可证(版本 2.0)进行分发和使用。许可证位于源代码树根目录中的 LICENSE-MIT 和 LICENSE-APACHE 文件。
  4. //
  5. // 不得利用本项目从事危害国家安全、扰乱社会秩序、侵犯他人合法权益等法律法规禁止的活动!任何基于本项目二次开发而产生的一切法律纠纷和责任,我们不承担任何责任!
  6. using OnceMi.AspNetCore.OSS;
  7. namespace Admin.NET.Core.Service;
  8. /// <summary>
  9. /// 多OSS文件提供者
  10. /// </summary>
  11. public class MultiOSSFileProvider : ICustomFileProvider, ITransient
  12. {
  13. private readonly SysFileProviderService _fileProviderService;
  14. private readonly IOSSServiceManager _ossServiceManager;
  15. private readonly OSSProviderOptions _ossProviderOptions;
  16. public MultiOSSFileProvider(SysFileProviderService fileProviderService,
  17. IOSSServiceManager ossServiceManager,
  18. IOptions<OSSProviderOptions> ossProviderOptions)
  19. {
  20. _fileProviderService = fileProviderService;
  21. _ossServiceManager = ossServiceManager;
  22. _ossProviderOptions = ossProviderOptions.Value;
  23. }
  24. /// <summary>
  25. /// 上传文件
  26. /// </summary>
  27. /// <param name="file">文件</param>
  28. /// <param name="sysFile">系统文件信息</param>
  29. /// <param name="path">文件存储位置</param>
  30. /// <param name="finalName">文件最终名称</param>
  31. /// <returns></returns>
  32. public async Task<SysFile> UploadFileAsync(IFormFile file, SysFile sysFile, string path, string finalName)
  33. {
  34. // 获取OSS配置(传入文件信息用于策略选择)
  35. var provider = await GetFileProvider(sysFile, file) ?? throw Oops.Oh("未找到可用的存储提供者配置");
  36. // 获取OSS服务
  37. var ossService = await _ossServiceManager.GetOSSServiceAsync(provider);
  38. // 设置文件信息
  39. sysFile.Provider = provider.Provider;
  40. sysFile.BucketName = provider.BucketName; // 保存原始存储桶名称
  41. var filePath = string.Concat(path, "/", finalName);
  42. // 上传文件
  43. await ossService.PutObjectAsync(provider.BucketName, filePath, file.OpenReadStream());
  44. // 生成外链地址
  45. sysFile.Url = GenerateFileUrl(provider, provider.BucketName, filePath);
  46. return sysFile;
  47. }
  48. /// <summary>
  49. /// 删除文件
  50. /// </summary>
  51. /// <param name="sysFile">系统文件信息</param>
  52. /// <returns></returns>
  53. public async Task DeleteFileAsync(SysFile sysFile)
  54. {
  55. // 获取OSS配置(统一方法)
  56. var provider = await GetFileProvider(sysFile) ?? throw Oops.Oh($"未找到存储提供者配置: {sysFile.Provider}-{sysFile.BucketName}");
  57. var ossService = await _ossServiceManager.GetOSSServiceAsync(provider);
  58. var filePath = string.Concat(sysFile.FilePath, "/", $"{sysFile.Id}{sysFile.Suffix}");
  59. await ossService.RemoveObjectAsync(provider.BucketName, filePath);
  60. }
  61. /// <summary>
  62. /// 获取文件流
  63. /// </summary>
  64. /// <param name="sysFile">系统文件信息</param>
  65. /// <param name="fileName">文件名</param>
  66. /// <returns></returns>
  67. public async Task<FileStreamResult> GetFileStreamResultAsync(SysFile sysFile, string fileName)
  68. {
  69. // 获取OSS配置(统一方法)
  70. var provider = await GetFileProvider(sysFile) ?? throw Oops.Oh($"未找到存储提供者配置: {sysFile.Provider}-{sysFile.BucketName}");
  71. var ossService = await _ossServiceManager.GetOSSServiceAsync(provider);
  72. var filePath = Path.Combine(sysFile.FilePath ?? "", sysFile.Id + sysFile.Suffix);
  73. var httpRemoteService = App.GetRequiredService<IHttpRemoteService>();
  74. var stream = await httpRemoteService.GetAsStreamAsync(await ossService.PresignedGetObjectAsync(provider.BucketName, filePath, 5));
  75. return new FileStreamResult(stream, "application/octet-stream") { FileDownloadName = fileName + sysFile.Suffix };
  76. }
  77. /// <summary>
  78. /// 下载文件Base64格式
  79. /// </summary>
  80. /// <param name="sysFile">系统文件信息</param>
  81. /// <returns></returns>
  82. public async Task<string> DownloadFileBase64Async(SysFile sysFile)
  83. {
  84. using var httpClient = new HttpClient();
  85. HttpResponseMessage response = await httpClient.GetAsync(sysFile.Url);
  86. if (response.IsSuccessStatusCode)
  87. {
  88. byte[] fileBytes = await response.Content.ReadAsByteArrayAsync();
  89. return Convert.ToBase64String(fileBytes);
  90. }
  91. throw Oops.Oh($"下载文件失败,状态码: {response.StatusCode}");
  92. }
  93. /// <summary>
  94. /// 获取文件提供者配置(统一方法,支持上传、删除、下载场景)
  95. /// </summary>
  96. /// <param name="sysFile">系统文件信息</param>
  97. /// <param name="file">上传的文件(可选,仅上传时传入)</param>
  98. /// <returns></returns>
  99. private async Task<SysFileProvider?> GetFileProvider(SysFile sysFile, IFormFile? file = null)
  100. {
  101. // 1. 如果已指定存储桶,直接使用
  102. if (!string.IsNullOrEmpty(sysFile.BucketName))
  103. {
  104. var provider = await _fileProviderService.GetFileProviderByBucket(sysFile.Provider, sysFile.BucketName);
  105. if (provider != null) return provider;
  106. // 如果数据库中找不到,尝试配置文件兜底
  107. if (_ossProviderOptions.Enabled && _ossProviderOptions.Bucket == sysFile.BucketName)
  108. {
  109. return await CreateProviderFromConfiguration();
  110. }
  111. }
  112. // 2. 如果有上传文件信息,使用策略选择(仅上传场景)
  113. if (file != null)
  114. {
  115. var uploadInput = new UploadFileInput
  116. {
  117. File = file,
  118. BucketName = sysFile.BucketName,
  119. FileType = sysFile.FileType
  120. };
  121. return await SelectProviderAsync(file, uploadInput);
  122. }
  123. // 3. 最后兜底:使用默认存储提供者
  124. return await _fileProviderService.GetDefaultProvider();
  125. }
  126. /// <summary>
  127. /// 选择合适的OSS存储提供者(内联版本)
  128. /// </summary>
  129. /// <param name="file">上传的文件</param>
  130. /// <param name="input">上传输入参数</param>
  131. /// <returns></returns>
  132. private async Task<SysFileProvider?> SelectProviderAsync(IFormFile file, UploadFileInput input)
  133. {
  134. // 1. 优先使用指定的提供者ID
  135. if (input.ProviderId.HasValue)
  136. {
  137. var provider = await _fileProviderService.GetFileProviderById(input.ProviderId.Value);
  138. if (provider != null) return provider;
  139. }
  140. // 2. 其次使用指定的存储桶名称
  141. if (!string.IsNullOrEmpty(input.BucketName))
  142. {
  143. var providers = await _fileProviderService.GetCachedFileProviders();
  144. var provider = providers.FirstOrDefault(p => p.BucketName == input.BucketName);
  145. if (provider != null) return provider;
  146. }
  147. // 3. 使用默认提供者
  148. var defaultProvider = await _fileProviderService.GetDefaultProvider();
  149. if (defaultProvider != null) return defaultProvider;
  150. // 4. 兜底:如果数据库中没有配置,尝试从配置文件创建默认提供者
  151. return await CreateProviderFromConfiguration();
  152. }
  153. /// <summary>
  154. /// 生成文件URL(内联版本)
  155. /// </summary>
  156. /// <param name="provider">存储提供者配置</param>
  157. /// <param name="bucketName">存储桶名称</param>
  158. /// <param name="filePath">文件路径</param>
  159. /// <returns></returns>
  160. private static string GenerateFileUrl(SysFileProvider provider, string bucketName, string filePath)
  161. {
  162. ArgumentNullException.ThrowIfNull(provider);
  163. ArgumentException.ThrowIfNullOrWhiteSpace(bucketName);
  164. ArgumentException.ThrowIfNullOrWhiteSpace(filePath);
  165. var protocol = provider.IsEnableHttps == true ? "https" : "http";
  166. // 如果有自定义域名,直接使用
  167. if (!string.IsNullOrWhiteSpace(provider.SinceDomain))
  168. {
  169. return $"{provider.SinceDomain.TrimEnd('/')}/{filePath.TrimStart('/')}";
  170. }
  171. // 根据不同提供者生成URL
  172. return provider.Provider.ToUpper() switch
  173. {
  174. "ALIYUN" => $"{protocol}://{bucketName}.{provider.Endpoint}/{filePath.TrimStart('/')}",
  175. "QCLOUD" => $"{protocol}://{bucketName}-{provider.Endpoint}.cos.{provider.Region}.myqcloud.com/{filePath.TrimStart('/')}",
  176. "MINIO" => $"{protocol}://{provider.Endpoint}/{bucketName}/{filePath.TrimStart('/')}",
  177. _ => throw Oops.Oh($"不支持的OSS提供者: {provider.Provider}")
  178. };
  179. }
  180. /// <summary>
  181. /// 从配置文件创建默认提供者(兜底机制)
  182. /// </summary>
  183. /// <returns></returns>
  184. private Task<SysFileProvider?> CreateProviderFromConfiguration()
  185. {
  186. try
  187. {
  188. // 检查是否启用了OSS配置
  189. if (!_ossProviderOptions.Enabled && !App.Configuration["MultiOSS:Enabled"].ToBoolean())
  190. return Task.FromResult<SysFileProvider?>(null);
  191. // 验证必要配置
  192. if (string.IsNullOrWhiteSpace(_ossProviderOptions.AccessKey) ||
  193. string.IsNullOrWhiteSpace(_ossProviderOptions.SecretKey) ||
  194. string.IsNullOrWhiteSpace(_ossProviderOptions.Bucket))
  195. {
  196. return Task.FromResult<SysFileProvider?>(null);
  197. }
  198. // 使用现有的OSSProviderOptions创建临时提供者配置(不保存到数据库)
  199. var provider = new SysFileProvider
  200. {
  201. Id = 0, // 临时ID
  202. Provider = Enum.GetName(_ossProviderOptions.Provider),
  203. BucketName = _ossProviderOptions.Bucket,
  204. AccessKey = _ossProviderOptions.AccessKey,
  205. SecretKey = _ossProviderOptions.SecretKey,
  206. Endpoint = _ossProviderOptions.Endpoint,
  207. Region = _ossProviderOptions.Region,
  208. IsEnableHttps = _ossProviderOptions.IsEnableHttps,
  209. IsEnableCache = _ossProviderOptions.IsEnableCache,
  210. IsEnable = true,
  211. IsDefault = true,
  212. SinceDomain = _ossProviderOptions.CustomHost,
  213. CreateTime = DateTime.Now
  214. };
  215. return Task.FromResult<SysFileProvider?>(provider);
  216. }
  217. catch (Exception)
  218. {
  219. // 配置读取失败,返回null
  220. return Task.FromResult<SysFileProvider?>(null);
  221. }
  222. }
  223. }