// Copyright 2018-2022 the Deno authors. All rights reserved. MIT license. use crate::colors; use crate::fs_util::canonicalize_path; use deno_core::error::AnyError; use deno_core::futures::Future; use log::info; use notify::event::Event as NotifyEvent; use notify::event::EventKind; use notify::Config; use notify::Error as NotifyError; use notify::RecommendedWatcher; use notify::RecursiveMode; use notify::Watcher; use std::collections::HashSet; use std::path::PathBuf; use std::sync::Arc; use std::time::Duration; use tokio::select; use tokio::sync::mpsc; use tokio::time::sleep; const CLEAR_SCREEN: &str = "\x1B[2J\x1B[1;1H"; const DEBOUNCE_INTERVAL: Duration = Duration::from_millis(200); struct DebouncedReceiver { receiver: mpsc::UnboundedReceiver>, } impl DebouncedReceiver { fn new_with_sender() -> (Arc>>, Self) { let (sender, receiver) = mpsc::unbounded_channel(); (Arc::new(sender), Self { receiver }) } async fn recv(&mut self) -> Option> { let mut received_items = self .receiver .recv() .await? .into_iter() .collect::>(); // prevent duplicates loop { tokio::select! { items = self.receiver.recv() => { received_items.extend(items?); } _ = sleep(DEBOUNCE_INTERVAL) => { return Some(received_items.into_iter().collect()); } } } } } async fn error_handler(watch_future: F) where F: Future>, { let result = watch_future.await; if let Err(err) = result { let msg = format!("{}: {}", colors::red_bold("error"), err.to_string(),); eprintln!("{}", msg); } } pub enum ResolutionResult { Restart { paths_to_watch: Vec, result: Result, }, Ignore, } async fn next_restart( resolver: &mut R, debounced_receiver: &mut DebouncedReceiver, ) -> (Vec, Result) where R: FnMut(Option>) -> F, F: Future>, { loop { let changed = debounced_receiver.recv().await; match resolver(changed).await { ResolutionResult::Ignore => { log::debug!("File change ignored") } ResolutionResult::Restart { paths_to_watch, result, } => { // Clear screen first eprint!("{}", CLEAR_SCREEN); info!( "{} File change detected! Restarting!", colors::intense_blue("Watcher"), ); return (paths_to_watch, result); } } } } /// Creates a file watcher, which will call `resolver` with every file change. /// /// - `resolver` is used for resolving file paths to be watched at every restarting /// of the watcher, and can also return a value to be passed to `operation`. /// It returns a [`ResolutionResult`], which can either instruct the watcher to restart or ignore the change. /// This always contains paths to watch; /// /// - `operation` is the actual operation we want to run every time the watcher detects file /// changes. For example, in the case where we would like to bundle, then `operation` would /// have the logic for it like bundling the code. /// /// - `job_name` is just used for printing watcher status to terminal. pub async fn watch_func( mut resolver: R, mut operation: O, job_name: &str, ) -> Result<(), AnyError> where R: FnMut(Option>) -> F1, O: FnMut(T) -> F2, F1: Future>, F2: Future>, { let (sender, mut receiver) = DebouncedReceiver::new_with_sender(); // Store previous data. If module resolution fails at some point, the watcher will try to // continue watching files using these data. let mut paths_to_watch; let mut resolution_result; match resolver(None).await { ResolutionResult::Ignore => { // The only situation where it makes sense to ignore the initial 'change' // is if the command isn't supposed to do anything until something changes, // e.g. a variant of `deno test` which doesn't run the entire test suite to start with, // but instead does nothing until you make a change. // // In that case, this is probably the correct output. info!( "{} Waiting for file changes...", colors::intense_blue("Watcher"), ); let (paths, result) = next_restart(&mut resolver, &mut receiver).await; paths_to_watch = paths; resolution_result = result; } ResolutionResult::Restart { paths_to_watch: paths, result, } => { paths_to_watch = paths; resolution_result = result; } }; // Clear screen first eprint!("{}", CLEAR_SCREEN); info!("{} {} started.", colors::intense_blue("Watcher"), job_name,); loop { let watcher = new_watcher(&paths_to_watch, sender.clone())?; match resolution_result { Ok(operation_arg) => { let fut = error_handler(operation(operation_arg)); select! { (paths, result) = next_restart(&mut resolver, &mut receiver) => { if result.is_ok() { paths_to_watch = paths; } resolution_result = result; continue; }, _ = fut => {}, }; info!( "{} {} finished. Restarting on file change...", colors::intense_blue("Watcher"), job_name, ); } Err(error) => { eprintln!("{}: {}", colors::red_bold("error"), error); info!( "{} {} failed. Restarting on file change...", colors::intense_blue("Watcher"), job_name, ); } } let (paths, result) = next_restart(&mut resolver, &mut receiver).await; if result.is_ok() { paths_to_watch = paths; } resolution_result = result; drop(watcher); } } fn new_watcher( paths: &[PathBuf], sender: Arc>>, ) -> Result { let mut watcher: RecommendedWatcher = Watcher::new(move |res: Result| { if let Ok(event) = res { if matches!( event.kind, EventKind::Create(_) | EventKind::Modify(_) | EventKind::Remove(_) ) { let paths = event .paths .iter() .filter_map(|path| canonicalize_path(path).ok()) .collect(); sender.send(paths).unwrap(); } } })?; watcher.configure(Config::PreciseEvents(true)).unwrap(); log::debug!("Watching paths: {:?}", paths); for path in paths { // Ignore any error e.g. `PathNotFound` let _ = watcher.watch(path, RecursiveMode::Recursive); } Ok(watcher) }