FileWatcher Class — spring-boot Architecture
Architecture documentation for the FileWatcher class in FileWatcher.java from the spring-boot codebase.
Entity Profile
Source Code
core/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/ssl/FileWatcher.java lines 53–281
class FileWatcher implements Closeable {
private static final Log logger = LogFactory.getLog(FileWatcher.class);
private final Duration quietPeriod;
private final Object lock = new Object();
private @Nullable WatcherThread thread;
/**
* Create a new {@link FileWatcher} instance.
* @param quietPeriod the duration that no file changes should occur before triggering
* actions
*/
FileWatcher(Duration quietPeriod) {
Assert.notNull(quietPeriod, "'quietPeriod' must not be null");
this.quietPeriod = quietPeriod;
}
/**
* Watch the given files or directories for changes.
* @param paths the files or directories to watch
* @param action the action to take when changes are detected
*/
void watch(Set<Path> paths, Runnable action) {
Assert.notNull(paths, "'paths' must not be null");
Assert.notNull(action, "'action' must not be null");
if (paths.isEmpty()) {
return;
}
synchronized (this.lock) {
try {
if (this.thread == null) {
this.thread = new WatcherThread();
this.thread.start();
}
this.thread.register(new Registration(getRegistrationPaths(paths), action));
}
catch (IOException ex) {
throw new UncheckedIOException("Failed to register paths for watching: " + paths, ex);
}
}
}
/**
* Retrieves all {@link Path Paths} that should be registered for the specified
* {@link Path}. If the path is a symlink, changes to the symlink should be monitored,
* not just the file it points to. For example, for the given {@code keystore.jks}
* path in the following directory structure:<pre>
* +- stores
* | +─ keystore.jks
* +- <em>data</em> -> stores
* +─ <em>keystore.jks</em> -> data/keystore.jks
* </pre> the resulting paths would include:
* <p>
* <ul>
* <li>{@code keystore.jks}</li>
* <li>{@code data/keystore.jks}</li>
* <li>{@code data}</li>
* <li>{@code stores/keystore.jks}</li>
* </ul>
* @param paths the source paths
* @return all possible {@link Path} instances to be registered
* @throws IOException if an I/O error occurs
*/
private Set<Path> getRegistrationPaths(Set<Path> paths) throws IOException {
Set<Path> result = new HashSet<>();
for (Path path : paths) {
collectRegistrationPaths(path, result);
}
return Collections.unmodifiableSet(result);
}
private void collectRegistrationPaths(Path path, Set<Path> result) throws IOException {
path = path.toAbsolutePath();
result.add(path);
Path parent = path.getParent();
if (parent != null && Files.isSymbolicLink(parent)) {
result.add(parent);
collectRegistrationPaths(resolveSiblingSymbolicLink(parent).resolve(path.getFileName()), result);
}
else if (Files.isSymbolicLink(path)) {
collectRegistrationPaths(resolveSiblingSymbolicLink(path), result);
}
}
private Path resolveSiblingSymbolicLink(Path path) throws IOException {
return path.resolveSibling(Files.readSymbolicLink(path));
}
@Override
public void close() throws IOException {
synchronized (this.lock) {
if (this.thread != null) {
this.thread.close();
this.thread.interrupt();
try {
this.thread.join();
}
catch (InterruptedException ex) {
Thread.currentThread().interrupt();
}
this.thread = null;
}
}
}
/**
* The watcher thread used to check for changes.
*/
private class WatcherThread extends Thread implements Closeable {
private final WatchService watchService = FileSystems.getDefault().newWatchService();
private final Map<WatchKey, List<Registration>> registrations = new ConcurrentHashMap<>();
private volatile boolean running = true;
WatcherThread() throws IOException {
setName("ssl-bundle-watcher");
setDaemon(true);
setUncaughtExceptionHandler(this::onThreadException);
}
private void onThreadException(Thread thread, Throwable throwable) {
logger.error("Uncaught exception in file watcher thread", throwable);
}
void register(Registration registration) throws IOException {
Set<Path> directories = new HashSet<>();
for (Path path : registration.paths()) {
if (!Files.isRegularFile(path) && !Files.isDirectory(path)) {
throw new IOException("'%s' is neither a file nor a directory".formatted(path));
}
Path directory = Files.isDirectory(path) ? path : path.getParent();
directories.add(directory);
}
for (Path directory : directories) {
WatchKey watchKey = register(directory);
this.registrations.computeIfAbsent(watchKey, (key) -> new CopyOnWriteArrayList<>()).add(registration);
}
}
private WatchKey register(Path directory) throws IOException {
logger.debug(LogMessage.format("Registering '%s'", directory));
return directory.register(this.watchService, StandardWatchEventKinds.ENTRY_CREATE,
StandardWatchEventKinds.ENTRY_MODIFY, StandardWatchEventKinds.ENTRY_DELETE);
}
@Override
public void run() {
logger.debug("Watch thread started");
Set<Runnable> actions = new HashSet<>();
while (this.running) {
try {
long timeout = FileWatcher.this.quietPeriod.toMillis();
WatchKey key = this.watchService.poll(timeout, TimeUnit.MILLISECONDS);
if (key == null) {
actions.forEach(this::runSafely);
actions.clear();
}
else {
accumulate(key, actions);
key.reset();
}
}
catch (InterruptedException ex) {
Thread.currentThread().interrupt();
}
catch (ClosedWatchServiceException ex) {
logger.debug("File watcher has been closed");
this.running = false;
}
}
logger.debug("Watch thread stopped");
}
private void runSafely(Runnable action) {
try {
action.run();
}
catch (Throwable ex) {
logger.error("Unexpected SSL reload error", ex);
}
}
private void accumulate(WatchKey key, Set<Runnable> actions) {
List<Registration> registrations = this.registrations.get(key);
Path directory = (Path) key.watchable();
for (WatchEvent<?> event : key.pollEvents()) {
Path file = directory.resolve((Path) event.context());
Assert.state(registrations != null, "'registrations' must not be null");
for (Registration registration : registrations) {
if (registration.manages(file)) {
actions.add(registration.action());
}
}
}
}
@Override
public void close() throws IOException {
this.running = false;
this.watchService.close();
}
}
/**
* An individual watch registration.
*
* @param paths the paths being registered
* @param action the action to take
*/
private record Registration(Set<Path> paths, Runnable action) {
boolean manages(Path file) {
Path absolutePath = file.toAbsolutePath();
return this.paths.contains(absolutePath) || isInDirectories(absolutePath);
}
private boolean isInDirectories(Path file) {
return this.paths.stream().filter(Files::isDirectory).anyMatch(file::startsWith);
}
}
}
Source
Analyze Your Own Codebase
Get architecture documentation, dependency graphs, and domain analysis for your codebase in minutes.
Try Supermodel Free