Compare commits

...

19 Commits
test ... main

Author SHA1 Message Date
孙小云 22c9821314 xx 2026-02-05 17:00:48 +08:00
孙小云 2ea8c6962d xx' 2026-02-05 16:59:56 +08:00
孙小云 f1f54a1b5b xx 2026-02-05 16:58:33 +08:00
孙小云 c2e66fbc81 xx 2026-02-05 16:52:16 +08:00
孙小云 b6344c1188 xx 2026-02-05 16:49:45 +08:00
孙小云 cacffaee44 xx 2026-02-05 16:47:13 +08:00
孙小云 7e9884440c xx 2026-02-05 16:41:55 +08:00
孙小云 ab99bb8b1b xx 2026-02-05 16:38:30 +08:00
孙小云 1f9f100430 xx 2026-02-05 16:29:54 +08:00
孙小云 579abd3c45 xx 2026-02-05 16:11:21 +08:00
孙小云 6ffad8c83b xx 2026-02-05 16:04:54 +08:00
孙小云 725df605f4 xx 2026-02-05 16:04:15 +08:00
孙小云 3f36041b4a xx 2026-02-05 16:03:53 +08:00
高大 747539cec7 fix:联调航线上传 2026-01-28 11:20:40 +08:00
孙小云 94d7cf232b Merge branch 'main' of http://th.local.t-aaron.com:13000/THENG/a-ruoyi-file 2026-01-26 16:58:15 +08:00
孙小云 5f01d5af81 xx 2026-01-26 16:58:00 +08:00
高大 1e18bf59b5 fix:优化航线管理航线上传逻辑 2026-01-24 17:18:09 +08:00
高大 8c13532322 feat:文件服务补充feign接口-增加流式调用 2026-01-24 16:13:02 +08:00
高大 72ba3c8daf 文件服务增加通过配置控制文件存储方式的配置 2026-01-24 15:36:41 +08:00
6 changed files with 82 additions and 53 deletions

View File

@ -1 +0,0 @@
dda

View File

@ -1,5 +1,10 @@
package com.ruoyi.file.controller; package com.ruoyi.file.controller;
import com.ruoyi.common.core.domain.R;
import com.ruoyi.common.core.utils.StringUtils;
import com.ruoyi.common.core.utils.file.FileUtils;
import com.ruoyi.file.service.ISysFileService;
import com.ruoyi.system.api.domain.SysFile;
import org.slf4j.Logger; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Autowired;
@ -7,11 +12,6 @@ import org.springframework.web.bind.annotation.DeleteMapping;
import org.springframework.web.bind.annotation.PostMapping; import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RestController; import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.multipart.MultipartFile; import org.springframework.web.multipart.MultipartFile;
import com.ruoyi.common.core.domain.R;
import com.ruoyi.common.core.utils.StringUtils;
import com.ruoyi.common.core.utils.file.FileUtils;
import com.ruoyi.file.service.ISysFileService;
import com.ruoyi.system.api.domain.SysFile;
/** /**
* 文件请求处理 * 文件请求处理
@ -19,8 +19,7 @@ import com.ruoyi.system.api.domain.SysFile;
* @author ruoyi * @author ruoyi
*/ */
@RestController @RestController
public class SysFileController public class SysFileController {
{
private static final Logger log = LoggerFactory.getLogger(SysFileController.class); private static final Logger log = LoggerFactory.getLogger(SysFileController.class);
@Autowired @Autowired
@ -30,19 +29,15 @@ public class SysFileController
* 文件上传请求 * 文件上传请求
*/ */
@PostMapping("upload") @PostMapping("upload")
public R<SysFile> upload(MultipartFile file) public R<SysFile> upload(MultipartFile file) {
{ try {
try
{
// 上传并返回访问地址 // 上传并返回访问地址
String url = sysFileService.uploadFile(file); String url = sysFileService.uploadFile(file);
SysFile sysFile = new SysFile(); SysFile sysFile = new SysFile();
sysFile.setName(FileUtils.getName(url)); sysFile.setName(FileUtils.getName(url));
sysFile.setUrl(url); sysFile.setUrl(url);
return R.ok(sysFile); return R.ok(sysFile);
} } catch (Exception e) {
catch (Exception e)
{
log.error("上传文件失败", e); log.error("上传文件失败", e);
return R.fail(e.getMessage()); return R.fail(e.getMessage());
} }
@ -52,21 +47,30 @@ public class SysFileController
* 文件删除请求 * 文件删除请求
*/ */
@DeleteMapping("delete") @DeleteMapping("delete")
public R<Boolean> delete(String fileUrl) public R<Boolean> delete(String fileUrl) {
{ try {
try if (!FileUtils.validateFilePath(fileUrl)) {
{
if (!FileUtils.validateFilePath(fileUrl))
{
throw new Exception(StringUtils.format("资源文件({})非法,不允许删除。 ", fileUrl)); throw new Exception(StringUtils.format("资源文件({})非法,不允许删除。 ", fileUrl));
} }
sysFileService.deleteFile(fileUrl); sysFileService.deleteFile(fileUrl);
return R.ok(); return R.ok();
} } catch (Exception e) {
catch (Exception e)
{
log.error("删除文件失败", e); log.error("删除文件失败", e);
return R.fail(e.getMessage()); return R.fail(e.getMessage());
} }
} }
/**
* 文件流上传请求
*/
@PostMapping("uploadStream")
public R<String> uploadFileByStream(String filename, String extension, String data) {
try {
return R.ok(sysFileService.uploadFileByData(filename, extension, data));
} catch (Exception e) {
log.error("上传文件失败", e);
return R.fail(e.getMessage());
}
}
} }

View File

@ -2,8 +2,6 @@ package com.ruoyi.file.service;
import org.springframework.web.multipart.MultipartFile; import org.springframework.web.multipart.MultipartFile;
import java.io.ByteArrayOutputStream;
/** /**
* 文件上传接口 * 文件上传接口
* *
@ -36,5 +34,5 @@ public interface ISysFileService {
* @param out 上传的流 * @param out 上传的流
* @return 访问地址 * @return 访问地址
*/ */
String uploadFileByStream(String filename, String extension, ByteArrayOutputStream out) throws Exception; String uploadFileByData(String filename, String extension, String out) throws Exception;
} }

View File

@ -4,7 +4,7 @@ import com.ruoyi.common.core.utils.StringUtils;
import com.ruoyi.common.core.utils.file.FileUtils; import com.ruoyi.common.core.utils.file.FileUtils;
import com.ruoyi.file.utils.FileUploadUtils; import com.ruoyi.file.utils.FileUploadUtils;
import org.springframework.beans.factory.annotation.Value; import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Primary; import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
import org.springframework.stereotype.Service; import org.springframework.stereotype.Service;
import org.springframework.web.multipart.MultipartFile; import org.springframework.web.multipart.MultipartFile;
@ -16,8 +16,8 @@ import java.io.ByteArrayOutputStream;
* *
* @author ruoyi * @author ruoyi
*/ */
@Primary
@Service @Service
@ConditionalOnProperty(name = "file.storage", havingValue = "local")
public class LocalSysFileServiceImpl implements ISysFileService { public class LocalSysFileServiceImpl implements ISysFileService {
/** /**
* 资源映射路径 前缀 * 资源映射路径 前缀
@ -71,8 +71,8 @@ public class LocalSysFileServiceImpl implements ISysFileService {
* @return 访问地址 * @return 访问地址
*/ */
@Override @Override
public String uploadFileByStream(String filename, String extension, ByteArrayOutputStream out) throws Exception { public String uploadFileByData(String filename, String extension, String out) throws Exception {
try (ByteArrayInputStream in = new ByteArrayInputStream(out.toByteArray())) { try (ByteArrayInputStream in = new ByteArrayInputStream(out.getBytes())) {
String fileName = FileUploadUtils.extractFilename(filename, extension); String fileName = FileUploadUtils.extractFilename(filename, extension);
return FileUploadUtils.uploadByStream(localFilePath, fileName, in); return FileUploadUtils.uploadByStream(localFilePath, fileName, in);
} catch (Exception e) { } catch (Exception e) {

View File

@ -8,6 +8,7 @@ import io.minio.MinioClient;
import io.minio.PutObjectArgs; import io.minio.PutObjectArgs;
import io.minio.RemoveObjectArgs; import io.minio.RemoveObjectArgs;
import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
import org.springframework.stereotype.Service; import org.springframework.stereotype.Service;
import org.springframework.web.multipart.MultipartFile; import org.springframework.web.multipart.MultipartFile;
@ -21,6 +22,7 @@ import java.io.InputStream;
* @author ruoyi * @author ruoyi
*/ */
@Service @Service
@ConditionalOnProperty(name = "file.storage", havingValue = "minio", matchIfMissing = true)
public class MinioSysFileServiceImpl implements ISysFileService { public class MinioSysFileServiceImpl implements ISysFileService {
@Autowired @Autowired
private MinioConfig minioConfig; private MinioConfig minioConfig;
@ -73,10 +75,18 @@ public class MinioSysFileServiceImpl implements ISysFileService {
} }
@Override @Override
public String uploadFileByStream(String filename, String extension, ByteArrayOutputStream out) throws Exception { public String uploadFileByData(String filename, String extension, String out) throws Exception {
try (ByteArrayInputStream in = new ByteArrayInputStream(out.toByteArray())) { try (ByteArrayInputStream in = new ByteArrayInputStream(out.getBytes())) {
String fileName = FileUploadUtils.extractFilename(filename, extension); String fileName = FileUploadUtils.extractFilename(filename, extension);
System.out.println("========== MinIO Upload Debug ==========");
System.out.println("原始文件名: " + filename);
System.out.println("传入的扩展名: [" + extension + "]");
System.out.println("生成的文件名: " + fileName);
String contentType = FileUploadUtils.getContentType(extension); // 获取文件类型 String contentType = FileUploadUtils.getContentType(extension); // 获取文件类型
System.out.println("获取到的 ContentType: " + contentType);
System.out.println("========================================");
PutObjectArgs args = PutObjectArgs.builder() PutObjectArgs args = PutObjectArgs.builder()
.bucket(minioConfig.getBucketName()) .bucket(minioConfig.getBucketName())
.object(fileName) .object(fileName)

View File

@ -175,55 +175,73 @@ public class FileUploadUtils {
public static String getContentType(String FilenameExtension) { public static String getContentType(String FilenameExtension) {
if (FilenameExtension.equalsIgnoreCase(".bmp")) { System.out.println("---------- getContentType Debug ----------");
System.out.println("传入的扩展名参数: [" + FilenameExtension + "]");
System.out.println("扩展名长度: " + (FilenameExtension != null ? FilenameExtension.length() : "null"));
// 统一处理如果没有点自动添加点
String extension = FilenameExtension;
if (extension != null && !extension.startsWith(".")) {
extension = "." + extension;
System.out.println("扩展名没有点,自动添加后: [" + extension + "]");
}
if (extension.equalsIgnoreCase(".bmp")) {
return "image/bmp"; return "image/bmp";
} }
if (FilenameExtension.equalsIgnoreCase(".gif")) { if (extension.equalsIgnoreCase(".gif")) {
return "image/gif"; return "image/gif";
} }
if (FilenameExtension.equalsIgnoreCase(".jpeg") || if (extension.equalsIgnoreCase(".jpeg") ||
FilenameExtension.equalsIgnoreCase(".jpg") || extension.equalsIgnoreCase(".jpg") ||
FilenameExtension.equalsIgnoreCase(".png")) { extension.equalsIgnoreCase(".png")) {
return "image/jpeg"; return "image/jpeg";
} }
if (FilenameExtension.equalsIgnoreCase(".html")) { if (extension.equalsIgnoreCase(".html")) {
return "text/html"; return "text/html";
} }
if (FilenameExtension.equalsIgnoreCase(".txt")) { if (extension.equalsIgnoreCase(".txt")) {
return "text/plain"; return "text/plain";
} }
if (FilenameExtension.equalsIgnoreCase(".vsd")) { if (extension.equalsIgnoreCase(".vsd")) {
return "application/vnd.visio"; return "application/vnd.visio";
} }
if (FilenameExtension.equalsIgnoreCase(".pptx") || if (extension.equalsIgnoreCase(".pptx") ||
FilenameExtension.equalsIgnoreCase(".ppt")) { extension.equalsIgnoreCase(".ppt")) {
return "application/vnd.ms-powerpoint"; return "application/vnd.ms-powerpoint";
} }
if (FilenameExtension.equalsIgnoreCase(".docx") || if (extension.equalsIgnoreCase(".docx") ||
FilenameExtension.equalsIgnoreCase(".doc")) { extension.equalsIgnoreCase(".doc")) {
return "application/msword"; return "application/msword";
} }
if (FilenameExtension.equalsIgnoreCase(".xml")) { if (extension.equalsIgnoreCase(".xml")) {
return "text/xml"; return "text/xml";
} }
//PDF //PDF
if (FilenameExtension.equalsIgnoreCase(".pdf")) { if (extension.equalsIgnoreCase(".pdf")) {
return "application/pdf"; return "application/pdf";
} }
//excel //excel
if (".xls".equalsIgnoreCase(FilenameExtension)) { if (".xls".equalsIgnoreCase(extension)) {
return "application/vnd.ms-excel"; return "application/vnd.ms-excel";
} }
if (".xlsx".equalsIgnoreCase(FilenameExtension)) { if (".xlsx".equalsIgnoreCase(extension)) {
return "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet"; return "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet";
} }
//waypoints 拓恒+大疆的航线文件类型 //waypoints 拓恒+大疆的航线文件类型
if (FilenameExtension.equalsIgnoreCase(".waypoints") || System.out.println("检查 waypoints 匹配:");
FilenameExtension.equalsIgnoreCase(".kmz")) { System.out.println(" 与 '.waypoints' 比较: " + extension.equalsIgnoreCase(".waypoints"));
System.out.println(" 与 '.kmz' 比较: " + extension.equalsIgnoreCase(".kmz"));
if (extension.equalsIgnoreCase(".waypoints") ||
extension.equalsIgnoreCase(".kmz")) {
System.out.println("匹配成功! 返回 application/octet-stream");
return "application/octet-stream"; return "application/octet-stream";
} }
return "image/jpeg"; System.out.println("未匹配任何类型,返回默认值 application/octet-stream");
System.out.println("------------------------------------------");
return "application/octet-stream";
} }