extern crate notify; use self::notify::Watcher; use clap::{App, ArgMatches, SubCommand}; use mdbook::errors::Result; use mdbook::utils; use mdbook::MDBook; use std::path::Path; use std::sync::mpsc::channel; use std::time::Duration; use {get_book_dir, open}; // Create clap subcommand arguments pub fn make_subcommand<'a, 'b>() -> App<'a, 'b> { SubCommand::with_name("watch") .about("Watches a book's files and rebuilds it on changes") .arg_from_usage( "-d, --dest-dir=[dest-dir] 'Output directory for the book{n}\ (If omitted, uses build.build-dir from book.toml or defaults to ./book)'", ) .arg_from_usage( "[dir] 'Root directory for the book{n}\ (Defaults to the Current Directory when omitted)'", ) .arg_from_usage("-o, --open 'Open the compiled book in a web browser'") } // Watch command implementation pub fn execute(args: &ArgMatches) -> Result<()> { let book_dir = get_book_dir(args); let book = MDBook::load(&book_dir)?; if args.is_present("open") { book.build()?; open(book.build_dir_for("html").join("index.html")); } trigger_on_change(&book, |path, book_dir| { info!("File changed: {:?}\nBuilding book...\n", path); let result = MDBook::load(&book_dir).and_then(|b| b.build()); if let Err(e) = result { error!("Unable to build the book"); utils::log_backtrace(&e); } }); Ok(()) } /// Calls the closure when a book source file is changed, blocking indefinitely. pub fn trigger_on_change(book: &MDBook, closure: F) where F: Fn(&Path, &Path), { use self::notify::DebouncedEvent::*; use self::notify::RecursiveMode::*; // Create a channel to receive the events. let (tx, rx) = channel(); let mut watcher = match notify::watcher(tx, Duration::from_secs(1)) { Ok(w) => w, Err(e) => { error!("Error while trying to watch the files:\n\n\t{:?}", e); ::std::process::exit(1) } }; // Add the source directory to the watcher if let Err(e) = watcher.watch(book.source_dir(), Recursive) { error!("Error while watching {:?}:\n {:?}", book.source_dir(), e); ::std::process::exit(1); }; let _ = watcher.watch(book.theme_dir(), Recursive); // Add the book.toml file to the watcher if it exists let _ = watcher.watch(book.root.join("book.toml"), NonRecursive); info!("Listening for changes..."); for event in rx.iter() { debug!("Received filesystem event: {:?}", event); match event { Create(path) | Write(path) | Remove(path) | Rename(_, path) => { closure(&path, &book.root); } _ => {} } } }