rustlings/src/main.rs

148 lines
4.3 KiB
Rust
Raw Normal View History

2024-04-01 12:38:01 -04:00
use anyhow::{bail, Context, Result};
2023-08-25 17:18:01 -04:00
use clap::{Parser, Subcommand};
2024-04-07 18:36:26 -04:00
use std::{path::Path, process::exit};
2019-01-09 14:33:43 -05:00
2024-04-04 21:04:53 -04:00
mod consts;
2024-03-28 16:06:36 -04:00
mod embedded;
mod exercise;
mod init;
2024-04-06 21:03:37 -04:00
mod list;
2024-04-09 13:37:39 -04:00
mod progress_bar;
2019-01-09 14:33:43 -05:00
mod run;
2024-04-07 13:01:08 -04:00
mod state_file;
2019-01-09 14:33:58 -05:00
mod verify;
mod watch;
2018-05-14 12:41:58 -04:00
2024-04-07 18:36:26 -04:00
use self::{
consts::WELCOME,
exercise::{Exercise, InfoFile},
list::list,
2024-04-07 18:36:26 -04:00
run::run,
state_file::StateFile,
verify::{verify, VerifyState},
watch::{watch, WatchExit},
2024-04-07 18:36:26 -04:00
};
2024-04-07 13:01:08 -04:00
/// Rustlings is a collection of small exercises to get you used to writing and reading Rust code
2023-08-25 17:18:01 -04:00
#[derive(Parser)]
#[command(version)]
struct Args {
2023-08-25 17:18:01 -04:00
#[command(subcommand)]
command: Option<Subcommands>,
}
2023-08-25 17:18:01 -04:00
#[derive(Subcommand)]
enum Subcommands {
/// Initialize Rustlings
Init,
2023-08-25 17:18:01 -04:00
/// Verify all exercises according to the recommended order
Verify,
2024-04-04 21:04:53 -04:00
/// Same as just running `rustlings` without a subcommand.
2024-04-04 15:06:11 -04:00
Watch,
2023-08-25 17:18:01 -04:00
/// Run/Test a single exercise
Run {
/// The name of the exercise
name: String,
},
2024-03-28 17:11:16 -04:00
/// Reset a single exercise
2023-08-25 17:18:01 -04:00
Reset {
/// The name of the exercise
name: String,
},
/// Return a hint for the given exercise
Hint {
/// The name of the exercise
name: String,
},
}
2024-04-09 15:16:27 -04:00
fn find_exercise(name: &str, exercises: &'static [Exercise]) -> Result<(usize, &'static Exercise)> {
2024-04-07 16:43:59 -04:00
if name == "next" {
for (ind, exercise) in exercises.iter().enumerate() {
if !exercise.looks_done()? {
return Ok((ind, exercise));
}
}
println!("🎉 Congratulations! You have done all the exercises!");
println!("🔚 There are no more exercises to do next!");
exit(0);
}
exercises
.iter()
.enumerate()
.find(|(_, exercise)| exercise.name == name)
.with_context(|| format!("No exercise found for '{name}'!"))
}
2024-03-24 22:46:56 -04:00
fn main() -> Result<()> {
2023-08-25 17:18:01 -04:00
let args = Args::parse();
2024-03-31 12:25:54 -04:00
which::which("cargo").context(
"Failed to find `cargo`.
2024-03-31 10:55:33 -04:00
Did you already install Rust?
2024-03-31 12:25:54 -04:00
Try running `cargo --version` to diagnose the problem.",
)?;
let mut info_file = InfoFile::parse()?;
info_file.exercises.shrink_to_fit();
// Leaking is not a problem since the exercises' slice is used until the end of the program.
let exercises = info_file.exercises.leak();
if matches!(args.command, Some(Subcommands::Init)) {
2024-04-09 15:16:27 -04:00
init::init(exercises).context("Initialization failed")?;
2024-03-28 20:52:05 -04:00
println!(
"\nDone initialization!\n
Run `cd rustlings` to go into the generated directory.
Then run `rustlings` for further instructions on getting started."
);
return Ok(());
} else if !Path::new("exercises").is_dir() {
println!(
"
{WELCOME}
The `exercises` directory wasn't found in the current directory.
2024-03-28 20:52:05 -04:00
If you are just starting with Rustlings, run the command `rustlings init` to initialize it."
);
exit(1);
}
2024-04-09 15:16:27 -04:00
let mut state_file = StateFile::read_or_default(exercises);
2024-04-04 21:04:53 -04:00
match args.command {
None | Some(Subcommands::Watch) => loop {
match watch(&mut state_file, exercises)? {
WatchExit::Shutdown => break,
// It is much easier to exit the watch mode, launch the list mode and then restart
// the watch mode instead of trying to pause the watch threads and correct the
// watch state.
WatchExit::List => list(&mut state_file, exercises)?,
}
},
// `Init` is handled above.
2024-04-04 21:04:53 -04:00
Some(Subcommands::Init) => (),
Some(Subcommands::Run { name }) => {
2024-04-09 15:16:27 -04:00
let (_, exercise) = find_exercise(&name, exercises)?;
2024-04-04 15:06:11 -04:00
run(exercise).unwrap_or_else(|_| exit(1));
}
2024-04-04 21:04:53 -04:00
Some(Subcommands::Reset { name }) => {
2024-04-09 15:16:27 -04:00
let (ind, exercise) = find_exercise(&name, exercises)?;
2024-04-07 16:43:59 -04:00
exercise.reset()?;
state_file.reset(ind)?;
2024-04-07 19:33:11 -04:00
println!("The exercise {exercise} has been reset!");
}
2024-04-04 21:04:53 -04:00
Some(Subcommands::Hint { name }) => {
2024-04-09 15:16:27 -04:00
let (_, exercise) = find_exercise(&name, exercises)?;
println!("{}", exercise.hint);
}
2024-04-09 15:16:27 -04:00
Some(Subcommands::Verify) => match verify(exercises, 0)? {
2024-04-01 12:38:01 -04:00
VerifyState::AllExercisesDone => println!("All exercises done!"),
VerifyState::Failed(exercise) => bail!("Exercise {exercise} failed"),
},
}
2024-03-24 22:46:56 -04:00
Ok(())
2018-05-06 12:59:50 -04:00
}