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

s-lib-watch_2.13.0.11.4-M3.source-code.WatchServiceWatcher.scala Maven / Gradle / Ivy

The newest version!
package os.watch

import java.nio.file._
import java.io.IOException
import java.nio.file.ClosedWatchServiceException
import java.util.concurrent.atomic.AtomicBoolean
import java.nio.file.StandardWatchEventKinds.{ENTRY_CREATE, ENTRY_DELETE, ENTRY_MODIFY, OVERFLOW}
import com.sun.nio.file.{ExtendedWatchEventModifier, SensitivityWatchEventModifier}

import scala.collection.mutable
import collection.JavaConverters._
import scala.util.Properties.isWin

class WatchServiceWatcher(
    roots: Seq[os.Path],
    onEvent: Set[os.Path] => Unit,
    logger: (String, Any) => Unit = (_, _) => ()
) extends Watcher {

  val nioWatchService = FileSystems.getDefault.newWatchService()
  val currentlyWatchedPaths = mutable.Map.empty[os.Path, WatchKey]
  val newlyWatchedPaths = mutable.Buffer.empty[os.Path]
  val bufferedEvents = mutable.Set.empty[os.Path]
  val isRunning = new AtomicBoolean(false)

  isRunning.set(true)

  roots.foreach(watchSinglePath)
  recursiveWatches()

  bufferedEvents.clear()
  def watchSinglePath(p: os.Path) = {
    val isDir = os.isDir(p, followLinks = false)
    logger("WATCH", (p, isDir))
    if (isDir) {
      // https://stackoverflow.com/a/6265860/4496364
      // on Windows we watch only the root directory
      val modifiers: Array[WatchEvent.Modifier] = if (isWin)
        Array(SensitivityWatchEventModifier.HIGH, ExtendedWatchEventModifier.FILE_TREE)
      else Array(SensitivityWatchEventModifier.HIGH)
      currentlyWatchedPaths.put(
        p,
        p.toNIO.register(
          nioWatchService,
          Array[WatchEvent.Kind[_]](ENTRY_CREATE, ENTRY_DELETE, ENTRY_MODIFY, OVERFLOW),
          modifiers: _*
        )
      )
      newlyWatchedPaths.append(p)
    }
    bufferedEvents.add(p)
  }

  def processEvent(watchKey: WatchKey) = {
    val p = os.Path(watchKey.watchable().asInstanceOf[java.nio.file.Path], os.pwd)
    logger("WATCH PATH", p)

    val events = watchKey.pollEvents().asScala

    logger("WATCH CONTEXTS", events.map(_.context()))

    logger("WATCH KINDS", events.map(_.kind()))

    for (e <- events) {
      bufferedEvents.add(p / e.context().toString)
    }

    for (e <- events if e.kind() == ENTRY_CREATE) {
      watchSinglePath(p / e.context().toString)
    }

    watchKey.reset()
  }

  def recursiveWatches() = {
    // no need to recursively watch each folder on windows
    // https://stackoverflow.com/a/64030685/4496364
    if (isWin) {
      // noop
    } else {
      while (newlyWatchedPaths.nonEmpty) {
        val top = newlyWatchedPaths.remove(newlyWatchedPaths.length - 1)
        val listing =
          try os.list(top)
          catch {
            case e: java.nio.file.NotDirectoryException => Nil
          }
        for (p <- listing) watchSinglePath(p)
        bufferedEvents.add(top)
      }
    }
  }

  def run(): Unit = {
    while (isRunning.get())
      try {
        logger("WATCH CURRENT", currentlyWatchedPaths)
        val watchKey0 = nioWatchService.take()
        if (watchKey0 != null) {
          logger("WATCH KEY0", watchKey0.watchable())
          processEvent(watchKey0)
          while ({
            nioWatchService.poll() match {
              case null => false
              case watchKey =>
                logger("WATCH KEY", watchKey.watchable())
                processEvent(watchKey)
                true
            }
          }) ()

          // cleanup stale watches, but do so before we register new ones
          // because when folders are moved, the old watch is moved as well
          // and we need to make sure we re-register the watch after disabling
          // it due to the old file path within the old folder no longer existing
          for (p <- currentlyWatchedPaths.keySet if !os.isDir(p, followLinks = false)) {
            logger("WATCH CANCEL", p)
            currentlyWatchedPaths.remove(p).foreach(_.cancel())
          }

          recursiveWatches()
          triggerListener()
        }

      } catch {
        case e: InterruptedException =>
          println("Interrupted, exiting: " + e)
          isRunning.set(false)
        case e: ClosedWatchServiceException =>
          println("Watcher closed, exiting: " + e)
          isRunning.set(false)
      }
  }

  def close(): Unit = {
    try {
      isRunning.set(false)
      nioWatchService.close()
    } catch {
      case e: IOException => println("Error closing watcher: " + e)
    }
  }

  private def triggerListener(): Unit = {
    logger("TRIGGER", bufferedEvents.toSet)
    onEvent(bufferedEvents.toSet)
    bufferedEvents.clear()
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy