2019-04-11 23:41:24 +03:00
|
|
|
use crate::exercise::{Exercise, ExerciseList};
|
2019-01-09 22:33:58 +03:00
|
|
|
use crate::run::run;
|
|
|
|
use crate::verify::verify;
|
feat: Replace clap with argh
I’ve been wanting to do this for a while, but always procrastinated on it. We’ve been using Clap since the 2.0 rewrite, but Clap is known to be a fairly heavy library. Since Rustlings is usually peoples’ first contact with a Rust compilation, I think it’s in our best interests that this complation is as fast as possible. In effect, replacing Clap with the smaller, structopt-style `argh` reduces the amount of crates needing to be compiled from 82 to 60.
I also think this makes the code way easier to read, we don’t need to use Clap’s methods anymore, but can switch over to using pure Rust methods, e.g., switches are booleans, options are Option<String>s or the like, and subcommands are just structs.
2021-04-20 13:46:49 +03:00
|
|
|
use argh::FromArgs;
|
2019-12-22 23:27:38 +03:00
|
|
|
use console::Emoji;
|
2019-01-09 22:33:58 +03:00
|
|
|
use notify::DebouncedEvent;
|
|
|
|
use notify::{RecommendedWatcher, RecursiveMode, Watcher};
|
2019-03-06 21:38:55 +03:00
|
|
|
use std::ffi::OsStr;
|
2019-04-11 23:41:24 +03:00
|
|
|
use std::fs;
|
2020-12-12 21:48:25 +03:00
|
|
|
use std::io::{self, prelude::*};
|
2019-03-17 05:15:09 +03:00
|
|
|
use std::path::Path;
|
2019-11-11 19:15:14 +03:00
|
|
|
use std::process::{Command, Stdio};
|
2021-09-14 13:34:40 +03:00
|
|
|
use std::sync::atomic::{AtomicBool, Ordering};
|
2021-09-25 12:23:05 +03:00
|
|
|
use std::sync::mpsc::{channel, RecvTimeoutError};
|
2019-11-18 20:11:22 +03:00
|
|
|
use std::sync::{Arc, Mutex};
|
|
|
|
use std::thread;
|
2018-11-26 14:41:39 +03:00
|
|
|
use std::time::Duration;
|
2019-01-09 22:33:43 +03:00
|
|
|
|
2020-02-20 22:11:53 +03:00
|
|
|
#[macro_use]
|
|
|
|
mod ui;
|
|
|
|
|
2019-04-11 23:41:24 +03:00
|
|
|
mod exercise;
|
2019-01-09 22:33:43 +03:00
|
|
|
mod run;
|
2019-01-09 22:33:58 +03:00
|
|
|
mod verify;
|
2018-05-14 19:41:58 +03:00
|
|
|
|
feat: Replace clap with argh
I’ve been wanting to do this for a while, but always procrastinated on it. We’ve been using Clap since the 2.0 rewrite, but Clap is known to be a fairly heavy library. Since Rustlings is usually peoples’ first contact with a Rust compilation, I think it’s in our best interests that this complation is as fast as possible. In effect, replacing Clap with the smaller, structopt-style `argh` reduces the amount of crates needing to be compiled from 82 to 60.
I also think this makes the code way easier to read, we don’t need to use Clap’s methods anymore, but can switch over to using pure Rust methods, e.g., switches are booleans, options are Option<String>s or the like, and subcommands are just structs.
2021-04-20 13:46:49 +03:00
|
|
|
// In sync with crate version
|
2021-09-25 12:23:05 +03:00
|
|
|
const VERSION: &str = "4.6.0";
|
feat: Replace clap with argh
I’ve been wanting to do this for a while, but always procrastinated on it. We’ve been using Clap since the 2.0 rewrite, but Clap is known to be a fairly heavy library. Since Rustlings is usually peoples’ first contact with a Rust compilation, I think it’s in our best interests that this complation is as fast as possible. In effect, replacing Clap with the smaller, structopt-style `argh` reduces the amount of crates needing to be compiled from 82 to 60.
I also think this makes the code way easier to read, we don’t need to use Clap’s methods anymore, but can switch over to using pure Rust methods, e.g., switches are booleans, options are Option<String>s or the like, and subcommands are just structs.
2021-04-20 13:46:49 +03:00
|
|
|
|
|
|
|
#[derive(FromArgs, PartialEq, Debug)]
|
|
|
|
/// Rustlings is a collection of small exercises to get you used to writing and reading Rust code
|
|
|
|
struct Args {
|
|
|
|
/// show outputs from the test exercises
|
|
|
|
#[argh(switch)]
|
|
|
|
nocapture: bool,
|
|
|
|
/// show the executable version
|
|
|
|
#[argh(switch, short = 'v')]
|
|
|
|
version: bool,
|
|
|
|
#[argh(subcommand)]
|
|
|
|
nested: Option<Subcommands>,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(FromArgs, PartialEq, Debug)]
|
|
|
|
#[argh(subcommand)]
|
|
|
|
enum Subcommands {
|
|
|
|
Verify(VerifyArgs),
|
|
|
|
Watch(WatchArgs),
|
|
|
|
Run(RunArgs),
|
|
|
|
Hint(HintArgs),
|
|
|
|
List(ListArgs),
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(FromArgs, PartialEq, Debug)]
|
|
|
|
#[argh(subcommand, name = "verify")]
|
|
|
|
/// Verifies all exercises according to the recommended order
|
|
|
|
struct VerifyArgs {}
|
|
|
|
|
|
|
|
#[derive(FromArgs, PartialEq, Debug)]
|
|
|
|
#[argh(subcommand, name = "watch")]
|
|
|
|
/// Reruns `verify` when files were edited
|
|
|
|
struct WatchArgs {}
|
|
|
|
|
|
|
|
#[derive(FromArgs, PartialEq, Debug)]
|
|
|
|
#[argh(subcommand, name = "run")]
|
|
|
|
/// Runs/Tests a single exercise
|
|
|
|
struct RunArgs {
|
|
|
|
#[argh(positional)]
|
|
|
|
/// the name of the exercise
|
|
|
|
name: String,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(FromArgs, PartialEq, Debug)]
|
|
|
|
#[argh(subcommand, name = "hint")]
|
|
|
|
/// Returns a hint for the given exercise
|
|
|
|
struct HintArgs {
|
|
|
|
#[argh(positional)]
|
|
|
|
/// the name of the exercise
|
|
|
|
name: String,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(FromArgs, PartialEq, Debug)]
|
|
|
|
#[argh(subcommand, name = "list")]
|
|
|
|
/// Lists the exercises available in Rustlings
|
|
|
|
struct ListArgs {
|
|
|
|
#[argh(switch, short = 'p')]
|
|
|
|
/// show only the paths of the exercises
|
|
|
|
paths: bool,
|
|
|
|
#[argh(switch, short = 'n')]
|
|
|
|
/// show only the names of the exercises
|
|
|
|
names: bool,
|
|
|
|
#[argh(option, short = 'f')]
|
|
|
|
/// provide a string to match exercise names
|
|
|
|
/// comma separated patterns are acceptable
|
|
|
|
filter: Option<String>,
|
|
|
|
#[argh(switch, short = 'u')]
|
|
|
|
/// display only exercises not yet solved
|
|
|
|
unsolved: bool,
|
|
|
|
#[argh(switch, short = 's')]
|
|
|
|
/// display only exercises that have been solved
|
|
|
|
solved: bool,
|
|
|
|
}
|
|
|
|
|
2018-11-09 22:31:14 +03:00
|
|
|
fn main() {
|
feat: Replace clap with argh
I’ve been wanting to do this for a while, but always procrastinated on it. We’ve been using Clap since the 2.0 rewrite, but Clap is known to be a fairly heavy library. Since Rustlings is usually peoples’ first contact with a Rust compilation, I think it’s in our best interests that this complation is as fast as possible. In effect, replacing Clap with the smaller, structopt-style `argh` reduces the amount of crates needing to be compiled from 82 to 60.
I also think this makes the code way easier to read, we don’t need to use Clap’s methods anymore, but can switch over to using pure Rust methods, e.g., switches are booleans, options are Option<String>s or the like, and subcommands are just structs.
2021-04-20 13:46:49 +03:00
|
|
|
let args: Args = argh::from_env();
|
|
|
|
|
|
|
|
if args.version {
|
|
|
|
println!("v{}", VERSION);
|
|
|
|
std::process::exit(0);
|
|
|
|
}
|
2018-11-14 22:12:20 +03:00
|
|
|
|
feat: Replace clap with argh
I’ve been wanting to do this for a while, but always procrastinated on it. We’ve been using Clap since the 2.0 rewrite, but Clap is known to be a fairly heavy library. Since Rustlings is usually peoples’ first contact with a Rust compilation, I think it’s in our best interests that this complation is as fast as possible. In effect, replacing Clap with the smaller, structopt-style `argh` reduces the amount of crates needing to be compiled from 82 to 60.
I also think this makes the code way easier to read, we don’t need to use Clap’s methods anymore, but can switch over to using pure Rust methods, e.g., switches are booleans, options are Option<String>s or the like, and subcommands are just structs.
2021-04-20 13:46:49 +03:00
|
|
|
if args.nested.is_none() {
|
2019-03-11 17:09:20 +03:00
|
|
|
println!();
|
2019-01-09 22:44:55 +03:00
|
|
|
println!(r#" welcome to... "#);
|
|
|
|
println!(r#" _ _ _ "#);
|
|
|
|
println!(r#" _ __ _ _ ___| |_| (_)_ __ __ _ ___ "#);
|
|
|
|
println!(r#" | '__| | | / __| __| | | '_ \ / _` / __| "#);
|
|
|
|
println!(r#" | | | |_| \__ \ |_| | | | | | (_| \__ \ "#);
|
|
|
|
println!(r#" |_| \__,_|___/\__|_|_|_| |_|\__, |___/ "#);
|
|
|
|
println!(r#" |___/ "#);
|
2019-03-11 17:09:20 +03:00
|
|
|
println!();
|
2019-01-09 22:44:55 +03:00
|
|
|
}
|
2018-11-14 22:12:20 +03:00
|
|
|
|
2019-03-17 05:15:09 +03:00
|
|
|
if !Path::new("info.toml").exists() {
|
|
|
|
println!(
|
|
|
|
"{} must be run from the rustlings directory",
|
|
|
|
std::env::current_exe().unwrap().to_str().unwrap()
|
|
|
|
);
|
2019-04-22 13:43:39 +03:00
|
|
|
println!("Try `cd rustlings/`!");
|
2019-03-17 05:15:09 +03:00
|
|
|
std::process::exit(1);
|
|
|
|
}
|
|
|
|
|
2019-11-11 19:15:14 +03:00
|
|
|
if !rustc_exists() {
|
|
|
|
println!("We cannot find `rustc`.");
|
|
|
|
println!("Try running `rustc --version` to diagnose your problem.");
|
|
|
|
println!("For instructions on how to install Rust, check the README.");
|
|
|
|
std::process::exit(1);
|
|
|
|
}
|
|
|
|
|
2019-04-11 23:41:24 +03:00
|
|
|
let toml_str = &fs::read_to_string("info.toml").unwrap();
|
|
|
|
let exercises = toml::from_str::<ExerciseList>(toml_str).unwrap().exercises;
|
feat: Replace clap with argh
I’ve been wanting to do this for a while, but always procrastinated on it. We’ve been using Clap since the 2.0 rewrite, but Clap is known to be a fairly heavy library. Since Rustlings is usually peoples’ first contact with a Rust compilation, I think it’s in our best interests that this complation is as fast as possible. In effect, replacing Clap with the smaller, structopt-style `argh` reduces the amount of crates needing to be compiled from 82 to 60.
I also think this makes the code way easier to read, we don’t need to use Clap’s methods anymore, but can switch over to using pure Rust methods, e.g., switches are booleans, options are Option<String>s or the like, and subcommands are just structs.
2021-04-20 13:46:49 +03:00
|
|
|
let verbose = args.nocapture;
|
2019-04-11 23:41:24 +03:00
|
|
|
|
feat: Replace clap with argh
I’ve been wanting to do this for a while, but always procrastinated on it. We’ve been using Clap since the 2.0 rewrite, but Clap is known to be a fairly heavy library. Since Rustlings is usually peoples’ first contact with a Rust compilation, I think it’s in our best interests that this complation is as fast as possible. In effect, replacing Clap with the smaller, structopt-style `argh` reduces the amount of crates needing to be compiled from 82 to 60.
I also think this makes the code way easier to read, we don’t need to use Clap’s methods anymore, but can switch over to using pure Rust methods, e.g., switches are booleans, options are Option<String>s or the like, and subcommands are just structs.
2021-04-20 13:46:49 +03:00
|
|
|
let command = args.nested.unwrap_or_else(|| {
|
|
|
|
let text = fs::read_to_string("default_out.txt").unwrap();
|
|
|
|
println!("{}", text);
|
|
|
|
std::process::exit(0);
|
|
|
|
});
|
|
|
|
match command {
|
|
|
|
Subcommands::List(subargs) => {
|
|
|
|
if !subargs.paths && !subargs.names {
|
|
|
|
println!("{:<17}\t{:<46}\t{:<7}", "Name", "Path", "Status");
|
|
|
|
}
|
|
|
|
let mut exercises_done: u16 = 0;
|
|
|
|
let filters = subargs.filter.clone().unwrap_or_default().to_lowercase();
|
|
|
|
exercises.iter().for_each(|e| {
|
|
|
|
let fname = format!("{}", e.path.display());
|
|
|
|
let filter_cond = filters
|
|
|
|
.split(',')
|
|
|
|
.filter(|f| !f.trim().is_empty())
|
|
|
|
.any(|f| e.name.contains(&f) || fname.contains(&f));
|
|
|
|
let status = if e.looks_done() {
|
|
|
|
exercises_done += 1;
|
|
|
|
"Done"
|
2020-12-12 21:48:25 +03:00
|
|
|
} else {
|
feat: Replace clap with argh
I’ve been wanting to do this for a while, but always procrastinated on it. We’ve been using Clap since the 2.0 rewrite, but Clap is known to be a fairly heavy library. Since Rustlings is usually peoples’ first contact with a Rust compilation, I think it’s in our best interests that this complation is as fast as possible. In effect, replacing Clap with the smaller, structopt-style `argh` reduces the amount of crates needing to be compiled from 82 to 60.
I also think this makes the code way easier to read, we don’t need to use Clap’s methods anymore, but can switch over to using pure Rust methods, e.g., switches are booleans, options are Option<String>s or the like, and subcommands are just structs.
2021-04-20 13:46:49 +03:00
|
|
|
"Pending"
|
2020-12-12 21:48:25 +03:00
|
|
|
};
|
feat: Replace clap with argh
I’ve been wanting to do this for a while, but always procrastinated on it. We’ve been using Clap since the 2.0 rewrite, but Clap is known to be a fairly heavy library. Since Rustlings is usually peoples’ first contact with a Rust compilation, I think it’s in our best interests that this complation is as fast as possible. In effect, replacing Clap with the smaller, structopt-style `argh` reduces the amount of crates needing to be compiled from 82 to 60.
I also think this makes the code way easier to read, we don’t need to use Clap’s methods anymore, but can switch over to using pure Rust methods, e.g., switches are booleans, options are Option<String>s or the like, and subcommands are just structs.
2021-04-20 13:46:49 +03:00
|
|
|
let solve_cond = {
|
|
|
|
(e.looks_done() && subargs.solved)
|
|
|
|
|| (!e.looks_done() && subargs.unsolved)
|
|
|
|
|| (!subargs.solved && !subargs.unsolved)
|
|
|
|
};
|
|
|
|
if solve_cond && (filter_cond || subargs.filter.is_none()) {
|
|
|
|
let line = if subargs.paths {
|
|
|
|
format!("{}\n", fname)
|
|
|
|
} else if subargs.names {
|
|
|
|
format!("{}\n", e.name)
|
|
|
|
} else {
|
|
|
|
format!("{:<17}\t{:<46}\t{:<7}\n", e.name, fname, status)
|
|
|
|
};
|
|
|
|
// Somehow using println! leads to the binary panicking
|
|
|
|
// when its output is piped.
|
|
|
|
// So, we're handling a Broken Pipe error and exiting with 0 anyway
|
|
|
|
let stdout = std::io::stdout();
|
|
|
|
{
|
|
|
|
let mut handle = stdout.lock();
|
|
|
|
handle.write_all(line.as_bytes()).unwrap_or_else(|e| {
|
|
|
|
match e.kind() {
|
|
|
|
std::io::ErrorKind::BrokenPipe => std::process::exit(0),
|
|
|
|
_ => std::process::exit(1),
|
|
|
|
};
|
|
|
|
});
|
|
|
|
}
|
2020-12-12 21:48:25 +03:00
|
|
|
}
|
feat: Replace clap with argh
I’ve been wanting to do this for a while, but always procrastinated on it. We’ve been using Clap since the 2.0 rewrite, but Clap is known to be a fairly heavy library. Since Rustlings is usually peoples’ first contact with a Rust compilation, I think it’s in our best interests that this complation is as fast as possible. In effect, replacing Clap with the smaller, structopt-style `argh` reduces the amount of crates needing to be compiled from 82 to 60.
I also think this makes the code way easier to read, we don’t need to use Clap’s methods anymore, but can switch over to using pure Rust methods, e.g., switches are booleans, options are Option<String>s or the like, and subcommands are just structs.
2021-04-20 13:46:49 +03:00
|
|
|
});
|
|
|
|
let percentage_progress = exercises_done as f32 / exercises.len() as f32 * 100.0;
|
|
|
|
println!(
|
|
|
|
"Progress: You completed {} / {} exercises ({:.2} %).",
|
|
|
|
exercises_done,
|
|
|
|
exercises.len(),
|
|
|
|
percentage_progress
|
|
|
|
);
|
|
|
|
std::process::exit(0);
|
|
|
|
}
|
2019-04-11 23:41:24 +03:00
|
|
|
|
feat: Replace clap with argh
I’ve been wanting to do this for a while, but always procrastinated on it. We’ve been using Clap since the 2.0 rewrite, but Clap is known to be a fairly heavy library. Since Rustlings is usually peoples’ first contact with a Rust compilation, I think it’s in our best interests that this complation is as fast as possible. In effect, replacing Clap with the smaller, structopt-style `argh` reduces the amount of crates needing to be compiled from 82 to 60.
I also think this makes the code way easier to read, we don’t need to use Clap’s methods anymore, but can switch over to using pure Rust methods, e.g., switches are booleans, options are Option<String>s or the like, and subcommands are just structs.
2021-04-20 13:46:49 +03:00
|
|
|
Subcommands::Run(subargs) => {
|
2021-04-21 17:21:56 +03:00
|
|
|
let exercise = find_exercise(&subargs.name, &exercises);
|
2018-11-23 17:18:43 +03:00
|
|
|
|
2021-08-24 15:06:30 +03:00
|
|
|
run(exercise, verbose).unwrap_or_else(|_| std::process::exit(1));
|
feat: Replace clap with argh
I’ve been wanting to do this for a while, but always procrastinated on it. We’ve been using Clap since the 2.0 rewrite, but Clap is known to be a fairly heavy library. Since Rustlings is usually peoples’ first contact with a Rust compilation, I think it’s in our best interests that this complation is as fast as possible. In effect, replacing Clap with the smaller, structopt-style `argh` reduces the amount of crates needing to be compiled from 82 to 60.
I also think this makes the code way easier to read, we don’t need to use Clap’s methods anymore, but can switch over to using pure Rust methods, e.g., switches are booleans, options are Option<String>s or the like, and subcommands are just structs.
2021-04-20 13:46:49 +03:00
|
|
|
}
|
2019-11-11 18:51:38 +03:00
|
|
|
|
feat: Replace clap with argh
I’ve been wanting to do this for a while, but always procrastinated on it. We’ve been using Clap since the 2.0 rewrite, but Clap is known to be a fairly heavy library. Since Rustlings is usually peoples’ first contact with a Rust compilation, I think it’s in our best interests that this complation is as fast as possible. In effect, replacing Clap with the smaller, structopt-style `argh` reduces the amount of crates needing to be compiled from 82 to 60.
I also think this makes the code way easier to read, we don’t need to use Clap’s methods anymore, but can switch over to using pure Rust methods, e.g., switches are booleans, options are Option<String>s or the like, and subcommands are just structs.
2021-04-20 13:46:49 +03:00
|
|
|
Subcommands::Hint(subargs) => {
|
2021-04-21 17:21:56 +03:00
|
|
|
let exercise = find_exercise(&subargs.name, &exercises);
|
2019-11-11 18:51:38 +03:00
|
|
|
|
feat: Replace clap with argh
I’ve been wanting to do this for a while, but always procrastinated on it. We’ve been using Clap since the 2.0 rewrite, but Clap is known to be a fairly heavy library. Since Rustlings is usually peoples’ first contact with a Rust compilation, I think it’s in our best interests that this complation is as fast as possible. In effect, replacing Clap with the smaller, structopt-style `argh` reduces the amount of crates needing to be compiled from 82 to 60.
I also think this makes the code way easier to read, we don’t need to use Clap’s methods anymore, but can switch over to using pure Rust methods, e.g., switches are booleans, options are Option<String>s or the like, and subcommands are just structs.
2021-04-20 13:46:49 +03:00
|
|
|
println!("{}", exercise.hint);
|
|
|
|
}
|
2019-11-11 18:51:38 +03:00
|
|
|
|
feat: Replace clap with argh
I’ve been wanting to do this for a while, but always procrastinated on it. We’ve been using Clap since the 2.0 rewrite, but Clap is known to be a fairly heavy library. Since Rustlings is usually peoples’ first contact with a Rust compilation, I think it’s in our best interests that this complation is as fast as possible. In effect, replacing Clap with the smaller, structopt-style `argh` reduces the amount of crates needing to be compiled from 82 to 60.
I also think this makes the code way easier to read, we don’t need to use Clap’s methods anymore, but can switch over to using pure Rust methods, e.g., switches are booleans, options are Option<String>s or the like, and subcommands are just structs.
2021-04-20 13:46:49 +03:00
|
|
|
Subcommands::Verify(_subargs) => {
|
|
|
|
verify(&exercises, verbose).unwrap_or_else(|_| std::process::exit(1));
|
|
|
|
}
|
2018-11-26 14:41:39 +03:00
|
|
|
|
2021-09-25 12:23:05 +03:00
|
|
|
Subcommands::Watch(_subargs) => match watch(&exercises, verbose) {
|
|
|
|
Err(e) => {
|
|
|
|
println!(
|
|
|
|
"Error: Could not watch your progress. Error message was {:?}.",
|
|
|
|
e
|
|
|
|
);
|
|
|
|
println!("Most likely you've run out of disk space or your 'inotify limit' has been reached.");
|
|
|
|
std::process::exit(1);
|
feat: Replace clap with argh
I’ve been wanting to do this for a while, but always procrastinated on it. We’ve been using Clap since the 2.0 rewrite, but Clap is known to be a fairly heavy library. Since Rustlings is usually peoples’ first contact with a Rust compilation, I think it’s in our best interests that this complation is as fast as possible. In effect, replacing Clap with the smaller, structopt-style `argh` reduces the amount of crates needing to be compiled from 82 to 60.
I also think this makes the code way easier to read, we don’t need to use Clap’s methods anymore, but can switch over to using pure Rust methods, e.g., switches are booleans, options are Option<String>s or the like, and subcommands are just structs.
2021-04-20 13:46:49 +03:00
|
|
|
}
|
2021-09-25 12:23:05 +03:00
|
|
|
Ok(WatchStatus::Finished) => {
|
|
|
|
println!(
|
|
|
|
"{emoji} All exercises completed! {emoji}",
|
|
|
|
emoji = Emoji("🎉", "★")
|
|
|
|
);
|
|
|
|
println!();
|
|
|
|
println!("+----------------------------------------------------+");
|
|
|
|
println!("| You made it to the Fe-nish line! |");
|
|
|
|
println!("+-------------------------- ------------------------+");
|
|
|
|
println!(" \\/ ");
|
|
|
|
println!(" ▒▒ ▒▒▒▒▒▒▒▒ ▒▒▒▒▒▒▒▒ ▒▒ ");
|
|
|
|
println!(" ▒▒▒▒ ▒▒ ▒▒ ▒▒ ▒▒ ▒▒ ▒▒ ▒▒▒▒ ");
|
|
|
|
println!(" ▒▒▒▒ ▒▒ ▒▒ ▒▒ ▒▒ ▒▒ ▒▒▒▒ ");
|
|
|
|
println!(" ░░▒▒▒▒░░▒▒ ▒▒ ▒▒ ▒▒ ▒▒░░▒▒▒▒ ");
|
|
|
|
println!(" ▓▓▓▓▓▓▓▓ ▓▓ ▓▓██ ▓▓ ▓▓██ ▓▓ ▓▓▓▓▓▓▓▓ ");
|
|
|
|
println!(" ▒▒▒▒ ▒▒ ████ ▒▒ ████ ▒▒░░ ▒▒▒▒ ");
|
|
|
|
println!(" ▒▒ ▒▒▒▒▒▒ ▒▒▒▒▒▒ ▒▒▒▒▒▒ ▒▒ ");
|
|
|
|
println!(" ▒▒▒▒▒▒▒▒▒▒▓▓▓▓▓▓▒▒▒▒▒▒▒▒▓▓▒▒▓▓▒▒▒▒▒▒▒▒ ");
|
|
|
|
println!(" ▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒ ");
|
|
|
|
println!(" ▒▒▒▒▒▒▒▒▒▒██▒▒▒▒▒▒██▒▒▒▒▒▒▒▒▒▒ ");
|
|
|
|
println!(" ▒▒ ▒▒▒▒▒▒▒▒▒▒██████▒▒▒▒▒▒▒▒▒▒ ▒▒ ");
|
|
|
|
println!(" ▒▒ ▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒ ▒▒ ");
|
|
|
|
println!(" ▒▒ ▒▒ ▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒ ▒▒ ▒▒ ");
|
|
|
|
println!(" ▒▒ ▒▒ ▒▒ ▒▒ ▒▒ ▒▒ ");
|
|
|
|
println!(" ▒▒ ▒▒ ▒▒ ▒▒ ");
|
|
|
|
println!();
|
|
|
|
println!("We hope you enjoyed learning about the various aspects of Rust!");
|
|
|
|
println!(
|
|
|
|
"If you noticed any issues, please don't hesitate to report them to our repo."
|
|
|
|
);
|
|
|
|
println!(
|
|
|
|
"You can also contribute your own exercises to help the greater community!"
|
|
|
|
);
|
|
|
|
println!();
|
|
|
|
println!("Before reporting an issue or contributing, please read our guidelines:");
|
|
|
|
println!("https://github.com/rust-lang/rustlings/blob/main/CONTRIBUTING.md");
|
|
|
|
}
|
|
|
|
Ok(WatchStatus::Unfinished) => {
|
|
|
|
println!("We hope you're enjoying learning about Rust!");
|
|
|
|
println!("If you want to continue working on the exercises at a later point, you can simply run `rustlings watch` again");
|
|
|
|
}
|
|
|
|
},
|
2018-11-26 13:10:38 +03:00
|
|
|
}
|
2018-05-06 19:59:50 +03:00
|
|
|
}
|
|
|
|
|
2021-09-25 12:23:05 +03:00
|
|
|
fn spawn_watch_shell(
|
|
|
|
failed_exercise_hint: &Arc<Mutex<Option<String>>>,
|
|
|
|
should_quit: Arc<AtomicBool>,
|
|
|
|
) {
|
2019-11-18 20:11:22 +03:00
|
|
|
let failed_exercise_hint = Arc::clone(failed_exercise_hint);
|
2021-09-21 11:36:11 +03:00
|
|
|
println!("Welcome to watch mode! You can type 'help' to get an overview of the commands you can use here.");
|
2019-11-18 20:11:22 +03:00
|
|
|
thread::spawn(move || loop {
|
|
|
|
let mut input = String::new();
|
|
|
|
match io::stdin().read_line(&mut input) {
|
|
|
|
Ok(_) => {
|
2020-07-23 21:23:27 +03:00
|
|
|
let input = input.trim();
|
2021-09-14 13:34:40 +03:00
|
|
|
if input == "hint" {
|
2019-11-18 20:11:22 +03:00
|
|
|
if let Some(hint) = &*failed_exercise_hint.lock().unwrap() {
|
|
|
|
println!("{}", hint);
|
|
|
|
}
|
2021-09-14 13:34:40 +03:00
|
|
|
} else if input == "clear" {
|
2020-07-23 21:23:27 +03:00
|
|
|
println!("\x1B[2J\x1B[1;1H");
|
2021-09-21 11:36:11 +03:00
|
|
|
} else if input.eq("quit") {
|
2021-09-14 13:34:40 +03:00
|
|
|
should_quit.store(true, Ordering::SeqCst);
|
2021-09-21 11:36:11 +03:00
|
|
|
println!("Bye!");
|
|
|
|
} else if input.eq("help") {
|
|
|
|
println!("Commands available to you in watch mode:");
|
|
|
|
println!(" hint - prints the current exercise's hint");
|
|
|
|
println!(" clear - clears the screen");
|
|
|
|
println!(" quit - quits watch mode");
|
|
|
|
println!(" help - displays this help message");
|
2021-09-24 14:04:30 +03:00
|
|
|
println!();
|
2021-09-21 11:36:11 +03:00
|
|
|
println!("Watch mode automatically re-evaluates the current exercise");
|
|
|
|
println!("when you edit a file's contents.")
|
2019-11-18 20:11:22 +03:00
|
|
|
} else {
|
|
|
|
println!("unknown command: {}", input);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Err(error) => println!("error reading command: {}", error),
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-04-21 17:21:56 +03:00
|
|
|
fn find_exercise<'a>(name: &str, exercises: &'a [Exercise]) -> &'a Exercise {
|
2021-06-30 13:05:49 +03:00
|
|
|
if name.eq("next") {
|
|
|
|
exercises
|
|
|
|
.iter()
|
|
|
|
.find(|e| !e.looks_done())
|
|
|
|
.unwrap_or_else(|| {
|
|
|
|
println!("🎉 Congratulations! You have done all the exercises!");
|
|
|
|
println!("🔚 There are no more exercises to do next!");
|
|
|
|
std::process::exit(1)
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
exercises
|
|
|
|
.iter()
|
|
|
|
.find(|e| e.name == name)
|
|
|
|
.unwrap_or_else(|| {
|
|
|
|
println!("No exercise found for '{}'!", name);
|
|
|
|
std::process::exit(1)
|
|
|
|
})
|
|
|
|
}
|
feat: Replace clap with argh
I’ve been wanting to do this for a while, but always procrastinated on it. We’ve been using Clap since the 2.0 rewrite, but Clap is known to be a fairly heavy library. Since Rustlings is usually peoples’ first contact with a Rust compilation, I think it’s in our best interests that this complation is as fast as possible. In effect, replacing Clap with the smaller, structopt-style `argh` reduces the amount of crates needing to be compiled from 82 to 60.
I also think this makes the code way easier to read, we don’t need to use Clap’s methods anymore, but can switch over to using pure Rust methods, e.g., switches are booleans, options are Option<String>s or the like, and subcommands are just structs.
2021-04-20 13:46:49 +03:00
|
|
|
}
|
|
|
|
|
2021-09-14 13:34:40 +03:00
|
|
|
enum WatchStatus {
|
|
|
|
Finished,
|
|
|
|
Unfinished,
|
|
|
|
}
|
|
|
|
|
|
|
|
fn watch(exercises: &[Exercise], verbose: bool) -> notify::Result<WatchStatus> {
|
2019-11-09 17:24:24 +03:00
|
|
|
/* Clears the terminal with an ANSI escape code.
|
|
|
|
Works in UNIX and newer Windows terminals. */
|
|
|
|
fn clear_screen() {
|
|
|
|
println!("\x1Bc");
|
|
|
|
}
|
|
|
|
|
2018-11-26 14:41:39 +03:00
|
|
|
let (tx, rx) = channel();
|
2021-09-14 13:34:40 +03:00
|
|
|
let should_quit = Arc::new(AtomicBool::new(false));
|
2018-11-26 14:41:39 +03:00
|
|
|
|
|
|
|
let mut watcher: RecommendedWatcher = Watcher::new(tx, Duration::from_secs(2))?;
|
2019-03-27 12:58:56 +03:00
|
|
|
watcher.watch(Path::new("./exercises"), RecursiveMode::Recursive)?;
|
2018-11-26 14:41:39 +03:00
|
|
|
|
2019-11-09 17:24:24 +03:00
|
|
|
clear_screen();
|
2018-11-26 14:41:39 +03:00
|
|
|
|
2019-11-18 20:11:22 +03:00
|
|
|
let to_owned_hint = |t: &Exercise| t.hint.to_owned();
|
2020-06-04 17:31:17 +03:00
|
|
|
let failed_exercise_hint = match verify(exercises.iter(), verbose) {
|
2021-09-14 13:34:40 +03:00
|
|
|
Ok(_) => return Ok(WatchStatus::Finished),
|
2019-12-22 23:27:38 +03:00
|
|
|
Err(exercise) => Arc::new(Mutex::new(Some(to_owned_hint(exercise)))),
|
|
|
|
};
|
2021-09-14 13:34:40 +03:00
|
|
|
spawn_watch_shell(&failed_exercise_hint, Arc::clone(&should_quit));
|
2018-11-26 14:41:39 +03:00
|
|
|
loop {
|
2021-09-14 13:34:40 +03:00
|
|
|
match rx.recv_timeout(Duration::from_secs(1)) {
|
2019-01-09 22:33:58 +03:00
|
|
|
Ok(event) => match event {
|
2019-03-06 21:38:55 +03:00
|
|
|
DebouncedEvent::Create(b) | DebouncedEvent::Chmod(b) | DebouncedEvent::Write(b) => {
|
2019-07-12 00:54:18 +03:00
|
|
|
if b.extension() == Some(OsStr::new("rs")) && b.exists() {
|
2019-04-11 23:41:24 +03:00
|
|
|
let filepath = b.as_path().canonicalize().unwrap();
|
2019-11-11 15:38:24 +03:00
|
|
|
let pending_exercises = exercises
|
2019-04-11 23:41:24 +03:00
|
|
|
.iter()
|
2020-12-12 21:48:25 +03:00
|
|
|
.skip_while(|e| !filepath.ends_with(&e.path))
|
|
|
|
// .filter(|e| filepath.ends_with(&e.path))
|
|
|
|
.chain(
|
|
|
|
exercises
|
|
|
|
.iter()
|
2021-04-18 16:37:41 +03:00
|
|
|
.filter(|e| !e.looks_done() && !filepath.ends_with(&e.path)),
|
2020-12-12 21:48:25 +03:00
|
|
|
);
|
2019-11-09 17:24:24 +03:00
|
|
|
clear_screen();
|
2020-06-04 17:31:17 +03:00
|
|
|
match verify(pending_exercises, verbose) {
|
2021-09-14 13:34:40 +03:00
|
|
|
Ok(_) => return Ok(WatchStatus::Finished),
|
2019-12-22 23:27:38 +03:00
|
|
|
Err(exercise) => {
|
|
|
|
let mut failed_exercise_hint = failed_exercise_hint.lock().unwrap();
|
|
|
|
*failed_exercise_hint = Some(to_owned_hint(exercise));
|
|
|
|
}
|
|
|
|
}
|
2019-03-06 21:38:55 +03:00
|
|
|
}
|
2018-11-26 14:41:39 +03:00
|
|
|
}
|
2019-01-09 22:33:58 +03:00
|
|
|
_ => {}
|
2018-11-26 14:41:39 +03:00
|
|
|
},
|
2021-09-14 13:34:40 +03:00
|
|
|
Err(RecvTimeoutError::Timeout) => {
|
|
|
|
// the timeout expired, just check the `should_quit` variable below then loop again
|
|
|
|
}
|
2018-11-26 14:41:39 +03:00
|
|
|
Err(e) => println!("watch error: {:?}", e),
|
|
|
|
}
|
2021-09-14 13:34:40 +03:00
|
|
|
// Check if we need to exit
|
|
|
|
if should_quit.load(Ordering::SeqCst) {
|
|
|
|
return Ok(WatchStatus::Unfinished);
|
|
|
|
}
|
2018-11-26 14:41:39 +03:00
|
|
|
}
|
|
|
|
}
|
2019-11-11 19:15:14 +03:00
|
|
|
|
|
|
|
fn rustc_exists() -> bool {
|
|
|
|
Command::new("rustc")
|
|
|
|
.args(&["--version"])
|
|
|
|
.stdout(Stdio::null())
|
|
|
|
.spawn()
|
|
|
|
.and_then(|mut child| child.wait())
|
|
|
|
.map(|status| status.success())
|
|
|
|
.unwrap_or(false)
|
|
|
|
}
|