use std::{ path::{PathBuf, Path}, fs, }; use crossbeam_channel::{Sender, Receiver}; use walkdir::WalkDir; use { thread_watcher::{ThreadWatcher, worker_chan}, }; #[derive(Debug)] pub struct FileEvent { pub path: PathBuf, pub kind: FileEventKind, } #[derive(Debug)] pub enum FileEventKind { Add(String), } pub fn roots_loader() -> ((Sender, Receiver<(PathBuf, Vec)>), ThreadWatcher) { let (interface, input_receiver, output_sender) = worker_chan::)>(128); let thread = ThreadWatcher::spawn("roots loader", move || { input_receiver .into_iter() .map(|path| { debug!("loading {} ...", path.as_path().display()); let events = load_root(path.as_path()); debug!("... loaded {}", path.as_path().display()); (path, events) }) .for_each(|it| output_sender.send(it)) }); (interface, thread) } fn load_root(path: &Path) -> Vec { let mut res = Vec::new(); for entry in WalkDir::new(path) { let entry = match entry { Ok(entry) => entry, Err(e) => { warn!("watcher error: {}", e); continue; } }; if !entry.file_type().is_file() { continue; } let path = entry.path(); if path.extension().and_then(|os| os.to_str()) != Some("rs") { continue; } let text = match fs::read_to_string(path) { Ok(text) => text, Err(e) => { warn!("watcher error: {}", e); continue; } }; res.push(FileEvent { path: path.to_owned(), kind: FileEventKind::Add(text), }) } res }