MultiOSSFileProvider.cs 11 KB

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