use std::{ path::{Path, PathBuf}, process::{Child, ChildStdin, Command, Stdio}, sync::{ atomic::{AtomicUsize, Ordering}, Arc, }, }; use anyhow::{anyhow, Context, Result}; use indoc::indoc; use tree_sitter::{Parser, Tree}; use tree_sitter_config::Config; use tree_sitter_loader::Config as LoaderConfig; const HTML_HEADER: &[u8] = b" "; #[must_use] pub fn lang_not_found_for_path(path: &Path, loader_config: &LoaderConfig) -> String { let path = path.display(); format!( indoc! {" No language found for path `{}` If a language should be associated with this file extension, please ensure the path to `{}` is inside one of the following directories as specified by your 'config.json':\n\n{}\n If the directory that contains the relevant grammar for `{}` is not listed above, please add the directory to the list of directories in your config file, {} "}, path, path, loader_config .parser_directories .iter() .enumerate() .map(|(i, d)| format!(" {}. {}", i + 1, d.display())) .collect::>() .join(" \n"), path, if let Ok(Some(config_path)) = Config::find_config_file() { format!("located at {}", config_path.display()) } else { String::from("which you need to create by running `tree-sitter init-config`") } ) } #[must_use] pub fn cancel_on_signal() -> Arc { let result = Arc::new(AtomicUsize::new(0)); ctrlc::set_handler({ let flag = result.clone(); move || { flag.store(1, Ordering::Relaxed); } }) .expect("Error setting Ctrl-C handler"); result } pub struct LogSession { path: PathBuf, dot_process: Option, dot_process_stdin: Option, open_log: bool, } pub fn print_tree_graph(tree: &Tree, path: &str, quiet: bool) -> Result<()> { let session = LogSession::new(path, quiet)?; tree.print_dot_graph(session.dot_process_stdin.as_ref().unwrap()); Ok(()) } pub fn log_graphs(parser: &mut Parser, path: &str, open_log: bool) -> Result { let session = LogSession::new(path, open_log)?; parser.print_dot_graphs(session.dot_process_stdin.as_ref().unwrap()); Ok(session) } impl LogSession { fn new(path: &str, open_log: bool) -> Result { use std::io::Write; let mut dot_file = std::fs::File::create(path)?; dot_file.write_all(HTML_HEADER)?; let mut dot_process = Command::new("dot") .arg("-Tsvg") .stdin(Stdio::piped()) .stdout(dot_file) .spawn() .with_context(|| { "Failed to run the `dot` command. Check that graphviz is installed." })?; let dot_stdin = dot_process .stdin .take() .ok_or_else(|| anyhow!("Failed to open stdin for `dot` process."))?; Ok(Self { path: PathBuf::from(path), dot_process: Some(dot_process), dot_process_stdin: Some(dot_stdin), open_log, }) } } impl Drop for LogSession { fn drop(&mut self) { use std::fs; drop(self.dot_process_stdin.take().unwrap()); let output = self.dot_process.take().unwrap().wait_with_output().unwrap(); if output.status.success() { if self.open_log && fs::metadata(&self.path).unwrap().len() > HTML_HEADER.len() as u64 { webbrowser::open(&self.path.to_string_lossy()).unwrap(); } } else { eprintln!( "Dot failed: {} {}", String::from_utf8_lossy(&output.stdout), String::from_utf8_lossy(&output.stderr) ); } } }