2019-08-03 05:42:33 +00:00
|
|
|
use crossbeam::channel::{select, unbounded, Receiver, Sender};
|
|
|
|
use lazy_static::lazy_static;
|
2020-01-27 23:59:59 +00:00
|
|
|
use notify::{Event, EventKind, RecommendedWatcher, RecursiveMode, Watcher as _};
|
2019-08-03 05:42:33 +00:00
|
|
|
use std::{
|
|
|
|
collections::HashMap,
|
|
|
|
path::PathBuf,
|
|
|
|
sync::{
|
|
|
|
atomic::{AtomicBool, Ordering},
|
|
|
|
Arc, Mutex, Weak,
|
|
|
|
},
|
|
|
|
thread,
|
|
|
|
};
|
2020-06-21 14:26:06 +00:00
|
|
|
use tracing::{error, warn};
|
2019-08-03 05:42:33 +00:00
|
|
|
|
|
|
|
type Handler = Box<dyn Fn() + Send>;
|
|
|
|
|
|
|
|
lazy_static! {
|
2019-08-03 22:33:59 +00:00
|
|
|
static ref WATCHER_TX: Mutex<Sender<(PathBuf, Handler, Weak<AtomicBool>)>> =
|
2019-08-03 05:42:33 +00:00
|
|
|
Mutex::new(Watcher::new().run());
|
|
|
|
}
|
|
|
|
|
2020-02-01 20:39:39 +00:00
|
|
|
// This will need to be adjusted when specifier mapping to asset location
|
|
|
|
// becomes more dynamic
|
2019-08-03 05:42:33 +00:00
|
|
|
struct Watcher {
|
|
|
|
watching: HashMap<PathBuf, (Handler, Vec<Weak<AtomicBool>>)>,
|
|
|
|
watcher: RecommendedWatcher,
|
|
|
|
event_rx: Receiver<Result<Event, notify::Error>>,
|
|
|
|
}
|
|
|
|
impl Watcher {
|
|
|
|
fn new() -> Self {
|
|
|
|
let (event_tx, event_rx) = unbounded();
|
|
|
|
Watcher {
|
|
|
|
watching: HashMap::new(),
|
2020-01-27 23:59:59 +00:00
|
|
|
watcher: notify::Watcher::new_immediate(move |event| {
|
|
|
|
let _ = event_tx.send(event);
|
|
|
|
})
|
|
|
|
.expect("Failed to create notify::Watcher"),
|
2019-08-03 05:42:33 +00:00
|
|
|
event_rx,
|
|
|
|
}
|
|
|
|
}
|
2020-02-01 20:39:39 +00:00
|
|
|
|
2019-08-03 05:42:33 +00:00
|
|
|
fn watch(&mut self, path: PathBuf, handler: Handler, signal: Weak<AtomicBool>) {
|
|
|
|
match self.watching.get_mut(&path) {
|
|
|
|
Some((_, ref mut v)) => {
|
|
|
|
if !v.iter().any(|s| match (s.upgrade(), signal.upgrade()) {
|
|
|
|
(Some(arc1), Some(arc2)) => Arc::ptr_eq(&arc1, &arc2),
|
|
|
|
_ => false,
|
|
|
|
}) {
|
|
|
|
v.push(signal);
|
|
|
|
}
|
2020-02-01 20:39:39 +00:00
|
|
|
},
|
2019-08-03 05:42:33 +00:00
|
|
|
None => {
|
2020-06-21 21:47:49 +00:00
|
|
|
if let Err(e) = self.watcher.watch(path.clone(), RecursiveMode::Recursive) {
|
|
|
|
warn!(?e, ?path, "Could not start watching file");
|
2019-08-03 22:33:59 +00:00
|
|
|
return;
|
|
|
|
}
|
2019-08-03 05:42:33 +00:00
|
|
|
self.watching.insert(path, (handler, vec![signal]));
|
2020-02-01 20:39:39 +00:00
|
|
|
},
|
2019-08-03 05:42:33 +00:00
|
|
|
}
|
|
|
|
}
|
2020-02-01 20:39:39 +00:00
|
|
|
|
2019-08-03 05:42:33 +00:00
|
|
|
fn handle_event(&mut self, event: Event) {
|
|
|
|
if let Event {
|
|
|
|
kind: EventKind::Modify(_),
|
|
|
|
paths,
|
|
|
|
..
|
|
|
|
} = event
|
|
|
|
{
|
|
|
|
for path in paths {
|
|
|
|
match self.watching.get_mut(&path) {
|
|
|
|
Some((reloader, ref mut signals)) => {
|
|
|
|
if !signals.is_empty() {
|
|
|
|
// Reload this file
|
|
|
|
reloader();
|
|
|
|
|
|
|
|
signals.retain(|signal| match signal.upgrade() {
|
|
|
|
Some(signal) => {
|
|
|
|
signal.store(true, Ordering::Release);
|
|
|
|
true
|
2020-02-01 20:39:39 +00:00
|
|
|
},
|
2019-08-03 05:42:33 +00:00
|
|
|
None => false,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
// If there is no one to signal stop watching this path
|
|
|
|
if signals.is_empty() {
|
2019-08-03 22:33:59 +00:00
|
|
|
if let Err(err) = self.watcher.unwatch(&path) {
|
|
|
|
warn!("Error unwatching: {}", err);
|
|
|
|
}
|
2019-08-03 05:42:33 +00:00
|
|
|
self.watching.remove(&path);
|
|
|
|
}
|
2020-02-01 20:39:39 +00:00
|
|
|
},
|
2019-08-03 05:42:33 +00:00
|
|
|
None => {
|
2020-02-01 20:39:39 +00:00
|
|
|
warn!(
|
|
|
|
"Watching {:#?} but there are no signals for this path. The path will \
|
|
|
|
be unwatched.",
|
|
|
|
path
|
|
|
|
);
|
2019-08-03 22:33:59 +00:00
|
|
|
if let Err(err) = self.watcher.unwatch(&path) {
|
|
|
|
warn!("Error unwatching: {}", err);
|
|
|
|
}
|
2020-02-01 20:39:39 +00:00
|
|
|
},
|
2019-08-03 05:42:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-02-01 20:39:39 +00:00
|
|
|
|
2020-06-10 19:47:36 +00:00
|
|
|
#[allow(clippy::drop_copy)] // TODO: Pending review in #587
|
|
|
|
#[allow(clippy::single_match)] // TODO: Pending review in #587
|
|
|
|
#[allow(clippy::zero_ptr)] // TODO: Pending review in #587
|
2019-08-03 05:42:33 +00:00
|
|
|
fn run(mut self) -> Sender<(PathBuf, Handler, Weak<AtomicBool>)> {
|
|
|
|
let (watch_tx, watch_rx) = unbounded();
|
|
|
|
|
|
|
|
thread::spawn(move || {
|
|
|
|
loop {
|
|
|
|
select! {
|
|
|
|
recv(watch_rx) -> res => match res {
|
|
|
|
Ok((path, handler, signal)) => self.watch(path, handler, signal),
|
|
|
|
// Disconnected
|
|
|
|
Err(_) => (),
|
|
|
|
},
|
|
|
|
recv(self.event_rx) -> res => match res {
|
|
|
|
Ok(Ok(event)) => self.handle_event(event),
|
|
|
|
// Notify Error
|
2020-06-21 21:47:49 +00:00
|
|
|
Ok(Err(e)) => error!(?e, "Notify error"),
|
2019-08-03 05:42:33 +00:00
|
|
|
// Disconnected
|
|
|
|
Err(_) => (),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
watch_tx
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct ReloadIndicator {
|
|
|
|
reloaded: Arc<AtomicBool>,
|
|
|
|
// Paths that have already been added
|
|
|
|
paths: Vec<PathBuf>,
|
|
|
|
}
|
|
|
|
impl ReloadIndicator {
|
2020-06-10 19:47:36 +00:00
|
|
|
#[allow(clippy::new_without_default)] // TODO: Pending review in #587
|
2019-08-03 05:42:33 +00:00
|
|
|
pub fn new() -> Self {
|
|
|
|
Self {
|
|
|
|
reloaded: Arc::new(AtomicBool::new(false)),
|
|
|
|
paths: Vec::new(),
|
|
|
|
}
|
|
|
|
}
|
2020-02-01 20:39:39 +00:00
|
|
|
|
2019-08-03 05:42:33 +00:00
|
|
|
pub fn add<F>(&mut self, path: PathBuf, reloader: F)
|
|
|
|
where
|
|
|
|
F: 'static + Fn() + Send,
|
|
|
|
{
|
|
|
|
// Check to see if this was already added
|
|
|
|
if self.paths.iter().any(|p| *p == path) {
|
|
|
|
// Nothing else needs to be done
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
self.paths.push(path.clone());
|
|
|
|
};
|
|
|
|
|
2019-08-03 22:33:59 +00:00
|
|
|
if WATCHER_TX
|
2019-08-03 05:42:33 +00:00
|
|
|
.lock()
|
|
|
|
.unwrap()
|
2019-08-03 22:33:59 +00:00
|
|
|
.send((path, Box::new(reloader), Arc::downgrade(&self.reloaded)))
|
|
|
|
.is_err()
|
|
|
|
{
|
|
|
|
error!("Could not add. Asset watcher channel disconnected.");
|
|
|
|
}
|
2019-08-03 05:42:33 +00:00
|
|
|
}
|
2020-02-01 20:39:39 +00:00
|
|
|
|
2019-08-03 05:42:33 +00:00
|
|
|
// Returns true if the watched file was changed
|
2020-02-01 20:39:39 +00:00
|
|
|
pub fn reloaded(&self) -> bool { self.reloaded.swap(false, Ordering::Acquire) }
|
2019-08-03 05:42:33 +00:00
|
|
|
}
|