aboutsummaryrefslogtreecommitdiff
path: root/crates/ra_vfs/src/watcher.rs
blob: cc05f949ee9e2e011d205cfbc4ea442fd6db5010 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
use std::{
    path::{Path, PathBuf},
    sync::mpsc,
    thread,
    time::Duration,
};

use crossbeam_channel::Receiver;
use drop_bomb::DropBomb;
use notify::{DebouncedEvent, RecommendedWatcher, RecursiveMode, Watcher as NotifyWatcher};

pub struct Watcher {
    receiver: Receiver<WatcherChange>,
    watcher: RecommendedWatcher,
    thread: thread::JoinHandle<()>,
    bomb: DropBomb,
}

#[derive(Debug)]
pub enum WatcherChange {
    Create(PathBuf),
    Write(PathBuf),
    Remove(PathBuf),
    Rename(PathBuf, PathBuf),
}

impl WatcherChange {
    fn from_debounced_event(ev: DebouncedEvent) -> Option<WatcherChange> {
        match ev {
            DebouncedEvent::NoticeWrite(_)
            | DebouncedEvent::NoticeRemove(_)
            | DebouncedEvent::Chmod(_)
            | DebouncedEvent::Rescan => {
                // ignore
                None
            }
            DebouncedEvent::Create(path) => Some(WatcherChange::Create(path)),
            DebouncedEvent::Write(path) => Some(WatcherChange::Write(path)),
            DebouncedEvent::Remove(path) => Some(WatcherChange::Remove(path)),
            DebouncedEvent::Rename(src, dst) => Some(WatcherChange::Rename(src, dst)),
            DebouncedEvent::Error(err, path) => {
                // TODO
                log::warn!("watch error {}, {:?}", err, path);
                None
            }
        }
    }
}

impl Watcher {
    pub fn new() -> Result<Watcher, Box<std::error::Error>> {
        let (input_sender, input_receiver) = mpsc::channel();
        let watcher = notify::watcher(input_sender, Duration::from_millis(250))?;
        let (output_sender, output_receiver) = crossbeam_channel::unbounded();
        let thread = thread::spawn(move || loop {
            match input_receiver.recv() {
                Ok(ev) => {
                    // forward relevant events only
                    if let Some(change) = WatcherChange::from_debounced_event(ev) {
                        output_sender.send(change).unwrap();
                    }
                }
                Err(err) => {
                    log::debug!("Watcher stopped ({})", err);
                    break;
                }
            }
        });
        Ok(Watcher {
            receiver: output_receiver,
            watcher,
            thread,
            bomb: DropBomb::new(format!("Watcher was not shutdown")),
        })
    }

    pub fn watch(&mut self, root: impl AsRef<Path>) -> Result<(), Box<std::error::Error>> {
        self.watcher.watch(root, RecursiveMode::Recursive)?;
        Ok(())
    }

    pub fn change_receiver(&self) -> &Receiver<WatcherChange> {
        &self.receiver
    }

    pub fn shutdown(mut self) -> thread::Result<()> {
        self.bomb.defuse();
        drop(self.watcher);
        let res = self.thread.join();
        match &res {
            Ok(()) => log::info!("... Watcher terminated with ok"),
            Err(_) => log::error!("... Watcher terminated with err"),
        }
        res
    }
}