SysFileServiceImpl.java 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388
  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. SysFileResult uploadResult = new SysFileResult();
  119. // target: 上传目标 (-1:本地, 1:腾讯云, 2:阿里云, 3.抖音云)
  120. if (target == -1) {
  121. uploadResult = uploadUtil.put(multipartFile);
  122. }
  123. if (target == 1) {
  124. // [腾讯云-上传对象]
  125. uploadResult = tencentCosService.putObject(multipartFile, null);
  126. }
  127. if (target == 3) {
  128. // [抖音云-上传对象]
  129. uploadResult = douyinTosService.putObject(multipartFile, null);
  130. }
  131. // [新增] 上传文件记录
  132. SysFile sysFileEntity = new SysFile();
  133. sysFileEntity.setCategory_id(category_id);
  134. sysFileEntity.setRequest_id(uploadResult.getRequest_id());
  135. sysFileEntity.setUser_id(httpRequestUtil.getUserId());
  136. sysFileEntity.setObject_key(uploadResult.getKey());
  137. // 文件名
  138. // sysFileEntity.setName(FileNameUtil.getName(uploadResult.getKey()));
  139. String filename = multipartFile.getOriginalFilename();
  140. String filename_prefix = StrUtil.subBefore(filename, '.', true);
  141. String filename_suffix = StringUtils.getFilenameExtension(filename);
  142. /*
  143. filename = 200x134.png
  144. filename_prefix = 200x134
  145. filename_suffix = png
  146. */
  147. // 查询如果有同名文件,则在文件中加入数量命名
  148. LambdaQueryWrapper<SysFile> wrapper = new LambdaQueryWrapper<>();
  149. // 构造正则表达式 (匹配同名、同名(数量))
  150. String regex = "^" + filename_prefix + "(\\(\\d+\\))?\\." + filename_suffix + "$";
  151. wrapper.apply("name REGEXP {0}", regex);
  152. Long count = sysFileDao.selectCount(wrapper);
  153. if (count > 0) {
  154. sysFileEntity.setName(filename_prefix + "(" + count + ")." + filename_suffix);
  155. } else {
  156. sysFileEntity.setName(filename_prefix + "." + filename_suffix);
  157. }
  158. sysFileEntity.setContent_type(multipartFile.getContentType());
  159. if (target != -1) {
  160. // 非本地的 (Cos, Tos, ..)
  161. sysFileEntity.setUrl(uploadResult.getDomain() + "/" + uploadResult.getKey());
  162. } else {
  163. // 本地的
  164. sysFileEntity.setUrl(uploadResult.getDomain());
  165. }
  166. sysFileEntity.setSize(multipartFile.getSize());
  167. if (StrUtil.isNotEmpty(uploadResult.getE_tag())) {
  168. sysFileEntity.setMd5(uploadResult.getE_tag().replace("\"", ""));
  169. }
  170. sysFileEntity.setTarget(target);
  171. // 获得公共配置
  172. List<SysCommon> sysCommonList = sysCommonService.getCommonByCategory("UPLOAD");
  173. AtomicReference<Integer> UPLOAD_TARGET = new AtomicReference<>();
  174. AtomicReference<Integer> UPLOAD_THUMB_SIZE = new AtomicReference<>();
  175. sysCommonList.stream().forEach(sysCommon -> {
  176. if (sysCommon.getTag().equals("UPLOAD_TARGET")) UPLOAD_TARGET.set(Convert.toInt(sysCommon.getValue()));
  177. if (sysCommon.getTag().equals("UPLOAD_THUMB_SIZE")) UPLOAD_THUMB_SIZE.set(Convert.toInt(sysCommon.getValue()));
  178. });
  179. System.out.println("[系统配置] 上传目标(-1:本地, 1:腾讯云, 2:阿里云, 3.抖音云): " + UPLOAD_TARGET);
  180. System.out.println("[系统配置] 图片的缩略图宽高尺寸 (px): " + UPLOAD_THUMB_SIZE);
  181. // 设置缩略图
  182. sysFileEntity = setThumbUrl(sysFileEntity, UPLOAD_THUMB_SIZE.get(), UPLOAD_THUMB_SIZE.get(), StyleEnums.THUMB_BACKGROUND.getValue());
  183. sysFileEntity.setCreate_time(DateUtil.now());
  184. sysFileEntity.setUpdate_time(DateUtil.now());
  185. sysFileDao.insert(sysFileEntity);
  186. return sysFileEntity;
  187. } catch (IOException e) {
  188. throw new RuntimeException(e);
  189. }
  190. }
  191. /**
  192. * 上传文件 (单文件大小不超过 n)
  193. * - target: 上传目标 (-1:本地, 1:腾讯云, 2:阿里云, 3.抖音云)
  194. */
  195. @Override
  196. public SysFile uploadSmall(MultipartFile multipartFile, Long category_id) {
  197. List<SysCommon> sysCommonList = sysCommonService.getCommonByCategory("UPLOAD");
  198. AtomicReference<Integer> UPLOAD_TARGET = new AtomicReference<>();
  199. AtomicReference<Long> UPLOAD_MAX_SIZE_MB = new AtomicReference<>();
  200. AtomicReference<Integer> UPLOAD_THUMB_SIZE = new AtomicReference<>();
  201. AtomicReference<Boolean> UPLOAD_MD5_DUPLICATE = new AtomicReference<>();
  202. sysCommonList.stream().forEach(sysCommon -> {
  203. if (sysCommon.getTag().equals("UPLOAD_TARGET")) UPLOAD_TARGET.set(Convert.toInt(sysCommon.getValue()));
  204. if (sysCommon.getTag().equals("UPLOAD_MAX_SIZE_MB")) UPLOAD_MAX_SIZE_MB.set(Convert.toLong(sysCommon.getValue()));
  205. if (sysCommon.getTag().equals("UPLOAD_THUMB_SIZE")) UPLOAD_THUMB_SIZE.set(Convert.toInt(sysCommon.getValue()));
  206. if (sysCommon.getTag().equals("UPLOAD_MD5_DUPLICATE")) UPLOAD_MD5_DUPLICATE.set(Convert.toBool(sysCommon.getValue()));
  207. });
  208. System.out.println("[系统配置] 上传目标(-1:本地, 1:腾讯云, 2:阿里云, 3.抖音云): " + UPLOAD_TARGET);
  209. System.out.println("[系统配置] 单文件上传大小限制(MB): " + UPLOAD_MAX_SIZE_MB);
  210. System.out.println("[系统配置] 图片的缩略图宽高尺寸 (px): " + UPLOAD_THUMB_SIZE);
  211. System.out.println("[系统配置] 是否启用文件MD5查重: " + UPLOAD_MD5_DUPLICATE);
  212. // 判断文件是否超过大小
  213. Long MAX_SIZE = UPLOAD_MAX_SIZE_MB.get() * 1024 * 1024;
  214. if (multipartFile.getSize() > MAX_SIZE) {
  215. throw new CustException("上传文件不能大于 " + MAX_SIZE / 1024 / 1024 + " MB,请使用大文件上传功能");
  216. }
  217. try {
  218. // [系统配置] 是否启用文件MD5查重
  219. if (UPLOAD_MD5_DUPLICATE.get()) {
  220. // - 是
  221. // 根据文件MD5 判断是否上传过文件
  222. // - 如果上传过,则仅更新记录,不再上传文件,仅更新文件分类
  223. // - 如果存在
  224. String md5 = DigestUtil.md5Hex(multipartFile.getInputStream());
  225. // 排除文件异常的情况(出现两个以上相同MD5的文件)
  226. List<SysFile> sysFileEntityList = sysFileDao.selectList(new LambdaQueryWrapper<SysFile>().eq(SysFile::getMd5, md5));
  227. if (sysFileEntityList != null && sysFileEntityList.size() > 1) {
  228. throw new CustException("存在 " + sysFileEntityList.size() + " 个相同Md5 (" + md5 + ") 的文件,请先排重后再上传");
  229. }
  230. SysFile sysFileEntity = (sysFileEntityList != null && sysFileEntityList.size() > 0) ? sysFileEntityList.get(0) : null;
  231. if (sysFileEntity == null) {
  232. // [方法] 上传事件
  233. sysFileEntity = uploadEvent(multipartFile, category_id, UPLOAD_TARGET.get());
  234. } else {
  235. // [更新] 上传文件记录 (更换文件分类)
  236. sysFileEntity.setCategory_id(category_id);
  237. sysFileEntity.setUpdate_time(DateUtil.now());
  238. sysFileDao.updateById(sysFileEntity);
  239. }
  240. sysFileEntity = setThumbUrl(sysFileEntity, UPLOAD_THUMB_SIZE.get(), UPLOAD_THUMB_SIZE.get(), StyleEnums.THUMB_BACKGROUND.getValue());
  241. return sysFileEntity;
  242. } else {
  243. // - 否
  244. // [方法] 上传事件
  245. SysFile sysFileEntity = uploadEvent(multipartFile, category_id, UPLOAD_TARGET.get());
  246. sysFileEntity = setThumbUrl(sysFileEntity, UPLOAD_THUMB_SIZE.get(), UPLOAD_THUMB_SIZE.get(), StyleEnums.THUMB_BACKGROUND.getValue());
  247. return sysFileEntity;
  248. }
  249. } catch (IOException e) {
  250. throw new RuntimeException(e);
  251. }
  252. }
  253. // target: 上传目标 (-1:本地, 1:腾讯云, 2:阿里云, 3.抖音云)
  254. private void deleteObject(String object_key, Integer target) {
  255. if (target == -1) {
  256. // [本地] 删除文件
  257. uploadUtil.delete(object_key);
  258. }
  259. if (target == 1) {
  260. // [腾讯云] 删除对象
  261. tencentCosService.deleteObject(object_key);
  262. System.out.println("Delete tencent cos object: " + object_key);
  263. }
  264. if (target == 3) {
  265. // [抖音云] 删除对象
  266. try {
  267. douyinTosService.deleteObject(object_key);
  268. } catch (IOException e) {
  269. throw new RuntimeException(e);
  270. }
  271. System.out.println("Delete douyin tos object: " + object_key);
  272. }
  273. }
  274. /**
  275. * 删除文件 (包括缩略图,如果有的话)
  276. */
  277. @Override
  278. public Map<String, Object> removeUploadFile(SysFile sysFile, SysFile querySysFile) {
  279. String object_key = querySysFile.getObject_key();
  280. // [Delete] 删除文件记录
  281. sysFileDao.delete(new LambdaQueryWrapper<SysFile>().eq(SysFile::getObject_key, object_key));
  282. // [异步任务] 创建一个 CompletableFuture 来执行异步任务
  283. CompletableFuture.runAsync(() -> {
  284. deleteObject(querySysFile.getObject_key(), querySysFile.getTarget());
  285. });
  286. return Map.of("object_key", object_key);
  287. }
  288. /**
  289. * 删除文件 (批量)
  290. */
  291. @Override
  292. public Map<String, Object> removeUploadFileBatch(SysFile sysFile, List<SysFile> querySysFileList) {
  293. // 判断是否存在
  294. List<String> object_keys = StrUtil.split(sysFile.getObject_keys(), ',', true, true);
  295. // [Delete] 批量删除
  296. sysFileDao.delete(new LambdaQueryWrapper<SysFile>().in(SysFile::getObject_key, object_keys));
  297. // [异步任务] 创建一个 CompletableFuture 来执行异步任务
  298. CompletableFuture.runAsync(() -> {
  299. querySysFileList.stream().forEach(entity -> {
  300. deleteObject(entity.getObject_key(), entity.getTarget());
  301. });
  302. });
  303. return Map.of("object_keys", object_keys);
  304. }
  305. /**
  306. * 编辑文件 (名称、文件分类)
  307. */
  308. @Override
  309. public Map<String, Object> updateUploadFile(SysFile sysFile) {
  310. // 查询文件是否存在
  311. LambdaQueryWrapper<SysFile> wrapper = new LambdaQueryWrapper<>();
  312. wrapper.eq(SysFile::getUser_id, sysFile.getUser_id());
  313. wrapper.eq(SysFile::getId, sysFile.getCategory_id());
  314. Boolean isFileExist = sysFileDao.exists(wrapper);
  315. if (!isFileExist) throw new CustException("文件不存在");
  316. // 查询文件分类是否存在
  317. if (sysFile.getCategory_id() != null) {
  318. LambdaQueryWrapper<SysFileCategory> wrapperCategory = new LambdaQueryWrapper<>();
  319. wrapperCategory.eq(SysFileCategory::getUser_id, sysFile.getUser_id());
  320. wrapperCategory.eq(SysFileCategory::getId, sysFile.getCategory_id());
  321. Boolean isFileCategoryExist = sysFileCategoryDao.exists(wrapperCategory);
  322. if (!isFileCategoryExist) throw new CustException("文件分类不存在");
  323. }
  324. // [DB] 更新文件
  325. sysFileDao.updateFile(sysFile);
  326. return Map.of("id", sysFile.getId());
  327. }
  328. }