com.swoval.files.DirectoryRegistry Maven / Gradle / Ivy
package com.swoval.files;
import static java.util.Map.Entry;
import com.swoval.functional.Filter;
import java.nio.file.Path;
import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
/**
* Tracks which directories the user wishes to monitor. This can be used to determine whether or not
* a path is part of the subtree specified by the set of paths registered by the user.
*/
interface DirectoryRegistry extends Filter, AutoCloseable {
/**
* Add the input directory to the list of registered directories.
*
* @param path the directory to register
* @param maxDepth controls how many levels of the children of the path should be monitored
* @return true if the directory has not been previously registered before or if the new maxDepth
* value is greater than the previous value.
*/
boolean addDirectory(final Path path, final int maxDepth);
/**
* The maximum depth of children of the path to accept.
*
* @param path the registered path
* @return the maximum depth of children if the path has been registered. Otherwise it returns
* Integer.MIN_VALUE.
*/
int maxDepthFor(final Path path);
/**
* Returns a map of Path -> maxDepth for each path.
*
* @return a map of Path -> maxDepth for each path.
*/
Map registered();
/**
* Remove the path from monitoring.
*
* @param path the path to stop monitoring.
*/
void removeDirectory(final Path path);
/**
* Returns true if this path is a prefix of a registered path. The intended use case is for the
* {@link NioPathWatcher} which always has the root directory as the base. This is so that we can
* ensure that if we register a directory that does not yet exist, that we will detect when the
* directory is created. For example, if we want to monitor '/foo/bar/baz', then we would accept
* '/foo/bar' as a valid prefix path, but we would not accept '/foo/buzz' as a valid prefix path.
*
* @param path the path to compare against the registered path
* @return true if the path is a prefix of a registered path.
*/
boolean acceptPrefix(final Path path);
@Override
void close();
}
class DirectoryRegistries {
private DirectoryRegistries() {}
static Filter toTypedPathFilter(final DirectoryRegistry registry) {
return new Filter() {
@Override
public boolean accept(final TypedPath typedPath) {
return registry.accept(typedPath.getPath());
}
};
}
}
class DirectoryRegistryImpl implements DirectoryRegistry {
private final LockableMap registeredDirectoriesByPath =
new LockableMap<>(new ConcurrentHashMap());
@Override
public boolean addDirectory(final Path path, final int maxDepth) {
if (registeredDirectoriesByPath.lock()) {
try {
final RegisteredDirectory registeredDirectory = registeredDirectoriesByPath.get(path);
if (registeredDirectory == null || maxDepth > registeredDirectory.maxDepth) {
registeredDirectoriesByPath.put(path, new RegisteredDirectory(path, maxDepth));
return true;
} else {
return false;
}
} finally {
registeredDirectoriesByPath.unlock();
}
} else {
return false;
}
}
@Override
public int maxDepthFor(final Path path) {
if (registeredDirectoriesByPath.lock()) {
try {
int maxDepth = Integer.MIN_VALUE;
final Iterator it = registeredDirectoriesByPath.values().iterator();
while (it.hasNext()) {
final RegisteredDirectory dir = it.next();
if (path.startsWith(dir.path)) {
final int depth = dir.path.equals(path) ? 0 : dir.path.relativize(path).getNameCount();
final int possibleMaxDepth = dir.maxDepth - depth;
if (possibleMaxDepth > maxDepth) {
maxDepth = possibleMaxDepth;
}
}
}
return maxDepth;
} finally {
registeredDirectoriesByPath.unlock();
}
} else {
return -1;
}
}
@Override
public Map registered() {
if (registeredDirectoriesByPath.lock()) {
try {
final Map result = new HashMap<>();
final Iterator it = registeredDirectoriesByPath.values().iterator();
while (it.hasNext()) {
final RegisteredDirectory dir = it.next();
result.put(dir.path, dir.maxDepth);
}
return result;
} finally {
registeredDirectoriesByPath.unlock();
}
} else {
return Collections.emptyMap();
}
}
@Override
public void removeDirectory(final Path path) {
if (registeredDirectoriesByPath.lock()) {
try {
registeredDirectoriesByPath.remove(path);
} finally {
registeredDirectoriesByPath.unlock();
}
}
}
private boolean acceptImpl(final Path path, final boolean acceptPrefix) {
if (registeredDirectoriesByPath.lock()) {
try {
boolean result = false;
final Iterator> it =
registeredDirectoriesByPath.iterator();
while (!result && it.hasNext()) {
final Entry entry = it.next();
final RegisteredDirectory registeredDirectory = entry.getValue();
final Path watchPath = entry.getKey();
if (acceptPrefix && watchPath.startsWith(path)) {
result = true;
} else if (path.startsWith(watchPath)) {
result = registeredDirectory.accept(path);
}
}
return result;
} finally {
registeredDirectoriesByPath.unlock();
}
} else {
return false;
}
}
@Override
public boolean accept(final Path path) {
return acceptImpl(path, false);
}
@Override
public boolean acceptPrefix(final Path path) {
return acceptImpl(path, true);
}
@Override
public void close() {
registeredDirectoriesByPath.clear();
}
@Override
public String toString() {
if (registeredDirectoriesByPath.lock()) {
try {
final StringBuilder result = new StringBuilder();
result.append("DirectoryRegistry:\n");
final Iterator it = registeredDirectoriesByPath.values().iterator();
while (it.hasNext()) {
result.append(" ");
result.append(it.next());
result.append('\n');
}
return result.toString();
} finally {
registeredDirectoriesByPath.unlock();
}
} else {
return "";
}
}
private static class RegisteredDirectory {
final Path path;
final int maxDepth;
final int compMaxDepth;
RegisteredDirectory(final Path path, final int maxDepth) {
this.path = path;
this.maxDepth = maxDepth;
compMaxDepth = maxDepth == Integer.MAX_VALUE ? maxDepth : maxDepth + 1;
}
public boolean accept(final Path path) {
return path.startsWith(this.path)
&& (path.equals(this.path) || this.path.relativize(path).getNameCount() <= compMaxDepth);
}
@Override
public String toString() {
return "RegisteredDirectory(path = " + path + ", depth = " + maxDepth + ")";
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy