org.eclipse.jetty.util.PathWatcher Maven / Gradle / Ivy
Show all versions of jetty-util Show documentation
//
// ========================================================================
// Copyright (c) 1995-2022 Mort Bay Consulting Pty Ltd and others.
// ------------------------------------------------------------------------
// All rights reserved. This program and the accompanying materials
// are made available under the terms of the Eclipse Public License v1.0
// and Apache License v2.0 which accompanies this distribution.
//
// The Eclipse Public License is available at
// http://www.eclipse.org/legal/epl-v10.html
//
// The Apache License v2.0 is available at
// http://www.opensource.org/licenses/apache2.0.php
//
// You may elect to redistribute this code under either of these licenses.
// ========================================================================
//
package org.eclipse.jetty.util;
import java.io.File;
import java.io.IOException;
import java.nio.file.ClosedWatchServiceException;
import java.nio.file.FileSystem;
import java.nio.file.FileSystems;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.PathMatcher;
import java.nio.file.WatchEvent;
import java.nio.file.WatchKey;
import java.nio.file.WatchService;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.EventListener;
import java.util.HashSet;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Scanner;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.TimeUnit;
import java.util.function.Predicate;
import java.util.stream.Stream;
import org.eclipse.jetty.util.component.AbstractLifeCycle;
import org.eclipse.jetty.util.log.Log;
import org.eclipse.jetty.util.log.Logger;
import static java.nio.file.StandardWatchEventKinds.ENTRY_CREATE;
import static java.nio.file.StandardWatchEventKinds.ENTRY_DELETE;
import static java.nio.file.StandardWatchEventKinds.ENTRY_MODIFY;
/**
* Watch a Path (and sub directories) for Path changes.
*
* Suitable replacement for the old {@link Scanner} implementation.
*
* Allows for configured Excludes and Includes using {@link FileSystem#getPathMatcher(String)} syntax.
*
* Reports activity via registered {@link Listener}s
*/
public class PathWatcher extends AbstractLifeCycle implements Runnable
{
public static class Config implements Predicate
{
public static final int UNLIMITED_DEPTH = -9999;
private static final String PATTERN_SEP;
static
{
String sep = File.separator;
if (File.separatorChar == '\\')
{
sep = "\\\\";
}
PATTERN_SEP = sep;
}
protected final Config parent;
protected final Path path;
protected final IncludeExcludeSet includeExclude;
protected int recurseDepth = 0; // 0 means no sub-directories are scanned
protected boolean excludeHidden = false;
protected long pauseUntil;
public Config(Path path)
{
this(path, null);
}
public Config(Path path, Config parent)
{
this.parent = parent;
this.includeExclude = parent == null ? new IncludeExcludeSet<>(PathMatcherSet.class) : parent.includeExclude;
Path dir = path;
if (!Files.exists(path))
throw new IllegalStateException("Path does not exist: " + path);
if (!Files.isDirectory(path))
{
dir = path.getParent();
includeExclude.include(new ExactPathMatcher(path));
setRecurseDepth(0);
}
this.path = dir;
}
public Config getParent()
{
return parent;
}
public void setPauseUntil(long time)
{
if (time > pauseUntil)
pauseUntil = time;
}
public boolean isPaused(long now)
{
if (pauseUntil == 0)
return false;
if (pauseUntil > now)
{
if (LOG.isDebugEnabled())
LOG.debug("PAUSED {}", this);
return true;
}
if (LOG.isDebugEnabled())
LOG.debug("unpaused {}", this);
pauseUntil = 0;
return false;
}
/**
* Add an exclude PathMatcher
*
* @param matcher the path matcher for this exclude
*/
public void addExclude(PathMatcher matcher)
{
includeExclude.exclude(matcher);
}
/**
* Add an exclude PathMatcher.
*
* Note: this pattern is FileSystem specific (so use "/" for Linux and OSX, and "\\" for Windows)
*
* @param syntaxAndPattern the PathMatcher syntax and pattern to use
* @see FileSystem#getPathMatcher(String) for detail on syntax and pattern
*/
public void addExclude(final String syntaxAndPattern)
{
if (LOG.isDebugEnabled())
LOG.debug("Adding exclude: [{}]", syntaxAndPattern);
addExclude(path.getFileSystem().getPathMatcher(syntaxAndPattern));
}
/**
* Add a glob:
syntax pattern exclude reference in a directory relative, os neutral, pattern.
*
*
* On Linux:
* Config config = new Config(Path("/home/user/example"));
* config.addExcludeGlobRelative("*.war") => "glob:/home/user/example/*.war"
*
* On Windows
* Config config = new Config(Path("D:/code/examples"));
* config.addExcludeGlobRelative("*.war") => "glob:D:\\code\\examples\\*.war"
*
*
*
* @param pattern the pattern, in unixy format, relative to config.dir
*/
public void addExcludeGlobRelative(String pattern)
{
addExclude(toGlobPattern(path, pattern));
}
/**
* Exclude hidden files and hidden directories
*/
public void addExcludeHidden()
{
if (!excludeHidden)
{
if (LOG.isDebugEnabled())
{
LOG.debug("Adding hidden files and directories to exclusions");
}
excludeHidden = true;
}
}
/**
* Add multiple exclude PathMatchers
*
* @param syntaxAndPatterns the list of PathMatcher syntax and patterns to use
* @see FileSystem#getPathMatcher(String) for detail on syntax and pattern
*/
public void addExcludes(List syntaxAndPatterns)
{
for (String syntaxAndPattern : syntaxAndPatterns)
{
addExclude(syntaxAndPattern);
}
}
/**
* Add an include PathMatcher
*
* @param matcher the path matcher for this include
*/
public void addInclude(PathMatcher matcher)
{
includeExclude.include(matcher);
}
/**
* Add an include PathMatcher
*
* @param syntaxAndPattern the PathMatcher syntax and pattern to use
* @see FileSystem#getPathMatcher(String) for detail on syntax and pattern
*/
public void addInclude(String syntaxAndPattern)
{
if (LOG.isDebugEnabled())
{
LOG.debug("Adding include: [{}]", syntaxAndPattern);
}
addInclude(path.getFileSystem().getPathMatcher(syntaxAndPattern));
}
/**
* Add a glob:
syntax pattern reference in a directory relative, os neutral, pattern.
*
*
* On Linux:
* Config config = new Config(Path("/home/user/example"));
* config.addIncludeGlobRelative("*.war") => "glob:/home/user/example/*.war"
*
* On Windows
* Config config = new Config(Path("D:/code/examples"));
* config.addIncludeGlobRelative("*.war") => "glob:D:\\code\\examples\\*.war"
*
*
*
* @param pattern the pattern, in unixy format, relative to config.dir
*/
public void addIncludeGlobRelative(String pattern)
{
addInclude(toGlobPattern(path, pattern));
}
/**
* Add multiple include PathMatchers
*
* @param syntaxAndPatterns the list of PathMatcher syntax and patterns to use
* @see FileSystem#getPathMatcher(String) for detail on syntax and pattern
*/
public void addIncludes(List syntaxAndPatterns)
{
for (String syntaxAndPattern : syntaxAndPatterns)
{
addInclude(syntaxAndPattern);
}
}
/**
* Build a new config from a this configuration.
*
* Useful for working with sub-directories that also need to be watched.
*
* @param dir the directory to build new Config from (using this config as source of includes/excludes)
* @return the new Config
*/
public Config asSubConfig(Path dir)
{
Config subconfig = new Config(dir, this);
if (dir == this.path)
throw new IllegalStateException("sub " + dir.toString() + " of " + this);
if (this.recurseDepth == UNLIMITED_DEPTH)
subconfig.recurseDepth = UNLIMITED_DEPTH;
else
subconfig.recurseDepth = this.recurseDepth - (dir.getNameCount() - this.path.getNameCount());
if (LOG.isDebugEnabled())
LOG.debug("subconfig {} of {}", subconfig, path);
return subconfig;
}
public int getRecurseDepth()
{
return recurseDepth;
}
public boolean isRecurseDepthUnlimited()
{
return this.recurseDepth == UNLIMITED_DEPTH;
}
public Path getPath()
{
return this.path;
}
public Path resolve(Path path)
{
if (Files.isDirectory(this.path))
return this.path.resolve(path);
if (Files.exists(this.path))
return this.path;
return path;
}
@Override
public boolean test(Path path)
{
if (excludeHidden && isHidden(path))
{
if (LOG.isDebugEnabled())
LOG.debug("test({}) -> [Hidden]", toShortPath(path));
return false;
}
if (!path.startsWith(this.path))
{
if (LOG.isDebugEnabled())
LOG.debug("test({}) -> [!child {}]", toShortPath(path), this.path);
return false;
}
if (recurseDepth != UNLIMITED_DEPTH)
{
int depth = path.getNameCount() - this.path.getNameCount() - 1;
if (depth > recurseDepth)
{
if (LOG.isDebugEnabled())
LOG.debug("test({}) -> [depth {}>{}]", toShortPath(path), depth, recurseDepth);
return false;
}
}
boolean matched = includeExclude.test(path);
if (LOG.isDebugEnabled())
LOG.debug("test({}) -> {}", toShortPath(path), matched);
return matched;
}
/**
* Set the recurse depth for the directory scanning.
*
* -999 indicates arbitrarily deep recursion, 0 indicates no recursion, 1 is only one directory deep, and so on.
*
* @param depth the number of directories deep to recurse
*/
public void setRecurseDepth(int depth)
{
this.recurseDepth = depth;
}
private String toGlobPattern(Path path, String subPattern)
{
StringBuilder s = new StringBuilder();
s.append("glob:");
boolean needDelim = false;
// Add root (aka "C:\" for Windows)
Path root = path.getRoot();
if (root != null)
{
if (LOG.isDebugEnabled())
{
LOG.debug("Path: {} -> Root: {}", path, root);
}
for (char c : root.toString().toCharArray())
{
if (c == '\\')
{
s.append(PATTERN_SEP);
}
else
{
s.append(c);
}
}
}
else
{
needDelim = true;
}
// Add the individual path segments
for (Path segment : path)
{
if (needDelim)
{
s.append(PATTERN_SEP);
}
s.append(segment);
needDelim = true;
}
// Add the sub pattern (if specified)
if ((subPattern != null) && (subPattern.length() > 0))
{
if (needDelim)
{
s.append(PATTERN_SEP);
}
for (char c : subPattern.toCharArray())
{
if (c == '/')
{
s.append(PATTERN_SEP);
}
else
{
s.append(c);
}
}
}
return s.toString();
}
DirAction handleDir(Path path)
{
try
{
if (!Files.isDirectory(path))
return DirAction.IGNORE;
if (excludeHidden && isHidden(path))
return DirAction.IGNORE;
if (getRecurseDepth() == 0)
return DirAction.WATCH;
return DirAction.ENTER;
}
catch (Exception e)
{
LOG.ignore(e);
return DirAction.IGNORE;
}
}
public boolean isHidden(Path path)
{
try
{
if (!path.startsWith(this.path))
return true;
for (int i = this.path.getNameCount(); i < path.getNameCount(); i++)
{
if (path.getName(i).toString().startsWith("."))
{
return true;
}
}
return Files.exists(path) && Files.isHidden(path);
}
catch (IOException e)
{
LOG.ignore(e);
return false;
}
}
public String toShortPath(Path path)
{
if (!path.startsWith(this.path))
return path.toString();
return this.path.relativize(path).toString();
}
@Override
public String toString()
{
StringBuilder s = new StringBuilder();
s.append(path).append(" [depth=");
if (recurseDepth == UNLIMITED_DEPTH)
s.append("UNLIMITED");
else
s.append(recurseDepth);
s.append(']');
return s.toString();
}
}
public enum DirAction
{
IGNORE, WATCH, ENTER
}
/**
* Listener for path change events
*/
public interface Listener extends EventListener
{
void onPathWatchEvent(PathWatchEvent event);
}
/**
* EventListListener
*
* Listener that reports accumulated events in one shot
*/
public interface EventListListener extends EventListener
{
void onPathWatchEvents(List events);
}
/**
* PathWatchEvent
*
* Represents a file event. Reported to registered listeners.
*/
public class PathWatchEvent
{
private final Path path;
private final PathWatchEventType type;
private final Config config;
long checked;
long modified;
long length;
public PathWatchEvent(Path path, PathWatchEventType type, Config config)
{
this.path = path;
this.type = type;
this.config = config;
checked = TimeUnit.NANOSECONDS.toMillis(System.nanoTime());
check();
}
public Config getConfig()
{
return config;
}
public PathWatchEvent(Path path, WatchEvent event, Config config)
{
this.path = path;
if (event.kind() == ENTRY_CREATE)
{
this.type = PathWatchEventType.ADDED;
}
else if (event.kind() == ENTRY_DELETE)
{
this.type = PathWatchEventType.DELETED;
}
else if (event.kind() == ENTRY_MODIFY)
{
this.type = PathWatchEventType.MODIFIED;
}
else
{
this.type = PathWatchEventType.UNKNOWN;
}
this.config = config;
checked = TimeUnit.NANOSECONDS.toMillis(System.nanoTime());
check();
}
private void check()
{
if (Files.exists(path))
{
try
{
modified = Files.getLastModifiedTime(path).toMillis();
length = Files.size(path);
}
catch (IOException e)
{
modified = -1;
length = -1;
}
}
else
{
modified = -1;
length = -1;
}
}
public boolean isQuiet(long now, long quietTime)
{
long lastModified = modified;
long lastLength = length;
check();
if (lastModified == modified && lastLength == length)
return (now - checked) >= quietTime;
checked = now;
return false;
}
public long toQuietCheck(long now, long quietTime)
{
long check = quietTime - (now - checked);
if (check <= 0)
return quietTime;
return check;
}
public void modified()
{
long now = TimeUnit.NANOSECONDS.toMillis(System.nanoTime());
checked = now;
check();
config.setPauseUntil(now + getUpdateQuietTimeMillis());
}
/**
* @see java.lang.Object#equals(java.lang.Object)
*/
@Override
public boolean equals(Object obj)
{
if (this == obj)
{
return true;
}
if (obj == null)
{
return false;
}
if (getClass() != obj.getClass())
{
return false;
}
PathWatchEvent other = (PathWatchEvent)obj;
if (path == null)
{
if (other.path != null)
{
return false;
}
}
else if (!path.equals(other.path))
{
return false;
}
return type == other.type;
}
public Path getPath()
{
return path;
}
public PathWatchEventType getType()
{
return type;
}
@Deprecated
public int getCount()
{
return 1;
}
/**
* @see java.lang.Object#hashCode()
*/
@Override
public int hashCode()
{
final int prime = 31;
int result = 1;
result = (prime * result) + ((path == null) ? 0 : path.hashCode());
result = (prime * result) + ((type == null) ? 0 : type.hashCode());
return result;
}
/**
* @see java.lang.Object#toString()
*/
@Override
public String toString()
{
return String.format("PathWatchEvent[%8s|%s]", type, path);
}
}
/**
* PathWatchEventType
*
* Type of an event
*/
public enum PathWatchEventType
{
ADDED, DELETED, MODIFIED, UNKNOWN
}
private static final boolean IS_WINDOWS;
static
{
String os = System.getProperty("os.name");
if (os == null)
{
IS_WINDOWS = false;
}
else
{
String osl = os.toLowerCase(Locale.ENGLISH);
IS_WINDOWS = osl.contains("windows");
}
}
static final Logger LOG = Log.getLogger(PathWatcher.class);
@SuppressWarnings("unchecked")
protected static WatchEvent cast(WatchEvent> event)
{
return (WatchEvent)event;
}
private static final WatchEvent.Kind>[] WATCH_EVENT_KINDS = {ENTRY_CREATE, ENTRY_DELETE, ENTRY_MODIFY};
private static final WatchEvent.Kind>[] WATCH_DIR_KINDS = {ENTRY_CREATE, ENTRY_DELETE};
private WatchService watchService;
private final List configs = new ArrayList<>();
private final Map keys = new ConcurrentHashMap<>();
private final List listeners = new CopyOnWriteArrayList<>(); //a listener may modify the listener list directly or by stopping the PathWatcher
private final Map pending = new LinkedHashMap<>(32, (float)0.75, false);
private final List events = new ArrayList<>();
/**
* Update Quiet Time - set to 1000 ms as default (a lower value in Windows is not supported)
*/
private long updateQuietTimeDuration = 1000;
private TimeUnit updateQuietTimeUnit = TimeUnit.MILLISECONDS;
private Thread thread;
private boolean _notifyExistingOnStart = true;
/**
* Construct new PathWatcher
*/
public PathWatcher()
{
}
public Collection getConfigs()
{
return configs;
}
/**
* Request watch on a the given path (either file or dir)
* using all Config defaults. In the case of a dir,
* the default is not to recurse into subdirs for watching.
*
* @param file the path to watch
*/
public void watch(final Path file)
{
//Make a config for the dir above it and
//include a match only for the given path
//using all defaults for the configuration
Path abs = file;
if (!abs.isAbsolute())
{
abs = file.toAbsolutePath();
}
//Check we don't already have a config for the parent directory.
//If we do, add in this filename.
Config config = null;
Path parent = abs.getParent();
for (Config c : configs)
{
if (c.getPath().equals(parent))
{
config = c;
break;
}
}
//Make a new config
if (config == null)
{
config = new Config(abs.getParent());
// the include for the directory itself
config.addIncludeGlobRelative("");
//add the include for the file
config.addIncludeGlobRelative(file.getFileName().toString());
watch(config);
}
else
//add the include for the file
config.addIncludeGlobRelative(file.getFileName().toString());
}
/**
* Request watch on a path with custom Config
* provided.
*
* @param config the configuration to watch
*/
public void watch(final Config config)
{
//Add a custom config
configs.add(config);
}
/**
* Add a listener for changes the watcher notices.
*
* @param listener change listener
*/
public void addListener(EventListener listener)
{
listeners.add(listener);
}
/**
* Append some info on the paths that we are watching.
*/
private void appendConfigId(StringBuilder s)
{
List dirs = new ArrayList<>();
for (Config config : keys.values())
{
dirs.add(config.path);
}
Collections.sort(dirs);
s.append("[");
if (dirs.size() > 0)
{
s.append(dirs.get(0));
if (dirs.size() > 1)
{
s.append(" (+").append(dirs.size() - 1).append(")");
}
}
else
{
s.append("");
}
s.append("]");
}
/**
* @see org.eclipse.jetty.util.component.AbstractLifeCycle#doStart()
*/
@Override
protected void doStart() throws Exception
{
//create a new watchservice
this.watchService = FileSystems.getDefault().newWatchService();
//ensure setting of quiet time is appropriate now we have a watcher
setUpdateQuietTime(getUpdateQuietTimeMillis(), TimeUnit.MILLISECONDS);
// Register all watched paths, walking dir hierarchies as needed, possibly generating
// fake add events if notifyExistingOnStart is true
for (Config c : configs)
{
registerTree(c.getPath(), c, isNotifyExistingOnStart());
}
// Start Thread for watcher take/pollKeys loop
StringBuilder threadId = new StringBuilder();
threadId.append("PathWatcher@");
threadId.append(Integer.toHexString(hashCode()));
if (LOG.isDebugEnabled())
LOG.debug("{} -> {}", this, threadId);
thread = new Thread(this, threadId.toString());
thread.setDaemon(true);
thread.start();
super.doStart();
}
/**
* @see org.eclipse.jetty.util.component.AbstractLifeCycle#doStop()
*/
@Override
protected void doStop() throws Exception
{
if (watchService != null)
watchService.close(); //will invalidate registered watch keys, interrupt thread in take or poll
watchService = null;
thread = null;
keys.clear();
pending.clear();
events.clear();
super.doStop();
}
/**
* Remove all current configs and listeners.
*/
public void reset()
{
if (!isStopped())
throw new IllegalStateException("PathWatcher must be stopped before reset.");
configs.clear();
listeners.clear();
}
/**
* Check to see if the watcher is in a state where it should generate
* watch events to the listeners. Used to determine if watcher should generate
* events for existing files and dirs on startup.
*
* @return true if the watcher should generate events to the listeners.
*/
protected boolean isNotifiable()
{
return (isStarted() || (!isStarted() && isNotifyExistingOnStart()));
}
/**
* Get an iterator over the listeners.
*
* @return iterator over the listeners.
*/
public Iterator getListeners()
{
return listeners.iterator();
}
/**
* Change the quiet time.
*
* @return the quiet time in millis
*/
public long getUpdateQuietTimeMillis()
{
return TimeUnit.MILLISECONDS.convert(updateQuietTimeDuration, updateQuietTimeUnit);
}
private void registerTree(Path dir, Config config, boolean notify) throws IOException
{
if (LOG.isDebugEnabled())
LOG.debug("registerTree {} {} {}", dir, config, notify);
if (!Files.isDirectory(dir))
throw new IllegalArgumentException(dir.toString());
register(dir, config);
final MultiException me = new MultiException();
try (Stream stream = Files.list(dir))
{
stream.forEach(p ->
{
if (LOG.isDebugEnabled())
LOG.debug("registerTree? {}", p);
try
{
if (notify && config.test(p))
pending.put(p, new PathWatchEvent(p, PathWatchEventType.ADDED, config));
switch (config.handleDir(p))
{
case ENTER:
registerTree(p, config.asSubConfig(p), notify);
break;
case WATCH:
registerDir(p, config);
break;
case IGNORE:
default:
break;
}
}
catch (IOException e)
{
me.add(e);
}
});
}
try
{
me.ifExceptionThrow();
}
catch (IOException e)
{
throw e;
}
catch (Throwable ex)
{
throw new IOException(ex);
}
}
private void registerDir(Path path, Config config) throws IOException
{
if (LOG.isDebugEnabled())
LOG.debug("registerDir {} {}", path, config);
if (!Files.isDirectory(path))
throw new IllegalArgumentException(path.toString());
register(path, config.asSubConfig(path), WATCH_DIR_KINDS);
}
protected void register(Path path, Config config) throws IOException
{
if (LOG.isDebugEnabled())
LOG.debug("Registering watch on {}", path);
register(path, config, WATCH_EVENT_KINDS);
}
private void register(Path path, Config config, WatchEvent.Kind>[] kinds) throws IOException
{
// Native Watcher
WatchKey key = path.register(watchService, kinds);
keys.put(key, config);
}
/**
* Delete a listener
*
* @param listener the listener to remove
* @return true if the listener existed and was removed
*/
public boolean removeListener(Listener listener)
{
return listeners.remove(listener);
}
/**
* Forever loop.
*
* Wait for the WatchService to report some filesystem events for the
* watched paths.
*
* When an event for a path first occurs, it is subjected to a quiet time.
* Subsequent events that arrive for the same path during this quiet time are
* accumulated and the timer reset. Only when the quiet time has expired are
* the accumulated events sent. MODIFY events are handled slightly differently -
* multiple MODIFY events arriving within a quiet time are coalesced into a
* single MODIFY event. Both the accumulation of events and coalescing of MODIFY
* events reduce the number and frequency of event reporting for "noisy" files (ie
* those that are undergoing rapid change).
*
* @see java.lang.Runnable#run()
*/
@Override
public void run()
{
// Start the java.nio watching
if (LOG.isDebugEnabled())
{
LOG.debug("Starting java.nio file watching with {}", watchService);
}
long waitTime = getUpdateQuietTimeMillis();
WatchService watch = watchService;
while (isRunning() && thread == Thread.currentThread())
{
WatchKey key;
try
{
// Reset all keys before watching
long now = TimeUnit.NANOSECONDS.toMillis(System.nanoTime());
for (Map.Entry e : keys.entrySet())
{
WatchKey k = e.getKey();
Config c = e.getValue();
if (!c.isPaused(now) && !k.reset())
{
keys.remove(k);
if (keys.isEmpty())
{
return; // all done, no longer monitoring anything
}
}
}
if (LOG.isDebugEnabled())
LOG.debug("Waiting for poll({})", waitTime);
key = waitTime < 0 ? watch.take() : waitTime > 0 ? watch.poll(waitTime, updateQuietTimeUnit) : watch.poll();
// handle all active keys
while (key != null)
{
handleKey(key);
key = watch.poll();
}
waitTime = processPending();
notifyEvents();
}
catch (ClosedWatchServiceException e)
{
// Normal shutdown of watcher
return;
}
catch (InterruptedException e)
{
if (isRunning())
{
LOG.warn(e);
}
else
{
LOG.ignore(e);
}
}
}
}
private void handleKey(WatchKey key)
{
Config config = keys.get(key);
if (config == null)
{
if (LOG.isDebugEnabled())
LOG.debug("WatchKey not recognized: {}", key);
return;
}
for (WatchEvent> event : key.pollEvents())
{
WatchEvent ev = cast(event);
Path name = ev.context();
Path path = config.resolve(name);
if (LOG.isDebugEnabled())
LOG.debug("handleKey? {} {} {}", ev.kind(), config.toShortPath(path), config);
// Ignore modified events on directories. These are handled as create/delete events of their contents
if (ev.kind() == ENTRY_MODIFY && Files.exists(path) && Files.isDirectory(path))
continue;
if (config.test(path))
handleWatchEvent(path, new PathWatchEvent(path, ev, config));
else if (config.getRecurseDepth() == -1)
{
// Convert a watched directory into a modify event on its parent
Path parent = path.getParent();
Config parentConfig = config.getParent();
handleWatchEvent(parent, new PathWatchEvent(parent, PathWatchEventType.MODIFIED, parentConfig));
continue;
}
if (ev.kind() == ENTRY_CREATE)
{
try
{
switch (config.handleDir(path))
{
case ENTER:
registerTree(path, config.asSubConfig(path), true);
break;
case WATCH:
registerDir(path, config);
break;
case IGNORE:
default:
break;
}
}
catch (IOException e)
{
LOG.warn(e);
}
}
}
}
/**
* Add an event reported by the WatchService to list of pending events
* that will be sent after their quiet time has expired.
*
* @param path the path to add to the pending list
* @param event the pending event
*/
public void handleWatchEvent(Path path, PathWatchEvent event)
{
PathWatchEvent existing = pending.get(path);
if (LOG.isDebugEnabled())
LOG.debug("handleWatchEvent {} {} <= {}", path, event, existing);
switch (event.getType())
{
case ADDED:
if (existing != null && existing.getType() == PathWatchEventType.MODIFIED)
events.add(new PathWatchEvent(path, PathWatchEventType.DELETED, existing.getConfig()));
pending.put(path, event);
break;
case MODIFIED:
if (existing == null)
pending.put(path, event);
else
existing.modified();
break;
case DELETED:
case UNKNOWN:
if (existing != null)
pending.remove(path);
events.add(event);
break;
}
}
private long processPending()
{
if (LOG.isDebugEnabled())
LOG.debug("processPending> {}", pending.values());
long now = TimeUnit.NANOSECONDS.toMillis(System.nanoTime());
long wait = Long.MAX_VALUE;
// pending map is maintained in LRU order
for (PathWatchEvent event : new ArrayList<>(pending.values()))
{
Path path = event.getPath();
// for directories, wait until parent is quiet
if (pending.containsKey(path.getParent()))
continue;
// if the path is quiet move to events
if (event.isQuiet(now, getUpdateQuietTimeMillis()))
{
if (LOG.isDebugEnabled())
LOG.debug("isQuiet {}", event);
pending.remove(path);
events.add(event);
}
else
{
long msToCheck = event.toQuietCheck(now, getUpdateQuietTimeMillis());
if (LOG.isDebugEnabled())
LOG.debug("pending {} {}", event, msToCheck);
if (msToCheck < wait)
wait = msToCheck;
}
}
if (LOG.isDebugEnabled())
LOG.debug("processPending< {}", pending.values());
return wait == Long.MAX_VALUE ? -1 : wait;
}
private void notifyEvents()
{
if (LOG.isDebugEnabled())
LOG.debug("notifyEvents {}", events.size());
if (events.isEmpty())
return;
boolean eventListeners = false;
for (EventListener listener : listeners)
{
if (listener instanceof EventListListener)
{
try
{
if (LOG.isDebugEnabled())
LOG.debug("notifyEvents {} {}", listener, events);
((EventListListener)listener).onPathWatchEvents(events);
}
catch (Throwable t)
{
LOG.warn(t);
}
}
else
eventListeners = true;
}
if (eventListeners)
{
for (PathWatchEvent event : events)
{
if (LOG.isDebugEnabled())
LOG.debug("notifyEvent {} {}", event, listeners);
for (EventListener listener : listeners)
{
if (listener instanceof Listener)
{
try
{
((Listener)listener).onPathWatchEvent(event);
}
catch (Throwable t)
{
LOG.warn(t);
}
}
}
}
}
events.clear();
}
/**
* Whether or not to issue notifications for directories and files that
* already exist when the watcher starts.
*
* @param notify true if existing paths should be notified or not
*/
public void setNotifyExistingOnStart(boolean notify)
{
_notifyExistingOnStart = notify;
}
public boolean isNotifyExistingOnStart()
{
return _notifyExistingOnStart;
}
/**
* Set the quiet time.
*
* @param duration the quiet time duration
* @param unit the quite time unit
*/
public void setUpdateQuietTime(long duration, TimeUnit unit)
{
long desiredMillis = unit.toMillis(duration);
if (IS_WINDOWS && (desiredMillis < 1000))
{
LOG.warn("Quiet Time is too low for Microsoft Windows: {} < 1000 ms (defaulting to 1000 ms)", desiredMillis);
this.updateQuietTimeDuration = 1000;
this.updateQuietTimeUnit = TimeUnit.MILLISECONDS;
return;
}
// All other OS and watch service combinations can use desired setting
this.updateQuietTimeDuration = duration;
this.updateQuietTimeUnit = unit;
}
@Override
public String toString()
{
StringBuilder s = new StringBuilder(this.getClass().getName());
appendConfigId(s);
return s.toString();
}
private static class ExactPathMatcher implements PathMatcher
{
private final Path path;
ExactPathMatcher(Path path)
{
this.path = path;
}
@Override
public boolean matches(Path path)
{
return this.path.equals(path);
}
}
public static class PathMatcherSet extends HashSet implements Predicate
{
@Override
public boolean test(Path path)
{
for (PathMatcher pm : this)
{
if (pm.matches(path))
return true;
}
return false;
}
}
}