2024-08-28 18:17:22 -04:00
|
|
|
use anyhow::{bail, Context, Result};
|
2024-04-13 23:13:27 -04:00
|
|
|
use std::{
|
2024-08-27 19:19:53 -04:00
|
|
|
env,
|
2024-08-28 19:56:45 -04:00
|
|
|
fs::{File, OpenOptions},
|
|
|
|
io::{self, Read, Seek, StdoutLock, Write},
|
2024-09-03 20:19:45 -04:00
|
|
|
path::{Path, MAIN_SEPARATOR_STR},
|
2024-04-18 06:41:17 -04:00
|
|
|
process::{Command, Stdio},
|
2024-10-02 17:42:50 -04:00
|
|
|
sync::{atomic::AtomicUsize, mpsc, Arc},
|
2024-07-28 11:39:46 -04:00
|
|
|
thread,
|
2024-04-13 23:13:27 -04:00
|
|
|
};
|
2024-04-10 20:51:02 -04:00
|
|
|
|
2024-04-24 19:56:01 -04:00
|
|
|
use crate::{
|
2024-04-29 19:41:08 -04:00
|
|
|
clear_terminal,
|
2024-08-01 09:23:54 -04:00
|
|
|
cmd::CmdRunner,
|
2024-08-08 17:46:21 -04:00
|
|
|
collections::hash_set_with_capacity,
|
2024-04-24 19:56:01 -04:00
|
|
|
embedded::EMBEDDED_FILES,
|
2024-07-28 14:30:23 -04:00
|
|
|
exercise::{Exercise, RunnableExercise},
|
2024-04-24 19:56:01 -04:00
|
|
|
info_file::ExerciseInfo,
|
2024-09-03 20:19:45 -04:00
|
|
|
term,
|
2024-04-24 19:56:01 -04:00
|
|
|
};
|
2024-04-10 20:51:02 -04:00
|
|
|
|
2024-04-13 23:13:27 -04:00
|
|
|
const STATE_FILE_NAME: &str = ".rustlings-state.txt";
|
2024-10-02 17:42:50 -04:00
|
|
|
const DEFAULT_CHECK_PARALLELISM: usize = 8;
|
2024-04-10 20:51:02 -04:00
|
|
|
|
2024-04-12 14:06:56 -04:00
|
|
|
#[must_use]
|
|
|
|
pub enum ExercisesProgress {
|
2024-05-12 20:32:25 -04:00
|
|
|
// All exercises are done.
|
2024-04-12 14:06:56 -04:00
|
|
|
AllDone,
|
2024-05-12 20:32:25 -04:00
|
|
|
// A new exercise is now pending.
|
|
|
|
NewPending,
|
2024-09-05 11:32:59 -04:00
|
|
|
// The current exercise is still pending.
|
|
|
|
CurrentPending,
|
2024-04-12 14:06:56 -04:00
|
|
|
}
|
|
|
|
|
2024-04-14 10:03:49 -04:00
|
|
|
pub enum StateFileStatus {
|
|
|
|
Read,
|
|
|
|
NotRead,
|
|
|
|
}
|
|
|
|
|
2024-10-02 14:45:55 -04:00
|
|
|
#[derive(Clone, Copy, PartialEq)]
|
|
|
|
enum AllExercisesResult {
|
|
|
|
Pending,
|
|
|
|
Success,
|
|
|
|
Failed,
|
|
|
|
Error,
|
2024-08-28 18:17:22 -04:00
|
|
|
}
|
|
|
|
|
2024-04-10 20:51:02 -04:00
|
|
|
pub struct AppState {
|
2024-04-13 19:15:43 -04:00
|
|
|
current_exercise_ind: usize,
|
|
|
|
exercises: Vec<Exercise>,
|
2024-04-29 11:01:47 -04:00
|
|
|
// Caches the number of done exercises to avoid iterating over all exercises every time.
|
2024-04-10 20:51:02 -04:00
|
|
|
n_done: u16,
|
2024-04-13 19:15:43 -04:00
|
|
|
final_message: String,
|
2024-08-28 19:56:45 -04:00
|
|
|
state_file: File,
|
2024-04-29 11:01:47 -04:00
|
|
|
// Preallocated buffer for reading and writing the state file.
|
2024-04-13 23:13:27 -04:00
|
|
|
file_buf: Vec<u8>,
|
2024-04-18 06:41:17 -04:00
|
|
|
official_exercises: bool,
|
2024-08-01 09:23:54 -04:00
|
|
|
cmd_runner: CmdRunner,
|
2024-08-27 19:19:53 -04:00
|
|
|
// Running in VS Code.
|
|
|
|
vs_code: bool,
|
2024-04-10 20:51:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
impl AppState {
|
2024-04-14 10:03:49 -04:00
|
|
|
pub fn new(
|
|
|
|
exercise_infos: Vec<ExerciseInfo>,
|
|
|
|
final_message: String,
|
2024-04-27 11:31:51 -04:00
|
|
|
) -> Result<(Self, StateFileStatus)> {
|
2024-08-01 09:23:54 -04:00
|
|
|
let cmd_runner = CmdRunner::build()?;
|
2024-08-28 19:56:45 -04:00
|
|
|
let mut state_file = OpenOptions::new()
|
|
|
|
.create(true)
|
|
|
|
.read(true)
|
|
|
|
.write(true)
|
|
|
|
.truncate(false)
|
|
|
|
.open(STATE_FILE_NAME)
|
|
|
|
.with_context(|| {
|
|
|
|
format!("Failed to open or create the state file {STATE_FILE_NAME}")
|
|
|
|
})?;
|
|
|
|
|
2024-09-03 20:19:45 -04:00
|
|
|
let dir_canonical_path = term::canonicalize("exercises");
|
2024-08-28 19:56:45 -04:00
|
|
|
let mut exercises = exercise_infos
|
2024-04-13 19:15:43 -04:00
|
|
|
.into_iter()
|
2024-05-13 15:36:20 -04:00
|
|
|
.map(|exercise_info| {
|
|
|
|
// Leaking to be able to borrow in the watch mode `Table`.
|
|
|
|
// Leaking is not a problem because the `AppState` instance lives until
|
|
|
|
// the end of the program.
|
|
|
|
let path = exercise_info.path().leak();
|
|
|
|
let name = exercise_info.name.leak();
|
|
|
|
let dir = exercise_info.dir.map(|dir| &*dir.leak());
|
2024-08-02 10:28:05 -04:00
|
|
|
let hint = exercise_info.hint.leak().trim_ascii();
|
2024-05-13 15:36:20 -04:00
|
|
|
|
2024-09-03 20:19:45 -04:00
|
|
|
let canonical_path = dir_canonical_path.as_deref().map(|dir_canonical_path| {
|
|
|
|
let mut canonical_path;
|
|
|
|
if let Some(dir) = dir {
|
|
|
|
canonical_path = String::with_capacity(
|
|
|
|
2 + dir_canonical_path.len() + dir.len() + name.len(),
|
|
|
|
);
|
|
|
|
canonical_path.push_str(dir_canonical_path);
|
|
|
|
canonical_path.push_str(MAIN_SEPARATOR_STR);
|
|
|
|
canonical_path.push_str(dir);
|
|
|
|
} else {
|
|
|
|
canonical_path =
|
|
|
|
String::with_capacity(1 + dir_canonical_path.len() + name.len());
|
|
|
|
canonical_path.push_str(dir_canonical_path);
|
|
|
|
}
|
|
|
|
|
|
|
|
canonical_path.push_str(MAIN_SEPARATOR_STR);
|
|
|
|
canonical_path.push_str(name);
|
|
|
|
canonical_path.push_str(".rs");
|
|
|
|
canonical_path
|
|
|
|
});
|
|
|
|
|
2024-05-13 15:36:20 -04:00
|
|
|
Exercise {
|
|
|
|
dir,
|
|
|
|
name,
|
|
|
|
path,
|
2024-09-03 20:19:45 -04:00
|
|
|
canonical_path,
|
2024-05-13 15:36:20 -04:00
|
|
|
test: exercise_info.test,
|
|
|
|
strict_clippy: exercise_info.strict_clippy,
|
|
|
|
hint,
|
2024-08-28 19:56:45 -04:00
|
|
|
// Updated below.
|
2024-05-13 15:36:20 -04:00
|
|
|
done: false,
|
|
|
|
}
|
|
|
|
})
|
2024-04-13 19:15:43 -04:00
|
|
|
.collect::<Vec<_>>();
|
|
|
|
|
2024-08-28 19:56:45 -04:00
|
|
|
let mut current_exercise_ind = 0;
|
|
|
|
let mut n_done = 0;
|
|
|
|
let mut file_buf = Vec::with_capacity(2048);
|
|
|
|
let state_file_status = 'block: {
|
|
|
|
if state_file.read_to_end(&mut file_buf).is_err() {
|
|
|
|
break 'block StateFileStatus::NotRead;
|
|
|
|
}
|
|
|
|
|
|
|
|
// See `Self::write` for more information about the file format.
|
|
|
|
let mut lines = file_buf.split(|c| *c == b'\n').skip(2);
|
|
|
|
|
|
|
|
let Some(current_exercise_name) = lines.next() else {
|
|
|
|
break 'block StateFileStatus::NotRead;
|
|
|
|
};
|
|
|
|
|
|
|
|
if current_exercise_name.is_empty() || lines.next().is_none() {
|
|
|
|
break 'block StateFileStatus::NotRead;
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut done_exercises = hash_set_with_capacity(exercises.len());
|
|
|
|
|
|
|
|
for done_exerise_name in lines {
|
|
|
|
if done_exerise_name.is_empty() {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
done_exercises.insert(done_exerise_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (ind, exercise) in exercises.iter_mut().enumerate() {
|
|
|
|
if done_exercises.contains(exercise.name.as_bytes()) {
|
|
|
|
exercise.done = true;
|
|
|
|
n_done += 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if exercise.name.as_bytes() == current_exercise_name {
|
|
|
|
current_exercise_ind = ind;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
StateFileStatus::Read
|
|
|
|
};
|
|
|
|
|
|
|
|
file_buf.clear();
|
|
|
|
file_buf.extend_from_slice(STATE_FILE_HEADER);
|
|
|
|
|
|
|
|
let slf = Self {
|
|
|
|
current_exercise_ind,
|
2024-04-10 20:51:02 -04:00
|
|
|
exercises,
|
2024-08-28 19:56:45 -04:00
|
|
|
n_done,
|
2024-04-14 10:03:49 -04:00
|
|
|
final_message,
|
2024-08-28 19:56:45 -04:00
|
|
|
state_file,
|
|
|
|
file_buf,
|
2024-04-18 06:41:17 -04:00
|
|
|
official_exercises: !Path::new("info.toml").exists(),
|
2024-08-01 09:23:54 -04:00
|
|
|
cmd_runner,
|
2024-08-27 19:19:53 -04:00
|
|
|
vs_code: env::var_os("TERM_PROGRAM").is_some_and(|v| v == "vscode"),
|
2024-04-13 23:13:27 -04:00
|
|
|
};
|
|
|
|
|
2024-04-27 11:31:51 -04:00
|
|
|
Ok((slf, state_file_status))
|
2024-04-10 20:51:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
pub fn current_exercise_ind(&self) -> usize {
|
2024-04-13 19:15:43 -04:00
|
|
|
self.current_exercise_ind
|
2024-04-10 20:51:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2024-04-13 19:15:43 -04:00
|
|
|
pub fn exercises(&self) -> &[Exercise] {
|
|
|
|
&self.exercises
|
2024-04-10 20:51:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
pub fn n_done(&self) -> u16 {
|
|
|
|
self.n_done
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2024-04-13 19:15:43 -04:00
|
|
|
pub fn current_exercise(&self) -> &Exercise {
|
|
|
|
&self.exercises[self.current_exercise_ind]
|
2024-04-10 20:51:02 -04:00
|
|
|
}
|
|
|
|
|
2024-04-26 22:14:59 -04:00
|
|
|
#[inline]
|
2024-08-01 09:23:54 -04:00
|
|
|
pub fn cmd_runner(&self) -> &CmdRunner {
|
|
|
|
&self.cmd_runner
|
2024-04-26 22:14:59 -04:00
|
|
|
}
|
|
|
|
|
2024-08-27 19:19:53 -04:00
|
|
|
#[inline]
|
|
|
|
pub fn vs_code(&self) -> bool {
|
|
|
|
self.vs_code
|
|
|
|
}
|
|
|
|
|
2024-04-29 11:01:47 -04:00
|
|
|
// Write the state file.
|
|
|
|
// The file's format is very simple:
|
|
|
|
// - The first line is a comment.
|
|
|
|
// - The second line is an empty line.
|
|
|
|
// - The third line is the name of the current exercise. It must end with `\n` even if there
|
|
|
|
// are no done exercises.
|
|
|
|
// - The fourth line is an empty line.
|
|
|
|
// - All remaining lines are the names of done exercises.
|
|
|
|
fn write(&mut self) -> Result<()> {
|
2024-08-28 19:56:45 -04:00
|
|
|
self.file_buf.truncate(STATE_FILE_HEADER.len());
|
2024-04-29 11:01:47 -04:00
|
|
|
|
|
|
|
self.file_buf
|
|
|
|
.extend_from_slice(self.current_exercise().name.as_bytes());
|
|
|
|
self.file_buf.push(b'\n');
|
|
|
|
|
|
|
|
for exercise in &self.exercises {
|
|
|
|
if exercise.done {
|
|
|
|
self.file_buf.push(b'\n');
|
|
|
|
self.file_buf.extend_from_slice(exercise.name.as_bytes());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-28 19:56:45 -04:00
|
|
|
self.state_file
|
|
|
|
.rewind()
|
|
|
|
.with_context(|| format!("Failed to rewind the state file {STATE_FILE_NAME}"))?;
|
|
|
|
self.state_file
|
|
|
|
.set_len(0)
|
|
|
|
.with_context(|| format!("Failed to truncate the state file {STATE_FILE_NAME}"))?;
|
|
|
|
self.state_file
|
|
|
|
.write_all(&self.file_buf)
|
2024-04-29 11:01:47 -04:00
|
|
|
.with_context(|| format!("Failed to write the state file {STATE_FILE_NAME}"))?;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2024-05-12 11:40:53 -04:00
|
|
|
pub fn set_current_exercise_ind(&mut self, exercise_ind: usize) -> Result<()> {
|
2024-05-13 11:06:11 -04:00
|
|
|
if exercise_ind == self.current_exercise_ind {
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
|
2024-05-12 11:40:53 -04:00
|
|
|
if exercise_ind >= self.exercises.len() {
|
2024-04-10 20:51:02 -04:00
|
|
|
bail!(BAD_INDEX_ERR);
|
|
|
|
}
|
|
|
|
|
2024-05-12 11:40:53 -04:00
|
|
|
self.current_exercise_ind = exercise_ind;
|
2024-04-10 20:51:02 -04:00
|
|
|
|
2024-04-13 23:13:27 -04:00
|
|
|
self.write()
|
2024-04-10 20:51:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn set_current_exercise_by_name(&mut self, name: &str) -> Result<()> {
|
2024-04-13 19:15:43 -04:00
|
|
|
// O(N) is fine since this method is used only once until the program exits.
|
|
|
|
// Building a hashmap would have more overhead.
|
|
|
|
self.current_exercise_ind = self
|
2024-04-10 20:51:02 -04:00
|
|
|
.exercises
|
|
|
|
.iter()
|
2024-04-13 19:15:43 -04:00
|
|
|
.position(|exercise| exercise.name == name)
|
2024-04-10 20:51:02 -04:00
|
|
|
.with_context(|| format!("No exercise found for '{name}'!"))?;
|
|
|
|
|
2024-04-13 23:13:27 -04:00
|
|
|
self.write()
|
2024-04-10 20:51:02 -04:00
|
|
|
}
|
|
|
|
|
2024-10-02 14:45:55 -04:00
|
|
|
// Set the status of an exercise without saving. Returns `true` if the
|
|
|
|
// status actually changed (and thus needs saving later)
|
|
|
|
pub fn set_status(&mut self, exercise_ind: usize, done: bool) -> Result<bool> {
|
2024-05-12 11:40:53 -04:00
|
|
|
let exercise = self
|
|
|
|
.exercises
|
|
|
|
.get_mut(exercise_ind)
|
|
|
|
.context(BAD_INDEX_ERR)?;
|
2024-04-13 19:15:43 -04:00
|
|
|
|
2024-10-02 14:45:55 -04:00
|
|
|
if exercise.done == done {
|
|
|
|
Ok(false)
|
|
|
|
} else {
|
|
|
|
exercise.done = done;
|
|
|
|
if done {
|
|
|
|
self.n_done += 1;
|
|
|
|
} else {
|
|
|
|
self.n_done -= 1;
|
|
|
|
}
|
|
|
|
Ok(true)
|
2024-04-10 20:51:02 -04:00
|
|
|
}
|
2024-10-02 14:45:55 -04:00
|
|
|
}
|
2024-04-10 20:51:02 -04:00
|
|
|
|
2024-10-02 14:45:55 -04:00
|
|
|
// Set the status of an exercise to "pending" and save
|
|
|
|
pub fn set_pending(&mut self, exercise_ind: usize) -> Result<()> {
|
|
|
|
if self.set_status(exercise_ind, false)? {
|
|
|
|
self.write()?;
|
|
|
|
}
|
2024-04-10 20:51:02 -04:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2024-04-29 11:01:47 -04:00
|
|
|
// Official exercises: Dump the original file from the binary.
|
|
|
|
// Third-party exercises: Reset the exercise file with `git stash`.
|
2024-05-12 11:40:53 -04:00
|
|
|
fn reset(&self, exercise_ind: usize, path: &str) -> Result<()> {
|
2024-04-18 06:41:17 -04:00
|
|
|
if self.official_exercises {
|
|
|
|
return EMBEDDED_FILES
|
2024-05-12 11:40:53 -04:00
|
|
|
.write_exercise_to_disk(exercise_ind, path)
|
2024-04-18 06:41:17 -04:00
|
|
|
.with_context(|| format!("Failed to reset the exercise {path}"));
|
|
|
|
}
|
|
|
|
|
|
|
|
let output = Command::new("git")
|
|
|
|
.arg("stash")
|
|
|
|
.arg("push")
|
|
|
|
.arg("--")
|
|
|
|
.arg(path)
|
|
|
|
.stdin(Stdio::null())
|
|
|
|
.stdout(Stdio::null())
|
|
|
|
.output()
|
|
|
|
.with_context(|| format!("Failed to run `git stash push -- {path}`"))?;
|
|
|
|
|
|
|
|
if !output.status.success() {
|
|
|
|
bail!(
|
|
|
|
"`git stash push -- {path}` didn't run successfully: {}",
|
|
|
|
String::from_utf8_lossy(&output.stderr),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn reset_current_exercise(&mut self) -> Result<&'static str> {
|
|
|
|
self.set_pending(self.current_exercise_ind)?;
|
2024-04-23 13:18:25 -04:00
|
|
|
let exercise = self.current_exercise();
|
2024-05-12 11:40:53 -04:00
|
|
|
self.reset(self.current_exercise_ind, exercise.path)?;
|
2024-04-18 06:41:17 -04:00
|
|
|
|
2024-04-23 13:18:25 -04:00
|
|
|
Ok(exercise.path)
|
2024-04-18 06:41:17 -04:00
|
|
|
}
|
|
|
|
|
2024-08-25 14:29:54 -04:00
|
|
|
// Reset the exercise by index and return its name.
|
2024-04-18 06:41:17 -04:00
|
|
|
pub fn reset_exercise_by_ind(&mut self, exercise_ind: usize) -> Result<&'static str> {
|
|
|
|
if exercise_ind >= self.exercises.len() {
|
|
|
|
bail!(BAD_INDEX_ERR);
|
|
|
|
}
|
|
|
|
|
|
|
|
self.set_pending(exercise_ind)?;
|
2024-04-23 13:18:25 -04:00
|
|
|
let exercise = &self.exercises[exercise_ind];
|
2024-05-12 11:40:53 -04:00
|
|
|
self.reset(exercise_ind, exercise.path)?;
|
2024-04-18 06:41:17 -04:00
|
|
|
|
2024-08-25 14:29:54 -04:00
|
|
|
Ok(exercise.name)
|
2024-04-18 06:41:17 -04:00
|
|
|
}
|
|
|
|
|
2024-04-29 11:01:47 -04:00
|
|
|
// Return the index of the next pending exercise or `None` if all exercises are done.
|
2024-04-12 12:57:04 -04:00
|
|
|
fn next_pending_exercise_ind(&self) -> Option<usize> {
|
2024-08-28 19:59:04 -04:00
|
|
|
let next_ind = self.current_exercise_ind + 1;
|
|
|
|
self.exercises
|
|
|
|
// If the exercise done isn't the last, search for pending exercises after it.
|
|
|
|
.get(next_ind..)
|
|
|
|
.and_then(|later_exercises| {
|
|
|
|
later_exercises
|
|
|
|
.iter()
|
|
|
|
.position(|exercise| !exercise.done)
|
|
|
|
.map(|ind| next_ind + ind)
|
|
|
|
})
|
|
|
|
// Search from the start.
|
|
|
|
.or_else(|| {
|
|
|
|
self.exercises[..self.current_exercise_ind]
|
|
|
|
.iter()
|
|
|
|
.position(|exercise| !exercise.done)
|
|
|
|
})
|
2024-04-10 20:51:02 -04:00
|
|
|
}
|
|
|
|
|
2024-09-01 14:31:09 -04:00
|
|
|
/// Official exercises: Dump the solution file from the binary and return its path.
|
2024-05-13 15:40:40 -04:00
|
|
|
/// Third-party exercises: Check if a solution file exists and return its path in that case.
|
2024-04-23 20:52:30 -04:00
|
|
|
pub fn current_solution_path(&self) -> Result<Option<String>> {
|
2024-08-01 09:23:54 -04:00
|
|
|
if cfg!(debug_assertions) {
|
2024-04-23 20:52:30 -04:00
|
|
|
return Ok(None);
|
2024-04-10 20:51:02 -04:00
|
|
|
}
|
|
|
|
|
2024-04-23 20:52:30 -04:00
|
|
|
let current_exercise = self.current_exercise();
|
|
|
|
|
|
|
|
if self.official_exercises {
|
2024-05-12 11:40:53 -04:00
|
|
|
EMBEDDED_FILES
|
|
|
|
.write_solution_to_disk(self.current_exercise_ind, current_exercise.name)
|
|
|
|
.map(Some)
|
2024-04-23 20:52:30 -04:00
|
|
|
} else {
|
2024-08-27 19:10:19 -04:00
|
|
|
let sol_path = current_exercise.sol_path();
|
|
|
|
|
|
|
|
if Path::new(&sol_path).exists() {
|
|
|
|
return Ok(Some(sol_path));
|
2024-04-23 20:52:30 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
Ok(None)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-28 18:17:22 -04:00
|
|
|
// Return the exercise index of the first pending exercise found.
|
2024-10-02 16:40:32 -04:00
|
|
|
pub fn check_all_exercises(
|
|
|
|
&mut self,
|
|
|
|
stdout: &mut StdoutLock,
|
|
|
|
final_check: bool,
|
|
|
|
) -> Result<Option<usize>> {
|
|
|
|
if !final_check {
|
|
|
|
stdout.write_all(INTERMEDIATE_CHECK_MSG)?;
|
|
|
|
} else {
|
|
|
|
stdout.write_all(FINAL_CHECK_MSG)?;
|
|
|
|
}
|
2024-07-28 11:39:46 -04:00
|
|
|
let n_exercises = self.exercises.len();
|
|
|
|
|
2024-10-02 14:45:55 -04:00
|
|
|
let (mut checked_count, mut results) = thread::scope(|s| {
|
2024-10-02 17:10:26 -04:00
|
|
|
let (tx, rx) = mpsc::channel();
|
2024-10-02 17:42:50 -04:00
|
|
|
let exercise_ind = Arc::new(AtomicUsize::default());
|
|
|
|
|
|
|
|
let num_core = thread::available_parallelism()
|
|
|
|
.map_or(DEFAULT_CHECK_PARALLELISM, |count| count.get());
|
|
|
|
(0..num_core).for_each(|_| {
|
|
|
|
let tx = tx.clone();
|
|
|
|
let exercise_ind = exercise_ind.clone();
|
|
|
|
let this = &self;
|
|
|
|
let _ = thread::Builder::new().spawn_scoped(s, move || {
|
|
|
|
loop {
|
|
|
|
let exercise_ind =
|
|
|
|
exercise_ind.fetch_add(1, std::sync::atomic::Ordering::AcqRel);
|
|
|
|
let Some(exercise) = this.exercises.get(exercise_ind) else {
|
|
|
|
// No more exercises
|
|
|
|
break;
|
|
|
|
};
|
|
|
|
if tx
|
|
|
|
.send((exercise_ind, exercise.run_exercise(None, &this.cmd_runner)))
|
|
|
|
.is_err()
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2024-10-02 17:10:26 -04:00
|
|
|
});
|
2024-10-02 17:42:50 -04:00
|
|
|
});
|
2024-10-02 17:10:26 -04:00
|
|
|
|
|
|
|
// Drop this `tx`, since the `rx` loop will not stop while there is
|
|
|
|
// at least one tx alive (i.e. we want the loop to block only while
|
|
|
|
// there are `tx` clones, i.e. threads)
|
|
|
|
drop(tx);
|
2024-07-28 11:39:46 -04:00
|
|
|
|
2024-10-02 14:45:55 -04:00
|
|
|
let mut results = vec![AllExercisesResult::Pending; n_exercises];
|
|
|
|
let mut checked_count = 0;
|
2024-10-02 17:10:26 -04:00
|
|
|
write!(stdout, "\rProgress: {checked_count}/{n_exercises}")?;
|
|
|
|
stdout.flush()?;
|
|
|
|
while let Ok((exercise_ind, result)) = rx.recv() {
|
|
|
|
results[exercise_ind] = result.map_or_else(
|
|
|
|
|_| AllExercisesResult::Error,
|
|
|
|
|success| {
|
|
|
|
checked_count += 1;
|
|
|
|
if success {
|
|
|
|
AllExercisesResult::Success
|
|
|
|
} else {
|
|
|
|
AllExercisesResult::Failed
|
|
|
|
}
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
2024-10-02 14:45:55 -04:00
|
|
|
write!(stdout, "\rProgress: {checked_count}/{n_exercises}")?;
|
2024-08-25 17:53:50 -04:00
|
|
|
stdout.flush()?;
|
2024-04-14 10:04:05 -04:00
|
|
|
}
|
2024-04-12 12:57:04 -04:00
|
|
|
|
2024-10-02 14:45:55 -04:00
|
|
|
Ok::<_, io::Error>((checked_count, results))
|
2024-07-28 11:39:46 -04:00
|
|
|
})?;
|
|
|
|
|
2024-10-02 14:45:55 -04:00
|
|
|
// If we got an error while checking all exercises in parallel,
|
|
|
|
// it could be because we exceeded the limit of open file descriptors.
|
|
|
|
// Therefore, re-try those one at a time (i.e. sequentially).
|
|
|
|
results
|
|
|
|
.iter_mut()
|
|
|
|
.enumerate()
|
|
|
|
.filter(|(_, result)| {
|
|
|
|
**result == AllExercisesResult::Pending || **result == AllExercisesResult::Error
|
|
|
|
})
|
|
|
|
.try_for_each(|(exercise_ind, result)| {
|
|
|
|
let exercise = self.exercises.get(exercise_ind).context(BAD_INDEX_ERR)?;
|
|
|
|
*result = match exercise
|
|
|
|
.run_exercise(None, &self.cmd_runner)
|
|
|
|
.context("Sequential retry")
|
|
|
|
{
|
|
|
|
Ok(true) => AllExercisesResult::Success,
|
|
|
|
Ok(false) => AllExercisesResult::Failed,
|
|
|
|
Err(err) => bail!(err),
|
|
|
|
};
|
|
|
|
checked_count += 1;
|
|
|
|
write!(stdout, "\rProgress: {checked_count}/{n_exercises}")?;
|
|
|
|
stdout.flush()?;
|
|
|
|
Ok(())
|
|
|
|
})?;
|
2024-08-28 18:17:22 -04:00
|
|
|
|
2024-10-02 14:45:55 -04:00
|
|
|
// Update the state of each exercise and return the first that failed
|
|
|
|
let first_fail = results
|
|
|
|
.iter()
|
|
|
|
.enumerate()
|
|
|
|
.filter_map(|(exercise_ind, result)| {
|
|
|
|
match result {
|
|
|
|
AllExercisesResult::Success => self
|
|
|
|
.set_status(exercise_ind, true)
|
|
|
|
.map_or_else(|err| Some(Err(err)), |_| None),
|
|
|
|
AllExercisesResult::Failed => self
|
|
|
|
.set_status(exercise_ind, false)
|
|
|
|
.map_or_else(|err| Some(Err(err)), |_| Some(Ok(exercise_ind))),
|
|
|
|
// The sequential check done earlier will have converted all
|
|
|
|
// exercises to Success/Failed, or bailed, so those are unreachable
|
|
|
|
AllExercisesResult::Pending | AllExercisesResult::Error => unreachable!(),
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.try_fold(None::<usize>, |current_min, index| {
|
|
|
|
match (current_min, index) {
|
|
|
|
(_, Err(err)) => Err(err),
|
|
|
|
(None, Ok(index)) => Ok(Some(index)),
|
|
|
|
(Some(current_min), Ok(index)) => Ok(Some(current_min.min(index))),
|
|
|
|
}
|
|
|
|
})?;
|
|
|
|
self.write()?;
|
2024-08-28 18:17:22 -04:00
|
|
|
|
2024-10-02 14:45:55 -04:00
|
|
|
Ok(first_fail)
|
2024-08-28 18:17:22 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Mark the current exercise as done and move on to the next pending exercise if one exists.
|
|
|
|
/// If all exercises are marked as done, run all of them to make sure that they are actually
|
|
|
|
/// done. If an exercise which is marked as done fails, mark it as pending and continue on it.
|
2024-09-24 10:12:44 -04:00
|
|
|
pub fn done_current_exercise<const CLEAR_BEFORE_FINAL_CHECK: bool>(
|
|
|
|
&mut self,
|
|
|
|
stdout: &mut StdoutLock,
|
|
|
|
) -> Result<ExercisesProgress> {
|
2024-08-28 18:17:22 -04:00
|
|
|
let exercise = &mut self.exercises[self.current_exercise_ind];
|
|
|
|
if !exercise.done {
|
|
|
|
exercise.done = true;
|
|
|
|
self.n_done += 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if let Some(ind) = self.next_pending_exercise_ind() {
|
|
|
|
self.set_current_exercise_ind(ind)?;
|
|
|
|
return Ok(ExercisesProgress::NewPending);
|
|
|
|
}
|
|
|
|
|
2024-09-24 10:12:44 -04:00
|
|
|
if CLEAR_BEFORE_FINAL_CHECK {
|
|
|
|
clear_terminal(stdout)?;
|
|
|
|
} else {
|
|
|
|
stdout.write_all(b"\n")?;
|
|
|
|
}
|
|
|
|
|
2024-10-02 16:40:32 -04:00
|
|
|
if let Some(pending_exercise_ind) = self.check_all_exercises(stdout, true)? {
|
2024-08-28 18:17:22 -04:00
|
|
|
stdout.write_all(b"\n\n")?;
|
|
|
|
|
2024-07-28 11:39:46 -04:00
|
|
|
self.current_exercise_ind = pending_exercise_ind;
|
2024-08-28 18:17:22 -04:00
|
|
|
self.exercises[pending_exercise_ind].done = false;
|
2024-10-02 14:45:55 -04:00
|
|
|
|
2024-07-28 11:39:46 -04:00
|
|
|
return Ok(ExercisesProgress::NewPending);
|
2024-04-29 11:01:47 -04:00
|
|
|
}
|
2024-04-13 23:13:27 -04:00
|
|
|
|
2024-07-07 07:55:39 -04:00
|
|
|
// Write that the last exercise is done.
|
|
|
|
self.write()?;
|
|
|
|
|
2024-10-02 16:40:32 -04:00
|
|
|
self.render_final_message(stdout)?;
|
|
|
|
|
|
|
|
Ok(ExercisesProgress::AllDone)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn render_final_message(&self, stdout: &mut StdoutLock) -> Result<()> {
|
2024-08-25 17:53:50 -04:00
|
|
|
clear_terminal(stdout)?;
|
|
|
|
stdout.write_all(FENISH_LINE.as_bytes())?;
|
2024-04-13 23:13:27 -04:00
|
|
|
|
2024-08-02 10:28:05 -04:00
|
|
|
let final_message = self.final_message.trim_ascii();
|
2024-04-29 11:01:47 -04:00
|
|
|
if !final_message.is_empty() {
|
2024-08-25 17:53:50 -04:00
|
|
|
stdout.write_all(final_message.as_bytes())?;
|
|
|
|
stdout.write_all(b"\n")?;
|
2024-04-13 23:13:27 -04:00
|
|
|
}
|
|
|
|
|
2024-10-02 16:40:32 -04:00
|
|
|
Ok(())
|
2024-04-13 23:13:27 -04:00
|
|
|
}
|
2024-04-10 20:51:02 -04:00
|
|
|
}
|
2024-04-12 12:57:04 -04:00
|
|
|
|
2024-08-28 19:56:45 -04:00
|
|
|
const BAD_INDEX_ERR: &str = "The current exercise index is higher than the number of exercises";
|
|
|
|
const STATE_FILE_HEADER: &[u8] = b"DON'T EDIT THIS FILE!\n\n";
|
2024-10-02 16:40:32 -04:00
|
|
|
const INTERMEDIATE_CHECK_MSG: &[u8] = b"Checking all exercises
|
|
|
|
";
|
2024-09-24 10:12:44 -04:00
|
|
|
const FINAL_CHECK_MSG: &[u8] = b"All exercises seem to be done.
|
2024-04-12 12:57:04 -04:00
|
|
|
Recompiling and running all exercises to make sure that all of them are actually done.
|
|
|
|
";
|
2024-04-18 05:21:39 -04:00
|
|
|
const FENISH_LINE: &str = "+----------------------------------------------------+
|
|
|
|
| You made it to the Fe-nish line! |
|
|
|
|
+-------------------------- ------------------------+
|
|
|
|
\\/\x1b[31m
|
|
|
|
▒▒ ▒▒▒▒▒▒▒▒ ▒▒▒▒▒▒▒▒ ▒▒
|
|
|
|
▒▒▒▒ ▒▒ ▒▒ ▒▒ ▒▒ ▒▒ ▒▒ ▒▒▒▒
|
|
|
|
▒▒▒▒ ▒▒ ▒▒ ▒▒ ▒▒ ▒▒ ▒▒▒▒
|
|
|
|
░░▒▒▒▒░░▒▒ ▒▒ ▒▒ ▒▒ ▒▒░░▒▒▒▒
|
|
|
|
▓▓▓▓▓▓▓▓ ▓▓ ▓▓██ ▓▓ ▓▓██ ▓▓ ▓▓▓▓▓▓▓▓
|
|
|
|
▒▒▒▒ ▒▒ ████ ▒▒ ████ ▒▒░░ ▒▒▒▒
|
|
|
|
▒▒ ▒▒▒▒▒▒ ▒▒▒▒▒▒ ▒▒▒▒▒▒ ▒▒
|
|
|
|
▒▒▒▒▒▒▒▒▒▒▓▓▓▓▓▓▒▒▒▒▒▒▒▒▓▓▓▓▓▓▒▒▒▒▒▒▒▒
|
|
|
|
▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒
|
|
|
|
▒▒▒▒▒▒▒▒▒▒██▒▒▒▒▒▒██▒▒▒▒▒▒▒▒▒▒
|
|
|
|
▒▒ ▒▒▒▒▒▒▒▒▒▒██████▒▒▒▒▒▒▒▒▒▒ ▒▒
|
|
|
|
▒▒ ▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒ ▒▒
|
|
|
|
▒▒ ▒▒ ▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒▒ ▒▒ ▒▒
|
|
|
|
▒▒ ▒▒ ▒▒ ▒▒ ▒▒ ▒▒
|
|
|
|
▒▒ ▒▒ ▒▒ ▒▒\x1b[0m
|
|
|
|
|
|
|
|
";
|
2024-04-29 20:14:20 -04:00
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
fn dummy_exercise() -> Exercise {
|
|
|
|
Exercise {
|
|
|
|
dir: None,
|
|
|
|
name: "0",
|
|
|
|
path: "exercises/0.rs",
|
2024-09-03 20:19:45 -04:00
|
|
|
canonical_path: None,
|
2024-04-29 20:14:20 -04:00
|
|
|
test: false,
|
|
|
|
strict_clippy: false,
|
2024-08-02 10:28:05 -04:00
|
|
|
hint: "",
|
2024-04-29 20:14:20 -04:00
|
|
|
done: false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn next_pending_exercise() {
|
|
|
|
let mut app_state = AppState {
|
|
|
|
current_exercise_ind: 0,
|
|
|
|
exercises: vec![dummy_exercise(), dummy_exercise(), dummy_exercise()],
|
|
|
|
n_done: 0,
|
|
|
|
final_message: String::new(),
|
2024-08-28 19:56:45 -04:00
|
|
|
state_file: tempfile::tempfile().unwrap(),
|
2024-04-29 20:14:20 -04:00
|
|
|
file_buf: Vec::new(),
|
|
|
|
official_exercises: true,
|
2024-08-01 09:23:54 -04:00
|
|
|
cmd_runner: CmdRunner::build().unwrap(),
|
2024-08-27 19:19:53 -04:00
|
|
|
vs_code: false,
|
2024-04-29 20:14:20 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
let mut assert = |done: [bool; 3], expected: [Option<usize>; 3]| {
|
|
|
|
for (exercise, done) in app_state.exercises.iter_mut().zip(done) {
|
|
|
|
exercise.done = done;
|
|
|
|
}
|
|
|
|
for (ind, expected) in expected.into_iter().enumerate() {
|
|
|
|
app_state.current_exercise_ind = ind;
|
|
|
|
assert_eq!(
|
|
|
|
app_state.next_pending_exercise_ind(),
|
|
|
|
expected,
|
|
|
|
"done={done:?}, ind={ind}",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
assert([true, true, true], [None, None, None]);
|
|
|
|
assert([false, false, false], [Some(1), Some(2), Some(0)]);
|
|
|
|
assert([false, true, true], [None, Some(0), Some(0)]);
|
|
|
|
assert([true, false, true], [Some(1), None, Some(1)]);
|
|
|
|
assert([true, true, false], [Some(2), Some(2), None]);
|
|
|
|
assert([true, false, false], [Some(1), Some(2), Some(1)]);
|
|
|
|
assert([false, true, false], [Some(2), Some(2), Some(0)]);
|
|
|
|
assert([false, false, true], [Some(1), Some(0), Some(0)]);
|
|
|
|
}
|
|
|
|
}
|