SysFileService.cs 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444
  1. // 麻省理工学院许可证
  2. //
  3. // 版权所有 (c) 2021-2023 zuohuaijun,大名科技(天津)有限公司 联系电话/微信:18020030720 QQ:515096995
  4. //
  5. // 特此免费授予获得本软件的任何人以处理本软件的权利,但须遵守以下条件:在所有副本或重要部分的软件中必须包括上述版权声明和本许可声明。
  6. //
  7. // 软件按“原样”提供,不提供任何形式的明示或暗示的保证,包括但不限于对适销性、适用性和非侵权的保证。
  8. // 在任何情况下,作者或版权持有人均不对任何索赔、损害或其他责任负责,无论是因合同、侵权或其他方式引起的,与软件或其使用或其他交易有关。
  9. using Aliyun.OSS.Util;
  10. using COSXML.Network;
  11. using Furion.RemoteRequest;
  12. using Furion.VirtualFileServer;
  13. using OnceMi.AspNetCore.OSS;
  14. namespace Admin.NET.Core.Service;
  15. /// <summary>
  16. /// 系统文件服务
  17. /// </summary>
  18. [ApiDescriptionSettings(Order = 410)]
  19. public class SysFileService : IDynamicApiController, ITransient
  20. {
  21. private readonly UserManager _userManager;
  22. private readonly SqlSugarRepository<SysFile> _sysFileRep;
  23. private readonly OSSProviderOptions _OSSProviderOptions;
  24. private readonly UploadOptions _uploadOptions;
  25. private readonly IOSSService _OSSService;
  26. public SysFileService(UserManager userManager,
  27. SqlSugarRepository<SysFile> sysFileRep,
  28. IOptions<OSSProviderOptions> oSSProviderOptions,
  29. IOptions<UploadOptions> uploadOptions,
  30. IOSSServiceFactory ossServiceFactory)
  31. {
  32. _userManager = userManager;
  33. _sysFileRep = sysFileRep;
  34. _OSSProviderOptions = oSSProviderOptions.Value;
  35. _uploadOptions = uploadOptions.Value;
  36. if (_OSSProviderOptions.IsEnable)
  37. _OSSService = ossServiceFactory.Create(Enum.GetName(_OSSProviderOptions.Provider));
  38. }
  39. /// <summary>
  40. /// 获取文件分页列表
  41. /// </summary>
  42. /// <param name="input"></param>
  43. /// <returns></returns>
  44. [DisplayName("获取文件分页列表")]
  45. public async Task<SqlSugarPagedList<SysFile>> Page(PageFileInput input)
  46. {
  47. return await _sysFileRep.AsQueryable()
  48. .WhereIF(!string.IsNullOrWhiteSpace(input.FileName), u => u.FileName.Contains(input.FileName.Trim()))
  49. .WhereIF(!string.IsNullOrWhiteSpace(input.StartTime.ToString()) && !string.IsNullOrWhiteSpace(input.EndTime.ToString()),
  50. u => u.CreateTime >= input.StartTime && u.CreateTime <= input.EndTime)
  51. .OrderBy(u => u.CreateTime, OrderByType.Desc)
  52. .ToPagedListAsync(input.Page, input.PageSize);
  53. }
  54. /// <summary>
  55. /// 上传文件
  56. /// </summary>
  57. /// <param name="file"></param>
  58. /// <param name="path"></param>
  59. /// <returns></returns>
  60. [DisplayName("上传文件")]
  61. public async Task<FileOutput> UploadFile([Required] IFormFile file, [FromQuery] string? path)
  62. {
  63. var sysFile = await HandleUploadFile(file, path);
  64. return new FileOutput
  65. {
  66. Id = sysFile.Id,
  67. Url = sysFile.Url, // string.IsNullOrWhiteSpace(sysFile.Url) ? _commonService.GetFileUrl(sysFile) : sysFile.Url,
  68. SizeKb = sysFile.SizeKb,
  69. Suffix = sysFile.Suffix,
  70. FilePath = sysFile.FilePath,
  71. FileName = sysFile.FileName
  72. };
  73. }
  74. /// <summary>
  75. /// 上传文件Base64
  76. /// </summary>
  77. /// <param name="strBase64"></param>
  78. /// <param name="fileName"></param>
  79. /// <param name="contentType"></param>
  80. /// <param name="path"></param>
  81. /// <returns></returns>
  82. private async Task<FileOutput> UploadFileFromBase64(string strBase64, string fileName, string contentType, string? path)
  83. {
  84. byte[] fileData = Convert.FromBase64String(strBase64);
  85. var ms = new MemoryStream();
  86. ms.Write(fileData);
  87. ms.Seek(0, SeekOrigin.Begin);
  88. if (string.IsNullOrEmpty(fileName))
  89. fileName = $"{YitIdHelper.NextId()}.jpg";
  90. if (string.IsNullOrEmpty(contentType))
  91. contentType = "image/jpg";
  92. IFormFile formFile = new FormFile(ms, 0, fileData.Length, "file", fileName)
  93. {
  94. Headers = new HeaderDictionary(),
  95. ContentType = contentType
  96. };
  97. return await UploadFile(formFile, path);
  98. }
  99. /// <summary>
  100. /// 上传文件Base64
  101. /// </summary>
  102. /// <param name="input"></param>
  103. /// <returns></returns>
  104. [DisplayName("上传文件Base64")]
  105. [HttpPost]
  106. public async Task<FileOutput> UploadFileFromBase64(UploadFileFromBase64Input input)
  107. {
  108. return await UploadFileFromBase64(input.FileDataBase64, input.FileName, input.ContentType, input.Path);
  109. }
  110. /// <summary>
  111. /// 上传多文件
  112. /// </summary>
  113. /// <param name="files"></param>
  114. /// <returns></returns>
  115. [DisplayName("上传多文件")]
  116. public async Task<List<FileOutput>> UploadFiles([Required] List<IFormFile> files)
  117. {
  118. var filelist = new List<FileOutput>();
  119. foreach (var file in files)
  120. {
  121. filelist.Add(await UploadFile(file, ""));
  122. }
  123. return filelist;
  124. }
  125. /// <summary>
  126. /// 根据文件Id或Url下载
  127. /// </summary>
  128. /// <param name="input"></param>
  129. /// <returns></returns>
  130. [DisplayName("根据文件Id或Url下载")]
  131. public async Task<IActionResult> DownloadFile(FileInput input)
  132. {
  133. var file = input.Id > 0 ? await GetFile(input) : await _sysFileRep.GetFirstAsync(u => u.Url == input.Url);
  134. var fileName = HttpUtility.UrlEncode(file.FileName, Encoding.GetEncoding("UTF-8"));
  135. if (_OSSProviderOptions.IsEnable)
  136. {
  137. var filePath = string.Concat(file.FilePath, "/", file.Id.ToString() + file.Suffix);
  138. var stream = await (await _OSSService.PresignedGetObjectAsync(file.BucketName.ToString(), filePath, 5)).GetAsStreamAsync();
  139. return new FileStreamResult(stream.Stream, "application/octet-stream") { FileDownloadName = fileName + file.Suffix };
  140. }
  141. else
  142. {
  143. var filePath = Path.Combine(file.FilePath, file.Id.ToString() + file.Suffix);
  144. var path = Path.Combine(App.WebHostEnvironment.WebRootPath, filePath);
  145. return new FileStreamResult(new FileStream(path, FileMode.Open), "application/octet-stream") { FileDownloadName = fileName + file.Suffix };
  146. }
  147. }
  148. /// <summary>
  149. /// 下载指定url的文件
  150. /// </summary>
  151. /// <param name="url"></param>
  152. /// <returns></returns>
  153. /// <exception cref="HttpRequestException"></exception>
  154. [Post]
  155. [AllowAnonymous]
  156. public async Task<string> DownloadSysFile([FromBody] string url)
  157. {
  158. if (_OSSProviderOptions.IsEnable)
  159. {
  160. using var httpClient = new System.Net.Http.HttpClient();
  161. // 发送 GET 请求以下载文件
  162. HttpResponseMessage response = await httpClient.GetAsync(url);
  163. if (response.IsSuccessStatusCode)
  164. {
  165. // 读取文件内容并将其转换为 Base64 字符串
  166. byte[] fileBytes = await response.Content.ReadAsByteArrayAsync();
  167. string base64File = Convert.ToBase64String(fileBytes);
  168. return base64File;
  169. }
  170. else
  171. {
  172. throw new HttpRequestException($"Request failed with status code: {response.StatusCode}");
  173. }
  174. }
  175. else
  176. {
  177. var fileRecord = _sysFileRep.AsQueryable().Where(u => u.Url == url).First();
  178. if (fileRecord == null)
  179. throw new HttpRequestException($"文件没有记录");
  180. var filePath = Path.Combine(App.WebHostEnvironment.WebRootPath, fileRecord.FilePath);
  181. if (!Directory.Exists(filePath))
  182. Directory.CreateDirectory(filePath);
  183. var realFile = Path.Combine(filePath, $"{fileRecord.Id}{fileRecord.Suffix}");
  184. if (!File.Exists(realFile))
  185. throw Oops.Oh($"文件[{realFile}]不在存");
  186. byte[] fileBytes = File.ReadAllBytes(realFile);
  187. string base64File = Convert.ToBase64String(fileBytes);
  188. return base64File;
  189. }
  190. }
  191. /// <summary>
  192. /// 删除文件
  193. /// </summary>
  194. /// <param name="input"></param>
  195. /// <returns></returns>
  196. [ApiDescriptionSettings(Name = "Delete"), HttpPost]
  197. [DisplayName("删除文件")]
  198. public async Task DeleteFile(DeleteFileInput input)
  199. {
  200. var file = await _sysFileRep.GetFirstAsync(u => u.Id == input.Id);
  201. if (file != null)
  202. {
  203. await _sysFileRep.DeleteAsync(file);
  204. if (_OSSProviderOptions.IsEnable)
  205. {
  206. await _OSSService.RemoveObjectAsync(file.BucketName.ToString(), string.Concat(file.FilePath, "/", $"{input.Id}{file.Suffix}"));
  207. }
  208. else
  209. {
  210. var filePath = Path.Combine(App.WebHostEnvironment.WebRootPath, file.FilePath, input.Id.ToString() + file.Suffix);
  211. if (File.Exists(filePath))
  212. File.Delete(filePath);
  213. }
  214. }
  215. }
  216. /// <summary>
  217. /// 更新文件
  218. /// </summary>
  219. /// <param name="input"></param>
  220. /// <returns></returns>
  221. [ApiDescriptionSettings(Name = "Update"), HttpPost]
  222. [DisplayName("更新文件")]
  223. public async Task UpdateFile(FileInput input)
  224. {
  225. var isExist = await _sysFileRep.IsAnyAsync(u => u.Id == input.Id);
  226. if (!isExist) throw Oops.Oh(ErrorCodeEnum.D8000);
  227. await _sysFileRep.UpdateAsync(u => new SysFile() { FileName = input.FileName }, u => u.Id == input.Id);
  228. }
  229. /// <summary>
  230. /// 获取文件
  231. /// </summary>
  232. /// <param name="input"></param>
  233. /// <returns></returns>
  234. private async Task<SysFile> GetFile([FromQuery] FileInput input)
  235. {
  236. var file = await _sysFileRep.GetFirstAsync(u => u.Id == input.Id);
  237. return file ?? throw Oops.Oh(ErrorCodeEnum.D8000);
  238. }
  239. /// <summary>
  240. /// 上传文件
  241. /// </summary>
  242. /// <param name="file">文件</param>
  243. /// <param name="savePath">路径</param>
  244. /// <returns></returns>
  245. private async Task<SysFile> HandleUploadFile(IFormFile file, string savePath)
  246. {
  247. if (file == null) throw Oops.Oh(ErrorCodeEnum.D8000);
  248. // 判断是否重复上传的文件
  249. var sizeKb = (long)(file.Length / 1024.0); // 大小KB
  250. var fileMd5 = string.Empty;
  251. if (_uploadOptions.EnableMd5)
  252. {
  253. using (var fileStream = file.OpenReadStream())
  254. {
  255. fileMd5 = OssUtils.ComputeContentMd5(fileStream, fileStream.Length);
  256. }
  257. var strSizeKb = sizeKb.ToString();
  258. var sysFile = await _sysFileRep.GetFirstAsync(u => u.FileMd5 == fileMd5 && (u.SizeKb == null || u.SizeKb == strSizeKb));
  259. /*
  260. * Mysql8中如果使用了 utf8mb4_general_ci 之外的编码会出错,尽量避免在条件里使用.ToString()
  261. * 因为Squsugar,并不是把变量转换为字符串来构造SQL语句,而是构造了CAST(123 AS CHAR)这样的语句,这样这个返回值是utf8mb4_general_ci,所以容易出错。
  262. * */
  263. //var sysFile = await _sysFileRep.GetFirstAsync(u => u.FileMd5 == fileMd5 && (u.SizeKb == null || u.SizeKb == sizeKb.ToString())); //在条件时使用ToString会导到ubf8mb4字符集的MySQL数据库出错的。
  264. if (sysFile != null) return sysFile;
  265. }
  266. var path = savePath;
  267. if (string.IsNullOrWhiteSpace(savePath))
  268. {
  269. path = _uploadOptions.Path;
  270. var reg = new Regex(@"(\{.+?})");
  271. var match = reg.Matches(path);
  272. match.ToList().ForEach(a =>
  273. {
  274. var str = DateTime.Now.ToString(a.ToString().Substring(1, a.Length - 2)); // 每天一个目录
  275. path = path.Replace(a.ToString(), str);
  276. });
  277. }
  278. if (!_uploadOptions.ContentType.Contains(file.ContentType))
  279. throw Oops.Oh(ErrorCodeEnum.D8001);
  280. if (sizeKb > _uploadOptions.MaxSize)
  281. throw Oops.Oh(ErrorCodeEnum.D8002);
  282. var suffix = Path.GetExtension(file.FileName).ToLower(); // 后缀
  283. if (string.IsNullOrWhiteSpace(suffix))
  284. {
  285. var contentTypeProvider = FS.GetFileExtensionContentTypeProvider();
  286. suffix = contentTypeProvider.Mappings.FirstOrDefault(u => u.Value == file.ContentType).Key;
  287. // 修改 image/jpeg 类型返回的 .jpe 后缀
  288. if (suffix == ".jpe")
  289. suffix = ".jpg";
  290. }
  291. if (string.IsNullOrWhiteSpace(suffix))
  292. throw Oops.Oh(ErrorCodeEnum.D8003);
  293. var newFile = new SysFile
  294. {
  295. Id = YitIdHelper.NextId(),
  296. // BucketName = _OSSProviderOptions.IsEnable ? _OSSProviderOptions.Provider.ToString() : "Local",
  297. // 阿里云对bucket名称有要求,1.只能包括小写字母,数字,短横线(-)2.必须以小写字母或者数字开头 3.长度必须在3-63字节之间
  298. // 无法使用Provider
  299. BucketName = _OSSProviderOptions.IsEnable ? _OSSProviderOptions.Bucket : "Local",
  300. FileName = Path.GetFileNameWithoutExtension(file.FileName),
  301. Suffix = suffix,
  302. SizeKb = sizeKb.ToString(),
  303. FilePath = path,
  304. FileMd5 = fileMd5,
  305. };
  306. var finalName = newFile.Id + suffix; // 文件最终名称
  307. if (_OSSProviderOptions.IsEnable)
  308. {
  309. newFile.Provider = Enum.GetName(_OSSProviderOptions.Provider);
  310. var filePath = string.Concat(path, "/", finalName);
  311. await _OSSService.PutObjectAsync(newFile.BucketName, filePath, file.OpenReadStream());
  312. // http://<你的bucket名字>.oss.aliyuncs.com/<你的object名字>
  313. // 生成外链地址 方便前端预览
  314. switch (_OSSProviderOptions.Provider)
  315. {
  316. case OSSProvider.Aliyun:
  317. newFile.Url = $"{(_OSSProviderOptions.IsEnableHttps ? "https" : "http")}://{newFile.BucketName}.{_OSSProviderOptions.Endpoint}/{filePath}";
  318. break;
  319. case OSSProvider.Minio:
  320. // 获取Minio文件的下载或者预览地址
  321. // newFile.Url = await GetMinioPreviewFileUrl(newFile.BucketName, filePath);// 这种方法生成的Url是有7天有效期的,不能这样使用
  322. // 需要在MinIO中的Buckets开通对 Anonymous 的readonly权限
  323. newFile.Url = $"{(_OSSProviderOptions.IsEnableHttps ? "https" : "http")}://{_OSSProviderOptions.Endpoint}/{newFile.BucketName}/{filePath}";
  324. break;
  325. }
  326. }
  327. else
  328. {
  329. newFile.Provider = ""; // 本地存储 Provider 显示为空
  330. var filePath = Path.Combine(App.WebHostEnvironment.WebRootPath, path);
  331. if (!Directory.Exists(filePath))
  332. Directory.CreateDirectory(filePath);
  333. var realFile = Path.Combine(filePath, finalName);
  334. //IDetector detector;
  335. using (var stream = File.Create(realFile))
  336. {
  337. await file.CopyToAsync(stream);
  338. //detector = stream.DetectFiletype();
  339. }
  340. //var realExt = detector.Extension; // 真实扩展名
  341. //// 二次校验扩展名
  342. //if (!string.Equals(realExt, suffix.Replace(".", ""), StringComparison.OrdinalIgnoreCase))
  343. //{
  344. // var delFilePath = Path.Combine(App.WebHostEnvironment.WebRootPath, realFile);
  345. // if (File.Exists(delFilePath))
  346. // File.Delete(delFilePath);
  347. // throw Oops.Oh(ErrorCodeEnum.D8001);
  348. //}
  349. // 生成外链
  350. string host = CommonUtil.GetLocalhost();
  351. if (!host.EndsWith("/"))
  352. host += "/";
  353. newFile.Url = $"{host}{newFile.FilePath}/{newFile.Id + newFile.Suffix}";
  354. }
  355. await _sysFileRep.AsInsertable(newFile).ExecuteCommandAsync();
  356. return newFile;
  357. }
  358. ///// <summary>
  359. ///// 获取Minio文件的下载或者预览地址
  360. ///// </summary>
  361. ///// <param name="bucketName">桶名</param>
  362. ///// <param name="fileName">文件名</param>
  363. ///// <returns></returns>
  364. //private async Task<string> GetMinioPreviewFileUrl(string bucketName, string fileName)
  365. //{
  366. // return await _OSSService.PresignedGetObjectAsync(bucketName, fileName, 7);
  367. //}
  368. /// <summary>
  369. /// 上传头像
  370. /// </summary>
  371. /// <param name="file"></param>
  372. /// <returns></returns>
  373. [DisplayName("上传头像")]
  374. public async Task<FileOutput> UploadAvatar([Required] IFormFile file)
  375. {
  376. var sysUserRep = _sysFileRep.ChangeRepository<SqlSugarRepository<SysUser>>();
  377. var user = sysUserRep.GetFirst(u => u.Id == _userManager.UserId);
  378. // 删除当前用户已有头像
  379. if (!string.IsNullOrWhiteSpace(user.Avatar))
  380. {
  381. var fileId = Path.GetFileNameWithoutExtension(user.Avatar);
  382. await DeleteFile(new DeleteFileInput { Id = long.Parse(fileId) });
  383. }
  384. var res = await UploadFile(file, "Upload/Avatar");
  385. var url = _OSSProviderOptions.IsEnable ? res.Url : $"{res.FilePath}/{res.Name}";
  386. await sysUserRep.UpdateAsync(u => new SysUser() { Avatar = url }, u => u.Id == user.Id);
  387. return res;
  388. }
  389. /// <summary>
  390. /// 上传电子签名
  391. /// </summary>
  392. /// <param name="file"></param>
  393. /// <returns></returns>
  394. [DisplayName("上传电子签名")]
  395. public async Task<FileOutput> UploadSignature([Required] IFormFile file)
  396. {
  397. var sysUserRep = _sysFileRep.ChangeRepository<SqlSugarRepository<SysUser>>();
  398. var user = sysUserRep.GetFirst(u => u.Id == _userManager.UserId);
  399. // 删除当前用户已有电子签名
  400. if (!string.IsNullOrWhiteSpace(user.Signature) && user.Signature.EndsWith(".png"))
  401. {
  402. var fileId = Path.GetFileNameWithoutExtension(user.Signature);
  403. await DeleteFile(new DeleteFileInput { Id = long.Parse(fileId) });
  404. }
  405. var res = await UploadFile(file, "Upload/Signature");
  406. var url = _OSSProviderOptions.IsEnable ? res.Url : $"{res.FilePath}/{res.Name}";
  407. await sysUserRep.UpdateAsync(u => new SysUser() { Signature = url }, u => u.Id == user.Id);
  408. return res;
  409. }
  410. }