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

org.infinispan.commons.io.FileWatcher Maven / Gradle / Ivy

There is a newer version: 15.1.0.Dev04
Show newest version
package org.infinispan.commons.io;

import java.io.FileNotFoundException;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.NoSuchFileException;
import java.nio.file.Path;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.function.Consumer;

import org.infinispan.commons.logging.Log;
import org.infinispan.commons.logging.LogFactory;

/**
 * @since 15.0
 */
public class FileWatcher implements Runnable, AutoCloseable {
   private static final Log log = LogFactory.getLog(FileWatcher.class);
   public static final int SLEEP = 2_000;
   private final Thread thread;
   private final ConcurrentHashMap watched;
   private boolean running = true;

   public FileWatcher() {
      watched = new ConcurrentHashMap<>();
      thread = new Thread(this, "FileWatcher");
      Runtime.getRuntime().addShutdownHook(new Thread(this::stop));
      thread.start();
   }

   public void unwatch(Path path) {
      watched.remove(path);
      log.debugf("Unwatched %s", path);
   }

   public void watch(Path path, Consumer callback) {
      watched.compute(path, (k, w) -> {
         if (w == null) {
            w = new Watched();
            try {
               w.lastModified = Files.getLastModifiedTime(path).toMillis();
            } catch (FileNotFoundException | NoSuchFileException e) {
               w.lastModified = -1;
               log.debugf("File not found %s", path);
            } catch (IOException e) {
               throw new RuntimeException(e);
            }
         }
         w.watchers.add(callback);
         return w;
      });
      log.debugf("Watching %s", path);
   }

   @Override
   public void run() {
      while (running) {
         try {
            Thread.sleep(SLEEP);
         } catch (InterruptedException e) {
            return;
         }
         if (!running) {
            return;
         }
         for (Map.Entry e : watched.entrySet()) {
            Watched w = e.getValue();
            try {
               long lastModified = Files.getLastModifiedTime(e.getKey()).toMillis();
               if (w.lastModified < lastModified) {
                  w.lastModified = lastModified;
                  for (Consumer c : w.watchers) {
                     c.accept(e.getKey());
                  }
               }
            } catch (FileNotFoundException | NoSuchFileException ex) {
               w.lastModified = -1;
            } catch (IOException ex) {
               throw new RuntimeException(ex);
            }
         }
      }
   }

   public void stop() {
      running = false;
      try {
         thread.join();
      } catch (InterruptedException e) {
         // Ignore
      }
   }

   @Override
   public void close() {
      stop();
   }

   static class Watched {
      long lastModified;
      List> watchers = new ArrayList<>(2);
   }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy