feat(rust): Support new rust-toolchain format (#1938)

* feat(rust): Support new rust-toolchain format

* Match file parsing with rustup and update link

* Use cargo to deserialize the rust-toolchain file

* Filter empty channel strings after extraction

* Use the option value instead of rewrapping
This commit is contained in:
Dominik Nakamura 2020-11-30 21:52:55 +09:00 committed by GitHub
parent 389e006c00
commit abfe4324e0
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 80 additions and 14 deletions

10
Cargo.lock generated
View File

@ -1122,18 +1122,18 @@ dependencies = [
[[package]] [[package]]
name = "serde" name = "serde"
version = "1.0.116" version = "1.0.117"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "96fe57af81d28386a513cbc6858332abc6117cfdb5999647c6444b8f43a370a5" checksum = "b88fa983de7720629c9387e9f517353ed404164b1e482c970a90c1a4aaf7dc1a"
dependencies = [ dependencies = [
"serde_derive", "serde_derive",
] ]
[[package]] [[package]]
name = "serde_derive" name = "serde_derive"
version = "1.0.116" version = "1.0.117"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "f630a6370fd8e457873b4bd2ffdae75408bc291ba72be773772a4c2a065d9ae8" checksum = "cbd1ae72adb44aab48f325a02444a5fc079349a8d804c1fc922aed3f7454c74e"
dependencies = [ dependencies = [
"proc-macro2", "proc-macro2",
"quote 1.0.7", "quote 1.0.7",
@ -1210,7 +1210,7 @@ dependencies = [
"rayon", "rayon",
"regex", "regex",
"rust-ini", "rust-ini",
"serde_derive", "serde",
"serde_json", "serde_json",
"shell-words", "shell-words",
"starship_module_config_derive", "starship_module_config_derive",

View File

@ -59,7 +59,7 @@ unicode-width = "0.1.8"
term_size = "0.3.2" term_size = "0.3.2"
quick-xml = "0.20.0" quick-xml = "0.20.0"
rand = "0.7.3" rand = "0.7.3"
serde_derive = "1.0.115" serde = { version = "1.0.117", features = ["derive"] }
indexmap = "1.6.0" indexmap = "1.6.0"
notify-rust = { version = "4.0.0", optional = true } notify-rust = { version = "4.0.0", optional = true }

View File

@ -2,6 +2,8 @@ use std::fs;
use std::path::Path; use std::path::Path;
use std::process::{Command, Output}; use std::process::{Command, Output};
use serde::Deserialize;
use super::{Context, Module, RootModuleConfig}; use super::{Context, Module, RootModuleConfig};
use crate::configs::rust::RustConfig; use crate::configs::rust::RustConfig;
@ -125,26 +127,47 @@ fn extract_toolchain_from_rustup_override_list(stdout: &str, cwd: &Path) -> Opti
fn find_rust_toolchain_file(context: &Context) -> Option<String> { fn find_rust_toolchain_file(context: &Context) -> Option<String> {
// Look for 'rust-toolchain' as rustup does. // Look for 'rust-toolchain' as rustup does.
// https://github.com/rust-lang/rustup.rs/blob/d84e6e50126bccd84649e42482fc35a11d019401/src/config.rs#L320-L358 // https://github.com/rust-lang/rustup/blob/89912c4cf51645b9c152ab7380fd07574fec43a3/src/config.rs#L546-L616
fn read_first_line(path: &Path) -> Option<String> { #[derive(Deserialize)]
let content = fs::read_to_string(path).ok()?; struct OverrideFile {
let line = content.lines().next()?; toolchain: ToolchainSection,
Some(line.trim().to_owned()) }
#[derive(Deserialize)]
struct ToolchainSection {
channel: Option<String>,
}
fn read_channel(path: &Path) -> Option<String> {
let contents = fs::read_to_string(path).ok()?;
match contents.lines().count() {
0 => None,
1 => Some(contents),
_ => {
toml::from_str::<OverrideFile>(&contents)
.ok()?
.toolchain
.channel
}
}
.filter(|c| !c.trim().is_empty())
.map(|c| c.trim().to_owned())
} }
if let Ok(true) = context if let Ok(true) = context
.dir_contents() .dir_contents()
.map(|dir| dir.has_file("rust-toolchain")) .map(|dir| dir.has_file("rust-toolchain"))
{ {
if let Some(toolchain) = read_first_line(Path::new("rust-toolchain")) { if let Some(toolchain) = read_channel(Path::new("rust-toolchain")) {
return Some(toolchain); return Some(toolchain);
} }
} }
let mut dir = &*context.current_dir; let mut dir = &*context.current_dir;
loop { loop {
if let Some(toolchain) = read_first_line(&dir.join("rust-toolchain")) { if let Some(toolchain) = read_channel(&dir.join("rust-toolchain")) {
return Some(toolchain); return Some(toolchain);
} }
dir = dir.parent()?; dir = dir.parent()?;
@ -200,6 +223,7 @@ enum RustupRunRustcVersionOutcome {
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use once_cell::sync::Lazy; use once_cell::sync::Lazy;
use std::io;
use std::process::{ExitStatus, Output}; use std::process::{ExitStatus, Output};
use super::*; use super::*;
@ -309,4 +333,46 @@ mod tests {
let version_without_hash = String::from("rustc 1.34.0"); let version_without_hash = String::from("rustc 1.34.0");
assert_eq!(format_rustc_version(version_without_hash), "v1.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()
}
} }