com.swoval.files.NioPathWatcher Maven / Gradle / Ivy
package com.swoval.files;
import static com.swoval.files.PathWatchers.Event.Kind.Create;
import static com.swoval.files.PathWatchers.Event.Kind.Delete;
import static com.swoval.files.PathWatchers.Event.Kind.Overflow;
import static com.swoval.functional.Filters.AllPass;
import static java.util.Map.Entry;
import com.swoval.files.FileTreeDataViews.CacheObserver;
import com.swoval.files.FileTreeDataViews.Converter;
import com.swoval.files.FileTreeViews.Observer;
import com.swoval.files.PathWatchers.Event;
import com.swoval.files.PathWatchers.Event.Kind;
import com.swoval.files.PathWatchers.Overflow;
import com.swoval.functional.Consumer;
import com.swoval.functional.Either;
import com.swoval.functional.Filter;
import com.swoval.functional.Filters;
import com.swoval.logging.Logger;
import com.swoval.logging.Loggers;
import com.swoval.logging.Loggers.Level;
import java.io.IOException;
import java.nio.file.FileSystems;
import java.nio.file.NoSuchFileException;
import java.nio.file.Path;
import java.util.ArrayList;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Set;
import java.util.concurrent.atomic.AtomicBoolean;
class RootDirectories extends LockableMap> {}
/** Provides a PathWatcher that is backed by a {@link java.nio.file.WatchService}. */
class NioPathWatcher implements PathWatcher, AutoCloseable {
private final AtomicBoolean closed = new AtomicBoolean(false);
private final Observers observers = new Observers<>();
private final RootDirectories rootDirectories = new RootDirectories();
private final DirectoryRegistry directoryRegistry;
private final Converter converter;
private final Logger logger;
private CacheObserver updateCacheObserver(final List events) {
return new CacheObserver() {
@Override
@SuppressWarnings("EmptyCatchBlock")
public void onCreate(final FileTreeDataViews.Entry newEntry) {
events.add(new Event(newEntry.getTypedPath(), Create));
try {
final Iterator it =
FileTreeViews.list(
newEntry.getTypedPath().getPath(),
0,
new Filter() {
@Override
public boolean accept(final TypedPath typedPath) {
return directoryRegistry.accept(typedPath.getPath());
}
})
.iterator();
while (it.hasNext()) {
final TypedPath tp = it.next();
events.add(new Event(tp, Create));
}
} catch (final IOException e) {
// This likely means the directory was deleted, which should be handle by the downstream
// NioPathWatcherService.
}
}
@Override
public void onDelete(final FileTreeDataViews.Entry oldEntry) {
if (oldEntry.getValue().isRight()) {
if (Loggers.shouldLog(logger, Level.DEBUG))
logger.debug(this + " closing key for " + oldEntry.getTypedPath().getPath());
oldEntry.getValue().get().close();
}
events.add(new Event(oldEntry.getTypedPath(), Delete));
}
@Override
public void onUpdate(
final FileTreeDataViews.Entry oldEntry,
final FileTreeDataViews.Entry newEntry) {}
@Override
public void onError(final IOException exception) {}
};
}
private final NioPathWatcherService service;
NioPathWatcher(
final DirectoryRegistry directoryRegistry,
final RegisterableWatchService watchService,
final Logger logger)
throws InterruptedException {
this.directoryRegistry = directoryRegistry;
this.logger = logger;
this.service =
new NioPathWatcherService(
new Consumer>() {
@Override
public void accept(final Either either) {
if (!closed.get()) {
if (either.isRight()) {
final Event event = either.get();
handleEvent(event);
} else {
handleOverflow(Either.leftProjection(either).getValue());
}
}
}
},
watchService,
logger);
this.converter =
new Converter() {
@Override
public WatchedDirectory apply(final TypedPath typedPath) {
return typedPath.isDirectory() && !typedPath.isSymbolicLink()
? Either.getOrElse(
service.register(typedPath.getPath()), WatchedDirectories.INVALID)
: WatchedDirectories.INVALID;
}
};
}
/**
* Similar to register, but tracks all of the new files found in the directory. It polls the
* directory until the contents stop changing to ensure that a callback is fired for each path in
* the newly created directory (up to the maxDepth). The assumption is that once the callback is
* fired for the path, it is safe to assume that no event for a new file in the directory is
* missed. Without the polling, it would be possible that a new file was created in the directory
* before we registered it with the watch service. If this happened, then no callback would be
* invoked for that file.
*
* @param typedPath The newly created directory to add
*/
void add(final TypedPath typedPath, final List events) {
if (directoryRegistry.maxDepthFor(typedPath.getPath()) >= 0
|| directoryRegistry.acceptPrefix(typedPath.getPath())) {
final CachedDirectory dir = getOrAdd(typedPath.getPath());
if (dir != null) {
update(dir, typedPath, events, true);
}
}
}
private void remove(final Path path, List events) {
final CachedDirectory root = rootDirectories.remove(path);
final CachedDirectory dir = root == null ? find(path) : root;
if (dir != null) remove(dir, path, events);
}
private void remove(
final CachedDirectory cachedDirectory,
final Path path,
final List events) {
final List> toCancel = cachedDirectory.remove(path);
if (path == null || path == cachedDirectory.getPath()) toCancel.add(cachedDirectory.getEntry());
final Iterator> it = toCancel.iterator();
while (it.hasNext()) {
final FileTreeDataViews.Entry entry = it.next();
final Either either = entry.getValue();
if (either.isRight()) {
if (events != null) {
final TypedPath typedPath =
TypedPaths.get(
entry.getTypedPath().getPath(),
TypedPaths.getKind(entry.getTypedPath()) | Entries.NONEXISTENT);
events.add(new Event(typedPath, Delete));
}
either.get().close();
}
}
}
@Override
public Either register(final Path path, final int maxDepth) {
final Path absolutePath = path.isAbsolute() ? path : path.toAbsolutePath();
final int existingMaxDepth = directoryRegistry.maxDepthFor(absolutePath);
boolean result = existingMaxDepth < maxDepth;
final TypedPath typedPath = TypedPaths.get(absolutePath);
if (result) {
directoryRegistry.addDirectory(absolutePath, maxDepth);
}
final CachedDirectory dir = getOrAdd(absolutePath);
final List events = new ArrayList<>();
if (dir != null) {
if (result) {
Path current = typedPath.getPath();
Path root = dir.getEntry().getTypedPath().getPath();
Path relative = root.relativize(current);
Path lastPath = root;
final List> entries =
dir.listEntries(root, relative.getNameCount() + 1, AllPass);
Iterator> it = entries.iterator();
while (it.hasNext()) {
FileTreeDataViews.Entry entry = it.next();
Path next = entry.getTypedPath().getPath();
if (current.startsWith(next)
&& (next.toString().length() > lastPath.toString().length())) {
lastPath = next;
}
}
final List> directories =
dir.listEntries(lastPath, -1, AllPass);
if (!directories.isEmpty() && directories.get(0).getValue().isRight()) {
update(dir, TypedPaths.get(lastPath), events, true);
}
} else {
final List> directories =
dir.listEntries(typedPath.getPath(), -1, AllPass);
if (!directories.isEmpty() && directories.get(0).getValue().isRight()) {
update(dir, typedPath, events, true);
}
}
}
/*
if (false) {
// It is not clear that any of these callbacks should be
// ran at all. If the path watcher is working correctly
// then presumably it has already reported all of the change
// events. Creation events just indicate that after the
// registration we have added new paths to watch. These
// should not be reported back to users. Out of paranoia,
// we are going to continue running the callbacks for non
// Creation events because at the time of writing I am not
// 100% sure that this won't break any downstream users of
// the library. That being said, it would probably be safe
// to remove this entire code block and not run any of the
// callbacks at all upon registration.
//
final List eventsMinusCreate = new ArrayList<>();
for (int i = 0; i < events.size(); i++) {
if (events.get(i).getKind() != Kind.Create) {
eventsMinusCreate.add(events.get(i));
}
}
runCallbacks(eventsMinusCreate);
}
*/
if (Loggers.shouldLog(logger, Level.DEBUG))
logger.debug(this + " registered " + path + " with max depth " + maxDepth);
return Either.right(result);
}
private CachedDirectory find(final Path rawPath) {
return find(rawPath, null);
}
private CachedDirectory find(final Path rawPath, final List toRemove) {
final Path path = rawPath == null ? getRoot() : rawPath;
assert (path != null);
if (rootDirectories.lock()) {
try {
final Iterator>> it =
rootDirectories.iterator();
CachedDirectory result = null;
while (result == null && it.hasNext()) {
final Entry> entry = it.next();
final Path root = entry.getKey();
if (path.startsWith(root)) {
result = entry.getValue();
} else if (toRemove != null && root.startsWith(path) && !path.equals(root)) {
toRemove.add(root);
}
}
return result;
} finally {
rootDirectories.unlock();
}
} else {
return null;
}
}
private Path getRoot() {
/* This may not make sense on windows which has multiple root directories, but at least it
* will return something.
*/
final Iterator it = FileSystems.getDefault().getRootDirectories().iterator();
return it.next();
}
private CachedDirectory findOrAddRoot(final Path rawPath) {
final List toRemove = new ArrayList<>();
CachedDirectory result = find(rawPath, toRemove);
if (result == null) {
/*
* We want to monitor the parent in case the file is deleted.
*/
final Path parent = rawPath.getParent();
Path path = parent == null ? getRoot() : parent;
assert (path != null);
boolean init = false;
while (!init && path != null) {
try {
result =
new CachedDirectoryImpl<>(
TypedPaths.get(path),
converter,
Integer.MAX_VALUE,
new Filter() {
@Override
public boolean accept(final TypedPath typedPath) {
return typedPath.isDirectory()
&& !typedPath.isSymbolicLink()
&& directoryRegistry.acceptPrefix(typedPath.getPath());
}
},
false)
.init();
init = true;
rootDirectories.put(path, result);
} catch (final IOException e) {
path = path.getParent();
}
}
}
final Iterator toRemoveIterator = toRemove.iterator();
while (toRemoveIterator.hasNext()) {
rootDirectories.remove(toRemoveIterator.next());
}
return result;
}
private CachedDirectory getOrAdd(final Path path) {
CachedDirectory result = null;
if (rootDirectories.lock()) {
try {
if (!closed.get()) {
result = findOrAddRoot(path);
}
} finally {
rootDirectories.unlock();
}
}
return result;
}
@Override
@SuppressWarnings("EmptyCatchBlock")
public void unregister(final Path path) {
final Path absolutePath = path.isAbsolute() ? path : path.toAbsolutePath();
directoryRegistry.removeDirectory(absolutePath);
if (rootDirectories.lock()) {
try {
final CachedDirectory dir = find(absolutePath);
if (dir != null) {
final int depth = dir.getPath().relativize(absolutePath).getNameCount();
List> toRemove =
dir.listEntries(
depth,
new Filter>() {
@Override
public boolean accept(final FileTreeDataViews.Entry entry) {
return !directoryRegistry.acceptPrefix(entry.getTypedPath().getPath());
}
});
final Iterator> it = toRemove.iterator();
while (it.hasNext()) {
final FileTreeDataViews.Entry entry = it.next();
if (!directoryRegistry.acceptPrefix(entry.getTypedPath().getPath())) {
remove(dir, entry.getTypedPath().getPath(), null);
}
}
rootDirectories.remove(dir.getPath());
}
} finally {
rootDirectories.unlock();
}
}
if (Loggers.shouldLog(logger, Level.DEBUG)) logger.debug(this + " unregistered " + path);
}
@Override
public void close() {
if (closed.compareAndSet(false, true)) {
service.close();
rootDirectories.clear();
}
}
@SuppressWarnings("EmptyCatchBlock")
private void update(
final CachedDirectory dir,
final TypedPath typedPath,
final List events,
final boolean rescanDirectories) {
try {
dir.update(typedPath, rescanDirectories).observe(updateCacheObserver(events));
} catch (final NoSuchFileException e) {
remove(dir, typedPath.getPath(), events);
final TypedPath newTypedPath = TypedPaths.get(typedPath.getPath());
events.add(new Event(newTypedPath, newTypedPath.exists() ? Kind.Modify : Kind.Delete));
final CachedDirectory root = rootDirectories.remove(typedPath.getPath());
if (root != null) {
final Iterator> it =
root.listEntries(Integer.MAX_VALUE, AllPass).iterator();
while (it.hasNext()) {
it.next().getValue().get().close();
}
}
} catch (final IOException e) {
}
}
private void handleOverflow(final Overflow overflow) {
final Path path = overflow.getPath();
if (Loggers.shouldLog(logger, Level.DEBUG))
logger.debug(this + " received overflow for " + path);
final List events = new ArrayList<>();
if (rootDirectories.lock()) {
try {
final CachedDirectory root = find(path);
if (root != null) {
try {
final Iterator it =
FileTreeViews.list(
path,
0,
new Filter() {
@Override
public boolean accept(TypedPath typedPath) {
return typedPath.isDirectory()
&& directoryRegistry.acceptPrefix(typedPath.getPath());
}
})
.iterator();
while (it.hasNext()) {
final TypedPath file = it.next();
add(file, events);
}
} catch (final IOException e) {
final List> removed = root.remove(path);
final Iterator> removedIt =
removed.iterator();
while (removedIt.hasNext()) {
events.add(
new Event(Entries.setExists(removedIt.next(), false).getTypedPath(), Delete));
}
}
}
} finally {
rootDirectories.unlock();
}
}
final TypedPath tp = TypedPaths.get(path);
events.add(new Event(tp, tp.exists() ? Overflow : Delete));
runCallbacks(events);
}
private void runCallbacks(final List events) {
final Iterator it = events.iterator();
final Set handled = new HashSet<>();
while (it.hasNext()) {
final Event event = it.next();
final Path path = event.getTypedPath().getPath();
if (directoryRegistry.accept(path) && handled.add(path)) {
observers.onNext(new Event(TypedPaths.get(path), event.getKind()));
}
}
}
private void handleEvent(final Event event) {
if (Loggers.shouldLog(logger, Level.DEBUG)) logger.debug(this + " received event " + event);
final List events = new ArrayList<>();
if (!closed.get() && rootDirectories.lock()) {
try {
if (directoryRegistry.acceptPrefix(event.getTypedPath().getPath())) {
final boolean isDelete = event.getKind() == Delete;
final TypedPath typedPath = TypedPaths.get(event.getTypedPath().getPath());
if (isDelete) remove(typedPath.getPath(), events);
if (typedPath.exists()) {
if (typedPath.isDirectory() && !typedPath.isSymbolicLink()) add(typedPath, events);
events.add(event);
} else if (!isDelete) remove(typedPath.getPath(), events);
else events.add(event);
}
} finally {
rootDirectories.unlock();
}
}
runCallbacks(events);
}
@Override
public int addObserver(final Observer super Event> observer) {
return observers.addObserver(observer);
}
@Override
public void removeObserver(final int handle) {
observers.removeObserver(handle);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy