jvmMain.jdk7.kotlin.io.path.PathTreeWalk.kt Maven / Gradle / Ivy
/*
* Copyright 2010-2021 JetBrains s.r.o. and Kotlin Programming Language contributors.
* Use of this source code is governed by the Apache 2.0 license that can be found in the license/LICENSE.txt file.
*/
package kotlin.io.path
import java.io.IOException
import java.nio.file.*
import java.nio.file.attribute.BasicFileAttributes
/**
* This class is intended to implement different file traversal methods.
* It allows to iterate through all files inside a given directory.
* The order in which sibling files are visited is unspecified.
*
* If the file located by this path is not a directory, the walker iterates only it.
* If the file located by this path does not exist, the walker iterates nothing, i.e. it's equivalent to an empty sequence.
*/
@ExperimentalPathApi
internal class PathTreeWalk(
private val start: Path,
private val options: Array
) : Sequence {
private val followLinks: Boolean
get() = options.contains(PathWalkOption.FOLLOW_LINKS)
private val linkOptions: Array
get() = LinkFollowing.toLinkOptions(followLinks)
private val includeDirectories: Boolean
get() = options.contains(PathWalkOption.INCLUDE_DIRECTORIES)
private val isBFS: Boolean
get() = options.contains(PathWalkOption.BREADTH_FIRST)
override fun iterator(): Iterator = if (isBFS) bfsIterator() else dfsIterator()
private suspend inline fun SequenceScope.yieldIfNeeded(
node: PathNode,
entriesReader: DirectoryEntriesReader,
entriesAction: (List) -> Unit
) {
val path = node.path
if (node.parent != null) {
// Check entries other than the starting path of traversal
path.checkFileName()
}
if (path.isDirectory(*linkOptions)) {
if (node.createsCycle())
throw FileSystemLoopException(path.toString())
if (includeDirectories)
yield(path)
if (path.isDirectory(*linkOptions)) // make sure the path was not deleted after it was yielded
entriesAction(entriesReader.readEntries(node))
} else if (path.exists(LinkOption.NOFOLLOW_LINKS)) {
yield(path)
}
}
private fun dfsIterator() = iterator {
// Stack of directory iterators, beginning from the start directory
val stack = ArrayDeque()
val entriesReader = DirectoryEntriesReader(followLinks)
val startNode = PathNode(start, keyOf(start, linkOptions), null)
yieldIfNeeded(startNode, entriesReader) { entries ->
startNode.contentIterator = entries.iterator()
stack.addLast(startNode)
}
while (stack.isNotEmpty()) {
val topNode = stack.last()
val topIterator = topNode.contentIterator!!
if (topIterator.hasNext()) {
val pathNode = topIterator.next()
yieldIfNeeded(pathNode, entriesReader) { entries ->
pathNode.contentIterator = entries.iterator()
stack.addLast(pathNode)
}
} else {
// There is nothing more on the top of the stack, go back
stack.removeLast()
}
}
}
private fun bfsIterator() = iterator {
// Queue of entries to be visited.
val queue = ArrayDeque()
val entriesReader = DirectoryEntriesReader(followLinks)
queue.addLast(PathNode(start, keyOf(start, linkOptions), null))
while (queue.isNotEmpty()) {
val pathNode = queue.removeFirst()
yieldIfNeeded(pathNode, entriesReader) { entries ->
queue.addAll(entries)
}
}
}
}
private fun keyOf(path: Path, linkOptions: Array): Any? {
return try {
path.readAttributes(*linkOptions).fileKey()
} catch (exception: Throwable) {
null
}
}
private class PathNode(val path: Path, val key: Any?, val parent: PathNode?) {
var contentIterator: Iterator? = null
}
private fun PathNode.createsCycle(): Boolean {
var ancestor = parent
while (ancestor != null) {
if (ancestor.key != null && key != null) {
if (ancestor.key == key)
return true
} else {
try {
if (ancestor.path.isSameFileAs(path))
return true
} catch (_: IOException) { // ignore
} catch (_: SecurityException) { // ignore
}
}
ancestor = ancestor.parent
}
return false
}
internal object LinkFollowing {
private val nofollowLinkOption = arrayOf(LinkOption.NOFOLLOW_LINKS)
private val followLinkOption = emptyArray()
private val nofollowVisitOption = emptySet()
private val followVisitOption = setOf(FileVisitOption.FOLLOW_LINKS)
fun toLinkOptions(followLinks: Boolean): Array =
if (followLinks) followLinkOption else nofollowLinkOption
fun toVisitOptions(followLinks: Boolean): Set =
if (followLinks) followVisitOption else nofollowVisitOption
}
private class DirectoryEntriesReader(val followLinks: Boolean) : SimpleFileVisitor() {
private var directoryNode: PathNode? = null
private var entries = ArrayDeque()
fun readEntries(directoryNode: PathNode): List {
this.directoryNode = directoryNode
Files.walkFileTree(directoryNode.path, LinkFollowing.toVisitOptions(followLinks), 1, this)
entries.removeFirst()
return entries.also { entries = ArrayDeque() }
}
override fun preVisitDirectory(dir: Path, attrs: BasicFileAttributes): FileVisitResult {
val directoryEntry = PathNode(dir, attrs.fileKey(), directoryNode)
entries.add(directoryEntry)
return super.preVisitDirectory(dir, attrs)
}
override fun visitFile(file: Path, attrs: BasicFileAttributes): FileVisitResult {
val fileEntry = PathNode(file, null, directoryNode)
entries.add(fileEntry)
return super.visitFile(file, attrs)
}
}