|
@@ -1,64 +1,54 @@
|
|
package com.ruoyi.common.utils.file;
|
|
package com.ruoyi.common.utils.file;
|
|
|
|
|
|
-import java.io.File;
|
|
|
|
-import java.io.FileInputStream;
|
|
|
|
-import java.io.FileNotFoundException;
|
|
|
|
-import java.io.FileOutputStream;
|
|
|
|
-import java.io.IOException;
|
|
|
|
-import java.io.OutputStream;
|
|
|
|
-import java.io.UnsupportedEncodingException;
|
|
|
|
-import java.net.URLEncoder;
|
|
|
|
-import java.nio.charset.StandardCharsets;
|
|
|
|
-import javax.servlet.http.HttpServletRequest;
|
|
|
|
-import javax.servlet.http.HttpServletResponse;
|
|
|
|
-import org.apache.commons.io.IOUtils;
|
|
|
|
-import org.apache.commons.lang3.ArrayUtils;
|
|
|
|
import com.ruoyi.common.config.RuoYiConfig;
|
|
import com.ruoyi.common.config.RuoYiConfig;
|
|
import com.ruoyi.common.utils.DateUtils;
|
|
import com.ruoyi.common.utils.DateUtils;
|
|
import com.ruoyi.common.utils.StringUtils;
|
|
import com.ruoyi.common.utils.StringUtils;
|
|
import com.ruoyi.common.utils.uuid.IdUtils;
|
|
import com.ruoyi.common.utils.uuid.IdUtils;
|
|
import org.apache.commons.io.FilenameUtils;
|
|
import org.apache.commons.io.FilenameUtils;
|
|
|
|
+import org.apache.commons.io.IOUtils;
|
|
|
|
+import org.apache.commons.lang3.ArrayUtils;
|
|
|
|
+import org.apache.tika.Tika;
|
|
|
|
+import org.springframework.web.multipart.MultipartFile;
|
|
|
|
+
|
|
|
|
+import javax.servlet.http.HttpServletRequest;
|
|
|
|
+import javax.servlet.http.HttpServletResponse;
|
|
|
|
+import java.io.*;
|
|
|
|
+import java.net.URLEncoder;
|
|
|
|
+import java.nio.charset.StandardCharsets;
|
|
|
|
+import java.util.ArrayList;
|
|
|
|
+import java.util.List;
|
|
|
|
|
|
/**
|
|
/**
|
|
* 文件处理工具类
|
|
* 文件处理工具类
|
|
- *
|
|
|
|
|
|
+ *
|
|
* @author ruoyi
|
|
* @author ruoyi
|
|
*/
|
|
*/
|
|
-public class FileUtils
|
|
|
|
-{
|
|
|
|
|
|
+public class FileUtils {
|
|
public static String FILENAME_PATTERN = "[a-zA-Z0-9_\\-\\|\\.\\u4e00-\\u9fa5]+";
|
|
public static String FILENAME_PATTERN = "[a-zA-Z0-9_\\-\\|\\.\\u4e00-\\u9fa5]+";
|
|
|
|
|
|
/**
|
|
/**
|
|
* 输出指定文件的byte数组
|
|
* 输出指定文件的byte数组
|
|
- *
|
|
|
|
|
|
+ *
|
|
* @param filePath 文件路径
|
|
* @param filePath 文件路径
|
|
- * @param os 输出流
|
|
|
|
|
|
+ * @param os 输出流
|
|
* @return
|
|
* @return
|
|
*/
|
|
*/
|
|
- public static void writeBytes(String filePath, OutputStream os) throws IOException
|
|
|
|
- {
|
|
|
|
|
|
+ public static void writeBytes(String filePath, OutputStream os) throws IOException {
|
|
FileInputStream fis = null;
|
|
FileInputStream fis = null;
|
|
- try
|
|
|
|
- {
|
|
|
|
|
|
+ try {
|
|
File file = new File(filePath);
|
|
File file = new File(filePath);
|
|
- if (!file.exists())
|
|
|
|
- {
|
|
|
|
|
|
+ if (!file.exists()) {
|
|
throw new FileNotFoundException(filePath);
|
|
throw new FileNotFoundException(filePath);
|
|
}
|
|
}
|
|
fis = new FileInputStream(file);
|
|
fis = new FileInputStream(file);
|
|
byte[] b = new byte[1024];
|
|
byte[] b = new byte[1024];
|
|
int length;
|
|
int length;
|
|
- while ((length = fis.read(b)) > 0)
|
|
|
|
- {
|
|
|
|
|
|
+ while ((length = fis.read(b)) > 0) {
|
|
os.write(b, 0, length);
|
|
os.write(b, 0, length);
|
|
}
|
|
}
|
|
- }
|
|
|
|
- catch (IOException e)
|
|
|
|
- {
|
|
|
|
|
|
+ } catch (IOException e) {
|
|
throw e;
|
|
throw e;
|
|
- }
|
|
|
|
- finally
|
|
|
|
- {
|
|
|
|
|
|
+ } finally {
|
|
IOUtils.close(os);
|
|
IOUtils.close(os);
|
|
IOUtils.close(fis);
|
|
IOUtils.close(fis);
|
|
}
|
|
}
|
|
@@ -71,33 +61,28 @@ public class FileUtils
|
|
* @return 目标文件
|
|
* @return 目标文件
|
|
* @throws IOException IO异常
|
|
* @throws IOException IO异常
|
|
*/
|
|
*/
|
|
- public static String writeImportBytes(byte[] data) throws IOException
|
|
|
|
- {
|
|
|
|
|
|
+ public static String writeImportBytes(byte[] data) throws IOException {
|
|
return writeBytes(data, RuoYiConfig.getImportPath());
|
|
return writeBytes(data, RuoYiConfig.getImportPath());
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
/**
|
|
* 写数据到文件中
|
|
* 写数据到文件中
|
|
*
|
|
*
|
|
- * @param data 数据
|
|
|
|
|
|
+ * @param data 数据
|
|
* @param uploadDir 目标文件
|
|
* @param uploadDir 目标文件
|
|
* @return 目标文件
|
|
* @return 目标文件
|
|
* @throws IOException IO异常
|
|
* @throws IOException IO异常
|
|
*/
|
|
*/
|
|
- public static String writeBytes(byte[] data, String uploadDir) throws IOException
|
|
|
|
- {
|
|
|
|
|
|
+ public static String writeBytes(byte[] data, String uploadDir) throws IOException {
|
|
FileOutputStream fos = null;
|
|
FileOutputStream fos = null;
|
|
String pathName = "";
|
|
String pathName = "";
|
|
- try
|
|
|
|
- {
|
|
|
|
|
|
+ try {
|
|
String extension = getFileExtendName(data);
|
|
String extension = getFileExtendName(data);
|
|
pathName = DateUtils.datePath() + "/" + IdUtils.fastUUID() + "." + extension;
|
|
pathName = DateUtils.datePath() + "/" + IdUtils.fastUUID() + "." + extension;
|
|
File file = FileUploadUtils.getAbsoluteFile(uploadDir, pathName);
|
|
File file = FileUploadUtils.getAbsoluteFile(uploadDir, pathName);
|
|
fos = new FileOutputStream(file);
|
|
fos = new FileOutputStream(file);
|
|
fos.write(data);
|
|
fos.write(data);
|
|
- }
|
|
|
|
- finally
|
|
|
|
- {
|
|
|
|
|
|
+ } finally {
|
|
IOUtils.close(fos);
|
|
IOUtils.close(fos);
|
|
}
|
|
}
|
|
return FileUploadUtils.getPathFileName(uploadDir, pathName);
|
|
return FileUploadUtils.getPathFileName(uploadDir, pathName);
|
|
@@ -105,17 +90,15 @@ public class FileUtils
|
|
|
|
|
|
/**
|
|
/**
|
|
* 删除文件
|
|
* 删除文件
|
|
- *
|
|
|
|
|
|
+ *
|
|
* @param filePath 文件
|
|
* @param filePath 文件
|
|
* @return
|
|
* @return
|
|
*/
|
|
*/
|
|
- public static boolean deleteFile(String filePath)
|
|
|
|
- {
|
|
|
|
|
|
+ public static boolean deleteFile(String filePath) {
|
|
boolean flag = false;
|
|
boolean flag = false;
|
|
File file = new File(filePath);
|
|
File file = new File(filePath);
|
|
// 路径为文件且不为空则进行删除
|
|
// 路径为文件且不为空则进行删除
|
|
- if (file.isFile() && file.exists())
|
|
|
|
- {
|
|
|
|
|
|
+ if (file.isFile() && file.exists()) {
|
|
flag = file.delete();
|
|
flag = file.delete();
|
|
}
|
|
}
|
|
return flag;
|
|
return flag;
|
|
@@ -123,32 +106,28 @@ public class FileUtils
|
|
|
|
|
|
/**
|
|
/**
|
|
* 文件名称验证
|
|
* 文件名称验证
|
|
- *
|
|
|
|
|
|
+ *
|
|
* @param filename 文件名称
|
|
* @param filename 文件名称
|
|
* @return true 正常 false 非法
|
|
* @return true 正常 false 非法
|
|
*/
|
|
*/
|
|
- public static boolean isValidFilename(String filename)
|
|
|
|
- {
|
|
|
|
|
|
+ public static boolean isValidFilename(String filename) {
|
|
return filename.matches(FILENAME_PATTERN);
|
|
return filename.matches(FILENAME_PATTERN);
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
/**
|
|
* 检查文件是否可下载
|
|
* 检查文件是否可下载
|
|
- *
|
|
|
|
|
|
+ *
|
|
* @param resource 需要下载的文件
|
|
* @param resource 需要下载的文件
|
|
* @return true 正常 false 非法
|
|
* @return true 正常 false 非法
|
|
*/
|
|
*/
|
|
- public static boolean checkAllowDownload(String resource)
|
|
|
|
- {
|
|
|
|
|
|
+ public static boolean checkAllowDownload(String resource) {
|
|
// 禁止目录上跳级别
|
|
// 禁止目录上跳级别
|
|
- if (StringUtils.contains(resource, ".."))
|
|
|
|
- {
|
|
|
|
|
|
+ if (StringUtils.contains(resource, "..")) {
|
|
return false;
|
|
return false;
|
|
}
|
|
}
|
|
|
|
|
|
// 检查允许下载的文件规则
|
|
// 检查允许下载的文件规则
|
|
- if (ArrayUtils.contains(MimeTypeUtils.DEFAULT_ALLOWED_EXTENSION, FileTypeUtils.getFileType(resource)))
|
|
|
|
- {
|
|
|
|
|
|
+ if (ArrayUtils.contains(MimeTypeUtils.DEFAULT_ALLOWED_EXTENSION, FileTypeUtils.getFileType(resource))) {
|
|
return true;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
|
|
@@ -158,33 +137,25 @@ public class FileUtils
|
|
|
|
|
|
/**
|
|
/**
|
|
* 下载文件名重新编码
|
|
* 下载文件名重新编码
|
|
- *
|
|
|
|
- * @param request 请求对象
|
|
|
|
|
|
+ *
|
|
|
|
+ * @param request 请求对象
|
|
* @param fileName 文件名
|
|
* @param fileName 文件名
|
|
* @return 编码后的文件名
|
|
* @return 编码后的文件名
|
|
*/
|
|
*/
|
|
- public static String setFileDownloadHeader(HttpServletRequest request, String fileName) throws UnsupportedEncodingException
|
|
|
|
- {
|
|
|
|
|
|
+ public static String setFileDownloadHeader(HttpServletRequest request, String fileName) throws UnsupportedEncodingException {
|
|
final String agent = request.getHeader("USER-AGENT");
|
|
final String agent = request.getHeader("USER-AGENT");
|
|
String filename = fileName;
|
|
String filename = fileName;
|
|
- if (agent.contains("MSIE"))
|
|
|
|
- {
|
|
|
|
|
|
+ if (agent.contains("MSIE")) {
|
|
// IE浏览器
|
|
// IE浏览器
|
|
filename = URLEncoder.encode(filename, "utf-8");
|
|
filename = URLEncoder.encode(filename, "utf-8");
|
|
filename = filename.replace("+", " ");
|
|
filename = filename.replace("+", " ");
|
|
- }
|
|
|
|
- else if (agent.contains("Firefox"))
|
|
|
|
- {
|
|
|
|
|
|
+ } else if (agent.contains("Firefox")) {
|
|
// 火狐浏览器
|
|
// 火狐浏览器
|
|
filename = new String(fileName.getBytes(), "ISO8859-1");
|
|
filename = new String(fileName.getBytes(), "ISO8859-1");
|
|
- }
|
|
|
|
- else if (agent.contains("Chrome"))
|
|
|
|
- {
|
|
|
|
|
|
+ } else if (agent.contains("Chrome")) {
|
|
// google浏览器
|
|
// google浏览器
|
|
filename = URLEncoder.encode(filename, "utf-8");
|
|
filename = URLEncoder.encode(filename, "utf-8");
|
|
- }
|
|
|
|
- else
|
|
|
|
- {
|
|
|
|
|
|
+ } else {
|
|
// 其它浏览器
|
|
// 其它浏览器
|
|
filename = URLEncoder.encode(filename, "utf-8");
|
|
filename = URLEncoder.encode(filename, "utf-8");
|
|
}
|
|
}
|
|
@@ -194,11 +165,10 @@ public class FileUtils
|
|
/**
|
|
/**
|
|
* 下载文件名重新编码
|
|
* 下载文件名重新编码
|
|
*
|
|
*
|
|
- * @param response 响应对象
|
|
|
|
|
|
+ * @param response 响应对象
|
|
* @param realFileName 真实文件名
|
|
* @param realFileName 真实文件名
|
|
*/
|
|
*/
|
|
- public static void setAttachmentResponseHeader(HttpServletResponse response, String realFileName) throws UnsupportedEncodingException
|
|
|
|
- {
|
|
|
|
|
|
+ public static void setAttachmentResponseHeader(HttpServletResponse response, String realFileName) throws UnsupportedEncodingException {
|
|
String percentEncodedFileName = percentEncode(realFileName);
|
|
String percentEncodedFileName = percentEncode(realFileName);
|
|
|
|
|
|
StringBuilder contentDispositionValue = new StringBuilder();
|
|
StringBuilder contentDispositionValue = new StringBuilder();
|
|
@@ -220,36 +190,27 @@ public class FileUtils
|
|
* @param s 需要百分号编码的字符串
|
|
* @param s 需要百分号编码的字符串
|
|
* @return 百分号编码后的字符串
|
|
* @return 百分号编码后的字符串
|
|
*/
|
|
*/
|
|
- public static String percentEncode(String s) throws UnsupportedEncodingException
|
|
|
|
- {
|
|
|
|
|
|
+ public static String percentEncode(String s) throws UnsupportedEncodingException {
|
|
String encode = URLEncoder.encode(s, StandardCharsets.UTF_8.toString());
|
|
String encode = URLEncoder.encode(s, StandardCharsets.UTF_8.toString());
|
|
return encode.replaceAll("\\+", "%20");
|
|
return encode.replaceAll("\\+", "%20");
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
/**
|
|
* 获取图像后缀
|
|
* 获取图像后缀
|
|
- *
|
|
|
|
|
|
+ *
|
|
* @param photoByte 图像数据
|
|
* @param photoByte 图像数据
|
|
* @return 后缀名
|
|
* @return 后缀名
|
|
*/
|
|
*/
|
|
- public static String getFileExtendName(byte[] photoByte)
|
|
|
|
- {
|
|
|
|
|
|
+ public static String getFileExtendName(byte[] photoByte) {
|
|
String strFileExtendName = "jpg";
|
|
String strFileExtendName = "jpg";
|
|
if ((photoByte[0] == 71) && (photoByte[1] == 73) && (photoByte[2] == 70) && (photoByte[3] == 56)
|
|
if ((photoByte[0] == 71) && (photoByte[1] == 73) && (photoByte[2] == 70) && (photoByte[3] == 56)
|
|
- && ((photoByte[4] == 55) || (photoByte[4] == 57)) && (photoByte[5] == 97))
|
|
|
|
- {
|
|
|
|
|
|
+ && ((photoByte[4] == 55) || (photoByte[4] == 57)) && (photoByte[5] == 97)) {
|
|
strFileExtendName = "gif";
|
|
strFileExtendName = "gif";
|
|
- }
|
|
|
|
- else if ((photoByte[6] == 74) && (photoByte[7] == 70) && (photoByte[8] == 73) && (photoByte[9] == 70))
|
|
|
|
- {
|
|
|
|
|
|
+ } else if ((photoByte[6] == 74) && (photoByte[7] == 70) && (photoByte[8] == 73) && (photoByte[9] == 70)) {
|
|
strFileExtendName = "jpg";
|
|
strFileExtendName = "jpg";
|
|
- }
|
|
|
|
- else if ((photoByte[0] == 66) && (photoByte[1] == 77))
|
|
|
|
- {
|
|
|
|
|
|
+ } else if ((photoByte[0] == 66) && (photoByte[1] == 77)) {
|
|
strFileExtendName = "bmp";
|
|
strFileExtendName = "bmp";
|
|
- }
|
|
|
|
- else if ((photoByte[1] == 80) && (photoByte[2] == 78) && (photoByte[3] == 71))
|
|
|
|
- {
|
|
|
|
|
|
+ } else if ((photoByte[1] == 80) && (photoByte[2] == 78) && (photoByte[3] == 71)) {
|
|
strFileExtendName = "png";
|
|
strFileExtendName = "png";
|
|
}
|
|
}
|
|
return strFileExtendName;
|
|
return strFileExtendName;
|
|
@@ -257,14 +218,12 @@ public class FileUtils
|
|
|
|
|
|
/**
|
|
/**
|
|
* 获取文件名称 /profile/upload/2022/04/16/ruoyi.png -- ruoyi.png
|
|
* 获取文件名称 /profile/upload/2022/04/16/ruoyi.png -- ruoyi.png
|
|
- *
|
|
|
|
|
|
+ *
|
|
* @param fileName 路径名称
|
|
* @param fileName 路径名称
|
|
* @return 没有文件路径的名称
|
|
* @return 没有文件路径的名称
|
|
*/
|
|
*/
|
|
- public static String getName(String fileName)
|
|
|
|
- {
|
|
|
|
- if (fileName == null)
|
|
|
|
- {
|
|
|
|
|
|
+ public static String getName(String fileName) {
|
|
|
|
+ if (fileName == null) {
|
|
return null;
|
|
return null;
|
|
}
|
|
}
|
|
int lastUnixPos = fileName.lastIndexOf('/');
|
|
int lastUnixPos = fileName.lastIndexOf('/');
|
|
@@ -275,17 +234,94 @@ public class FileUtils
|
|
|
|
|
|
/**
|
|
/**
|
|
* 获取不带后缀文件名称 /profile/upload/2022/04/16/ruoyi.png -- ruoyi
|
|
* 获取不带后缀文件名称 /profile/upload/2022/04/16/ruoyi.png -- ruoyi
|
|
- *
|
|
|
|
|
|
+ *
|
|
* @param fileName 路径名称
|
|
* @param fileName 路径名称
|
|
* @return 没有文件路径和后缀的名称
|
|
* @return 没有文件路径和后缀的名称
|
|
*/
|
|
*/
|
|
- public static String getNameNotSuffix(String fileName)
|
|
|
|
- {
|
|
|
|
- if (fileName == null)
|
|
|
|
- {
|
|
|
|
|
|
+ public static String getNameNotSuffix(String fileName) {
|
|
|
|
+ if (fileName == null) {
|
|
return null;
|
|
return null;
|
|
}
|
|
}
|
|
String baseName = FilenameUtils.getBaseName(fileName);
|
|
String baseName = FilenameUtils.getBaseName(fileName);
|
|
return baseName;
|
|
return baseName;
|
|
}
|
|
}
|
|
|
|
+
|
|
|
|
+ /**
|
|
|
|
+ * 区分上传的文件是什么类型
|
|
|
|
+ *
|
|
|
|
+ * @param file 文件流
|
|
|
|
+ * @return 类型
|
|
|
|
+ */
|
|
|
|
+ public static String getFileType(MultipartFile file) {
|
|
|
|
+ Tika tika = new Tika();
|
|
|
|
+ try {
|
|
|
|
+ String mimeType = tika.detect(file.getInputStream());
|
|
|
|
+ if (mimeType.startsWith("image/")) {
|
|
|
|
+ return "image";
|
|
|
|
+ } else if (mimeType.startsWith("audio/")) {
|
|
|
|
+ return "audio";
|
|
|
|
+ } else if (mimeType.startsWith("video/")) {
|
|
|
|
+ return "video";
|
|
|
|
+ } else {
|
|
|
|
+ return "other";
|
|
|
|
+ }
|
|
|
|
+ } catch (IOException e) {
|
|
|
|
+ e.printStackTrace();
|
|
|
|
+ return "other";
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /**
|
|
|
|
+ * 截取视频的指定时间帧,生成图片文件
|
|
|
|
+ *
|
|
|
|
+ * @param source 源文件
|
|
|
|
+ * @param file 图片文件
|
|
|
|
+ * @param time 截图时间 HH:mm:ss.[SSS]
|
|
|
|
+ * @throws IOException
|
|
|
|
+ * @throws InterruptedException
|
|
|
|
+ */
|
|
|
|
+ public static boolean screenShots(String source, String file, String time) throws IOException, InterruptedException {
|
|
|
|
+
|
|
|
|
+ List<String> commands = new ArrayList<>();
|
|
|
|
+ commands.add("ffmpeg");
|
|
|
|
+ commands.add("-i");
|
|
|
|
+ commands.add(source);
|
|
|
|
+ commands.add("-ss");
|
|
|
|
+ commands.add(time);
|
|
|
|
+ commands.add("-y");
|
|
|
|
+ commands.add("-q:v");
|
|
|
|
+ commands.add("1");
|
|
|
|
+ commands.add("-frames:v");
|
|
|
|
+ commands.add("1");
|
|
|
|
+ commands.add("-f");
|
|
|
|
+ commands.add("image2");
|
|
|
|
+ commands.add(file);
|
|
|
|
+
|
|
|
|
+ Process process = new ProcessBuilder(commands)
|
|
|
|
+ .start();
|
|
|
|
+
|
|
|
|
+ // 读取进程标准输出
|
|
|
|
+ new Thread(() -> {
|
|
|
|
+ try (BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(process.getInputStream()))) {
|
|
|
|
+ String line = null;
|
|
|
|
+ while ((line = bufferedReader.readLine()) != null) {
|
|
|
|
+ System.out.println(line);
|
|
|
|
+ }
|
|
|
|
+ } catch (IOException e) {
|
|
|
|
+ }
|
|
|
|
+ }).start();
|
|
|
|
+
|
|
|
|
+ // 读取进程异常输出
|
|
|
|
+ new Thread(() -> {
|
|
|
|
+ try (BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(process.getErrorStream()))) {
|
|
|
|
+ String line = null;
|
|
|
|
+ while ((line = bufferedReader.readLine()) != null) {
|
|
|
|
+ System.out.println(line);
|
|
|
|
+ }
|
|
|
|
+ } catch (IOException e) {
|
|
|
|
+ }
|
|
|
|
+ }).start();
|
|
|
|
+
|
|
|
|
+ return process.waitFor() == 0;
|
|
|
|
+ }
|
|
}
|
|
}
|