From ff3e880c1708a88a5797b0de0cf66388025bc3d3 Mon Sep 17 00:00:00 2001 From: Zach Lute Date: Sat, 5 Oct 2019 08:44:26 -0700 Subject: [PATCH 1/2] Added aliases to subcommand typo suggestions. Fixes #7278. Also adds tests for alias suggestions. --- src/bin/cargo/main.rs | 20 +++++++++++++-- tests/testsuite/cargo_command.rs | 43 ++++++++++++++++++++++++++++++++ 2 files changed, 61 insertions(+), 2 deletions(-) diff --git a/src/bin/cargo/main.rs b/src/bin/cargo/main.rs index 9ed135cf123..a2bd1c16c57 100644 --- a/src/bin/cargo/main.rs +++ b/src/bin/cargo/main.rs @@ -113,6 +113,17 @@ fn list_commands(config: &Config) -> BTreeSet { commands } +/// List all runnable aliases +fn list_aliases(config: &Config) -> Vec { + match config.get_table("alias") { + Ok(table) => match table { + Some(aliases) => aliases.val.keys().map(|a| a.to_string()).collect(), + None => Vec::new(), + }, + Err(_) => Vec::new(), + } +} + fn execute_external_subcommand(config: &Config, cmd: &str, args: &[&str]) -> CliResult { let command_exe = format!("cargo-{}{}", cmd, env::consts::EXE_SUFFIX); let path = search_directories(config) @@ -122,8 +133,13 @@ fn execute_external_subcommand(config: &Config, cmd: &str, args: &[&str]) -> Cli let command = match path { Some(command) => command, None => { - let cmds = list_commands(config); - let did_you_mean = closest_msg(cmd, cmds.iter(), |c| c.name()); + let commands: Vec = list_commands(config) + .iter() + .map(|c| c.name().to_string()) + .collect(); + let aliases = list_aliases(config); + let suggestions = commands.iter().chain(aliases.iter()); + let did_you_mean = closest_msg(cmd, suggestions, |c| c); let err = failure::format_err!("no such subcommand: `{}`{}", cmd, did_you_mean); return Err(CliError::new(err, 101)); } diff --git a/tests/testsuite/cargo_command.rs b/tests/testsuite/cargo_command.rs index e0f4cda2e6b..a09751acc77 100644 --- a/tests/testsuite/cargo_command.rs +++ b/tests/testsuite/cargo_command.rs @@ -168,6 +168,49 @@ error: no such subcommand: `biuld` .run(); } +#[cargo_test] +fn find_closest_alias() { + let root = paths::root(); + let my_home = root.join("my_home"); + fs::create_dir(&my_home).unwrap(); + File::create(&my_home.join("config")) + .unwrap() + .write_all( + br#" + [alias] + myalias = "build" + "#, + ) + .unwrap(); + + cargo_process("myalais") + .env("CARGO_HOME", &my_home) + .with_status(101) + .with_stderr_contains( + "\ +error: no such subcommand: `myalais` + +Did you mean `myalias`? +", + ) + .run(); + + // But, if no alias is defined, it must not suggest one! + cargo_process("myalais") + .with_status(101) + .with_stderr_contains( + "\ +error: no such subcommand: `myalais` +", + ) + .with_stderr_does_not_contain( + "\ +Did you mean `myalias`? +", + ) + .run(); +} + // If a subcommand is more than an edit distance of 3 away, we don't make a suggestion. #[cargo_test] fn find_closest_dont_correct_nonsense() { From 0f157f52b0af83eb255c87c49f06d826f1818783 Mon Sep 17 00:00:00 2001 From: Zach Lute Date: Sat, 9 Nov 2019 07:54:58 -0800 Subject: [PATCH 2/2] Replaced config.get_table() with a more succinct config.get() when listing aliases. --- src/bin/cargo/main.rs | 9 +++------ 1 file changed, 3 insertions(+), 6 deletions(-) diff --git a/src/bin/cargo/main.rs b/src/bin/cargo/main.rs index a2bd1c16c57..d4cd4e235bd 100644 --- a/src/bin/cargo/main.rs +++ b/src/bin/cargo/main.rs @@ -4,7 +4,7 @@ #![warn(clippy::needless_borrow)] #![warn(clippy::redundant_clone)] -use std::collections::BTreeSet; +use std::collections::{BTreeMap, BTreeSet}; use std::env; use std::fs; use std::path::{Path, PathBuf}; @@ -115,11 +115,8 @@ fn list_commands(config: &Config) -> BTreeSet { /// List all runnable aliases fn list_aliases(config: &Config) -> Vec { - match config.get_table("alias") { - Ok(table) => match table { - Some(aliases) => aliases.val.keys().map(|a| a.to_string()).collect(), - None => Vec::new(), - }, + match config.get::>("alias") { + Ok(aliases) => aliases.keys().map(|a| a.to_string()).collect(), Err(_) => Vec::new(), } }