Compare commits

..

No commits in common. "2b7caf6fcb7fc128bc5c0be93913d3482c87b35b" and "9faa5d3aa48f7a94ed87e61ad6ea659579f1311a" have entirely different histories.

5 changed files with 15 additions and 38 deletions

View file

@ -10,7 +10,4 @@ disallowed-methods = [
"std::collections::HashSet::with_capacity", "std::collections::HashSet::with_capacity",
# Inefficient. Use `.queue(…)` instead. # Inefficient. Use `.queue(…)` instead.
"crossterm::style::style", "crossterm::style::style",
# Use `thread::Builder::spawn` instead and handle the error.
"std::thread::spawn",
"std::thread::Scope::spawn",
] ]

View file

@ -217,5 +217,3 @@ empty_loop = "forbid"
infinite_loop = "deny" infinite_loop = "deny"
# You shouldn't leak memory while still learning Rust # You shouldn't leak memory while still learning Rust
mem_forget = "deny" mem_forget = "deny"
# Currently, there are no disallowed methods. This line avoids problems when developing Rustlings.
disallowed_methods = "allow"

View file

@ -388,20 +388,13 @@ impl AppState {
let handles = self let handles = self
.exercises .exercises
.iter() .iter()
.map(|exercise| { .map(|exercise| s.spawn(|| exercise.run_exercise(None, &self.cmd_runner)))
thread::Builder::new()
.spawn_scoped(s, || exercise.run_exercise(None, &self.cmd_runner))
})
.collect::<Vec<_>>(); .collect::<Vec<_>>();
for (exercise_ind, spawn_res) in handles.into_iter().enumerate() { for (exercise_ind, handle) in handles.into_iter().enumerate() {
write!(stdout, "\rProgress: {exercise_ind}/{n_exercises}")?; write!(stdout, "\rProgress: {exercise_ind}/{n_exercises}")?;
stdout.flush()?; stdout.flush()?;
let Ok(handle) = spawn_res else {
return Ok(AllExercisesCheck::CheckedUntil(exercise_ind));
};
let Ok(success) = handle.join().unwrap() else { let Ok(success) = handle.join().unwrap() else {
return Ok(AllExercisesCheck::CheckedUntil(exercise_ind)); return Ok(AllExercisesCheck::CheckedUntil(exercise_ind));
}; };

View file

@ -41,10 +41,10 @@ fn check_cargo_toml(
if old_bins != new_bins { if old_bins != new_bins {
if cfg!(debug_assertions) { if cfg!(debug_assertions) {
bail!("The file `dev/Cargo.toml` is outdated. Run `cargo run -- dev update` to update it. Then run `cargo run -- dev check` again"); bail!("The file `dev/Cargo.toml` is outdated. Please run `cargo run -- dev update` to update it. Then run `cargo run -- dev check` again");
} }
bail!("The file `Cargo.toml` is outdated. Run `rustlings dev update` to update it. Then run `rustlings dev check` again"); bail!("The file `Cargo.toml` is outdated. Please run `rustlings dev update` to update it. Then run `rustlings dev check` again");
} }
Ok(()) Ok(())
@ -185,14 +185,12 @@ fn check_exercises_unsolved(
return None; return None;
} }
Some( Some((
thread::Builder::new() exercise_info.name.as_str(),
.spawn(|| exercise_info.run_exercise(None, cmd_runner)) thread::spawn(|| exercise_info.run_exercise(None, cmd_runner)),
.map(|handle| (exercise_info.name.as_str(), handle)), ))
)
}) })
.collect::<Result<Vec<_>, _>>() .collect::<Vec<_>>();
.context("Failed to spawn a thread to check if an exercise is already solved")?;
let n_handles = handles.len(); let n_handles = handles.len();
write!(stdout, "Progress: 0/{n_handles}")?; write!(stdout, "Progress: 0/{n_handles}")?;
@ -228,9 +226,7 @@ fn check_exercises(info_file: &'static InfoFile, cmd_runner: &'static CmdRunner)
Ordering::Equal => (), Ordering::Equal => (),
} }
let handle = thread::Builder::new() let handle = thread::spawn(move || check_exercises_unsolved(info_file, cmd_runner));
.spawn(move || check_exercises_unsolved(info_file, cmd_runner))
.context("Failed to spawn a thread to check if any exercise is already solved")?;
let info_file_paths = check_info_file_exercises(info_file)?; let info_file_paths = check_info_file_exercises(info_file)?;
check_unexpected_files("exercises", &info_file_paths)?; check_unexpected_files("exercises", &info_file_paths)?;
@ -257,7 +253,7 @@ fn check_solutions(
.exercises .exercises
.iter() .iter()
.map(|exercise_info| { .map(|exercise_info| {
thread::Builder::new().spawn(move || { thread::spawn(move || {
let sol_path = exercise_info.sol_path(); let sol_path = exercise_info.sol_path();
if !Path::new(&sol_path).exists() { if !Path::new(&sol_path).exists() {
if require_solutions { if require_solutions {
@ -278,8 +274,7 @@ fn check_solutions(
} }
}) })
}) })
.collect::<Result<Vec<_>, _>>() .collect::<Vec<_>>();
.context("Failed to spawn a thread to check a solution")?;
let mut sol_paths = hash_set_with_capacity(info_file.exercises.len()); let mut sol_paths = hash_set_with_capacity(info_file.exercises.len());
let mut fmt_cmd = Command::new("rustfmt"); let mut fmt_cmd = Command::new("rustfmt");
@ -327,11 +322,7 @@ fn check_solutions(
} }
stdout.write_all(b"\n")?; stdout.write_all(b"\n")?;
let handle = thread::Builder::new() let handle = thread::spawn(move || check_unexpected_files("solutions", &sol_paths));
.spawn(move || check_unexpected_files("solutions", &sol_paths))
.context(
"Failed to spawn a thread to check for unexpected files in the solutions directory",
)?;
if !fmt_cmd if !fmt_cmd
.status() .status()

View file

@ -1,4 +1,4 @@
use anyhow::{Context, Error, Result}; use anyhow::{Error, Result};
use notify_debouncer_mini::{ use notify_debouncer_mini::{
new_debouncer, new_debouncer,
notify::{self, RecursiveMode}, notify::{self, RecursiveMode},
@ -77,9 +77,7 @@ fn run_watch(
let mut stdout = io::stdout().lock(); let mut stdout = io::stdout().lock();
watch_state.run_current_exercise(&mut stdout)?; watch_state.run_current_exercise(&mut stdout)?;
thread::Builder::new() thread::spawn(move || terminal_event_handler(tx, manual_run));
.spawn(move || terminal_event_handler(tx, manual_run))
.context("Failed to spawn a thread to handle terminal events")?;
while let Ok(event) = rx.recv() { while let Ok(event) = rx.recv() {
match event { match event {