PathUtil增加loopFiles重载,可选是否追踪软链

This commit is contained in:
Looly 2023-10-27 22:43:11 +08:00
parent fcd3d9f7d0
commit d119d2dc5e
2 changed files with 65 additions and 14 deletions

View File

@ -10,6 +10,7 @@
* 【http 】 UserAgent增加百度浏览器识别issue#I847JY@Gitee * 【http 】 UserAgent增加百度浏览器识别issue#I847JY@Gitee
* 【core 】 ReflectUtil.getFieldsValue增加Filter重载pr#1090@Gitee * 【core 】 ReflectUtil.getFieldsValue增加Filter重载pr#1090@Gitee
* 【core 】 Snowflake增加方法根据传入时间戳计算ID起终点pr#1096@Gitee * 【core 】 Snowflake增加方法根据传入时间戳计算ID起终点pr#1096@Gitee
* 【core 】 PathUtil增加loopFiles重载可选是否追踪软链issue#3353@Github
### 🐞Bug修复 ### 🐞Bug修复
* 【cron 】 修复Cron表达式range解析错误问题issue#I82CSH@Gitee * 【cron 】 修复Cron表达式range解析错误问题issue#I82CSH@Gitee

View File

@ -1,5 +1,6 @@
package cn.hutool.core.io.file; package cn.hutool.core.io.file;
import cn.hutool.core.collection.CollUtil;
import cn.hutool.core.io.IORuntimeException; import cn.hutool.core.io.IORuntimeException;
import cn.hutool.core.io.IoUtil; import cn.hutool.core.io.IoUtil;
import cn.hutool.core.io.file.visitor.CopyVisitor; import cn.hutool.core.io.file.visitor.CopyVisitor;
@ -31,6 +32,7 @@ import java.nio.file.attribute.BasicFileAttributes;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.EnumSet; import java.util.EnumSet;
import java.util.List; import java.util.List;
import java.util.Set;
/** /**
* NIO中Path对象操作封装 * NIO中Path对象操作封装
@ -78,11 +80,26 @@ public class PathUtil {
* @since 5.4.1 * @since 5.4.1
*/ */
public static List<File> loopFiles(Path path, int maxDepth, FileFilter fileFilter) { public static List<File> loopFiles(Path path, int maxDepth, FileFilter fileFilter) {
return loopFiles(path, maxDepth, false, fileFilter);
}
/**
* 递归遍历目录以及子目录中的所有文件<br>
* 如果提供path为文件直接返回过滤结果
*
* @param path 当前遍历文件或目录
* @param maxDepth 遍历最大深度-1表示遍历到没有目录为止
* @param isFollowLinks 是否跟踪软链快捷方式
* @param fileFilter 文件过滤规则对象选择要保留的文件只对文件有效不过滤目录null表示接收全部文件
* @return 文件列表
* @since 5.4.1
*/
public static List<File> loopFiles(final Path path, final int maxDepth, final boolean isFollowLinks, final FileFilter fileFilter) {
final List<File> fileList = new ArrayList<>(); final List<File> fileList = new ArrayList<>();
if (null == path || false == Files.exists(path)) { if (!exists(path, isFollowLinks)) {
return fileList; return fileList;
} else if (false == isDirectory(path)) { } else if (!isDirectory(path, isFollowLinks)) {
final File file = path.toFile(); final File file = path.toFile();
if (null == fileFilter || fileFilter.accept(file)) { if (null == fileFilter || fileFilter.accept(file)) {
fileList.add(file); fileList.add(file);
@ -90,10 +107,10 @@ public class PathUtil {
return fileList; return fileList;
} }
walkFiles(path, maxDepth, new SimpleFileVisitor<Path>() { walkFiles(path, maxDepth, isFollowLinks, new SimpleFileVisitor<Path>() {
@Override @Override
public FileVisitResult visitFile(Path path, BasicFileAttributes attrs) { public FileVisitResult visitFile(final Path path, final BasicFileAttributes attrs) {
final File file = path.toFile(); final File file = path.toFile();
if (null == fileFilter || fileFilter.accept(file)) { if (null == fileFilter || fileFilter.accept(file)) {
fileList.add(file); fileList.add(file);
@ -127,14 +144,28 @@ public class PathUtil {
* @since 4.6.3 * @since 4.6.3
*/ */
public static void walkFiles(Path start, int maxDepth, FileVisitor<? super Path> visitor) { public static void walkFiles(Path start, int maxDepth, FileVisitor<? super Path> visitor) {
walkFiles(start, maxDepth, false, visitor);
}
/**
* 遍历指定path下的文件并做处理
*
* @param start 起始路径必须为目录
* @param maxDepth 最大遍历深度-1表示不限制深度
* @param visitor {@link FileVisitor} 接口用于自定义在访问文件时访问目录前后等节点做的操作
* @param isFollowLinks 是否追踪到软链对应的真实地址
* @see Files#walkFileTree(Path, java.util.Set, int, FileVisitor)
* @since 5.8.23
*/
public static void walkFiles(final Path start, int maxDepth, final boolean isFollowLinks, final FileVisitor<? super Path> visitor) {
if (maxDepth < 0) { if (maxDepth < 0) {
// < 0 表示遍历到最底层 // < 0 表示遍历到最底层
maxDepth = Integer.MAX_VALUE; maxDepth = Integer.MAX_VALUE;
} }
try { try {
Files.walkFileTree(start, EnumSet.noneOf(FileVisitOption.class), maxDepth, visitor); Files.walkFileTree(start, getFileVisitOption(isFollowLinks), maxDepth, visitor);
} catch (IOException e) { } catch (final IOException e) {
throw new IORuntimeException(e); throw new IORuntimeException(e);
} }
} }
@ -281,8 +312,7 @@ public class PathUtil {
if (null == path) { if (null == path) {
return false; return false;
} }
final LinkOption[] options = isFollowLinks ? new LinkOption[0] : new LinkOption[]{LinkOption.NOFOLLOW_LINKS}; return Files.isDirectory(path, getLinkOptions(isFollowLinks));
return Files.isDirectory(path, options);
} }
/** /**
@ -366,9 +396,8 @@ public class PathUtil {
return null; return null;
} }
final LinkOption[] options = isFollowLinks ? new LinkOption[0] : new LinkOption[]{LinkOption.NOFOLLOW_LINKS};
try { try {
return Files.readAttributes(path, BasicFileAttributes.class, options); return Files.readAttributes(path, BasicFileAttributes.class, getLinkOptions(isFollowLinks));
} catch (IOException e) { } catch (IOException e) {
throw new IORuntimeException(e); throw new IORuntimeException(e);
} }
@ -539,8 +568,7 @@ public class PathUtil {
if (null == path) { if (null == path) {
return false; return false;
} }
final LinkOption[] options = isFollowLinks ? new LinkOption[0] : new LinkOption[]{LinkOption.NOFOLLOW_LINKS}; return Files.isRegularFile(path, getLinkOptions(isFollowLinks));
return Files.isRegularFile(path, options);
} }
/** /**
@ -563,8 +591,7 @@ public class PathUtil {
* @since 5.5.3 * @since 5.5.3
*/ */
public static boolean exists(Path path, boolean isFollowLinks) { public static boolean exists(Path path, boolean isFollowLinks) {
final LinkOption[] options = isFollowLinks ? new LinkOption[0] : new LinkOption[]{LinkOption.NOFOLLOW_LINKS}; return Files.exists(path, getLinkOptions(isFollowLinks));
return Files.exists(path, options);
} }
/** /**
@ -712,4 +739,27 @@ public class PathUtil {
} }
} }
} }
/**
* 构建是否追踪软链的选项
*
* @param isFollowLinks 是否追踪软链
* @return 选项
* @since 5.8.23
*/
public static LinkOption[] getLinkOptions(final boolean isFollowLinks) {
return isFollowLinks ? new LinkOption[0] : new LinkOption[]{LinkOption.NOFOLLOW_LINKS};
}
/**
* 构建是否追踪软链的选项
*
* @param isFollowLinks 是否追踪软链
* @return 选项
* @since 5.8.23
*/
public static Set<FileVisitOption> getFileVisitOption(final boolean isFollowLinks) {
return isFollowLinks ? EnumSet.of(FileVisitOption.FOLLOW_LINKS) :
EnumSet.noneOf(FileVisitOption.class);
}
} }