SysFileServiceImpl.java 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423
  1. package com.backendsys.modules.upload.service.impl;
  2. import cn.hutool.core.codec.Base64;
  3. import cn.hutool.core.convert.Convert;
  4. import cn.hutool.core.date.DateUtil;
  5. import cn.hutool.core.util.StrUtil;
  6. import cn.hutool.crypto.digest.DigestUtil;
  7. import cn.hutool.json.JSONArray;
  8. import com.backendsys.exception.CustException;
  9. import com.backendsys.modules.common.config.security.utils.HttpRequestUtil;
  10. import com.backendsys.modules.common.config.security.utils.SecurityUtil;
  11. import com.backendsys.modules.sdk.douyincloud.tos.service.DouyinTosService;
  12. import com.backendsys.modules.sdk.tencentcloud.cos.service.TencentCosService;
  13. import com.backendsys.modules.system.entity.SysCommon;
  14. import com.backendsys.modules.system.service.SysCommonService;
  15. import com.backendsys.modules.upload.dao.SysFileCategoryDao;
  16. import com.backendsys.modules.upload.dao.SysFileDao;
  17. import com.backendsys.modules.upload.entity.SysFile;
  18. import com.backendsys.modules.upload.entity.SysFileCategory;
  19. import com.backendsys.modules.upload.entity.SysFileResult;
  20. import com.backendsys.modules.upload.enums.StyleEnums;
  21. import com.backendsys.modules.upload.service.SysFileService;
  22. import com.backendsys.modules.upload.utils.UploadUtil;
  23. import com.backendsys.utils.response.PageEntity;
  24. import com.backendsys.utils.response.PageInfoResult;
  25. import com.backendsys.utils.v2.PageUtils;
  26. import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
  27. import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
  28. import org.springframework.beans.factory.annotation.Autowired;
  29. import org.springframework.stereotype.Service;
  30. import org.springframework.util.StringUtils;
  31. import org.springframework.web.multipart.MultipartFile;
  32. import java.io.IOException;
  33. import java.util.*;
  34. import java.util.concurrent.CompletableFuture;
  35. import java.util.concurrent.atomic.AtomicReference;
  36. import java.util.stream.Collectors;
  37. @Service
  38. public class SysFileServiceImpl extends ServiceImpl<SysFileDao, SysFile> implements SysFileService {
  39. @Autowired
  40. private HttpRequestUtil httpRequestUtil;
  41. @Autowired
  42. private TencentCosService tencentCosService;
  43. @Autowired
  44. private DouyinTosService douyinTosService;
  45. @Autowired
  46. private UploadUtil uploadUtil;
  47. @Autowired
  48. private SysFileDao sysFileDao;
  49. @Autowired
  50. private SysFileCategoryDao sysFileCategoryDao;
  51. @Autowired
  52. private SysCommonService sysCommonService;
  53. // [方法] 设置缩略图 (参数)
  54. // 不同的云环境 (target),缩略图配置也不一样
  55. // -1:本地:
  56. // 1:腾讯云: https://cloud.tencent.com/document/product/436/113295
  57. // 2:阿里云:
  58. // 3:抖音云: https://www.volcengine.com/docs/6349/153626
  59. private SysFile setThumbUrl(SysFile sysFile, Integer width, Integer height, String backgroundColor) {
  60. if (sysFile.getContent_type() != null) {
  61. if (sysFile.getContent_type().toLowerCase().contains("image")) {
  62. if (StrUtil.isEmpty(sysFile.getUpload_id())) {
  63. // 本地上传
  64. if (sysFile.getTarget() == -1) {
  65. sysFile.setUrl_thumb(sysFile.getUrl() + "?w=" + width + "&h=" + height);
  66. }
  67. // 腾讯云 (color值通过base64加密, #f8f8f8)
  68. if (sysFile.getTarget() == 1) {
  69. backgroundColor = "#" + backgroundColor;
  70. System.out.println("base64 encode: " + Base64.encode(backgroundColor));
  71. sysFile.setUrl_thumb(sysFile.getUrl() + "?imageMogr2/thumbnail/" + width + "x" + height + "/pad/1/color/" + Base64.encode(backgroundColor));
  72. }
  73. // 抖音云
  74. if (sysFile.getTarget() == 3) {
  75. sysFile.setUrl_thumb(sysFile.getUrl() + "?x-tos-process=image/resize,w_" + width + ",h_" + height + ",m_pad,color_" + backgroundColor);
  76. }
  77. }
  78. }
  79. }
  80. return sysFile;
  81. }
  82. /**
  83. * 获取文件列表 (total 有问题)
  84. */
  85. @Override
  86. public PageEntity selectUploadFileList(SysFile sysFile) {
  87. PageUtils.startPage(); // 分页
  88. List<SysFile> sysFileList = sysFileDao.selectUploadFileList(sysFile);
  89. // 完成分页渲染
  90. PageEntity pageEntity = new PageInfoResult(sysFileList).toEntity();
  91. // -- 完成分页渲染之后,再做列表格式化 -----------------------------------
  92. // [Common] 根据 Tag 获得 Options
  93. JSONArray COMMON_OPTIONS = sysCommonService.getCommonOptionByTag("UPLOAD_TARGET");
  94. // 遍历列表,赋值公共值翻译
  95. sysFileList = sysFileList.stream().map(item -> {
  96. // 查询出 上传存储介质(target) 的翻译
  97. String target_label = Convert.toStr(sysCommonService.getLabelByValue(COMMON_OPTIONS, item.getTarget()));
  98. item.setTarget_label(target_label);
  99. return item;
  100. }).collect(Collectors.toList());
  101. List<Object> objectList = sysFileList.stream().map(file -> (Object) file).collect(Collectors.toList());
  102. pageEntity.setList(objectList);
  103. // -----------------------------------------------------------------
  104. return pageEntity;
  105. }
  106. @Override
  107. public Map<String, Object> selectUploadTarget() {
  108. Integer UPLOAD_TARGET = Convert.toInt(sysCommonService.getCommonByTag("UPLOAD_TARGET"));
  109. String target_label = StyleEnums.targetToLabel(UPLOAD_TARGET);
  110. Map<String, Object> resp = new LinkedHashMap<>();
  111. resp.put("target", UPLOAD_TARGET);
  112. resp.put("target_label", target_label);
  113. return resp;
  114. }
  115. // [方法] 上传事件
  116. private SysFile uploadEvent(MultipartFile multipartFile, Long category_id, Integer target) {
  117. try {
  118. String filename = multipartFile.getOriginalFilename();
  119. if (filename.length() > 50) throw new CustException("文件名长度不能超过 50 字符");
  120. SysFileResult uploadResult = new SysFileResult();
  121. // target: 上传目标 (-1:本地, 1:腾讯云, 2:阿里云, 3.抖音云)
  122. if (target == -1) {
  123. uploadResult = uploadUtil.put(multipartFile);
  124. }
  125. if (target == 1) {
  126. // [腾讯云-上传对象]
  127. uploadResult = tencentCosService.putObject(multipartFile, null);
  128. }
  129. if (target == 3) {
  130. // [抖音云-上传对象]
  131. uploadResult = douyinTosService.putObject(multipartFile, null);
  132. }
  133. // [新增] 上传文件记录
  134. SysFile sysFileEntity = new SysFile();
  135. sysFileEntity.setCategory_id(category_id);
  136. sysFileEntity.setRequest_id(uploadResult.getRequest_id());
  137. sysFileEntity.setUser_id(httpRequestUtil.getUserId());
  138. sysFileEntity.setObject_key(uploadResult.getKey());
  139. // 文件名
  140. // sysFileEntity.setName(FileNameUtil.getName(uploadResult.getKey()));
  141. String filename_prefix = StrUtil.subBefore(filename, '.', true);
  142. String filename_suffix = StringUtils.getFilenameExtension(filename);
  143. /*
  144. filename = 200x134.png
  145. filename_prefix = 200x134
  146. filename_suffix = png
  147. */
  148. // 查询如果有同名文件,则在文件中加入数量命名
  149. LambdaQueryWrapper<SysFile> wrapper = new LambdaQueryWrapper<>();
  150. // 构造正则表达式 (匹配同名、同名(数量))
  151. String regex = "^" + filename_prefix + "(\\(\\d+\\))?\\." + filename_suffix + "$";
  152. wrapper.apply("name REGEXP {0}", regex);
  153. Long count = sysFileDao.selectCount(wrapper);
  154. if (count > 0) {
  155. sysFileEntity.setName(filename_prefix + "(" + count + ")." + filename_suffix);
  156. } else {
  157. sysFileEntity.setName(filename_prefix + "." + filename_suffix);
  158. }
  159. sysFileEntity.setContent_type(multipartFile.getContentType());
  160. if (target != -1) {
  161. // 非本地的 (Cos, Tos, ..)
  162. sysFileEntity.setUrl(uploadResult.getDomain() + "/" + uploadResult.getKey());
  163. } else {
  164. // 本地的
  165. sysFileEntity.setUrl(uploadResult.getDomain());
  166. }
  167. sysFileEntity.setSize(multipartFile.getSize());
  168. if (StrUtil.isNotEmpty(uploadResult.getE_tag())) {
  169. sysFileEntity.setMd5(uploadResult.getE_tag().replace("\"", ""));
  170. }
  171. sysFileEntity.setTarget(target);
  172. // 获得公共配置
  173. List<SysCommon> sysCommonList = sysCommonService.getCommonByCategory("UPLOAD");
  174. AtomicReference<Integer> UPLOAD_TARGET = new AtomicReference<>();
  175. AtomicReference<Integer> UPLOAD_THUMB_SIZE = new AtomicReference<>();
  176. sysCommonList.stream().forEach(sysCommon -> {
  177. if (sysCommon.getTag().equals("UPLOAD_TARGET")) UPLOAD_TARGET.set(Convert.toInt(sysCommon.getValue()));
  178. if (sysCommon.getTag().equals("UPLOAD_THUMB_SIZE")) UPLOAD_THUMB_SIZE.set(Convert.toInt(sysCommon.getValue()));
  179. });
  180. System.out.println("[系统配置] 上传目标(-1:本地, 1:腾讯云, 2:阿里云, 3.抖音云): " + UPLOAD_TARGET);
  181. System.out.println("[系统配置] 图片的缩略图宽高尺寸 (px): " + UPLOAD_THUMB_SIZE);
  182. // 设置缩略图
  183. sysFileEntity = setThumbUrl(sysFileEntity, UPLOAD_THUMB_SIZE.get(), UPLOAD_THUMB_SIZE.get(), StyleEnums.THUMB_BACKGROUND.getValue());
  184. sysFileEntity.setCreate_time(DateUtil.now());
  185. sysFileEntity.setUpdate_time(DateUtil.now());
  186. sysFileDao.insert(sysFileEntity);
  187. return sysFileEntity;
  188. } catch (IOException e) {
  189. throw new RuntimeException(e);
  190. }
  191. }
  192. /**
  193. * 上传文件 (单文件大小不超过 n)
  194. * - target: 上传目标 (-1:本地, 1:腾讯云, 2:阿里云, 3.抖音云)
  195. */
  196. @Override
  197. public SysFile uploadSmall(MultipartFile multipartFile, Long category_id) {
  198. List<SysCommon> sysCommonList = sysCommonService.getCommonByCategory("UPLOAD");
  199. AtomicReference<Integer> UPLOAD_TARGET = new AtomicReference<>();
  200. AtomicReference<Long> UPLOAD_MAX_SIZE_MB = new AtomicReference<>();
  201. AtomicReference<Integer> UPLOAD_THUMB_SIZE = new AtomicReference<>();
  202. AtomicReference<Boolean> UPLOAD_MD5_DUPLICATE = new AtomicReference<>();
  203. sysCommonList.stream().forEach(sysCommon -> {
  204. if (sysCommon.getTag().equals("UPLOAD_TARGET")) UPLOAD_TARGET.set(Convert.toInt(sysCommon.getValue()));
  205. if (sysCommon.getTag().equals("UPLOAD_MAX_SIZE_MB")) UPLOAD_MAX_SIZE_MB.set(Convert.toLong(sysCommon.getValue()));
  206. if (sysCommon.getTag().equals("UPLOAD_THUMB_SIZE")) UPLOAD_THUMB_SIZE.set(Convert.toInt(sysCommon.getValue()));
  207. if (sysCommon.getTag().equals("UPLOAD_MD5_DUPLICATE")) UPLOAD_MD5_DUPLICATE.set(Convert.toBool(sysCommon.getValue()));
  208. });
  209. System.out.println("[系统配置] 上传目标(-1:本地, 1:腾讯云, 2:阿里云, 3.抖音云): " + UPLOAD_TARGET);
  210. System.out.println("[系统配置] 单文件上传大小限制(MB): " + UPLOAD_MAX_SIZE_MB);
  211. System.out.println("[系统配置] 图片的缩略图宽高尺寸 (px): " + UPLOAD_THUMB_SIZE);
  212. System.out.println("[系统配置] 是否启用文件MD5查重: " + UPLOAD_MD5_DUPLICATE);
  213. // 判断文件是否超过大小
  214. Long MAX_SIZE = UPLOAD_MAX_SIZE_MB.get() * 1024 * 1024;
  215. if (multipartFile.getSize() > MAX_SIZE) {
  216. throw new CustException("上传文件不能大于 " + MAX_SIZE / 1024 / 1024 + " MB,请使用大文件上传功能");
  217. }
  218. try {
  219. // [系统配置] 是否启用文件MD5查重
  220. if (UPLOAD_MD5_DUPLICATE.get()) {
  221. // - 是
  222. // 根据文件MD5 判断是否上传过文件
  223. // - 如果上传过,则仅更新记录,不再上传文件,仅更新文件分类
  224. // - 如果存在
  225. String md5 = DigestUtil.md5Hex(multipartFile.getInputStream());
  226. // 排除文件异常的情况(出现两个以上相同MD5的文件)
  227. List<SysFile> sysFileEntityList = sysFileDao.selectList(new LambdaQueryWrapper<SysFile>().eq(SysFile::getMd5, md5));
  228. if (sysFileEntityList != null && sysFileEntityList.size() > 1) {
  229. throw new CustException("存在 " + sysFileEntityList.size() + " 个相同Md5 (" + md5 + ") 的文件,请先排重后再上传");
  230. }
  231. SysFile sysFileEntity = (sysFileEntityList != null && sysFileEntityList.size() > 0) ? sysFileEntityList.get(0) : null;
  232. if (sysFileEntity == null) {
  233. // [方法] 上传事件
  234. sysFileEntity = uploadEvent(multipartFile, category_id, UPLOAD_TARGET.get());
  235. } else {
  236. // [更新] 上传文件记录 (更换文件分类)
  237. sysFileEntity.setCategory_id(category_id);
  238. sysFileEntity.setUpdate_time(DateUtil.now());
  239. sysFileDao.updateById(sysFileEntity);
  240. }
  241. sysFileEntity = setThumbUrl(sysFileEntity, UPLOAD_THUMB_SIZE.get(), UPLOAD_THUMB_SIZE.get(), StyleEnums.THUMB_BACKGROUND.getValue());
  242. return sysFileEntity;
  243. } else {
  244. // - 否
  245. // [方法] 上传事件
  246. SysFile sysFileEntity = uploadEvent(multipartFile, category_id, UPLOAD_TARGET.get());
  247. sysFileEntity = setThumbUrl(sysFileEntity, UPLOAD_THUMB_SIZE.get(), UPLOAD_THUMB_SIZE.get(), StyleEnums.THUMB_BACKGROUND.getValue());
  248. return sysFileEntity;
  249. }
  250. } catch (IOException e) {
  251. throw new RuntimeException(e);
  252. }
  253. }
  254. // target: 上传目标 (-1:本地, 1:腾讯云, 2:阿里云, 3.抖音云)
  255. private void deleteObject(String object_key, Integer target) {
  256. if (target == -1) {
  257. // [本地] 删除文件
  258. uploadUtil.delete(object_key);
  259. }
  260. if (target == 1) {
  261. // [腾讯云] 删除对象
  262. tencentCosService.deleteObject(object_key);
  263. System.out.println("Delete tencent cos object: " + object_key);
  264. }
  265. if (target == 3) {
  266. // [抖音云] 删除对象
  267. try {
  268. douyinTosService.deleteObject(object_key);
  269. } catch (IOException e) {
  270. throw new RuntimeException(e);
  271. }
  272. System.out.println("Delete douyin tos object: " + object_key);
  273. }
  274. }
  275. /**
  276. * 删除文件 (包括缩略图,如果有的话)
  277. */
  278. @Override
  279. public Map<String, Object> removeUploadFile(SysFile sysFile, SysFile querySysFile) {
  280. String object_key = querySysFile.getObject_key();
  281. // [Delete] 删除文件记录
  282. sysFileDao.delete(new LambdaQueryWrapper<SysFile>().eq(SysFile::getObject_key, object_key));
  283. // [异步任务] 创建一个 CompletableFuture 来执行异步任务
  284. CompletableFuture.runAsync(() -> {
  285. deleteObject(querySysFile.getObject_key(), querySysFile.getTarget());
  286. });
  287. return Map.of("object_key", object_key);
  288. }
  289. /**
  290. * 删除文件 (批量)
  291. */
  292. @Override
  293. public Map<String, Object> removeUploadFileBatch(SysFile sysFile, List<SysFile> querySysFileList) {
  294. // 判断是否存在
  295. List<String> object_keys = sysFile.getObject_keys();
  296. // [Delete] 批量删除
  297. sysFileDao.delete(new LambdaQueryWrapper<SysFile>().in(SysFile::getObject_key, object_keys));
  298. // [异步任务] 创建一个 CompletableFuture 来执行异步任务
  299. CompletableFuture.runAsync(() -> {
  300. querySysFileList.stream().forEach(entity -> {
  301. deleteObject(entity.getObject_key(), entity.getTarget());
  302. });
  303. });
  304. return Map.of("object_keys", object_keys);
  305. }
  306. /**
  307. * 编辑文件 (名称、文件分类)
  308. * - 判断文件、文件分类是否存在
  309. * - 判断文件、文件分类是否当前用户所有
  310. */
  311. @Override
  312. public Map<String, Object> updateUploadFile(SysFile sysFile) {
  313. // 查询文件是否存在
  314. LambdaQueryWrapper<SysFile> wrapper = new LambdaQueryWrapper<>();
  315. wrapper.eq(SysFile::getUser_id, sysFile.getUser_id());
  316. wrapper.eq(SysFile::getId, sysFile.getId());
  317. Boolean isFileExist = sysFileDao.exists(wrapper);
  318. if (!isFileExist) throw new CustException("文件不存在");
  319. // 查询文件分类是否存在
  320. if (sysFile.getCategory_id() != null) {
  321. LambdaQueryWrapper<SysFileCategory> wrapperCategory = new LambdaQueryWrapper<>();
  322. wrapperCategory.eq(SysFileCategory::getUser_id, sysFile.getUser_id());
  323. wrapperCategory.eq(SysFileCategory::getId, sysFile.getCategory_id());
  324. Boolean isFileCategoryExist = sysFileCategoryDao.exists(wrapperCategory);
  325. if (!isFileCategoryExist) throw new CustException("文件分类不存在");
  326. }
  327. // [DB] 更新文件
  328. sysFileDao.updateFile(sysFile);
  329. return Map.of("id", sysFile.getId());
  330. }
  331. /**
  332. * 编辑文件 (批量)
  333. * - 文件、文件分类必须存在,且是该用户所有
  334. */
  335. @Override
  336. public Map<String, Object> updateUploadFileBatch(SysFile sysFile) {
  337. List<Long> ids = sysFile.getIds();
  338. Long category_id = sysFile.getCategory_id();
  339. // 判断 文件分类ID 是否存在
  340. Boolean isExist = sysFileCategoryDao.exists(new LambdaQueryWrapper<SysFileCategory>().eq(SysFileCategory::getId, category_id));
  341. if (!isExist) throw new CustException("文件分类不存在");
  342. // [DB] 批量更新 分类ID
  343. SysFile entity = new SysFile();
  344. entity.setCategory_id(category_id);
  345. // 查询条件 (当前用户, ids)
  346. LambdaQueryWrapper<SysFile> wrapperFile = new LambdaQueryWrapper<>();
  347. wrapperFile.eq(SysFile::getUser_id, sysFile.getUser_id());
  348. wrapperFile.in(SysFile::getId, ids);
  349. sysFileDao.update(entity, wrapperFile);
  350. return Map.of("ids", ids);
  351. }
  352. }