|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*/ |
|
|
|
package sun.nio.fs; |
|
|
|
import java.nio.file.ClosedWatchServiceException; |
|
import java.nio.file.DirectoryIteratorException; |
|
import java.nio.file.DirectoryStream; |
|
import java.nio.file.Files; |
|
import java.nio.file.LinkOption; |
|
import java.nio.file.NotDirectoryException; |
|
import java.nio.file.Path; |
|
import java.nio.file.StandardWatchEventKinds; |
|
import java.nio.file.WatchEvent; |
|
import java.nio.file.WatchKey; |
|
import java.nio.file.attribute.BasicFileAttributes; |
|
import java.security.AccessController; |
|
import java.security.PrivilegedAction; |
|
import java.security.PrivilegedExceptionAction; |
|
import java.security.PrivilegedActionException; |
|
import java.io.IOException; |
|
import java.util.HashMap; |
|
import java.util.HashSet; |
|
import java.util.Iterator; |
|
import java.util.Map; |
|
import java.util.Set; |
|
import java.util.concurrent.Executors; |
|
import java.util.concurrent.ScheduledExecutorService; |
|
import java.util.concurrent.ScheduledFuture; |
|
import java.util.concurrent.ThreadFactory; |
|
import java.util.concurrent.TimeUnit; |
|
|
|
/** |
|
* Simple WatchService implementation that uses periodic tasks to poll |
|
* registered directories for changes. This implementation is for use on |
|
* operating systems that do not have native file change notification support. |
|
*/ |
|
|
|
class PollingWatchService |
|
extends AbstractWatchService |
|
{ |
|
|
|
private final Map<Object, PollingWatchKey> map = new HashMap<>(); |
|
|
|
|
|
private final ScheduledExecutorService scheduledExecutor; |
|
|
|
PollingWatchService() { |
|
|
|
scheduledExecutor = Executors |
|
.newSingleThreadScheduledExecutor(new ThreadFactory() { |
|
@Override |
|
public Thread newThread(Runnable r) { |
|
Thread t = new Thread(null, r, "FileSystemWatcher", 0, false); |
|
t.setDaemon(true); |
|
return t; |
|
}}); |
|
} |
|
|
|
|
|
|
|
*/ |
|
@SuppressWarnings("removal") |
|
@Override |
|
WatchKey register(final Path path, |
|
WatchEvent.Kind<?>[] events, |
|
WatchEvent.Modifier... modifiers) |
|
throws IOException |
|
{ |
|
|
|
final Set<WatchEvent.Kind<?>> eventSet = new HashSet<>(events.length); |
|
for (WatchEvent.Kind<?> event: events) { |
|
|
|
if (event == StandardWatchEventKinds.ENTRY_CREATE || |
|
event == StandardWatchEventKinds.ENTRY_MODIFY || |
|
event == StandardWatchEventKinds.ENTRY_DELETE) |
|
{ |
|
eventSet.add(event); |
|
continue; |
|
} |
|
|
|
|
|
if (event == StandardWatchEventKinds.OVERFLOW) { |
|
continue; |
|
} |
|
|
|
|
|
if (event == null) |
|
throw new NullPointerException("An element in event set is 'null'"); |
|
throw new UnsupportedOperationException(event.name()); |
|
} |
|
if (eventSet.isEmpty()) |
|
throw new IllegalArgumentException("No events to register"); |
|
|
|
|
|
int sensitivity = 10; |
|
if (modifiers.length > 0) { |
|
for (WatchEvent.Modifier modifier: modifiers) { |
|
if (modifier == null) |
|
throw new NullPointerException(); |
|
|
|
if (ExtendedOptions.SENSITIVITY_HIGH.matches(modifier)) { |
|
sensitivity = ExtendedOptions.SENSITIVITY_HIGH.parameter(); |
|
} else if (ExtendedOptions.SENSITIVITY_MEDIUM.matches(modifier)) { |
|
sensitivity = ExtendedOptions.SENSITIVITY_MEDIUM.parameter(); |
|
} else if (ExtendedOptions.SENSITIVITY_LOW.matches(modifier)) { |
|
sensitivity = ExtendedOptions.SENSITIVITY_LOW.parameter(); |
|
} else { |
|
throw new UnsupportedOperationException("Modifier not supported"); |
|
} |
|
} |
|
} |
|
|
|
|
|
if (!isOpen()) |
|
throw new ClosedWatchServiceException(); |
|
|
|
// registration is done in privileged block as it requires the |
|
|
|
try { |
|
int value = sensitivity; |
|
return AccessController.doPrivileged( |
|
new PrivilegedExceptionAction<PollingWatchKey>() { |
|
@Override |
|
public PollingWatchKey run() throws IOException { |
|
return doPrivilegedRegister(path, eventSet, value); |
|
} |
|
}); |
|
} catch (PrivilegedActionException pae) { |
|
Throwable cause = pae.getCause(); |
|
if (cause instanceof IOException ioe) |
|
throw ioe; |
|
throw new AssertionError(pae); |
|
} |
|
} |
|
|
|
// registers directory returning a new key if not already registered or |
|
|
|
private PollingWatchKey doPrivilegedRegister(Path path, |
|
Set<? extends WatchEvent.Kind<?>> events, |
|
int sensitivityInSeconds) |
|
throws IOException |
|
{ |
|
|
|
BasicFileAttributes attrs = Files.readAttributes(path, BasicFileAttributes.class); |
|
if (!attrs.isDirectory()) { |
|
throw new NotDirectoryException(path.toString()); |
|
} |
|
Object fileKey = attrs.fileKey(); |
|
if (fileKey == null) |
|
throw new AssertionError("File keys must be supported"); |
|
|
|
|
|
synchronized (closeLock()) { |
|
if (!isOpen()) |
|
throw new ClosedWatchServiceException(); |
|
|
|
PollingWatchKey watchKey; |
|
synchronized (map) { |
|
watchKey = map.get(fileKey); |
|
if (watchKey == null) { |
|
|
|
watchKey = new PollingWatchKey(path, this, fileKey); |
|
map.put(fileKey, watchKey); |
|
} else { |
|
|
|
watchKey.disable(); |
|
} |
|
} |
|
watchKey.enable(events, sensitivityInSeconds); |
|
return watchKey; |
|
} |
|
|
|
} |
|
|
|
@SuppressWarnings("removal") |
|
@Override |
|
void implClose() throws IOException { |
|
synchronized (map) { |
|
for (Map.Entry<Object, PollingWatchKey> entry: map.entrySet()) { |
|
PollingWatchKey watchKey = entry.getValue(); |
|
watchKey.disable(); |
|
watchKey.invalidate(); |
|
} |
|
map.clear(); |
|
} |
|
AccessController.doPrivileged(new PrivilegedAction<Void>() { |
|
@Override |
|
public Void run() { |
|
scheduledExecutor.shutdown(); |
|
return null; |
|
} |
|
}); |
|
} |
|
|
|
|
|
|
|
*/ |
|
private static class CacheEntry { |
|
private long lastModified; |
|
private int lastTickCount; |
|
|
|
CacheEntry(long lastModified, int lastTickCount) { |
|
this.lastModified = lastModified; |
|
this.lastTickCount = lastTickCount; |
|
} |
|
|
|
int lastTickCount() { |
|
return lastTickCount; |
|
} |
|
|
|
long lastModified() { |
|
return lastModified; |
|
} |
|
|
|
void update(long lastModified, int tickCount) { |
|
this.lastModified = lastModified; |
|
this.lastTickCount = tickCount; |
|
} |
|
} |
|
|
|
|
|
|
|
|
|
|
|
*/ |
|
private class PollingWatchKey extends AbstractWatchKey { |
|
private final Object fileKey; |
|
|
|
|
|
private Set<? extends WatchEvent.Kind<?>> events; |
|
|
|
|
|
private ScheduledFuture<?> poller; |
|
|
|
|
|
private volatile boolean valid; |
|
|
|
|
|
private int tickCount; |
|
|
|
|
|
private Map<Path,CacheEntry> entries; |
|
|
|
PollingWatchKey(Path dir, PollingWatchService watcher, Object fileKey) |
|
throws IOException |
|
{ |
|
super(dir, watcher); |
|
this.fileKey = fileKey; |
|
this.valid = true; |
|
this.tickCount = 0; |
|
this.entries = new HashMap<Path,CacheEntry>(); |
|
|
|
|
|
try (DirectoryStream<Path> stream = Files.newDirectoryStream(dir)) { |
|
for (Path entry: stream) { |
|
|
|
long lastModified = |
|
Files.getLastModifiedTime(entry, LinkOption.NOFOLLOW_LINKS).toMillis(); |
|
entries.put(entry.getFileName(), new CacheEntry(lastModified, tickCount)); |
|
} |
|
} catch (DirectoryIteratorException e) { |
|
throw e.getCause(); |
|
} |
|
} |
|
|
|
Object fileKey() { |
|
return fileKey; |
|
} |
|
|
|
@Override |
|
public boolean isValid() { |
|
return valid; |
|
} |
|
|
|
void invalidate() { |
|
valid = false; |
|
} |
|
|
|
|
|
void enable(Set<? extends WatchEvent.Kind<?>> events, long period) { |
|
synchronized (this) { |
|
|
|
this.events = events; |
|
|
|
|
|
Runnable thunk = new Runnable() { public void run() { poll(); }}; |
|
this.poller = scheduledExecutor |
|
.scheduleAtFixedRate(thunk, period, period, TimeUnit.SECONDS); |
|
} |
|
} |
|
|
|
|
|
void disable() { |
|
synchronized (this) { |
|
if (poller != null) |
|
poller.cancel(false); |
|
} |
|
} |
|
|
|
@Override |
|
public void cancel() { |
|
valid = false; |
|
synchronized (map) { |
|
map.remove(fileKey()); |
|
} |
|
disable(); |
|
} |
|
|
|
|
|
|
|
|
|
*/ |
|
synchronized void poll() { |
|
if (!valid) { |
|
return; |
|
} |
|
|
|
|
|
tickCount++; |
|
|
|
|
|
DirectoryStream<Path> stream = null; |
|
try { |
|
stream = Files.newDirectoryStream(watchable()); |
|
} catch (IOException x) { |
|
|
|
cancel(); |
|
signal(); |
|
return; |
|
} |
|
|
|
|
|
try { |
|
for (Path entry: stream) { |
|
long lastModified = 0L; |
|
try { |
|
lastModified = |
|
Files.getLastModifiedTime(entry, LinkOption.NOFOLLOW_LINKS).toMillis(); |
|
} catch (IOException x) { |
|
// unable to get attributes of entry. If file has just |
|
// been deleted then we'll report it as deleted on the |
|
|
|
continue; |
|
} |
|
|
|
|
|
CacheEntry e = entries.get(entry.getFileName()); |
|
if (e == null) { |
|
|
|
entries.put(entry.getFileName(), |
|
new CacheEntry(lastModified, tickCount)); |
|
|
|
|
|
if (events.contains(StandardWatchEventKinds.ENTRY_CREATE)) { |
|
signalEvent(StandardWatchEventKinds.ENTRY_CREATE, entry.getFileName()); |
|
continue; |
|
} else { |
|
// if ENTRY_CREATE is not enabled and ENTRY_MODIFY is |
|
// enabled then queue event to avoid missing out on |
|
// modifications to the file immediately after it is |
|
|
|
if (events.contains(StandardWatchEventKinds.ENTRY_MODIFY)) { |
|
signalEvent(StandardWatchEventKinds.ENTRY_MODIFY, entry.getFileName()); |
|
} |
|
} |
|
continue; |
|
} |
|
|
|
|
|
if (e.lastModified != lastModified) { |
|
if (events.contains(StandardWatchEventKinds.ENTRY_MODIFY)) { |
|
signalEvent(StandardWatchEventKinds.ENTRY_MODIFY, |
|
entry.getFileName()); |
|
} |
|
} |
|
|
|
e.update(lastModified, tickCount); |
|
|
|
} |
|
} catch (DirectoryIteratorException e) { |
|
// ignore for now; if the directory is no longer accessible |
|
// then the key will be cancelled on the next poll |
|
} finally { |
|
|
|
|
|
try { |
|
stream.close(); |
|
} catch (IOException x) { |
|
// ignore |
|
} |
|
} |
|
|
|
|
|
Iterator<Map.Entry<Path,CacheEntry>> i = entries.entrySet().iterator(); |
|
while (i.hasNext()) { |
|
Map.Entry<Path,CacheEntry> mapEntry = i.next(); |
|
CacheEntry entry = mapEntry.getValue(); |
|
if (entry.lastTickCount() != tickCount) { |
|
Path name = mapEntry.getKey(); |
|
|
|
i.remove(); |
|
if (events.contains(StandardWatchEventKinds.ENTRY_DELETE)) { |
|
signalEvent(StandardWatchEventKinds.ENTRY_DELETE, name); |
|
} |
|
} |
|
} |
|
} |
|
} |
|
} |