From 7968f54648bc48f1009cac04fb5c10bce0212642 Mon Sep 17 00:00:00 2001 From: shanmu Date: Wed, 14 Aug 2024 01:17:50 +0800 Subject: [PATCH] feat: Add custom completer for completing target triple --- src/cargo/util/command_prelude.rs | 59 ++++++++++++++++++++++++++++++- 1 file changed, 58 insertions(+), 1 deletion(-) diff --git a/src/cargo/util/command_prelude.rs b/src/cargo/util/command_prelude.rs index 4c8656bae18..4c4fb1cb397 100644 --- a/src/cargo/util/command_prelude.rs +++ b/src/cargo/util/command_prelude.rs @@ -272,7 +272,8 @@ pub trait CommandExt: Sized { }; self._arg( optional_multi_opt("target", "TRIPLE", target) - .help_heading(heading::COMPILATION_OPTIONS), + .help_heading(heading::COMPILATION_OPTIONS) + .add(clap_complete::ArgValueCandidates::new(get_target_triples)), ) ._arg(unsupported_short_arg) } @@ -1067,6 +1068,62 @@ fn get_targets_from_metadata() -> CargoResult> { Ok(targets) } +fn get_target_triples() -> Vec { + let mut candidates = Vec::new(); + + if is_rustup() { + if let Ok(targets) = get_target_triples_from_rustup() { + candidates.extend(targets); + } + } else { + if let Ok(targets) = get_target_triples_from_rustc() { + candidates.extend(targets); + } + } + + candidates +} + +fn get_target_triples_from_rustup() -> CargoResult> { + let output = std::process::Command::new("rustup") + .arg("target") + .arg("list") + .output()?; + + if !output.status.success() { + return Ok(vec![]); + } + + let stdout = String::from_utf8(output.stdout)?; + + Ok(stdout + .lines() + .map(|line| { + let target = line.split_once(' '); + match target { + None => clap_complete::CompletionCandidate::new(line.to_owned()).hide(true), + Some((target, _installed)) => clap_complete::CompletionCandidate::new(target), + } + }) + .collect()) +} + +fn get_target_triples_from_rustc() -> CargoResult> { + let cwd = std::env::current_dir()?; + let gctx = GlobalContext::new(shell::Shell::new(), cwd.clone(), cargo_home_with_cwd(&cwd)?); + let ws = Workspace::new(&find_root_manifest_for_wd(&PathBuf::from(&cwd))?, &gctx); + + let rustc = gctx.load_global_rustc(ws.as_ref().ok())?; + + let (stdout, _stderr) = + rustc.cached_output(rustc.process().arg("--print").arg("target-list"), 0)?; + + Ok(stdout + .lines() + .map(|line| clap_complete::CompletionCandidate::new(line.to_owned())) + .collect()) +} + #[track_caller] pub fn ignore_unknown(r: Result) -> T { match r {