SysFileServiceImpl.java 19 KB

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