Unnamed repository; edit this file 'description' to name the repository.
Diffstat (limited to 'helix-term/src/health.rs')
| -rw-r--r-- | helix-term/src/health.rs | 419 |
1 files changed, 91 insertions, 328 deletions
diff --git a/helix-term/src/health.rs b/helix-term/src/health.rs index 52112bf9..f13d35f0 100644 --- a/helix-term/src/health.rs +++ b/helix-term/src/health.rs @@ -1,42 +1,24 @@ -use crate::config::{Config, ConfigLoadError}; -use helix_core::config::{default_lang_config, user_lang_config}; +use crossterm::style::{Color, Print, Stylize}; +use helix_core::config::{default_syntax_loader, user_syntax_loader}; use helix_loader::grammar::load_runtime_file; -use std::{ - collections::HashSet, - io::{IsTerminal, Write}, -}; -use termina::{ - style::{ColorSpec, StyleExt as _, Stylized}, - Terminal as _, -}; #[derive(Copy, Clone)] pub enum TsFeature { Highlight, TextObject, AutoIndent, - Tags, - RainbowBracket, } impl TsFeature { pub fn all() -> &'static [Self] { - &[ - Self::Highlight, - Self::TextObject, - Self::AutoIndent, - Self::Tags, - Self::RainbowBracket, - ] + &[Self::Highlight, Self::TextObject, Self::AutoIndent] } pub fn runtime_filename(&self) -> &'static str { match *self { Self::Highlight => "highlights.scm", Self::TextObject => "textobjects.scm", - Self::AutoIndent => "indents.scm", - Self::Tags => "tags.scm", - Self::RainbowBracket => "rainbows.scm", + Self::AutoIndent => "indents.toml", } } @@ -45,8 +27,6 @@ impl TsFeature { Self::Highlight => "Syntax Highlighting", Self::TextObject => "Treesitter Textobjects", Self::AutoIndent => "Auto Indent", - Self::Tags => "Code Navigation Tags", - Self::RainbowBracket => "Rainbow Brackets", } } @@ -55,250 +35,121 @@ impl TsFeature { Self::Highlight => "Highlight", Self::TextObject => "Textobject", Self::AutoIndent => "Indent", - Self::Tags => "Tags", - Self::RainbowBracket => "Rainbow", } } } /// Display general diagnostics. -pub fn general() -> std::io::Result<()> { - let stdout = std::io::stdout(); - let mut stdout = stdout.lock(); - +pub fn general() { let config_file = helix_loader::config_file(); let lang_file = helix_loader::lang_config_file(); let log_file = helix_loader::log_file(); - let rt_dirs = helix_loader::runtime_dirs(); + let rt_dir = helix_loader::runtime_dir(); if config_file.exists() { - writeln!(stdout, "Config file: {}", config_file.display())?; + println!("Config file: {}", config_file.display()); } else { - writeln!(stdout, "Config file: default")?; + println!("Config file: default") } if lang_file.exists() { - writeln!(stdout, "Language file: {}", lang_file.display())?; + println!("Language file: {}", lang_file.display()); } else { - writeln!(stdout, "Language file: default")?; + println!("Language file: default") } - writeln!(stdout, "Log file: {}", log_file.display())?; - writeln!( - stdout, - "Runtime directories: {}", - rt_dirs - .iter() - .map(|d| d.to_string_lossy()) - .collect::<Vec<_>>() - .join(";") - )?; - for rt_dir in rt_dirs.iter() { - if let Ok(path) = std::fs::read_link(rt_dir) { - let msg = format!( - "Runtime directory {} is symlinked to: {}", - rt_dir.display(), - path.display() - ); - writeln!(stdout, "{}", msg.yellow())?; - } - if !rt_dir.exists() { - let msg = format!("Runtime directory does not exist: {}", rt_dir.display()); - writeln!(stdout, "{}", msg.yellow())?; - } else if rt_dir.read_dir().ok().map(|it| it.count()) == Some(0) { - let msg = format!("Runtime directory is empty: {}", rt_dir.display()); - writeln!(stdout, "{}", msg.yellow())?; - } - } - - Ok(()) -} - -pub fn clipboard() -> std::io::Result<()> { - let stdout = std::io::stdout(); - let mut stdout = stdout.lock(); + println!("Log file: {}", log_file.display()); + println!("Runtime directory: {}", rt_dir.display()); - let config = match Config::load_default() { - Ok(config) => config, - Err(ConfigLoadError::Error(err)) if err.kind() == std::io::ErrorKind::NotFound => { - Config::default() - } - Err(err) => { - writeln!(stdout, "{}", "Configuration file malformed".red())?; - writeln!(stdout, "{}", err)?; - return Ok(()); - } - }; - - match config.editor.clipboard_provider.name().as_ref() { - "none" => { - writeln!( - stdout, - "{}", - "System clipboard provider: Not installed".red() - )?; - writeln!( - stdout, - " {}", - "For troubleshooting system clipboard issues, refer".red() - )?; - writeln!(stdout, " {}", - "https://github.com/helix-editor/helix/wiki/Troubleshooting#copypaste-fromto-system-clipboard-not-working" - .red().underlined())?; - } - name => writeln!(stdout, "System clipboard provider: {}", name)?, + if let Ok(path) = std::fs::read_link(&rt_dir) { + let msg = format!("Runtime directory is symlinked to {}", path.display()); + println!("{}", msg.yellow()); + } + if !rt_dir.exists() { + println!("{}", "Runtime directory does not exist.".red()); + } + if rt_dir.read_dir().ok().map(|it| it.count()) == Some(0) { + println!("{}", "Runtime directory is empty.".red()); } - - Ok(()) -} - -pub fn languages_all() -> std::io::Result<()> { - languages(None) -} - -pub fn languages_selection() -> std::io::Result<()> { - let selection = helix_loader::grammar::get_grammar_names().unwrap_or_default(); - languages(selection) } -fn languages(selection: Option<HashSet<String>>) -> std::io::Result<()> { - let stdout = std::io::stdout(); - let mut stdout = stdout.lock(); +pub fn languages_all() { + let mut syn_loader_conf = user_syntax_loader().unwrap_or_else(|err| { + eprintln!("{}: {}", "Error parsing user language config".red(), err); + eprintln!("{}", "Using default language config".yellow()); + default_syntax_loader() + }); - let mut syn_loader_conf = match user_lang_config() { - Ok(conf) => conf, - Err(err) => { - let stderr = std::io::stderr(); - let mut stderr = stderr.lock(); - - writeln!( - stderr, - "{}: {}", - "Error parsing user language config".red(), - err - )?; - writeln!(stderr, "{}", "Using default language config".yellow())?; - default_lang_config() - } - }; - - let mut headings = vec!["Language", "Language servers", "Debug adapter", "Formatter"]; + let mut headings = vec!["Language", "LSP", "DAP"]; for feat in TsFeature::all() { headings.push(feat.short_title()) } - let terminal_cols = termina::PlatformTerminal::new() - .and_then(|terminal| terminal.get_dimensions()) - .map(|size| size.cols) - .unwrap_or(80); + let terminal_cols = crossterm::terminal::size().map(|(c, _)| c).unwrap_or(80); let column_width = terminal_cols as usize / headings.len(); - let is_terminal = std::io::stdout().is_terminal(); - let fit = |s: &str| -> Stylized<'static> { - format!( + let column = |item: &str, color: Color| { + let data = format!( "{:column_width$}", - s.get(..column_width - 2) - .map(|s| format!("{}…", s)) - .unwrap_or_else(|| s.to_string()) + item.get(..column_width - 2) + .map(|s| format!("{s}…")) + .unwrap_or_else(|| item.to_string()) ) - .stylized() + .stylize() + .with(color); + + // We can't directly use println!() because of + // https://github.com/crossterm-rs/crossterm/issues/589 + let _ = crossterm::execute!(std::io::stdout(), Print(data)); }; - let color = |s: Stylized<'static>, c: ColorSpec| if is_terminal { s.foreground(c) } else { s }; - let bold = |s: Stylized<'static>| if is_terminal { s.bold() } else { s }; for heading in headings { - write!(stdout, "{}", bold(fit(heading)))?; + column(heading, Color::White); } - writeln!(stdout)?; + println!(); syn_loader_conf .language .sort_unstable_by_key(|l| l.language_id.clone()); - let check_binary_with_name = |cmd: Option<(&str, &str)>| match cmd { - Some((name, cmd)) => match helix_stdx::env::which(cmd) { - Ok(_) => color(fit(&format!("✓ {}", name)), ColorSpec::BRIGHT_GREEN), - Err(_) => color(fit(&format!("✘ {}", name)), ColorSpec::BRIGHT_RED), + let check_binary = |cmd: Option<String>| match cmd { + Some(cmd) => match which::which(&cmd) { + Ok(_) => column(&cmd, Color::Green), + Err(_) => column(&cmd, Color::Red), }, - None => color(fit("None"), ColorSpec::BRIGHT_YELLOW), + None => column("None", Color::Yellow), }; - let check_binary = |cmd: Option<&str>| check_binary_with_name(cmd.map(|cmd| (cmd, cmd))); - for lang in &syn_loader_conf.language { - if selection - .as_ref() - .is_some_and(|s| !s.contains(&lang.language_id)) - { - continue; - } - - write!(stdout, "{}", fit(&lang.language_id))?; + column(&lang.language_id, Color::Reset); - let mut cmds = lang.language_servers.iter().filter_map(|ls| { - syn_loader_conf - .language_server - .get(&ls.name) - .map(|config| (ls.name.as_str(), config.command.as_str())) - }); - write!(stdout, "{}", check_binary_with_name(cmds.next()))?; - - let dap = lang.debugger.as_ref().map(|dap| dap.command.as_str()); - write!(stdout, "{}", check_binary(dap))?; - - let formatter = lang - .formatter + let lsp = lang + .language_server .as_ref() - .map(|formatter| formatter.command.as_str()); - write!(stdout, "{}", check_binary(formatter))?; + .map(|lsp| lsp.command.to_string()); + check_binary(lsp); + + let dap = lang.debugger.as_ref().map(|dap| dap.command.to_string()); + check_binary(dap); for ts_feat in TsFeature::all() { match load_runtime_file(&lang.language_id, ts_feat.runtime_filename()).is_ok() { - true => write!(stdout, "{}", color(fit("✓"), ColorSpec::BRIGHT_GREEN))?, - false => write!(stdout, "{}", color(fit("✘"), ColorSpec::BRIGHT_RED))?, + true => column("Found", Color::Green), + false => column("Not Found", Color::Red), } } - writeln!(stdout)?; - - for cmd in cmds { - write!(stdout, "{}", fit(""))?; - writeln!(stdout, "{}", check_binary_with_name(Some(cmd)))?; - } + println!(); } - - if selection.is_some() { - writeln!( - stdout, - "\nThis list is filtered according to the 'use-grammars' option in languages.toml file.\n\ - To see the full list, use the '--health all' or '--health all-languages' option." - )?; - } - - Ok(()) } /// Display diagnostics pertaining to a particular language (LSP, /// highlight queries, etc). -pub fn language(lang_str: String) -> std::io::Result<()> { - let stdout = std::io::stdout(); - let mut stdout = stdout.lock(); - - let syn_loader_conf = match user_lang_config() { - Ok(conf) => conf, - Err(err) => { - let stderr = std::io::stderr(); - let mut stderr = stderr.lock(); - - writeln!( - stderr, - "{}: {}", - "Error parsing user language config".red(), - err - )?; - writeln!(stderr, "{}", "Using default language config".yellow())?; - default_lang_config() - } - }; +pub fn language(lang_str: String) { + let syn_loader_conf = user_syntax_loader().unwrap_or_else(|err| { + eprintln!("{}: {}", "Error parsing user language config".red(), err); + eprintln!("{}", "Using default language config".yellow()); + default_syntax_loader() + }); let lang = match syn_loader_conf .language @@ -307,8 +158,8 @@ pub fn language(lang_str: String) -> std::io::Result<()> { { Some(l) => l, None => { - let msg = format!("Language '{}' not found", lang_str); - writeln!(stdout, "{}", msg.red())?; + let msg = format!("Language '{lang_str}' not found"); + println!("{}", msg.red()); let suggestions: Vec<&str> = syn_loader_conf .language .iter() @@ -317,140 +168,52 @@ pub fn language(lang_str: String) -> std::io::Result<()> { .collect(); if !suggestions.is_empty() { let suggestions = suggestions.join(", "); - writeln!( - stdout, - "Did you mean one of these: {} ?", - suggestions.yellow() - )?; + println!("Did you mean one of these: {} ?", suggestions.yellow()); } - return Ok(()); + return; } }; - probe_protocols( + probe_protocol( "language server", - lang.language_servers.iter().filter_map(|ls| { - syn_loader_conf - .language_server - .get(&ls.name) - .map(|config| (ls.name.as_str(), config.command.as_str())) - }), - )?; + lang.language_server + .as_ref() + .map(|lsp| lsp.command.to_string()), + ); probe_protocol( "debug adapter", lang.debugger.as_ref().map(|dap| dap.command.to_string()), - )?; - - probe_protocol( - "formatter", - lang.formatter - .as_ref() - .map(|formatter| formatter.command.to_string()), - )?; - - probe_parser(lang.grammar.as_ref().unwrap_or(&lang.language_id))?; + ); for ts_feat in TsFeature::all() { - probe_treesitter_feature(&lang_str, *ts_feat)? - } - - Ok(()) -} - -fn probe_parser(grammar_name: &str) -> std::io::Result<()> { - let stdout = std::io::stdout(); - let mut stdout = stdout.lock(); - - write!(stdout, "Tree-sitter parser: ")?; - - match helix_loader::grammar::get_language(grammar_name) { - Ok(Some(_)) => writeln!(stdout, "{}", "✓".green()), - Ok(None) | Err(_) => writeln!(stdout, "{}", "None".yellow()), - } -} - -/// Display diagnostics about multiple LSPs and DAPs. -fn probe_protocols<'a, I: Iterator<Item = (&'a str, &'a str)> + 'a>( - protocol_name: &str, - server_cmds: I, -) -> std::io::Result<()> { - let stdout = std::io::stdout(); - let mut stdout = stdout.lock(); - let mut server_cmds = server_cmds.peekable(); - - write!(stdout, "Configured {}s:", protocol_name)?; - if server_cmds.peek().is_none() { - writeln!(stdout, "{}", " None".yellow())?; - return Ok(()); + probe_treesitter_feature(&lang_str, *ts_feat) } - writeln!(stdout)?; - - for (name, cmd) in server_cmds { - let (diag, icon) = match helix_stdx::env::which(cmd) { - Ok(path) => (path.display().to_string().green(), "✓".green()), - Err(_) => (format!("'{}' not found in $PATH", cmd).red(), "✘".red()), - }; - writeln!(stdout, " {} {}: {}", icon, name, diag)?; - } - - Ok(()) } /// Display diagnostics about LSP and DAP. -fn probe_protocol(protocol_name: &str, server_cmd: Option<String>) -> std::io::Result<()> { - let stdout = std::io::stdout(); - let mut stdout = stdout.lock(); - - write!(stdout, "Configured {}:", protocol_name)?; - let Some(cmd) = server_cmd else { - writeln!(stdout, "{}", " None".yellow())?; - return Ok(()); - }; - writeln!(stdout)?; - - let (diag, icon) = match helix_stdx::env::which(&cmd) { - Ok(path) => (path.display().to_string().green(), "✓".green()), - Err(_) => (format!("'{}' not found in $PATH", cmd).red(), "✘".red()), +fn probe_protocol(protocol_name: &str, server_cmd: Option<String>) { + let cmd_name = match server_cmd { + Some(ref cmd) => cmd.as_str().green(), + None => "None".yellow(), }; - writeln!(stdout, " {} {}", icon, diag)?; + println!("Configured {}: {}", protocol_name, cmd_name); - Ok(()) + if let Some(cmd) = server_cmd { + let path = match which::which(&cmd) { + Ok(path) => path.display().to_string().green(), + Err(_) => "Not found in $PATH".to_string().red(), + }; + println!("Binary for {}: {}", protocol_name, path); + } } /// Display diagnostics about a feature that requires tree-sitter /// query files (highlights, textobjects, etc). -fn probe_treesitter_feature(lang: &str, feature: TsFeature) -> std::io::Result<()> { - let stdout = std::io::stdout(); - let mut stdout = stdout.lock(); - +fn probe_treesitter_feature(lang: &str, feature: TsFeature) { let found = match load_runtime_file(lang, feature.runtime_filename()).is_ok() { - true => "✓".green(), - false => "✘".red(), + true => "Found".green(), + false => "Not found".red(), }; - writeln!(stdout, "{} queries: {}", feature.short_title(), found)?; - - Ok(()) -} - -pub fn print_health(health_arg: Option<String>) -> std::io::Result<()> { - match health_arg.as_deref() { - Some("languages") => languages_selection()?, - Some("all-languages") => languages_all()?, - Some("clipboard") => clipboard()?, - None => { - general()?; - clipboard()?; - writeln!(std::io::stdout().lock())?; - languages_selection()?; - } - Some("all") => { - general()?; - clipboard()?; - writeln!(std::io::stdout().lock())?; - languages_all()?; - } - Some(lang) => language(lang.to_string())?, - } - Ok(()) + println!("{} queries: {}", feature.short_title(), found); } |