use std::fs; use std::path::Path; use std::process::{Command, Output}; use serde::Deserialize; use super::{Context, Module, RootModuleConfig}; use crate::configs::rust::RustConfig; use crate::formatter::StringFormatter; /// Creates a module with the current Rust version /// /// Will display the Rust version if any of the following criteria are met: /// - Current directory contains a file with a `.rs` extension /// - Current directory contains a `Cargo.toml` file pub fn module<'a>(context: &'a Context) -> Option> { let is_rs_project = context .try_begin_scan()? .set_files(&["Cargo.toml"]) .set_extensions(&["rs"]) .is_match(); if !is_rs_project { return None; } let mut module = context.new_module("rust"); let config = RustConfig::try_load(module.config); let parsed = StringFormatter::new(config.format).and_then(|formatter| { formatter .map_meta(|var, _| match var { "symbol" => Some(config.symbol), _ => None, }) .map_style(|variable| match variable { "style" => Some(Ok(config.style)), _ => None, }) .map(|variable| match variable { // This may result in multiple calls to `get_module_version` when a user have // multiple `$version` variables defined in `format`. "version" => get_module_version(context).map(Ok), _ => None, }) .parse(None) }); module.set_segments(match parsed { Ok(segments) => segments, Err(error) => { log::warn!("Error in module `rust`:\n{}", error); return None; } }); Some(module) } fn get_module_version(context: &Context) -> Option { // `$CARGO_HOME/bin/rustc(.exe) --version` may attempt installing a rustup toolchain. // https://github.com/starship/starship/issues/417 // // To display appropriate versions preventing `rustc` from downloading toolchains, we have to // check // 1. `$RUSTUP_TOOLCHAIN` // 2. `rustup override list` // 3. `rust-toolchain` in `.` or parent directories // as `rustup` does. // https://github.com/rust-lang/rustup.rs/tree/eb694fcada7becc5d9d160bf7c623abe84f8971d#override-precedence // // Probably we have no other way to know whether any toolchain override is specified for the // current directory. The following commands also cause toolchain installations. // - `rustup show` // - `rustup show active-toolchain` // - `rustup which` let module_version = if let Some(toolchain) = env_rustup_toolchain(context) .or_else(|| execute_rustup_override_list(&context.current_dir)) .or_else(|| find_rust_toolchain_file(&context)) { match execute_rustup_run_rustc_version(&toolchain) { RustupRunRustcVersionOutcome::RustcVersion(stdout) => format_rustc_version(stdout), RustupRunRustcVersionOutcome::ToolchainName(toolchain) => toolchain, RustupRunRustcVersionOutcome::RustupNotWorking => { // If `rustup` is not in `$PATH` or cannot be executed for other reasons, we can // safely execute `rustc --version`. format_rustc_version(execute_rustc_version()?) } RustupRunRustcVersionOutcome::Err => return None, } } else { format_rustc_version(execute_rustc_version()?) }; Some(module_version) } fn env_rustup_toolchain(context: &Context) -> Option { let val = context.get_env("RUSTUP_TOOLCHAIN")?; Some(val.trim().to_owned()) } fn execute_rustup_override_list(cwd: &Path) -> Option { let Output { stdout, .. } = Command::new("rustup") .args(&["override", "list"]) .output() .ok()?; let stdout = String::from_utf8(stdout).ok()?; extract_toolchain_from_rustup_override_list(&stdout, cwd) } fn extract_toolchain_from_rustup_override_list(stdout: &str, cwd: &Path) -> Option { if stdout == "no overrides\n" { return None; } stdout .lines() .flat_map(|line| { let mut words = line.split_whitespace(); let dir = words.next()?; let toolchain = words.next()?; Some((dir, toolchain)) }) .find(|(dir, _)| cwd.starts_with(dir)) .map(|(_, toolchain)| toolchain.to_owned()) } fn find_rust_toolchain_file(context: &Context) -> Option { // Look for 'rust-toolchain' as rustup does. // https://github.com/rust-lang/rustup/blob/89912c4cf51645b9c152ab7380fd07574fec43a3/src/config.rs#L546-L616 #[derive(Deserialize)] struct OverrideFile { toolchain: ToolchainSection, } #[derive(Deserialize)] struct ToolchainSection { channel: Option, } fn read_channel(path: &Path) -> Option { let contents = fs::read_to_string(path).ok()?; match contents.lines().count() { 0 => None, 1 => Some(contents), _ => { toml::from_str::(&contents) .ok()? .toolchain .channel } } .filter(|c| !c.trim().is_empty()) .map(|c| c.trim().to_owned()) } if let Ok(true) = context .dir_contents() .map(|dir| dir.has_file("rust-toolchain")) { if let Some(toolchain) = read_channel(Path::new("rust-toolchain")) { return Some(toolchain); } } let mut dir = &*context.current_dir; loop { if let Some(toolchain) = read_channel(&dir.join("rust-toolchain")) { return Some(toolchain); } dir = dir.parent()?; } } fn execute_rustup_run_rustc_version(toolchain: &str) -> RustupRunRustcVersionOutcome { Command::new("rustup") .args(&["run", toolchain, "rustc", "--version"]) .output() .map(extract_toolchain_from_rustup_run_rustc_version) .unwrap_or(RustupRunRustcVersionOutcome::RustupNotWorking) } fn extract_toolchain_from_rustup_run_rustc_version(output: Output) -> RustupRunRustcVersionOutcome { if output.status.success() { if let Ok(output) = String::from_utf8(output.stdout) { return RustupRunRustcVersionOutcome::RustcVersion(output); } } else if let Ok(stderr) = String::from_utf8(output.stderr) { if stderr.starts_with("error: toolchain '") && stderr.ends_with("' is not installed\n") { let stderr = stderr ["error: toolchain '".len()..stderr.len() - "' is not installed\n".len()] .to_owned(); return RustupRunRustcVersionOutcome::ToolchainName(stderr); } } RustupRunRustcVersionOutcome::Err } fn execute_rustc_version() -> Option { match Command::new("rustc").arg("--version").output() { Ok(output) => Some(String::from_utf8(output.stdout).unwrap()), Err(_) => None, } } fn format_rustc_version(mut rustc_stdout: String) -> String { let offset = &rustc_stdout.find('(').unwrap_or_else(|| rustc_stdout.len()); let formatted_version: String = rustc_stdout.drain(..offset).collect(); format!("v{}", formatted_version.replace("rustc", "").trim()) } #[derive(Debug, PartialEq)] enum RustupRunRustcVersionOutcome { RustcVersion(String), ToolchainName(String), RustupNotWorking, Err, } #[cfg(test)] mod tests { use once_cell::sync::Lazy; use std::io; use std::process::{ExitStatus, Output}; use super::*; #[test] fn test_extract_toolchain_from_rustup_override_list() { static NO_OVERRIDES_INPUT: &str = "no overrides\n"; static NO_OVERRIDES_CWD: &str = ""; assert_eq!( extract_toolchain_from_rustup_override_list( NO_OVERRIDES_INPUT, NO_OVERRIDES_CWD.as_ref(), ), None, ); static OVERRIDES_INPUT: &str = "/home/user/src/a beta-x86_64-unknown-linux-gnu\n\ /home/user/src/b nightly-x86_64-unknown-linux-gnu\n"; static OVERRIDES_CWD_A: &str = "/home/user/src/a/src"; static OVERRIDES_CWD_B: &str = "/home/user/src/b/tests"; static OVERRIDES_CWD_C: &str = "/home/user/src/c/examples"; assert_eq!( extract_toolchain_from_rustup_override_list(OVERRIDES_INPUT, OVERRIDES_CWD_A.as_ref()), Some("beta-x86_64-unknown-linux-gnu".to_owned()), ); assert_eq!( extract_toolchain_from_rustup_override_list(OVERRIDES_INPUT, OVERRIDES_CWD_B.as_ref()), Some("nightly-x86_64-unknown-linux-gnu".to_owned()), ); assert_eq!( extract_toolchain_from_rustup_override_list(OVERRIDES_INPUT, OVERRIDES_CWD_C.as_ref()), None, ); } #[cfg(any(unix, windows))] #[test] fn test_extract_toolchain_from_rustup_run_rustc_version() { #[cfg(unix)] use std::os::unix::process::ExitStatusExt as _; #[cfg(windows)] use std::os::windows::process::ExitStatusExt as _; static RUSTC_VERSION: Lazy = Lazy::new(|| Output { status: ExitStatus::from_raw(0), stdout: b"rustc 1.34.0\n"[..].to_owned(), stderr: vec![], }); assert_eq!( extract_toolchain_from_rustup_run_rustc_version(RUSTC_VERSION.clone()), RustupRunRustcVersionOutcome::RustcVersion("rustc 1.34.0\n".to_owned()), ); static TOOLCHAIN_NAME: Lazy = Lazy::new(|| Output { status: ExitStatus::from_raw(1), stdout: vec![], stderr: b"error: toolchain 'channel-triple' is not installed\n"[..].to_owned(), }); assert_eq!( extract_toolchain_from_rustup_run_rustc_version(TOOLCHAIN_NAME.clone()), RustupRunRustcVersionOutcome::ToolchainName("channel-triple".to_owned()), ); static INVALID_STDOUT: Lazy = Lazy::new(|| Output { status: ExitStatus::from_raw(0), stdout: b"\xc3\x28"[..].to_owned(), stderr: vec![], }); assert_eq!( extract_toolchain_from_rustup_run_rustc_version(INVALID_STDOUT.clone()), RustupRunRustcVersionOutcome::Err, ); static INVALID_STDERR: Lazy = Lazy::new(|| Output { status: ExitStatus::from_raw(1), stdout: vec![], stderr: b"\xc3\x28"[..].to_owned(), }); assert_eq!( extract_toolchain_from_rustup_run_rustc_version(INVALID_STDERR.clone()), RustupRunRustcVersionOutcome::Err, ); static UNEXPECTED_FORMAT_OF_ERROR: Lazy = Lazy::new(|| Output { status: ExitStatus::from_raw(1), stdout: vec![], stderr: b"error:"[..].to_owned(), }); assert_eq!( extract_toolchain_from_rustup_run_rustc_version(UNEXPECTED_FORMAT_OF_ERROR.clone()), RustupRunRustcVersionOutcome::Err, ); } #[test] fn test_format_rustc_version() { let nightly_input = String::from("rustc 1.34.0-nightly (b139669f3 2019-04-10)"); assert_eq!(format_rustc_version(nightly_input), "v1.34.0-nightly"); let beta_input = String::from("rustc 1.34.0-beta.1 (2bc1d406d 2019-04-10)"); assert_eq!(format_rustc_version(beta_input), "v1.34.0-beta.1"); let stable_input = String::from("rustc 1.34.0 (91856ed52 2019-04-10)"); assert_eq!(format_rustc_version(stable_input), "v1.34.0"); let version_without_hash = String::from("rustc 1.34.0"); assert_eq!(format_rustc_version(version_without_hash), "v1.34.0"); } #[test] fn test_find_rust_toolchain_file() -> io::Result<()> { let dir = tempfile::tempdir()?; fs::write(dir.path().join("rust-toolchain"), "1.34.0")?; let context = Context::new_with_dir(Default::default(), dir.path()); assert_eq!( find_rust_toolchain_file(&context), Some("1.34.0".to_owned()) ); dir.close()?; let dir = tempfile::tempdir()?; fs::write( dir.path().join("rust-toolchain"), "[toolchain]\nchannel = \"1.34.0\"", )?; let context = Context::new_with_dir(Default::default(), dir.path()); assert_eq!( find_rust_toolchain_file(&context), Some("1.34.0".to_owned()) ); dir.close()?; let dir = tempfile::tempdir()?; fs::write( dir.path().join("rust-toolchain"), "\n\n[toolchain]\n\n\nchannel = \"1.34.0\"", )?; let context = Context::new_with_dir(Default::default(), dir.path()); assert_eq!( find_rust_toolchain_file(&context), Some("1.34.0".to_owned()) ); dir.close() } }