s-lib-watch_3.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