watchexec/src/watcher.rs

62 lines
1.7 KiB
Rust

use notify::{raw_watcher, PollWatcher, RecommendedWatcher, RecursiveMode};
use std::convert::TryFrom;
use std::path::PathBuf;
use std::sync::mpsc::Sender;
use std::time::Duration;
/// Thin wrapper over the notify crate
///
/// `PollWatcher` and `RecommendedWatcher` are distinct types, but watchexec
/// really just wants to handle them without regard to the exact type
/// (e.g. polymorphically). This has the nice side effect of separating out
/// all coupling to the notify crate into this module.
pub struct Watcher {
watcher_impl: WatcherImpl,
}
pub use notify::Error;
pub use notify::RawEvent as Event;
enum WatcherImpl {
Recommended(RecommendedWatcher),
Poll(PollWatcher),
}
impl Watcher {
pub fn new(
tx: Sender<Event>,
paths: &[PathBuf],
poll: bool,
interval: Duration,
) -> Result<Self, Error> {
use notify::Watcher;
let imp = if poll {
let mut watcher = PollWatcher::with_delay_ms(
tx,
u32::try_from(interval.as_millis()).unwrap_or(u32::MAX),
)?;
for path in paths {
watcher.watch(path, RecursiveMode::Recursive)?;
debug!("Watching {:?}", path);
}
WatcherImpl::Poll(watcher)
} else {
let mut watcher = raw_watcher(tx)?;
for path in paths {
watcher.watch(path, RecursiveMode::Recursive)?;
debug!("Watching {:?}", path);
}
WatcherImpl::Recommended(watcher)
};
Ok(Self { watcher_impl: imp })
}
pub fn is_polling(&self) -> bool {
matches!(self.watcher_impl, WatcherImpl::Poll(_))
}
}