2024-03-29 03:29:41 +03:00
|
|
|
use anyhow::{bail, Context, Result};
|
2024-07-25 17:26:48 +03:00
|
|
|
use ratatui::crossterm::style::Stylize;
|
2024-03-29 03:29:41 +03:00
|
|
|
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},
|
2024-08-08 13:51:27 +03:00
|
|
|
path::{Path, PathBuf},
|
2024-04-25 16:41:52 +03:00
|
|
|
process::{Command, Stdio},
|
2024-03-29 03:29:41 +03:00
|
|
|
};
|
|
|
|
|
2024-08-08 03:45:18 +03:00
|
|
|
use crate::{
|
|
|
|
cargo_toml::updated_cargo_toml, embedded::EMBEDDED_FILES, info_file::InfoFile,
|
|
|
|
term::press_enter_prompt,
|
|
|
|
};
|
2024-04-04 16:44:48 +03:00
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2024-08-08 03:45:18 +03:00
|
|
|
let mut stdout = io::stdout().lock();
|
|
|
|
let mut init_git = true;
|
|
|
|
|
2024-08-08 13:51:27 +03:00
|
|
|
let manifest_path = Command::new("cargo")
|
|
|
|
.args(["locate-project", "--message-format=plain"])
|
|
|
|
.output()?;
|
|
|
|
if manifest_path.status.success() {
|
|
|
|
let manifest_path: PathBuf = String::from_utf8_lossy(&manifest_path.stdout).trim().into();
|
|
|
|
|
2024-08-08 03:45:18 +03:00
|
|
|
if Path::new("exercises").exists() && Path::new("solutions").exists() {
|
|
|
|
bail!(IN_INITIALIZED_DIR_ERR);
|
2024-03-29 03:29:41 +03:00
|
|
|
}
|
2024-08-08 13:51:27 +03:00
|
|
|
if fs::read_to_string(manifest_path)?.contains("[workspace]") {
|
|
|
|
// make sure "rustlings" is added to `workspace.members` by making
|
|
|
|
// cargo initialize a new project
|
|
|
|
let output = Command::new("cargo").args(["new", "rustlings"]).output()?;
|
|
|
|
if !output.status.success() {
|
|
|
|
bail!("Failed to initilize new workspace member");
|
|
|
|
}
|
|
|
|
fs::remove_dir_all("rustlings")?;
|
|
|
|
init_git = false;
|
|
|
|
} else {
|
|
|
|
bail!(IN_NON_WORKSPACE_CARGO_PROJECT_ERR);
|
|
|
|
}
|
2024-03-29 03:29:41 +03:00
|
|
|
}
|
|
|
|
|
2024-08-08 03:45:18 +03:00
|
|
|
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)?;
|
|
|
|
|
|
|
|
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/`")?;
|
2024-03-29 03:29:41 +03:00
|
|
|
|
2024-04-23 20:18:25 +03:00
|
|
|
let info_file = InfoFile::parse()?;
|
2024-03-29 03:29:41 +03:00
|
|
|
EMBEDDED_FILES
|
2024-04-23 20:18:25 +03:00
|
|
|
.init_exercises_dir(&info_file.exercises)
|
2024-03-29 03:29:41 +03:00
|
|
|
.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");
|
2024-04-21 21:22:01 +03:00
|
|
|
// 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")?;
|
2024-04-21 21:22:01 +03:00
|
|
|
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-03-29 03:29:41 +03:00
|
|
|
|
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-03-29 03:29:41 +03:00
|
|
|
|
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())
|
|
|
|
.stderr(Stdio::null())
|
|
|
|
.status();
|
|
|
|
}
|
2024-04-25 16:41:52 +03:00
|
|
|
|
2024-08-08 03:45:18 +03:00
|
|
|
writeln!(
|
|
|
|
stdout,
|
2024-04-25 16:51:12 +03:00
|
|
|
"\n{}\n\n{}",
|
|
|
|
"Initialization done ✓".green(),
|
|
|
|
POST_INIT_MSG.bold(),
|
2024-08-08 03:45:18 +03:00
|
|
|
)?;
|
2024-04-16 04:15:14 +03:00
|
|
|
|
2024-03-29 03:29:41 +03:00
|
|
|
Ok(())
|
|
|
|
}
|
2024-04-12 02:24:01 +03:00
|
|
|
|
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-08-08 01:20:04 +03:00
|
|
|
const GITIGNORE: &[u8] = b"Cargo.lock
|
|
|
|
target/
|
|
|
|
.vscode/
|
2024-04-12 19:57:39 +03:00
|
|
|
";
|
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 another 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-08 13:51:27 +03:00
|
|
|
const IN_NON_WORKSPACE_CARGO_PROJECT_ERR: &str = "\
|
|
|
|
The current directory is already part of a cargo project.
|
|
|
|
Please initialize rustlings in a different directory.";
|
2024-08-08 03:45:18 +03:00
|
|
|
|
2024-04-25 16:51:12 +03:00
|
|
|
const POST_INIT_MSG: &str = "Run `cd rustlings` to go into the generated directory.
|
2024-04-16 04:15:14 +03:00
|
|
|
Then run `rustlings` to get started.";
|