mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-13 13:48:50 +00:00
handle recursive watching ourselves
This commit is contained in:
parent
e69b620f0d
commit
f181e36a44
4 changed files with 78 additions and 16 deletions
32
Cargo.lock
generated
32
Cargo.lock
generated
|
@ -446,6 +446,18 @@ name = "glob"
|
||||||
version = "0.2.11"
|
version = "0.2.11"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "globset"
|
||||||
|
version = "0.4.2"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
dependencies = [
|
||||||
|
"aho-corasick 0.6.9 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"fnv 1.0.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"log 0.4.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"memchr 2.1.2 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"regex 1.1.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "heck"
|
name = "heck"
|
||||||
version = "0.3.1"
|
version = "0.3.1"
|
||||||
|
@ -469,6 +481,23 @@ dependencies = [
|
||||||
"unicode-normalization 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)",
|
"unicode-normalization 0.1.8 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "ignore"
|
||||||
|
version = "0.4.6"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
dependencies = [
|
||||||
|
"crossbeam-channel 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"globset 0.4.2 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"lazy_static 1.2.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"log 0.4.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"memchr 2.1.2 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"regex 1.1.0 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"same-file 1.0.4 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"thread_local 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"walkdir 2.2.7 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"winapi-util 0.1.1 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "im"
|
name = "im"
|
||||||
version = "12.3.0"
|
version = "12.3.0"
|
||||||
|
@ -1010,6 +1039,7 @@ dependencies = [
|
||||||
"crossbeam-channel 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
"crossbeam-channel 0.3.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"drop_bomb 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)",
|
"drop_bomb 0.1.4 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"flexi_logger 0.10.5 (registry+https://github.com/rust-lang/crates.io-index)",
|
"flexi_logger 0.10.5 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
|
"ignore 0.4.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"log 0.4.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
"log 0.4.6 (registry+https://github.com/rust-lang/crates.io-index)",
|
||||||
"notify 4.0.6 (git+https://github.com/vemoo/notify/?branch=v4-legacy)",
|
"notify 4.0.6 (git+https://github.com/vemoo/notify/?branch=v4-legacy)",
|
||||||
"ra_arena 0.1.0",
|
"ra_arena 0.1.0",
|
||||||
|
@ -1868,9 +1898,11 @@ dependencies = [
|
||||||
"checksum futures 0.1.25 (registry+https://github.com/rust-lang/crates.io-index)" = "49e7653e374fe0d0c12de4250f0bdb60680b8c80eed558c5c7538eec9c89e21b"
|
"checksum futures 0.1.25 (registry+https://github.com/rust-lang/crates.io-index)" = "49e7653e374fe0d0c12de4250f0bdb60680b8c80eed558c5c7538eec9c89e21b"
|
||||||
"checksum generic-array 0.9.0 (registry+https://github.com/rust-lang/crates.io-index)" = "ef25c5683767570c2bbd7deba372926a55eaae9982d7726ee2a1050239d45b9d"
|
"checksum generic-array 0.9.0 (registry+https://github.com/rust-lang/crates.io-index)" = "ef25c5683767570c2bbd7deba372926a55eaae9982d7726ee2a1050239d45b9d"
|
||||||
"checksum glob 0.2.11 (registry+https://github.com/rust-lang/crates.io-index)" = "8be18de09a56b60ed0edf84bc9df007e30040691af7acd1c41874faac5895bfb"
|
"checksum glob 0.2.11 (registry+https://github.com/rust-lang/crates.io-index)" = "8be18de09a56b60ed0edf84bc9df007e30040691af7acd1c41874faac5895bfb"
|
||||||
|
"checksum globset 0.4.2 (registry+https://github.com/rust-lang/crates.io-index)" = "4743617a7464bbda3c8aec8558ff2f9429047e025771037df561d383337ff865"
|
||||||
"checksum heck 0.3.1 (registry+https://github.com/rust-lang/crates.io-index)" = "20564e78d53d2bb135c343b3f47714a56af2061f1c928fdb541dc7b9fdd94205"
|
"checksum heck 0.3.1 (registry+https://github.com/rust-lang/crates.io-index)" = "20564e78d53d2bb135c343b3f47714a56af2061f1c928fdb541dc7b9fdd94205"
|
||||||
"checksum humansize 1.1.0 (registry+https://github.com/rust-lang/crates.io-index)" = "b6cab2627acfc432780848602f3f558f7e9dd427352224b0d9324025796d2a5e"
|
"checksum humansize 1.1.0 (registry+https://github.com/rust-lang/crates.io-index)" = "b6cab2627acfc432780848602f3f558f7e9dd427352224b0d9324025796d2a5e"
|
||||||
"checksum idna 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)" = "38f09e0f0b1fb55fdee1f17470ad800da77af5186a1a76c026b679358b7e844e"
|
"checksum idna 0.1.5 (registry+https://github.com/rust-lang/crates.io-index)" = "38f09e0f0b1fb55fdee1f17470ad800da77af5186a1a76c026b679358b7e844e"
|
||||||
|
"checksum ignore 0.4.6 (registry+https://github.com/rust-lang/crates.io-index)" = "ad03ca67dc12474ecd91fdb94d758cbd20cb4e7a78ebe831df26a9b7511e1162"
|
||||||
"checksum im 12.3.0 (registry+https://github.com/rust-lang/crates.io-index)" = "0627d417829c1d763d602687634869f254fc79f7e22dea6c824dab993db857e4"
|
"checksum im 12.3.0 (registry+https://github.com/rust-lang/crates.io-index)" = "0627d417829c1d763d602687634869f254fc79f7e22dea6c824dab993db857e4"
|
||||||
"checksum indexmap 1.0.2 (registry+https://github.com/rust-lang/crates.io-index)" = "7e81a7c05f79578dbc15793d8b619db9ba32b4577003ef3af1a91c416798c58d"
|
"checksum indexmap 1.0.2 (registry+https://github.com/rust-lang/crates.io-index)" = "7e81a7c05f79578dbc15793d8b619db9ba32b4577003ef3af1a91c416798c58d"
|
||||||
"checksum inotify 0.6.1 (registry+https://github.com/rust-lang/crates.io-index)" = "40b54539f3910d6f84fbf9a643efd6e3aa6e4f001426c0329576128255994718"
|
"checksum inotify 0.6.1 (registry+https://github.com/rust-lang/crates.io-index)" = "40b54539f3910d6f84fbf9a643efd6e3aa6e4f001426c0329576128255994718"
|
||||||
|
|
|
@ -12,6 +12,7 @@ crossbeam-channel = "0.3.5"
|
||||||
log = "0.4.6"
|
log = "0.4.6"
|
||||||
# until https://github.com/passcod/notify/issues/169 is fixed
|
# until https://github.com/passcod/notify/issues/169 is fixed
|
||||||
notify = { git = "https://github.com/vemoo/notify/", branch = "v4-legacy" }
|
notify = { git = "https://github.com/vemoo/notify/", branch = "v4-legacy" }
|
||||||
|
ignore = "0.4"
|
||||||
drop_bomb = "0.1.0"
|
drop_bomb = "0.1.0"
|
||||||
|
|
||||||
thread_worker = { path = "../thread_worker" }
|
thread_worker = { path = "../thread_worker" }
|
||||||
|
|
|
@ -141,9 +141,7 @@ impl Vfs {
|
||||||
};
|
};
|
||||||
res.worker.inp.send(task).unwrap();
|
res.worker.inp.send(task).unwrap();
|
||||||
if let Some(ref mut watcher) = res.watcher {
|
if let Some(ref mut watcher) = res.watcher {
|
||||||
if let Err(e) = watcher.watch(path) {
|
watcher.watch(path);
|
||||||
log::warn!("could not watch \"{}\": {}", path.display(), e);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
let roots = res.roots.iter().map(|(id, _)| id).collect();
|
let roots = res.roots.iter().map(|(id, _)| id).collect();
|
||||||
|
|
|
@ -1,17 +1,17 @@
|
||||||
|
use crate::io;
|
||||||
|
use crossbeam_channel::Sender;
|
||||||
|
use drop_bomb::DropBomb;
|
||||||
|
use ignore;
|
||||||
|
use notify::{DebouncedEvent, RecommendedWatcher, RecursiveMode, Watcher as NotifyWatcher};
|
||||||
use std::{
|
use std::{
|
||||||
path::{Path, PathBuf},
|
path::{Path, PathBuf},
|
||||||
sync::mpsc,
|
sync::{mpsc, Arc, Mutex},
|
||||||
thread,
|
thread,
|
||||||
time::Duration,
|
time::Duration,
|
||||||
};
|
};
|
||||||
|
|
||||||
use crate::io;
|
|
||||||
use crossbeam_channel::Sender;
|
|
||||||
use drop_bomb::DropBomb;
|
|
||||||
use notify::{DebouncedEvent, RecommendedWatcher, RecursiveMode, Watcher as NotifyWatcher};
|
|
||||||
|
|
||||||
pub struct Watcher {
|
pub struct Watcher {
|
||||||
watcher: RecommendedWatcher,
|
watcher: Arc<Mutex<RecommendedWatcher>>,
|
||||||
thread: thread::JoinHandle<()>,
|
thread: thread::JoinHandle<()>,
|
||||||
bomb: DropBomb,
|
bomb: DropBomb,
|
||||||
}
|
}
|
||||||
|
@ -24,9 +24,10 @@ pub enum WatcherChange {
|
||||||
Rescan,
|
Rescan,
|
||||||
}
|
}
|
||||||
|
|
||||||
fn send_change_events(
|
fn handle_change_event(
|
||||||
ev: DebouncedEvent,
|
ev: DebouncedEvent,
|
||||||
sender: &Sender<io::Task>,
|
sender: &Sender<io::Task>,
|
||||||
|
watcher: &Arc<Mutex<RecommendedWatcher>>,
|
||||||
) -> Result<(), Box<std::error::Error>> {
|
) -> Result<(), Box<std::error::Error>> {
|
||||||
match ev {
|
match ev {
|
||||||
DebouncedEvent::NoticeWrite(_)
|
DebouncedEvent::NoticeWrite(_)
|
||||||
|
@ -38,6 +39,9 @@ fn send_change_events(
|
||||||
sender.send(io::Task::HandleChange(WatcherChange::Rescan))?;
|
sender.send(io::Task::HandleChange(WatcherChange::Rescan))?;
|
||||||
}
|
}
|
||||||
DebouncedEvent::Create(path) => {
|
DebouncedEvent::Create(path) => {
|
||||||
|
if path.is_dir() {
|
||||||
|
watch_recursive(watcher, &path);
|
||||||
|
}
|
||||||
sender.send(io::Task::HandleChange(WatcherChange::Create(path)))?;
|
sender.send(io::Task::HandleChange(WatcherChange::Create(path)))?;
|
||||||
}
|
}
|
||||||
DebouncedEvent::Write(path) => {
|
DebouncedEvent::Write(path) => {
|
||||||
|
@ -58,17 +62,45 @@ fn send_change_events(
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn watch_one(watcher: &mut RecommendedWatcher, path: &Path) {
|
||||||
|
match watcher.watch(path, RecursiveMode::NonRecursive) {
|
||||||
|
Ok(()) => log::debug!("watching \"{}\"", path.display()),
|
||||||
|
Err(e) => log::warn!("could not watch \"{}\": {}", path.display(), e),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn watch_recursive(watcher: &Arc<Mutex<RecommendedWatcher>>, path: &Path) {
|
||||||
|
log::debug!("watch_recursive \"{}\"", path.display());
|
||||||
|
let mut w = watcher.lock().unwrap();
|
||||||
|
// TODO it seems path itself isn't checked against ignores
|
||||||
|
// check if path should be ignored before walking it
|
||||||
|
for res in ignore::Walk::new(path) {
|
||||||
|
match res {
|
||||||
|
Ok(entry) => {
|
||||||
|
if entry.path().is_dir() {
|
||||||
|
watch_one(&mut w, entry.path());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(e) => log::warn!("watcher error: {}", e),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl Watcher {
|
impl Watcher {
|
||||||
pub(crate) fn start(
|
pub(crate) fn start(
|
||||||
output_sender: Sender<io::Task>,
|
output_sender: Sender<io::Task>,
|
||||||
) -> Result<Watcher, Box<std::error::Error>> {
|
) -> Result<Watcher, Box<std::error::Error>> {
|
||||||
let (input_sender, input_receiver) = mpsc::channel();
|
let (input_sender, input_receiver) = mpsc::channel();
|
||||||
let watcher = notify::watcher(input_sender, Duration::from_millis(250))?;
|
let watcher = Arc::new(Mutex::new(notify::watcher(
|
||||||
|
input_sender,
|
||||||
|
Duration::from_millis(250),
|
||||||
|
)?));
|
||||||
|
let w = watcher.clone();
|
||||||
let thread = thread::spawn(move || {
|
let thread = thread::spawn(move || {
|
||||||
input_receiver
|
input_receiver
|
||||||
.into_iter()
|
.into_iter()
|
||||||
// forward relevant events only
|
// forward relevant events only
|
||||||
.try_for_each(|change| send_change_events(change, &output_sender))
|
.try_for_each(|change| handle_change_event(change, &output_sender, &w))
|
||||||
.unwrap()
|
.unwrap()
|
||||||
});
|
});
|
||||||
Ok(Watcher {
|
Ok(Watcher {
|
||||||
|
@ -78,9 +110,8 @@ impl Watcher {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn watch(&mut self, root: impl AsRef<Path>) -> Result<(), Box<std::error::Error>> {
|
pub fn watch(&mut self, root: impl AsRef<Path>) {
|
||||||
self.watcher.watch(root, RecursiveMode::Recursive)?;
|
watch_recursive(&self.watcher, root.as_ref());
|
||||||
Ok(())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn shutdown(mut self) -> thread::Result<()> {
|
pub fn shutdown(mut self) -> thread::Result<()> {
|
||||||
|
|
Loading…
Reference in a new issue