All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.eclipse.jetty.util.PathWatcher Maven / Gradle / Ivy

//
//  ========================================================================
//  Copyright (c) 1995-2018 Mort Bay Consulting Pty. Ltd.
//  ------------------------------------------------------------------------
//  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 static java.nio.file.StandardWatchEventKinds.ENTRY_CREATE;
import static java.nio.file.StandardWatchEventKinds.ENTRY_DELETE;
import static java.nio.file.StandardWatchEventKinds.ENTRY_MODIFY;

import java.io.File;
import java.io.IOException;
import java.lang.reflect.Field;
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.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.EventListener;
import java.util.HashMap;
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.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;

/**
 * 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 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; } if (type != other.type) { return false; } return true; } 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 static 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 WatchEvent.Modifier watchModifiers[]; private boolean nativeWatchService; private final List configs = new ArrayList<>(); private final Map keys = new HashMap<>(); 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. * * @param s */ 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 createWatchService(); //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(); } /** * Create a fresh WatchService and determine if it is a * native implementation or not. * * @throws IOException */ private void createWatchService () throws IOException { //create a watch service this.watchService = FileSystems.getDefault().newWatchService(); WatchEvent.Modifier modifiers[] = null; boolean nativeService = true; // Try to determine native behavior // See http://stackoverflow.com/questions/9588737/is-java-7-watchservice-slow-for-anyone-else try { ClassLoader cl = Thread.currentThread().getContextClassLoader(); Class pollingWatchServiceClass = Class.forName("sun.nio.fs.PollingWatchService",false,cl); if (pollingWatchServiceClass.isAssignableFrom(this.watchService.getClass())) { nativeService = false; LOG.info("Using Non-Native Java {}",pollingWatchServiceClass.getName()); Class c = Class.forName("com.sun.nio.file.SensitivityWatchEventModifier"); Field f = c.getField("HIGH"); modifiers = new WatchEvent.Modifier[] { (WatchEvent.Modifier)f.get(c) }; } } catch (Throwable t) { // Unknown JVM environment, assuming native. LOG.ignore(t); } this.watchModifiers = modifiers; this.nativeWatchService = nativeService; } /** * 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 th) { throw new IOException(th); } } 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,watchModifiers==null?null:Arrays.asList(watchModifiers)); register(path,config,WATCH_EVENT_KINDS); } private void register(Path path, Config config, WatchEvent.Kind[] kinds) throws IOException { if(watchModifiers != null) { // Java Watcher WatchKey key = path.register(watchService,kinds,watchModifiers); keys.put(key,config); } else { // 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 wait_time = 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({})", wait_time); key = wait_time<0?watch.take():wait_time>0?watch.poll(wait_time,updateQuietTimeUnit):watch.poll(); // handle all active keys while (key!=null) { handleKey(key); key = watch.poll(); } wait_time = 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 ms_to_check = event.toQuietCheck(now,getUpdateQuietTimeMillis()); if (LOG.isDebugEnabled()) LOG.debug("pending {} {}",event, ms_to_check); if (ms_to_check implements Predicate { @Override public boolean test(Path path) { for (PathMatcher pm: this) if (pm.matches(path)) return true; return false; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy