Use Profile enum for x.py setup

This commit is contained in:
Antoine Martin 2020-10-05 17:27:42 +02:00
parent 6e06388a7f
commit d3d3397121
3 changed files with 75 additions and 38 deletions

View File

@ -12,6 +12,7 @@ use getopts::Options;
use crate::builder::Builder;
use crate::config::{Config, TargetSelection};
use crate::setup::Profile;
use crate::{Build, DocTests};
/// Deserialized version of all flags for this compile.
@ -94,7 +95,7 @@ pub enum Subcommand {
paths: Vec<PathBuf>,
},
Setup {
path: String,
profile: Profile,
},
}
@ -533,18 +534,19 @@ Arguments:
Subcommand::Run { paths }
}
"setup" => {
let path = if paths.len() > 1 {
let profile = if paths.len() > 1 {
println!("\nat most one profile can be passed to setup\n");
usage(1, &opts, verbose, &subcommand_help)
} else if let Some(path) = paths.pop() {
t!(path.into_os_string().into_string().map_err(|path| format!(
"{} is not a valid UTF8 string",
path.to_string_lossy()
)))
let profile_string = t!(path.into_os_string().into_string().map_err(
|path| format!("{} is not a valid UTF8 string", path.to_string_lossy())
));
profile_string.parse().expect("unknown profile")
} else {
t!(crate::setup::interactive_path())
};
Subcommand::Setup { path }
Subcommand::Setup { profile }
}
_ => {
usage(1, &opts, verbose, &subcommand_help);

View File

@ -471,8 +471,8 @@ impl Build {
return clean::clean(self, all);
}
if let Subcommand::Setup { path: include_name } = &self.config.cmd {
return setup::setup(&self.config.src, include_name);
if let Subcommand::Setup { profile } = &self.config.cmd {
return setup::setup(&self.config.src, *profile);
}
{

View File

@ -1,11 +1,56 @@
use crate::t;
use std::path::{Path, PathBuf};
use std::str::FromStr;
use std::{
env, fs,
env, fmt, fs,
io::{self, Write},
};
pub fn setup(src_path: &Path, include_name: &str) {
#[derive(Clone, Copy, Eq, PartialEq)]
pub enum Profile {
Compiler,
Codegen,
Library,
User,
}
impl Profile {
fn include_path(&self, src_path: &Path) -> PathBuf {
PathBuf::from(format!("{}/src/bootstrap/defaults/config.{}.toml", src_path.display(), self))
}
}
#[derive(Debug)]
pub struct ProfileErr {
pub name: String,
}
impl FromStr for Profile {
type Err = ProfileErr;
fn from_str(s: &str) -> Result<Self, Self::Err> {
match s {
"a" | "lib" | "library" => Ok(Profile::Library),
"b" | "compiler" => Ok(Profile::Compiler),
"c" | "llvm" | "codegen" => Ok(Profile::Codegen),
"d" | "maintainer" | "user" => Ok(Profile::User),
_ => Err(ProfileErr { name: s.to_string() }),
}
}
}
impl fmt::Display for Profile {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
Profile::Compiler => write!(f, "compiler"),
Profile::Codegen => write!(f, "codegen"),
Profile::Library => write!(f, "library"),
Profile::User => write!(f, "user"),
}
}
}
pub fn setup(src_path: &Path, profile: Profile) {
let cfg_file = env::var_os("BOOTSTRAP_CONFIG").map(PathBuf::from);
if cfg_file.as_ref().map_or(false, |f| f.exists()) {
@ -14,15 +59,10 @@ pub fn setup(src_path: &Path, include_name: &str) {
"error: you asked `x.py` to setup a new config file, but one already exists at `{}`",
file.display()
);
println!("help: try adding `profile = \"{}\"` at the top of {}", profile, file.display());
println!(
"help: try adding `profile = \"{}\"` at the top of {}",
include_name,
file.display()
);
println!(
"note: this will use the configuration in {}/src/bootstrap/defaults/config.{}.toml",
src_path.display(),
include_name
"note: this will use the configuration in {}",
profile.include_path(src_path).display()
);
std::process::exit(1);
}
@ -31,19 +71,17 @@ pub fn setup(src_path: &Path, include_name: &str) {
let settings = format!(
"# Includes one of the default files in src/bootstrap/defaults\n\
profile = \"{}\"\n",
include_name
profile
);
t!(fs::write(path, settings));
let include_path =
format!("{}/src/bootstrap/defaults/config.{}.toml", src_path.display(), include_name);
println!("`x.py` will now use the configuration at {}", include_path);
let include_path = profile.include_path(src_path);
println!("`x.py` will now use the configuration at {}", include_path.display());
let suggestions = match include_name {
"llvm" | "codegen" | "compiler" => &["check", "build", "test"][..],
"library" => &["check", "build", "test library/std", "doc"],
"maintainer" | "user" => &["dist", "build"],
_ => return,
let suggestions = match profile {
Profile::Codegen | Profile::Compiler => &["check", "build", "test"][..],
Profile::Library => &["check", "build", "test library/std", "doc"],
Profile::User => &["dist", "build"],
};
println!("To get started, try one of the following commands:");
@ -51,7 +89,7 @@ pub fn setup(src_path: &Path, include_name: &str) {
println!("- `x.py {}`", cmd);
}
if include_name != "user" {
if profile != Profile::User {
println!(
"For more suggestions, see https://rustc-dev-guide.rust-lang.org/building/suggested.html"
);
@ -59,7 +97,7 @@ pub fn setup(src_path: &Path, include_name: &str) {
}
// Used to get the path for `Subcommand::Setup`
pub fn interactive_path() -> io::Result<String> {
pub fn interactive_path() -> io::Result<Profile> {
let mut input = String::new();
println!(
"Welcome to the Rust project! What do you want to do with x.py?
@ -72,17 +110,14 @@ d) Install Rust from source"
print!("Please choose one (a/b/c/d): ");
io::stdout().flush()?;
io::stdin().read_line(&mut input)?;
break match input.trim().to_lowercase().as_str() {
"a" | "lib" | "library" => "library",
"b" | "compiler" => "compiler",
"c" | "llvm" => "llvm",
"d" | "user" | "maintainer" => "maintainer",
_ => {
println!("error: unrecognized option '{}'", input.trim());
break match input.trim().to_lowercase().parse() {
Ok(profile) => profile,
Err(ProfileErr { name }) => {
println!("error: unrecognized option '{}'", name);
println!("note: press Ctrl+C to exit");
continue;
}
};
};
Ok(template.to_owned())
Ok(template)
}