rustlings/src/init.rs

200 lines
7.3 KiB
Rust
Raw Normal View History

use anyhow::{bail, Context, Result};
2024-08-26 00:53:50 +03:00
use crossterm::{
style::{Attribute, Color, ResetColor, SetAttribute, SetForegroundColor},
QueueableCommand,
};
use serde::Deserialize;
use std::{
env::set_current_dir,
2024-04-16 04:08:45 +03:00
fs::{self, create_dir},
2024-08-08 03:45:18 +03:00
io::{self, Write},
path::{Path, PathBuf},
2024-04-25 16:41:52 +03:00
process::{Command, Stdio},
};
2024-08-08 03:45:18 +03:00
use crate::{
2024-08-28 02:10:19 +03:00
cargo_toml::updated_cargo_toml, embedded::EMBEDDED_FILES, exercise::RunnableExercise,
info_file::InfoFile, term::press_enter_prompt,
2024-08-08 03:45:18 +03:00
};
2024-04-04 16:44:48 +03:00
#[derive(Deserialize)]
struct CargoLocateProject {
root: PathBuf,
}
2024-04-17 16:55:50 +03:00
pub fn init() -> Result<()> {
2024-08-08 03:45:18 +03:00
let rustlings_dir = Path::new("rustlings");
if rustlings_dir.exists() {
bail!(RUSTLINGS_DIR_ALREADY_EXISTS_ERR);
2024-03-29 03:51:08 +03:00
}
let locate_project_output = Command::new("cargo")
.arg("locate-project")
.arg("-q")
.arg("--workspace")
.stdin(Stdio::null())
2024-08-09 02:27:31 +03:00
.stderr(Stdio::null())
.output()
.context(CARGO_LOCATE_PROJECT_ERR)?;
2024-08-08 03:45:18 +03:00
let mut stdout = io::stdout().lock();
let mut init_git = true;
if locate_project_output.status.success() {
2024-08-08 03:45:18 +03:00
if Path::new("exercises").exists() && Path::new("solutions").exists() {
bail!(IN_INITIALIZED_DIR_ERR);
}
let workspace_manifest =
serde_json::de::from_slice::<CargoLocateProject>(&locate_project_output.stdout)
.context(
"Failed to read the field `root` from the output of `cargo locate-project …`",
)?
.root;
let workspace_manifest_content = fs::read_to_string(&workspace_manifest)
.with_context(|| format!("Failed to read the file {}", workspace_manifest.display()))?;
if !workspace_manifest_content.contains("[workspace]\n")
&& !workspace_manifest_content.contains("workspace.")
{
bail!("The current directory is already part of a Cargo project.\nPlease initialize Rustlings in a different directory");
}
stdout.write_all(b"This command will create the directory `rustlings/` as a member of this Cargo workspace.\nPress ENTER to continue ")?;
press_enter_prompt(&mut stdout)?;
// Make sure "rustlings" is added to `workspace.members` by making
// Cargo initialize a new project.
let status = Command::new("cargo")
.arg("new")
.arg("-q")
.arg("--vcs")
.arg("none")
.arg("rustlings")
.stdin(Stdio::null())
.stdout(Stdio::null())
.status()?;
if !status.success() {
2024-08-09 02:08:52 +03:00
bail!("Failed to initialize a new Cargo workspace member.\nPlease initialize Rustlings in a different directory");
}
2024-08-09 02:16:45 +03:00
stdout.write_all(b"The directory `rustlings` has been added to `workspace.members` in the `Cargo.toml` file of this Cargo workspace.\n")?;
fs::remove_dir_all("rustlings")
.context("Failed to remove the temporary directory `rustlings/`")?;
init_git = false;
} else {
stdout.write_all(b"This command will create the directory `rustlings/` which will contain the exercises.\nPress ENTER to continue ")?;
press_enter_prompt(&mut stdout)?;
}
2024-08-08 03:45:18 +03:00
create_dir(rustlings_dir).context("Failed to create the `rustlings/` directory")?;
set_current_dir(rustlings_dir)
2024-05-14 01:35:12 +03:00
.context("Failed to change the current directory to `rustlings/`")?;
let info_file = InfoFile::parse()?;
EMBEDDED_FILES
.init_exercises_dir(&info_file.exercises)
.context("Failed to initialize the `rustlings/exercises` directory")?;
2024-05-25 19:19:30 +03:00
create_dir("solutions").context("Failed to create the `solutions/` directory")?;
2024-08-08 01:20:20 +03:00
fs::write(
"solutions/README.md",
include_bytes!("../solutions/README.md"),
)
.context("Failed to create the file rustlings/solutions/README.md")?;
2024-05-25 19:19:30 +03:00
for dir in EMBEDDED_FILES.exercise_dirs {
let mut dir_path = String::with_capacity(10 + dir.name.len());
dir_path.push_str("solutions/");
dir_path.push_str(dir.name);
create_dir(&dir_path)
.with_context(|| format!("Failed to create the directory {dir_path}"))?;
}
for exercise_info in &info_file.exercises {
let solution_path = exercise_info.sol_path();
fs::write(&solution_path, INIT_SOLUTION_FILE)
.with_context(|| format!("Failed to create the file {solution_path}"))?;
}
2024-04-25 20:58:55 +03:00
let current_cargo_toml = include_str!("../dev-Cargo.toml");
// Skip the first line (comment).
let newline_ind = current_cargo_toml
.as_bytes()
.iter()
.position(|c| *c == b'\n')
2024-05-14 01:35:12 +03:00
.context("The embedded `Cargo.toml` is empty or contains only one line")?;
let current_cargo_toml = current_cargo_toml
.get(newline_ind + 1..)
.context("The embedded `Cargo.toml` contains only one line")?;
let updated_cargo_toml = updated_cargo_toml(&info_file.exercises, current_cargo_toml, b"")
.context("Failed to generate `Cargo.toml`")?;
fs::write("Cargo.toml", updated_cargo_toml)
2024-04-14 02:15:43 +03:00
.context("Failed to create the file `rustlings/Cargo.toml`")?;
2024-09-13 17:38:53 +03:00
fs::write("rust-analyzer.toml", RUST_ANALYZER_TOML)
2024-09-12 16:26:40 +03:00
.context("Failed to create the file `rustlings/rust-analyzer.toml`")?;
2024-04-16 04:08:45 +03:00
fs::write(".gitignore", GITIGNORE)
.context("Failed to create the file `rustlings/.gitignore`")?;
2024-03-31 04:04:41 +03:00
2024-04-16 04:08:45 +03:00
create_dir(".vscode").context("Failed to create the directory `rustlings/.vscode`")?;
fs::write(".vscode/extensions.json", VS_CODE_EXTENSIONS_JSON)
.context("Failed to create the file `rustlings/.vscode/extensions.json`")?;
2024-08-08 03:45:18 +03:00
if init_git {
// Ignore any Git error because Git initialization is not required.
let _ = Command::new("git")
.arg("init")
.stdin(Stdio::null())
2024-08-17 15:40:09 +03:00
.stdout(Stdio::null())
2024-08-08 03:45:18 +03:00
.stderr(Stdio::null())
.status();
}
2024-04-25 16:41:52 +03:00
2024-08-26 00:53:50 +03:00
stdout.queue(SetForegroundColor(Color::Green))?;
2024-08-26 01:24:39 +03:00
stdout.write_all("Initialization done ✓".as_bytes())?;
stdout.queue(ResetColor)?;
stdout.write_all(b"\n\n")?;
stdout.queue(SetAttribute(Attribute::Bold))?;
2024-08-26 00:53:50 +03:00
stdout.write_all(POST_INIT_MSG)?;
stdout.queue(ResetColor)?;
2024-04-16 04:15:14 +03:00
Ok(())
}
2024-04-12 02:24:01 +03:00
const CARGO_LOCATE_PROJECT_ERR: &str = "Failed to run the command `cargo locate-project …`
Did you already install Rust?
Try running `cargo --version` to diagnose the problem.";
2024-05-25 19:19:30 +03:00
const INIT_SOLUTION_FILE: &[u8] = b"fn main() {
// DON'T EDIT THIS SOLUTION FILE!
// It will be automatically filled after you finish the exercise.
}
";
2024-09-13 17:38:53 +03:00
pub const RUST_ANALYZER_TOML: &[u8] = br#"check.command = "clippy"
check.extraArgs = ["--profile", "test"]
"#;
2024-09-12 16:46:09 +03:00
const GITIGNORE: &[u8] = b"Cargo.lock
target/
.vscode/
";
2024-04-12 02:24:01 +03:00
2024-04-16 04:08:45 +03:00
pub const VS_CODE_EXTENSIONS_JSON: &[u8] = br#"{"recommendations":["rust-lang.rust-analyzer"]}"#;
2024-04-12 02:24:01 +03:00
2024-08-08 03:45:18 +03:00
const IN_INITIALIZED_DIR_ERR: &str = "It looks like Rustlings is already initialized in this directory.
2024-04-12 02:24:01 +03:00
2024-05-14 01:35:12 +03:00
If you already initialized Rustlings, run the command `rustlings` for instructions on getting started with the exercises.
Otherwise, please run `rustlings init` again in a different directory.";
2024-04-12 02:24:01 +03:00
const RUSTLINGS_DIR_ALREADY_EXISTS_ERR: &str =
"A directory with the name `rustlings` already exists in the current directory.
You probably already initialized Rustlings.
Run `cd rustlings`
Then run `rustlings` again";
2024-04-16 04:15:14 +03:00
2024-08-26 00:53:50 +03:00
const POST_INIT_MSG: &[u8] = b"Run `cd rustlings` to go into the generated directory.
Then run `rustlings` to get started.
";