Browse Source

Add: 添加通用文件上传及文件预览+配置yml及config

main
简小朤 1 month ago
parent
commit
552d61d8a1
  1. 3
      .gitignore
  2. 4
      coin-app/src/main/java/com/coin/app/controller/common/QRCodeController.java
  3. 2
      coin-app/src/main/java/com/coin/app/controller/deal/DealController.java
  4. 8
      coin-app/src/main/resources/application.yml
  5. 2
      coin-system/src/main/java/com/coin/app/service/IAppDealService.java
  6. 2
      coin-system/src/main/java/com/coin/app/service/QRCodeService.java
  7. 10
      coin-system/src/main/java/com/coin/app/service/impl/AppDealServiceImpl.java
  8. 12
      coin-system/src/main/java/com/coin/app/service/impl/QRCodeServiceImpl.java
  9. 22
      coin-system/src/main/java/com/coin/system/config/FileUploadConfig.java
  10. 29
      coin-system/src/main/java/com/coin/system/config/MultipartConfig.java
  11. 30
      coin-system/src/main/java/com/coin/system/controller/FileController.java
  12. 21
      coin-system/src/main/java/com/coin/system/service/FileService.java
  13. 95
      coin-system/src/main/java/com/coin/system/service/impl/FileServiceImpl.java
  14. 2
      coin-system/src/main/java/com/coin/system/vo/DealItemVO.java

3
.gitignore

@ -14,6 +14,9 @@ target/
# log
/logs/
# upload file
/file/
### STS ###
.apt_generated
.classpath

4
coin-app/src/main/java/com/coin/app/controller/common/QRCodeController.java

@ -1,6 +1,6 @@
package com.coin.app.controller.common;
import com.coin.app.domain.vo.DealItemVO;
import com.coin.system.vo.DealItemVO;
import com.coin.app.service.QRCodeService;
import com.coin.common.core.domain.R;
import lombok.RequiredArgsConstructor;
@ -18,7 +18,7 @@ import org.springframework.web.bind.annotation.RestController;
@RestController
@RequestMapping("/appQRCode")
public class QRCodeController {
private final QRCodeService qrCodeService;
/**

2
coin-app/src/main/java/com/coin/app/controller/deal/DealController.java

@ -1,6 +1,6 @@
package com.coin.app.controller.deal;
import com.coin.app.domain.vo.DealItemVO;
import com.coin.system.vo.DealItemVO;
import com.coin.app.service.IAppDealService;
import com.coin.business.domain.DealTransfer;
import com.coin.business.domain.DealCashout;

8
coin-app/src/main/resources/application.yml

@ -139,6 +139,7 @@ security:
# app 端接口
- /appDeal/**
- /appTransRecord/**
- /file/**
# MyBatisPlus配置
# https://baomidou.com/config/
@ -271,3 +272,10 @@ management:
show-details: ALWAYS
logfile:
external-file: ./logs/sys-console.log
# 文件上传配置
file:
upload:
path: /Users/jian/coin/fundHold/coin/file
maxSize: 10
sourcePath: /file/preview

2
coin-system/src/main/java/com/coin/app/service/IAppDealService.java

@ -1,6 +1,6 @@
package com.coin.app.service;
import com.coin.app.domain.vo.DealItemVO;
import com.coin.system.vo.DealItemVO;
import com.coin.business.domain.DealTransfer;
import com.coin.business.domain.DealCashout;
import com.coin.business.domain.DealPay;

2
coin-system/src/main/java/com/coin/app/service/QRCodeService.java

@ -1,6 +1,6 @@
package com.coin.app.service;
import com.coin.app.domain.vo.DealItemVO;
import com.coin.system.vo.DealItemVO;
/**
* 二维码服务接口

10
coin-system/src/main/java/com/coin/app/service/impl/AppDealServiceImpl.java

@ -1,6 +1,6 @@
package com.coin.app.service.impl;
import com.coin.app.domain.vo.DealItemVO;
import com.coin.system.vo.DealItemVO;
import com.coin.app.service.IAppDealService;
import com.coin.business.domain.DealTransfer;
import com.coin.business.domain.DealCashout;
@ -49,7 +49,7 @@ public class AppDealServiceImpl implements IAppDealService {
transfer.setToAccount(dealItem.getToAccount());
transfer.setAmount(BigDecimal.valueOf(dealItem.getAmount()));
transfer.setBillCode(BusinessCodeUtils.generateBusinessCode(BusinessCodeUtils.BusinessType.BT,Long.parseLong(dealItem.getFromId()),Long.parseLong(dealItem.getToId()),dealItem.getAmount()));
if (dealTransferMapper.insert(transfer) <= 0) {
throw new ServiceException("转账订单创建失败");
}
@ -71,7 +71,7 @@ public class AppDealServiceImpl implements IAppDealService {
cashOut.setToAccount(dealItem.getToAccount());
cashOut.setAmount(BigDecimal.valueOf(dealItem.getAmount()));
cashOut.setBillCode(BusinessCodeUtils.generateBusinessCode(BusinessCodeUtils.BusinessType.BC,Long.parseLong(dealItem.getFromId()),Long.parseLong(dealItem.getToId()),dealItem.getAmount()));
if (dealCashoutMapper.insert(cashOut) <= 0) {
throw new ServiceException("提现订单创建失败");
}
@ -95,7 +95,7 @@ public class AppDealServiceImpl implements IAppDealService {
pay.setToAccount(payCodeInfo.get("merchantAccount"));
pay.setAmount(BigDecimal.valueOf(dealItem.getAmount()));
pay.setBillCode(BusinessCodeUtils.generateBusinessCode(BusinessCodeUtils.BusinessType.BP,Long.parseLong(dealItem.getFromId()),Long.parseLong(dealItem.getToId()),dealItem.getAmount()));
if (dealPayMapper.insert(pay) <= 0) {
throw new ServiceException("支付订单创建失败");
}
@ -123,7 +123,7 @@ public class AppDealServiceImpl implements IAppDealService {
receive.setAmount(BigDecimal.valueOf(dealItem.getAmount()));
}
receive.setBillCode(BusinessCodeUtils.generateBusinessCode(BusinessCodeUtils.BusinessType.BR,Long.parseLong(dealItem.getFromId()),Long.parseLong(dealItem.getToId()),dealItem.getAmount()));
if (dealReceiveMapper.insert(receive) <= 0) {
throw new ServiceException("收款订单创建失败");
}

12
coin-system/src/main/java/com/coin/app/service/impl/QRCodeServiceImpl.java

@ -1,6 +1,6 @@
package com.coin.app.service.impl;
import com.coin.app.domain.vo.DealItemVO;
import com.coin.system.vo.DealItemVO;
import com.coin.app.service.QRCodeService;
import com.coin.common.exception.ServiceException;
import com.coin.common.utils.code.QRCodeUtils;
@ -43,7 +43,7 @@ public class QRCodeServiceImpl implements QRCodeService {
if (dealItem == null || dealItem.getFromId() == null || dealItem.getFromAccount() == null) {
throw new ServiceException("交易信息不完整");
}
// 根据是否包含金额,调用不同的生成方法
if (dealItem.getAmount() != null && dealItem.getAmount() > 0) {
// 有金额的情况,需要将 Double 转换为 BigDecimal
@ -85,8 +85,8 @@ public class QRCodeServiceImpl implements QRCodeService {
public String parseReceiveCode(String code) {
try {
Map<String, String> result = QRCodeUtils.parseReceiveCode(code);
return String.format("商户ID: %s, 商户账户: %s%s",
result.get("merchantId"),
return String.format("商户ID: %s, 商户账户: %s%s",
result.get("merchantId"),
result.get("merchantAccount"),
"true".equals(result.get("hasAmount")) ? ", 金额: " + result.get("amount") : "");
} catch (IllegalArgumentException e) {
@ -98,8 +98,8 @@ public class QRCodeServiceImpl implements QRCodeService {
public String parsePayCode(String code) {
try {
Map<String, String> result = QRCodeUtils.parsePayCode(code);
return String.format("商户ID: %s, 商户账户: %s",
result.get("merchantId"),
return String.format("商户ID: %s, 商户账户: %s",
result.get("merchantId"),
result.get("merchantAccount"));
} catch (IllegalArgumentException e) {
throw new ServiceException("解析付款码失败: " + e.getMessage());

22
coin-system/src/main/java/com/coin/system/config/FileUploadConfig.java

@ -0,0 +1,22 @@
package com.coin.system.config;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.ResourceHandlerRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
@Configuration
public class FileUploadConfig implements WebMvcConfigurer {
@Value("${file.upload.path}")
private String uploadPath;
@Value("${file.upload.sourcePath}")
private String sourcePath;
@Override
public void addResourceHandlers(ResourceHandlerRegistry registry) {
registry.addResourceHandler(sourcePath+"/**")
.addResourceLocations("file:" + uploadPath + "/");
}
}

29
coin-system/src/main/java/com/coin/system/config/MultipartConfig.java

@ -0,0 +1,29 @@
package com.coin.system.config;
import org.springframework.boot.web.servlet.MultipartConfigFactory;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.util.unit.DataSize;
import org.springframework.web.multipart.MultipartResolver;
import org.springframework.web.multipart.support.StandardServletMultipartResolver;
import javax.servlet.MultipartConfigElement;
@Configuration
public class MultipartConfig {
@Bean
public MultipartResolver multipartResolver() {
return new StandardServletMultipartResolver();
}
@Bean
public MultipartConfigElement multipartConfigElement() {
MultipartConfigFactory factory = new MultipartConfigFactory();
// 设置单个文件最大大小为10MB
factory.setMaxFileSize(DataSize.ofMegabytes(10));
// 设置总上传数据最大大小为20MB
factory.setMaxRequestSize(DataSize.ofMegabytes(20));
return factory.createMultipartConfig();
}
}

30
coin-system/src/main/java/com/coin/system/controller/FileController.java

@ -0,0 +1,30 @@
package com.coin.system.controller;
import com.coin.system.service.FileService;
import com.coin.common.core.domain.R;
import lombok.RequiredArgsConstructor;
import org.springframework.web.bind.annotation.*;
import org.springframework.web.multipart.MultipartFile;
import javax.servlet.http.HttpServletResponse;
/**
* 文件通用接口
*/
@RestController
@RequestMapping("/file")
@RequiredArgsConstructor
public class FileController {
private final FileService fileService;
@PostMapping("/upload")
public R<String> upload(@RequestParam("file") MultipartFile file) {
return R.ok("上传成功!",fileService.uploadFile(file));
}
@GetMapping("/preview/{filePath}")
public void preview(@PathVariable String filePath, HttpServletResponse response) {
fileService.previewFile(filePath, response);
}
}

21
coin-system/src/main/java/com/coin/system/service/FileService.java

@ -0,0 +1,21 @@
package com.coin.system.service;
import org.springframework.web.multipart.MultipartFile;
import javax.servlet.http.HttpServletResponse;
public interface FileService {
/**
* 上传文件
* @param file 文件
* @return 文件访问路径
*/
String uploadFile(MultipartFile file);
/**
* 预览文件
* @param filePath 文件路径
* @param response HTTP响应
*/
void previewFile(String filePath, HttpServletResponse response);
}

95
coin-system/src/main/java/com/coin/system/service/impl/FileServiceImpl.java

@ -0,0 +1,95 @@
package com.coin.system.service.impl;
import com.coin.system.service.FileService;
import com.coin.common.exception.ServiceException;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.io.FilenameUtils;
import org.apache.commons.lang3.StringUtils;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Service;
import org.springframework.web.multipart.MultipartFile;
import javax.servlet.http.HttpServletResponse;
import java.io.*;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.time.LocalDate;
import java.time.format.DateTimeFormatter;
import java.util.UUID;
@Slf4j
@Service
public class FileServiceImpl implements FileService {
@Value("${file.upload.path}")
private String uploadPath;
@Value("${file.upload.maxSize}")
private long maxFileSize; // 默认最大10MB
@Override
public String uploadFile(MultipartFile file) {
if (file.isEmpty()) {
throw new ServiceException("上传文件不能为空");
}
// 检查文件大小
if (file.getSize() > maxFileSize * 1024 * 1024) {
throw new ServiceException("文件大小超过限制");
}
try {
// 生成文件存储路径(按日期分类)
String datePath = LocalDate.now().format(DateTimeFormatter.ofPattern("yyyy/MM/dd"));
Path uploadDir = Paths.get(uploadPath, datePath);
Files.createDirectories(uploadDir);
// 生成唯一文件名
String originalFilename = file.getOriginalFilename();
String extension = FilenameUtils.getExtension(originalFilename);
String newFileName = UUID.randomUUID().toString() + "." + extension;
Path targetPath = uploadDir.resolve(newFileName);
// 使用NIO进行文件写入
Files.copy(file.getInputStream(), targetPath);
// 返回相对路径
return datePath + "/" + newFileName;
} catch (IOException e) {
log.error("文件上传失败", e);
throw new ServiceException("文件上传失败");
}
}
@Override
public void previewFile(String filePath, HttpServletResponse response) {
if (StringUtils.isEmpty(filePath)) {
throw new ServiceException("文件路径不能为空");
}
Path path = Paths.get(uploadPath, filePath);
if (!Files.exists(path)) {
throw new ServiceException("文件不存在");
}
try {
String contentType = Files.probeContentType(path);
response.setContentType(contentType);
// 使用缓冲流提高性能
try (InputStream in = new BufferedInputStream(Files.newInputStream(path));
OutputStream out = new BufferedOutputStream(response.getOutputStream())) {
byte[] buffer = new byte[4096];
int bytesRead;
while ((bytesRead = in.read(buffer)) != -1) {
out.write(buffer, 0, bytesRead);
}
out.flush();
}
} catch (IOException e) {
log.error("文件预览失败", e);
throw new ServiceException("文件预览失败");
}
}
}

2
coin-system/src/main/java/com/coin/app/domain/vo/DealItemVO.java → coin-system/src/main/java/com/coin/system/vo/DealItemVO.java

@ -1,4 +1,4 @@
package com.coin.app.domain.vo;
package com.coin.system.vo;
import lombok.AllArgsConstructor;
import lombok.Data;
Loading…
Cancel
Save