mirror of
https://github.com/rust-lang/rustlings.git
synced 2025-01-09 20:03:24 +03:00
Compare commits
20 commits
d988054ad8
...
d0fcd8ae8a
Author | SHA1 | Date | |
---|---|---|---|
d0fcd8ae8a | |||
7c46e7ac69 | |||
1db5de9653 | |||
b5fc06bd56 | |||
7c4d33654f | |||
05729b27a0 | |||
0bf3f7e01f | |||
bd5503a0d3 | |||
25e855a009 | |||
c2501ae733 | |||
3a4f2bebb4 | |||
394ca402a8 | |||
db25cc9157 | |||
93f8d1610d | |||
99c9ab467b | |||
db43efe3ec | |||
9a4ee47c52 | |||
0a674a158d | |||
3bd26c7a24 | |||
8c31d38fa1 |
28
.gitignore
vendored
28
.gitignore
vendored
|
@ -1,19 +1,27 @@
|
||||||
|
# Cargo
|
||||||
target/
|
target/
|
||||||
/tests/fixture/*/Cargo.lock
|
/tests/fixture/*/Cargo.lock
|
||||||
/dev/Cargo.lock
|
/dev/Cargo.lock
|
||||||
|
|
||||||
*.swp
|
# State file
|
||||||
**/*.rs.bk
|
.rustlings-state.json
|
||||||
|
|
||||||
|
# oranda
|
||||||
|
public/
|
||||||
|
.netlify
|
||||||
|
|
||||||
|
# OS
|
||||||
.DS_Store
|
.DS_Store
|
||||||
*.pdb
|
.direnv/
|
||||||
|
|
||||||
|
# Editor
|
||||||
|
*.swp
|
||||||
.idea
|
.idea
|
||||||
|
*.iml
|
||||||
|
|
||||||
|
# VS Code extension recommendations
|
||||||
.vscode/*
|
.vscode/*
|
||||||
!.vscode/extensions.json
|
!.vscode/extensions.json
|
||||||
*.iml
|
|
||||||
*.o
|
|
||||||
public/
|
|
||||||
.direnv/
|
|
||||||
.ignore
|
|
||||||
|
|
||||||
# Local Netlify folder
|
# Ignore file for editors like Helix
|
||||||
.netlify
|
.ignore
|
||||||
|
|
|
@ -75,7 +75,6 @@ pub fn include_files(_: TokenStream) -> TokenStream {
|
||||||
|
|
||||||
quote! {
|
quote! {
|
||||||
EmbeddedFiles {
|
EmbeddedFiles {
|
||||||
info_toml_content: ::std::include_str!("../info.toml"),
|
|
||||||
exercises_dir: ExercisesDir {
|
exercises_dir: ExercisesDir {
|
||||||
readme: EmbeddedFile {
|
readme: EmbeddedFile {
|
||||||
path: "exercises/README.md",
|
path: "exercises/README.md",
|
||||||
|
|
|
@ -65,7 +65,6 @@ struct ExercisesDir {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct EmbeddedFiles {
|
pub struct EmbeddedFiles {
|
||||||
pub info_toml_content: &'static str,
|
|
||||||
exercises_dir: ExercisesDir,
|
exercises_dir: ExercisesDir,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,16 +1,21 @@
|
||||||
use anyhow::{Context, Result};
|
use anyhow::{Context, Result};
|
||||||
use serde::Deserialize;
|
use serde::Deserialize;
|
||||||
use std::fmt::{self, Debug, Display, Formatter};
|
use std::{
|
||||||
use std::fs::{self, File};
|
array,
|
||||||
use std::io::{self, BufRead, BufReader};
|
fmt::{self, Debug, Display, Formatter},
|
||||||
use std::path::PathBuf;
|
fs::{self, File},
|
||||||
use std::process::{Command, Output};
|
io::{self, BufRead, BufReader},
|
||||||
use std::{array, mem};
|
mem,
|
||||||
use winnow::ascii::{space0, Caseless};
|
path::PathBuf,
|
||||||
use winnow::combinator::opt;
|
process::{Command, Output},
|
||||||
use winnow::Parser;
|
};
|
||||||
|
use winnow::{
|
||||||
|
ascii::{space0, Caseless},
|
||||||
|
combinator::opt,
|
||||||
|
Parser,
|
||||||
|
};
|
||||||
|
|
||||||
use crate::embedded::EMBEDDED_FILES;
|
use crate::embedded::{WriteStrategy, EMBEDDED_FILES};
|
||||||
|
|
||||||
// The number of context lines above and below a highlighted line.
|
// The number of context lines above and below a highlighted line.
|
||||||
const CONTEXT: usize = 2;
|
const CONTEXT: usize = 2;
|
||||||
|
@ -41,18 +46,18 @@ pub enum Mode {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Deserialize)]
|
#[derive(Deserialize)]
|
||||||
pub struct ExerciseList {
|
pub struct InfoFile {
|
||||||
pub exercises: Vec<Exercise>,
|
pub exercises: Vec<Exercise>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ExerciseList {
|
impl InfoFile {
|
||||||
pub fn parse() -> Result<Self> {
|
pub fn parse() -> Result<Self> {
|
||||||
// Read a local `info.toml` if it exists.
|
// Read a local `info.toml` if it exists.
|
||||||
// Mainly to let the tests work for now.
|
// Mainly to let the tests work for now.
|
||||||
if let Ok(file_content) = fs::read_to_string("info.toml") {
|
if let Ok(file_content) = fs::read_to_string("info.toml") {
|
||||||
toml_edit::de::from_str(&file_content)
|
toml_edit::de::from_str(&file_content)
|
||||||
} else {
|
} else {
|
||||||
toml_edit::de::from_str(EMBEDDED_FILES.info_toml_content)
|
toml_edit::de::from_str(include_str!("../info.toml"))
|
||||||
}
|
}
|
||||||
.context("Failed to parse `info.toml`")
|
.context("Failed to parse `info.toml`")
|
||||||
}
|
}
|
||||||
|
@ -220,6 +225,12 @@ impl Exercise {
|
||||||
pub fn looks_done(&self) -> Result<bool> {
|
pub fn looks_done(&self) -> Result<bool> {
|
||||||
self.state().map(|state| state == State::Done)
|
self.state().map(|state| state == State::Done)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn reset(&self) -> Result<()> {
|
||||||
|
EMBEDDED_FILES
|
||||||
|
.write_exercise_to_disk(&self.path, WriteStrategy::Overwrite)
|
||||||
|
.with_context(|| format!("Failed to reset the exercise {self}"))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Display for Exercise {
|
impl Display for Exercise {
|
||||||
|
|
|
@ -36,7 +36,8 @@ publish = false
|
||||||
}
|
}
|
||||||
|
|
||||||
fn create_gitignore() -> io::Result<()> {
|
fn create_gitignore() -> io::Result<()> {
|
||||||
let gitignore = b"/target";
|
let gitignore = b"/target
|
||||||
|
/.rustlings-state.json";
|
||||||
OpenOptions::new()
|
OpenOptions::new()
|
||||||
.create_new(true)
|
.create_new(true)
|
||||||
.write(true)
|
.write(true)
|
||||||
|
@ -56,7 +57,7 @@ fn create_vscode_dir() -> Result<()> {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn init_rustlings(exercises: &[Exercise]) -> Result<()> {
|
pub fn init(exercises: &[Exercise]) -> Result<()> {
|
||||||
if Path::new("exercises").is_dir() && Path::new("Cargo.toml").is_file() {
|
if Path::new("exercises").is_dir() && Path::new("Cargo.toml").is_file() {
|
||||||
bail!(
|
bail!(
|
||||||
"A directory with the name `exercises` and a file with the name `Cargo.toml` already exist
|
"A directory with the name `exercises` and a file with the name `Cargo.toml` already exist
|
||||||
|
|
190
src/list.rs
190
src/list.rs
|
@ -4,151 +4,16 @@ use crossterm::{
|
||||||
terminal::{disable_raw_mode, enable_raw_mode, EnterAlternateScreen, LeaveAlternateScreen},
|
terminal::{disable_raw_mode, enable_raw_mode, EnterAlternateScreen, LeaveAlternateScreen},
|
||||||
ExecutableCommand,
|
ExecutableCommand,
|
||||||
};
|
};
|
||||||
use ratatui::{
|
use ratatui::{backend::CrosstermBackend, Terminal};
|
||||||
backend::CrosstermBackend,
|
use std::{fmt::Write, io};
|
||||||
layout::{Constraint, Rect},
|
|
||||||
style::{Style, Stylize},
|
|
||||||
text::Span,
|
|
||||||
widgets::{Block, Borders, HighlightSpacing, Row, Table, TableState},
|
|
||||||
Frame, Terminal,
|
|
||||||
};
|
|
||||||
use std::io;
|
|
||||||
|
|
||||||
use crate::{exercise::Exercise, state::State};
|
mod state;
|
||||||
|
|
||||||
struct UiState<'a> {
|
use crate::{exercise::Exercise, state_file::StateFile};
|
||||||
pub table: Table<'a>,
|
|
||||||
selected: usize,
|
|
||||||
table_state: TableState,
|
|
||||||
last_ind: usize,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a> UiState<'a> {
|
use self::state::{Filter, UiState};
|
||||||
pub fn rows<'s, 'i>(
|
|
||||||
state: &'s State,
|
|
||||||
exercises: &'a [Exercise],
|
|
||||||
) -> impl Iterator<Item = Row<'a>> + 'i
|
|
||||||
where
|
|
||||||
's: 'i,
|
|
||||||
'a: 'i,
|
|
||||||
{
|
|
||||||
exercises
|
|
||||||
.iter()
|
|
||||||
.zip(state.progress())
|
|
||||||
.enumerate()
|
|
||||||
.map(|(ind, (exercise, done))| {
|
|
||||||
let next = if ind == state.next_exercise_ind() {
|
|
||||||
">>>>".bold().red()
|
|
||||||
} else {
|
|
||||||
Span::default()
|
|
||||||
};
|
|
||||||
|
|
||||||
let exercise_state = if *done {
|
pub fn list(state_file: &mut StateFile, exercises: &[Exercise]) -> Result<()> {
|
||||||
"DONE".green()
|
|
||||||
} else {
|
|
||||||
"PENDING".yellow()
|
|
||||||
};
|
|
||||||
|
|
||||||
Row::new([
|
|
||||||
next,
|
|
||||||
exercise_state,
|
|
||||||
Span::raw(&exercise.name),
|
|
||||||
Span::raw(exercise.path.to_string_lossy()),
|
|
||||||
])
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn new(state: &State, exercises: &'a [Exercise]) -> Self {
|
|
||||||
let header = Row::new(["Next", "State", "Name", "Path"]);
|
|
||||||
|
|
||||||
let max_name_len = exercises
|
|
||||||
.iter()
|
|
||||||
.map(|exercise| exercise.name.len())
|
|
||||||
.max()
|
|
||||||
.unwrap_or(4) as u16;
|
|
||||||
|
|
||||||
let widths = [
|
|
||||||
Constraint::Length(4),
|
|
||||||
Constraint::Length(7),
|
|
||||||
Constraint::Length(max_name_len),
|
|
||||||
Constraint::Fill(1),
|
|
||||||
];
|
|
||||||
|
|
||||||
let rows = Self::rows(state, exercises);
|
|
||||||
|
|
||||||
let table = Table::new(rows, widths)
|
|
||||||
.header(header)
|
|
||||||
.column_spacing(2)
|
|
||||||
.highlight_spacing(HighlightSpacing::Always)
|
|
||||||
.highlight_style(Style::new().bg(ratatui::style::Color::Rgb(50, 50, 50)))
|
|
||||||
.highlight_symbol("🦀")
|
|
||||||
.block(Block::default().borders(Borders::BOTTOM));
|
|
||||||
|
|
||||||
let selected = 0;
|
|
||||||
let table_state = TableState::default().with_selected(Some(selected));
|
|
||||||
let last_ind = exercises.len() - 1;
|
|
||||||
|
|
||||||
Self {
|
|
||||||
table,
|
|
||||||
selected,
|
|
||||||
table_state,
|
|
||||||
last_ind,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn select(&mut self, ind: usize) {
|
|
||||||
self.selected = ind;
|
|
||||||
self.table_state.select(Some(ind));
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn select_next(&mut self) {
|
|
||||||
self.select(self.selected.saturating_add(1).min(self.last_ind));
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn select_previous(&mut self) {
|
|
||||||
self.select(self.selected.saturating_sub(1));
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
pub fn select_first(&mut self) {
|
|
||||||
self.select(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
pub fn select_last(&mut self) {
|
|
||||||
self.select(self.last_ind);
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn draw(&mut self, frame: &mut Frame) {
|
|
||||||
let area = frame.size();
|
|
||||||
|
|
||||||
frame.render_stateful_widget(
|
|
||||||
&self.table,
|
|
||||||
Rect {
|
|
||||||
x: 0,
|
|
||||||
y: 0,
|
|
||||||
width: area.width,
|
|
||||||
height: area.height - 1,
|
|
||||||
},
|
|
||||||
&mut self.table_state,
|
|
||||||
);
|
|
||||||
|
|
||||||
// Help footer
|
|
||||||
let footer =
|
|
||||||
"↓/j ↑/k home/g end/G │ Filter <d>one/<p>ending │ <r>eset │ <c>ontinue at │ <q>uit";
|
|
||||||
frame.render_widget(
|
|
||||||
Span::raw(footer),
|
|
||||||
Rect {
|
|
||||||
x: 0,
|
|
||||||
y: area.height - 1,
|
|
||||||
width: area.width,
|
|
||||||
height: 1,
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn list(state: &mut State, exercises: &[Exercise]) -> Result<()> {
|
|
||||||
let mut stdout = io::stdout().lock();
|
let mut stdout = io::stdout().lock();
|
||||||
stdout.execute(EnterAlternateScreen)?;
|
stdout.execute(EnterAlternateScreen)?;
|
||||||
enable_raw_mode()?;
|
enable_raw_mode()?;
|
||||||
|
@ -156,7 +21,7 @@ pub fn list(state: &mut State, exercises: &[Exercise]) -> Result<()> {
|
||||||
let mut terminal = Terminal::new(CrosstermBackend::new(&mut stdout))?;
|
let mut terminal = Terminal::new(CrosstermBackend::new(&mut stdout))?;
|
||||||
terminal.clear()?;
|
terminal.clear()?;
|
||||||
|
|
||||||
let mut ui_state = UiState::new(state, exercises);
|
let mut ui_state = UiState::new(state_file, exercises);
|
||||||
|
|
||||||
'outer: loop {
|
'outer: loop {
|
||||||
terminal.draw(|frame| ui_state.draw(frame))?;
|
terminal.draw(|frame| ui_state.draw(frame))?;
|
||||||
|
@ -177,15 +42,52 @@ pub fn list(state: &mut State, exercises: &[Exercise]) -> Result<()> {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
ui_state.message.clear();
|
||||||
|
|
||||||
match key.code {
|
match key.code {
|
||||||
KeyCode::Char('q') => break,
|
KeyCode::Char('q') => break,
|
||||||
KeyCode::Down | KeyCode::Char('j') => ui_state.select_next(),
|
KeyCode::Down | KeyCode::Char('j') => ui_state.select_next(),
|
||||||
KeyCode::Up | KeyCode::Char('k') => ui_state.select_previous(),
|
KeyCode::Up | KeyCode::Char('k') => ui_state.select_previous(),
|
||||||
KeyCode::Home | KeyCode::Char('g') => ui_state.select_first(),
|
KeyCode::Home | KeyCode::Char('g') => ui_state.select_first(),
|
||||||
KeyCode::End | KeyCode::Char('G') => ui_state.select_last(),
|
KeyCode::End | KeyCode::Char('G') => ui_state.select_last(),
|
||||||
|
KeyCode::Char('d') => {
|
||||||
|
let message = if ui_state.filter == Filter::Done {
|
||||||
|
ui_state.filter = Filter::None;
|
||||||
|
"Disabled filter DONE"
|
||||||
|
} else {
|
||||||
|
ui_state.filter = Filter::Done;
|
||||||
|
"Enabled filter DONE │ Press d again to disable the filter"
|
||||||
|
};
|
||||||
|
|
||||||
|
ui_state = ui_state.with_updated_rows(state_file);
|
||||||
|
ui_state.message.push_str(message);
|
||||||
|
}
|
||||||
|
KeyCode::Char('p') => {
|
||||||
|
let message = if ui_state.filter == Filter::Pending {
|
||||||
|
ui_state.filter = Filter::None;
|
||||||
|
"Disabled filter PENDING"
|
||||||
|
} else {
|
||||||
|
ui_state.filter = Filter::Pending;
|
||||||
|
"Enabled filter PENDING │ Press p again to disable the filter"
|
||||||
|
};
|
||||||
|
|
||||||
|
ui_state = ui_state.with_updated_rows(state_file);
|
||||||
|
ui_state.message.push_str(message);
|
||||||
|
}
|
||||||
|
KeyCode::Char('r') => {
|
||||||
|
let selected = ui_state.selected();
|
||||||
|
let exercise = &exercises[selected];
|
||||||
|
exercise.reset()?;
|
||||||
|
state_file.reset(selected)?;
|
||||||
|
|
||||||
|
ui_state = ui_state.with_updated_rows(state_file);
|
||||||
|
ui_state
|
||||||
|
.message
|
||||||
|
.write_fmt(format_args!("The exercise {exercise} has been reset!"))?;
|
||||||
|
}
|
||||||
KeyCode::Char('c') => {
|
KeyCode::Char('c') => {
|
||||||
state.set_next_exercise_ind(ui_state.selected)?;
|
state_file.set_next_exercise_ind(ui_state.selected())?;
|
||||||
ui_state.table = ui_state.table.rows(UiState::rows(state, exercises));
|
ui_state = ui_state.with_updated_rows(state_file);
|
||||||
}
|
}
|
||||||
_ => (),
|
_ => (),
|
||||||
}
|
}
|
||||||
|
|
175
src/list/state.rs
Normal file
175
src/list/state.rs
Normal file
|
@ -0,0 +1,175 @@
|
||||||
|
use ratatui::{
|
||||||
|
layout::{Constraint, Rect},
|
||||||
|
style::{Style, Stylize},
|
||||||
|
text::Span,
|
||||||
|
widgets::{Block, Borders, HighlightSpacing, Row, Table, TableState},
|
||||||
|
Frame,
|
||||||
|
};
|
||||||
|
|
||||||
|
use crate::{exercise::Exercise, state_file::StateFile};
|
||||||
|
|
||||||
|
#[derive(Copy, Clone, PartialEq, Eq)]
|
||||||
|
pub enum Filter {
|
||||||
|
Done,
|
||||||
|
Pending,
|
||||||
|
None,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct UiState<'a> {
|
||||||
|
pub table: Table<'a>,
|
||||||
|
pub message: String,
|
||||||
|
pub filter: Filter,
|
||||||
|
exercises: &'a [Exercise],
|
||||||
|
selected: usize,
|
||||||
|
table_state: TableState,
|
||||||
|
last_ind: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> UiState<'a> {
|
||||||
|
pub fn with_updated_rows(mut self, state_file: &StateFile) -> Self {
|
||||||
|
let mut rows_counter: usize = 0;
|
||||||
|
let rows = self
|
||||||
|
.exercises
|
||||||
|
.iter()
|
||||||
|
.zip(state_file.progress().iter().copied())
|
||||||
|
.enumerate()
|
||||||
|
.filter_map(|(ind, (exercise, done))| {
|
||||||
|
match (self.filter, done) {
|
||||||
|
(Filter::Done, false) | (Filter::Pending, true) => return None,
|
||||||
|
_ => (),
|
||||||
|
}
|
||||||
|
|
||||||
|
rows_counter += 1;
|
||||||
|
|
||||||
|
let next = if ind == state_file.next_exercise_ind() {
|
||||||
|
">>>>".bold().red()
|
||||||
|
} else {
|
||||||
|
Span::default()
|
||||||
|
};
|
||||||
|
|
||||||
|
let exercise_state = if done {
|
||||||
|
"DONE".green()
|
||||||
|
} else {
|
||||||
|
"PENDING".yellow()
|
||||||
|
};
|
||||||
|
|
||||||
|
Some(Row::new([
|
||||||
|
next,
|
||||||
|
exercise_state,
|
||||||
|
Span::raw(&exercise.name),
|
||||||
|
Span::raw(exercise.path.to_string_lossy()),
|
||||||
|
]))
|
||||||
|
});
|
||||||
|
|
||||||
|
self.table = self.table.rows(rows);
|
||||||
|
|
||||||
|
self.last_ind = rows_counter.saturating_sub(1);
|
||||||
|
self.select(self.selected.min(self.last_ind));
|
||||||
|
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn new(state_file: &StateFile, exercises: &'a [Exercise]) -> Self {
|
||||||
|
let header = Row::new(["Next", "State", "Name", "Path"]);
|
||||||
|
|
||||||
|
let max_name_len = exercises
|
||||||
|
.iter()
|
||||||
|
.map(|exercise| exercise.name.len())
|
||||||
|
.max()
|
||||||
|
.unwrap_or(4) as u16;
|
||||||
|
|
||||||
|
let widths = [
|
||||||
|
Constraint::Length(4),
|
||||||
|
Constraint::Length(7),
|
||||||
|
Constraint::Length(max_name_len),
|
||||||
|
Constraint::Fill(1),
|
||||||
|
];
|
||||||
|
|
||||||
|
let table = Table::default()
|
||||||
|
.widths(widths)
|
||||||
|
.header(header)
|
||||||
|
.column_spacing(2)
|
||||||
|
.highlight_spacing(HighlightSpacing::Always)
|
||||||
|
.highlight_style(Style::new().bg(ratatui::style::Color::Rgb(50, 50, 50)))
|
||||||
|
.highlight_symbol("🦀")
|
||||||
|
.block(Block::default().borders(Borders::BOTTOM));
|
||||||
|
|
||||||
|
let selected = state_file.next_exercise_ind();
|
||||||
|
let table_state = TableState::default()
|
||||||
|
.with_offset(selected.saturating_sub(10))
|
||||||
|
.with_selected(Some(selected));
|
||||||
|
|
||||||
|
let slf = Self {
|
||||||
|
table,
|
||||||
|
message: String::with_capacity(128),
|
||||||
|
filter: Filter::None,
|
||||||
|
exercises,
|
||||||
|
selected,
|
||||||
|
table_state,
|
||||||
|
last_ind: 0,
|
||||||
|
};
|
||||||
|
|
||||||
|
slf.with_updated_rows(state_file)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub fn selected(&self) -> usize {
|
||||||
|
self.selected
|
||||||
|
}
|
||||||
|
|
||||||
|
fn select(&mut self, ind: usize) {
|
||||||
|
self.selected = ind;
|
||||||
|
self.table_state.select(Some(ind));
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn select_next(&mut self) {
|
||||||
|
self.select(self.selected.saturating_add(1).min(self.last_ind));
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn select_previous(&mut self) {
|
||||||
|
self.select(self.selected.saturating_sub(1));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub fn select_first(&mut self) {
|
||||||
|
self.select(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub fn select_last(&mut self) {
|
||||||
|
self.select(self.last_ind);
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn draw(&mut self, frame: &mut Frame) {
|
||||||
|
let area = frame.size();
|
||||||
|
|
||||||
|
frame.render_stateful_widget(
|
||||||
|
&self.table,
|
||||||
|
Rect {
|
||||||
|
x: 0,
|
||||||
|
y: 0,
|
||||||
|
width: area.width,
|
||||||
|
height: area.height - 1,
|
||||||
|
},
|
||||||
|
&mut self.table_state,
|
||||||
|
);
|
||||||
|
|
||||||
|
let message = if self.message.is_empty() {
|
||||||
|
// Help footer.
|
||||||
|
Span::raw(
|
||||||
|
"↓/j ↑/k home/g end/G │ filter <d>one/<p>ending │ <r>eset │ <c>ontinue at │ <q>uit",
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
self.message.as_str().blue()
|
||||||
|
};
|
||||||
|
frame.render_widget(
|
||||||
|
message,
|
||||||
|
Rect {
|
||||||
|
x: 0,
|
||||||
|
y: area.height - 1,
|
||||||
|
width: area.width,
|
||||||
|
height: 1,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
82
src/main.rs
82
src/main.rs
|
@ -1,14 +1,6 @@
|
||||||
use crate::consts::WELCOME;
|
|
||||||
use crate::embedded::{WriteStrategy, EMBEDDED_FILES};
|
|
||||||
use crate::exercise::{Exercise, ExerciseList};
|
|
||||||
use crate::run::run;
|
|
||||||
use crate::verify::verify;
|
|
||||||
use anyhow::{bail, Context, Result};
|
use anyhow::{bail, Context, Result};
|
||||||
use clap::{Parser, Subcommand};
|
use clap::{Parser, Subcommand};
|
||||||
use state::State;
|
use std::{path::Path, process::exit};
|
||||||
use std::path::Path;
|
|
||||||
use std::process::exit;
|
|
||||||
use verify::VerifyState;
|
|
||||||
|
|
||||||
mod consts;
|
mod consts;
|
||||||
mod embedded;
|
mod embedded;
|
||||||
|
@ -16,10 +8,18 @@ mod exercise;
|
||||||
mod init;
|
mod init;
|
||||||
mod list;
|
mod list;
|
||||||
mod run;
|
mod run;
|
||||||
mod state;
|
mod state_file;
|
||||||
mod verify;
|
mod verify;
|
||||||
mod watch;
|
mod watch;
|
||||||
|
|
||||||
|
use self::{
|
||||||
|
consts::WELCOME,
|
||||||
|
exercise::{Exercise, InfoFile},
|
||||||
|
run::run,
|
||||||
|
state_file::StateFile,
|
||||||
|
verify::{verify, VerifyState},
|
||||||
|
};
|
||||||
|
|
||||||
/// Rustlings is a collection of small exercises to get you used to writing and reading Rust code
|
/// Rustlings is a collection of small exercises to get you used to writing and reading Rust code
|
||||||
#[derive(Parser)]
|
#[derive(Parser)]
|
||||||
#[command(version)]
|
#[command(version)]
|
||||||
|
@ -55,6 +55,26 @@ enum Subcommands {
|
||||||
List,
|
List,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn find_exercise<'a>(name: &str, exercises: &'a [Exercise]) -> Result<(usize, &'a Exercise)> {
|
||||||
|
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}'!"))
|
||||||
|
}
|
||||||
|
|
||||||
fn main() -> Result<()> {
|
fn main() -> Result<()> {
|
||||||
let args = Args::parse();
|
let args = Args::parse();
|
||||||
|
|
||||||
|
@ -64,10 +84,10 @@ Did you already install Rust?
|
||||||
Try running `cargo --version` to diagnose the problem.",
|
Try running `cargo --version` to diagnose the problem.",
|
||||||
)?;
|
)?;
|
||||||
|
|
||||||
let exercises = ExerciseList::parse()?.exercises;
|
let exercises = InfoFile::parse()?.exercises;
|
||||||
|
|
||||||
if matches!(args.command, Some(Subcommands::Init)) {
|
if matches!(args.command, Some(Subcommands::Init)) {
|
||||||
init::init_rustlings(&exercises).context("Initialization failed")?;
|
init::init(&exercises).context("Initialization failed")?;
|
||||||
println!(
|
println!(
|
||||||
"\nDone initialization!\n
|
"\nDone initialization!\n
|
||||||
Run `cd rustlings` to go into the generated directory.
|
Run `cd rustlings` to go into the generated directory.
|
||||||
|
@ -85,30 +105,29 @@ If you are just starting with Rustlings, run the command `rustlings init` to ini
|
||||||
exit(1);
|
exit(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut state = State::read_or_default(&exercises);
|
let mut state_file = StateFile::read_or_default(&exercises);
|
||||||
|
|
||||||
match args.command {
|
match args.command {
|
||||||
None | Some(Subcommands::Watch) => {
|
None | Some(Subcommands::Watch) => {
|
||||||
watch::watch(&state, &exercises)?;
|
watch::watch(&state_file, &exercises)?;
|
||||||
}
|
}
|
||||||
// `Init` is handled above.
|
// `Init` is handled above.
|
||||||
Some(Subcommands::Init) => (),
|
Some(Subcommands::Init) => (),
|
||||||
Some(Subcommands::List) => {
|
Some(Subcommands::List) => {
|
||||||
list::list(&mut state, &exercises)?;
|
list::list(&mut state_file, &exercises)?;
|
||||||
}
|
}
|
||||||
Some(Subcommands::Run { name }) => {
|
Some(Subcommands::Run { name }) => {
|
||||||
let exercise = find_exercise(&name, &exercises)?;
|
let (_, exercise) = find_exercise(&name, &exercises)?;
|
||||||
run(exercise).unwrap_or_else(|_| exit(1));
|
run(exercise).unwrap_or_else(|_| exit(1));
|
||||||
}
|
}
|
||||||
Some(Subcommands::Reset { name }) => {
|
Some(Subcommands::Reset { name }) => {
|
||||||
let exercise = find_exercise(&name, &exercises)?;
|
let (ind, exercise) = find_exercise(&name, &exercises)?;
|
||||||
EMBEDDED_FILES
|
exercise.reset()?;
|
||||||
.write_exercise_to_disk(&exercise.path, WriteStrategy::Overwrite)
|
state_file.reset(ind)?;
|
||||||
.with_context(|| format!("Failed to reset the exercise {exercise}"))?;
|
println!("The exercise {exercise} has been reset!");
|
||||||
println!("The file {} has been reset!", exercise.path.display());
|
|
||||||
}
|
}
|
||||||
Some(Subcommands::Hint { name }) => {
|
Some(Subcommands::Hint { name }) => {
|
||||||
let exercise = find_exercise(&name, &exercises)?;
|
let (_, exercise) = find_exercise(&name, &exercises)?;
|
||||||
println!("{}", exercise.hint);
|
println!("{}", exercise.hint);
|
||||||
}
|
}
|
||||||
Some(Subcommands::Verify) => match verify(&exercises, 0)? {
|
Some(Subcommands::Verify) => match verify(&exercises, 0)? {
|
||||||
|
@ -119,22 +138,3 @@ If you are just starting with Rustlings, run the command `rustlings init` to ini
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn find_exercise<'a>(name: &str, exercises: &'a [Exercise]) -> Result<&'a Exercise> {
|
|
||||||
if name == "next" {
|
|
||||||
for exercise in exercises {
|
|
||||||
if !exercise.looks_done()? {
|
|
||||||
return Ok(exercise);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
println!("🎉 Congratulations! You have done all the exercises!");
|
|
||||||
println!("🔚 There are no more exercises to do next!");
|
|
||||||
exit(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
exercises
|
|
||||||
.iter()
|
|
||||||
.find(|e| e.name == name)
|
|
||||||
.with_context(|| format!("No exercise found for '{name}'!"))
|
|
||||||
}
|
|
||||||
|
|
|
@ -5,14 +5,16 @@ use std::fs;
|
||||||
use crate::exercise::Exercise;
|
use crate::exercise::Exercise;
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize)]
|
#[derive(Serialize, Deserialize)]
|
||||||
pub struct State {
|
pub struct StateFile {
|
||||||
next_exercise_ind: usize,
|
next_exercise_ind: usize,
|
||||||
progress: Vec<bool>,
|
progress: Vec<bool>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl State {
|
const BAD_INDEX_ERR: &str = "The next exercise index is higher than the number of exercises";
|
||||||
|
|
||||||
|
impl StateFile {
|
||||||
fn read(exercises: &[Exercise]) -> Option<Self> {
|
fn read(exercises: &[Exercise]) -> Option<Self> {
|
||||||
let file_content = fs::read(".rustlings.json").ok()?;
|
let file_content = fs::read(".rustlings-state.json").ok()?;
|
||||||
|
|
||||||
let slf: Self = serde_json::de::from_slice(&file_content).ok()?;
|
let slf: Self = serde_json::de::from_slice(&file_content).ok()?;
|
||||||
|
|
||||||
|
@ -34,6 +36,8 @@ impl State {
|
||||||
// TODO: Capacity
|
// TODO: Capacity
|
||||||
let mut buf = Vec::with_capacity(1024);
|
let mut buf = Vec::with_capacity(1024);
|
||||||
serde_json::ser::to_writer(&mut buf, self).context("Failed to serialize the state")?;
|
serde_json::ser::to_writer(&mut buf, self).context("Failed to serialize the state")?;
|
||||||
|
fs::write(".rustlings-state.json", buf)
|
||||||
|
.context("Failed to write the state file `.rustlings-state.json`")?;
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
@ -45,9 +49,8 @@ impl State {
|
||||||
|
|
||||||
pub fn set_next_exercise_ind(&mut self, ind: usize) -> Result<()> {
|
pub fn set_next_exercise_ind(&mut self, ind: usize) -> Result<()> {
|
||||||
if ind >= self.progress.len() {
|
if ind >= self.progress.len() {
|
||||||
bail!("The next exercise index is higher than the number of exercises");
|
bail!(BAD_INDEX_ERR);
|
||||||
}
|
}
|
||||||
|
|
||||||
self.next_exercise_ind = ind;
|
self.next_exercise_ind = ind;
|
||||||
self.write()
|
self.write()
|
||||||
}
|
}
|
||||||
|
@ -56,4 +59,10 @@ impl State {
|
||||||
pub fn progress(&self) -> &[bool] {
|
pub fn progress(&self) -> &[bool] {
|
||||||
&self.progress
|
&self.progress
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn reset(&mut self, ind: usize) -> Result<()> {
|
||||||
|
let done = self.progress.get_mut(ind).context(BAD_INDEX_ERR)?;
|
||||||
|
*done = false;
|
||||||
|
self.write()
|
||||||
|
}
|
||||||
}
|
}
|
183
src/watch.rs
183
src/watch.rs
|
@ -1,25 +1,18 @@
|
||||||
use anyhow::Result;
|
use anyhow::Result;
|
||||||
use crossterm::{
|
use notify_debouncer_mini::{new_debouncer, notify::RecursiveMode};
|
||||||
style::{Attribute, ContentStyle, Stylize},
|
|
||||||
terminal::{Clear, ClearType},
|
|
||||||
ExecutableCommand,
|
|
||||||
};
|
|
||||||
use notify_debouncer_mini::{
|
|
||||||
new_debouncer, notify::RecursiveMode, DebounceEventResult, DebouncedEventKind,
|
|
||||||
};
|
|
||||||
use std::{
|
use std::{
|
||||||
fmt::Write as _,
|
io::{self, BufRead, Write},
|
||||||
io::{self, BufRead, StdoutLock, Write},
|
|
||||||
path::Path,
|
path::Path,
|
||||||
sync::mpsc::{channel, sync_channel, Receiver},
|
sync::mpsc::{channel, sync_channel},
|
||||||
thread,
|
thread,
|
||||||
time::Duration,
|
time::Duration,
|
||||||
};
|
};
|
||||||
|
|
||||||
use crate::{
|
mod state;
|
||||||
exercise::{self, Exercise},
|
|
||||||
state::State,
|
use crate::{exercise::Exercise, state_file::StateFile};
|
||||||
};
|
|
||||||
|
use self::state::WatchState;
|
||||||
|
|
||||||
enum Event {
|
enum Event {
|
||||||
Hint,
|
Hint,
|
||||||
|
@ -27,160 +20,14 @@ enum Event {
|
||||||
Quit,
|
Quit,
|
||||||
}
|
}
|
||||||
|
|
||||||
struct WatchState<'a> {
|
pub fn watch(state_file: &StateFile, exercises: &[Exercise]) -> Result<()> {
|
||||||
writer: StdoutLock<'a>,
|
|
||||||
rx: Receiver<DebounceEventResult>,
|
|
||||||
exercises: &'a [Exercise],
|
|
||||||
exercise: &'a Exercise,
|
|
||||||
current_exercise_ind: usize,
|
|
||||||
stdout: Option<Vec<u8>>,
|
|
||||||
stderr: Option<Vec<u8>>,
|
|
||||||
message: Option<String>,
|
|
||||||
prompt: Vec<u8>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a> WatchState<'a> {
|
|
||||||
fn run_exercise(&mut self) -> Result<bool> {
|
|
||||||
let output = self.exercise.run()?;
|
|
||||||
|
|
||||||
if !output.status.success() {
|
|
||||||
self.stdout = Some(output.stdout);
|
|
||||||
self.stderr = Some(output.stderr);
|
|
||||||
return Ok(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
if let exercise::State::Pending(context) = self.exercise.state()? {
|
|
||||||
let mut message = format!(
|
|
||||||
"
|
|
||||||
You can keep working on this exercise or jump into the next one by removing the {} comment:
|
|
||||||
|
|
||||||
",
|
|
||||||
"`I AM NOT DONE`".bold(),
|
|
||||||
);
|
|
||||||
|
|
||||||
for context_line in context {
|
|
||||||
let formatted_line = if context_line.important {
|
|
||||||
context_line.line.bold()
|
|
||||||
} else {
|
|
||||||
context_line.line.stylize()
|
|
||||||
};
|
|
||||||
|
|
||||||
writeln!(
|
|
||||||
message,
|
|
||||||
"{:>2} {} {}",
|
|
||||||
ContentStyle {
|
|
||||||
foreground_color: Some(crossterm::style::Color::Blue),
|
|
||||||
background_color: None,
|
|
||||||
underline_color: None,
|
|
||||||
attributes: Attribute::Bold.into()
|
|
||||||
}
|
|
||||||
.apply(context_line.number),
|
|
||||||
"|".blue(),
|
|
||||||
formatted_line,
|
|
||||||
)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
self.stdout = Some(output.stdout);
|
|
||||||
self.message = Some(message);
|
|
||||||
return Ok(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(true)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn try_recv_event(&mut self) -> Result<()> {
|
|
||||||
let Ok(events) = self.rx.recv_timeout(Duration::from_millis(100)) else {
|
|
||||||
return Ok(());
|
|
||||||
};
|
|
||||||
|
|
||||||
if let Some(current_exercise_ind) = events?
|
|
||||||
.iter()
|
|
||||||
.filter_map(|event| {
|
|
||||||
if event.kind != DebouncedEventKind::Any
|
|
||||||
|| !event.path.extension().is_some_and(|ext| ext == "rs")
|
|
||||||
{
|
|
||||||
return None;
|
|
||||||
}
|
|
||||||
|
|
||||||
self.exercises
|
|
||||||
.iter()
|
|
||||||
.position(|exercise| event.path.ends_with(&exercise.path))
|
|
||||||
})
|
|
||||||
.min()
|
|
||||||
{
|
|
||||||
self.current_exercise_ind = current_exercise_ind;
|
|
||||||
} else {
|
|
||||||
return Ok(());
|
|
||||||
};
|
|
||||||
|
|
||||||
while self.current_exercise_ind < self.exercises.len() {
|
|
||||||
self.exercise = &self.exercises[self.current_exercise_ind];
|
|
||||||
if !self.run_exercise()? {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
self.current_exercise_ind += 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn prompt(&mut self) -> io::Result<()> {
|
|
||||||
self.writer.write_all(&self.prompt)?;
|
|
||||||
self.writer.flush()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn render(&mut self) -> Result<()> {
|
|
||||||
self.writer.execute(Clear(ClearType::All))?;
|
|
||||||
|
|
||||||
if let Some(stdout) = &self.stdout {
|
|
||||||
self.writer.write_all(stdout)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
if let Some(stderr) = &self.stderr {
|
|
||||||
self.writer.write_all(stderr)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
if let Some(message) = &self.message {
|
|
||||||
self.writer.write_all(message.as_bytes())?;
|
|
||||||
}
|
|
||||||
|
|
||||||
self.prompt()?;
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn watch(state: &State, exercises: &[Exercise]) -> Result<()> {
|
|
||||||
let (tx, rx) = channel();
|
let (tx, rx) = channel();
|
||||||
let mut debouncer = new_debouncer(Duration::from_secs(1), tx)?;
|
let mut debouncer = new_debouncer(Duration::from_secs(1), tx)?;
|
||||||
debouncer
|
debouncer
|
||||||
.watcher()
|
.watcher()
|
||||||
.watch(Path::new("exercises"), RecursiveMode::Recursive)?;
|
.watch(Path::new("exercises"), RecursiveMode::Recursive)?;
|
||||||
|
|
||||||
let current_exercise_ind = state.next_exercise_ind();
|
let mut watch_state = WatchState::new(state_file, exercises, rx);
|
||||||
|
|
||||||
let exercise = &exercises[current_exercise_ind];
|
|
||||||
|
|
||||||
let writer = io::stdout().lock();
|
|
||||||
|
|
||||||
let mut watch_state = WatchState {
|
|
||||||
writer,
|
|
||||||
rx,
|
|
||||||
exercises,
|
|
||||||
exercise,
|
|
||||||
current_exercise_ind,
|
|
||||||
stdout: None,
|
|
||||||
stderr: None,
|
|
||||||
message: None,
|
|
||||||
prompt: format!(
|
|
||||||
"\n\n{}int/{}lear/{}uit? ",
|
|
||||||
"h".bold(),
|
|
||||||
"c".bold(),
|
|
||||||
"q".bold()
|
|
||||||
)
|
|
||||||
.into_bytes(),
|
|
||||||
};
|
|
||||||
|
|
||||||
watch_state.run_exercise()?;
|
watch_state.run_exercise()?;
|
||||||
watch_state.render()?;
|
watch_state.render()?;
|
||||||
|
@ -214,24 +61,20 @@ pub fn watch(state: &State, exercises: &[Exercise]) -> Result<()> {
|
||||||
if let Ok(event) = rx.try_recv() {
|
if let Ok(event) = rx.try_recv() {
|
||||||
match event {
|
match event {
|
||||||
Some(Event::Hint) => {
|
Some(Event::Hint) => {
|
||||||
watch_state
|
watch_state.show_hint()?;
|
||||||
.writer
|
|
||||||
.write_all(watch_state.exercise.hint.as_bytes())?;
|
|
||||||
watch_state.prompt()?;
|
|
||||||
}
|
}
|
||||||
Some(Event::Clear) => {
|
Some(Event::Clear) => {
|
||||||
watch_state.render()?;
|
watch_state.render()?;
|
||||||
}
|
}
|
||||||
Some(Event::Quit) => break,
|
Some(Event::Quit) => break,
|
||||||
None => {
|
None => {
|
||||||
watch_state.writer.write_all(b"Invalid command")?;
|
watch_state.handle_invalid_cmd()?;
|
||||||
watch_state.prompt()?;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
watch_state.writer.write_all(b"
|
watch_state.into_writer().write_all(b"
|
||||||
We hope you're enjoying learning Rust!
|
We hope you're enjoying learning Rust!
|
||||||
If you want to continue working on the exercises at a later point, you can simply run `rustlings` again.
|
If you want to continue working on the exercises at a later point, you can simply run `rustlings` again.
|
||||||
")?;
|
")?;
|
||||||
|
|
186
src/watch/state.rs
Normal file
186
src/watch/state.rs
Normal file
|
@ -0,0 +1,186 @@
|
||||||
|
use anyhow::Result;
|
||||||
|
use crossterm::{
|
||||||
|
style::{Attribute, ContentStyle, Stylize},
|
||||||
|
terminal::{Clear, ClearType},
|
||||||
|
ExecutableCommand,
|
||||||
|
};
|
||||||
|
use notify_debouncer_mini::{DebounceEventResult, DebouncedEventKind};
|
||||||
|
use std::{
|
||||||
|
fmt::Write as _,
|
||||||
|
io::{self, StdoutLock, Write as _},
|
||||||
|
sync::mpsc::Receiver,
|
||||||
|
time::Duration,
|
||||||
|
};
|
||||||
|
|
||||||
|
use crate::{
|
||||||
|
exercise::{Exercise, State},
|
||||||
|
state_file::StateFile,
|
||||||
|
};
|
||||||
|
|
||||||
|
pub struct WatchState<'a> {
|
||||||
|
writer: StdoutLock<'a>,
|
||||||
|
rx: Receiver<DebounceEventResult>,
|
||||||
|
exercises: &'a [Exercise],
|
||||||
|
exercise: &'a Exercise,
|
||||||
|
current_exercise_ind: usize,
|
||||||
|
stdout: Option<Vec<u8>>,
|
||||||
|
stderr: Option<Vec<u8>>,
|
||||||
|
message: Option<String>,
|
||||||
|
prompt: Vec<u8>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> WatchState<'a> {
|
||||||
|
pub fn new(
|
||||||
|
state_file: &StateFile,
|
||||||
|
exercises: &'a [Exercise],
|
||||||
|
rx: Receiver<DebounceEventResult>,
|
||||||
|
) -> Self {
|
||||||
|
let current_exercise_ind = state_file.next_exercise_ind();
|
||||||
|
let exercise = &exercises[current_exercise_ind];
|
||||||
|
|
||||||
|
let writer = io::stdout().lock();
|
||||||
|
|
||||||
|
let prompt = format!(
|
||||||
|
"\n\n{}int/{}lear/{}uit? ",
|
||||||
|
"h".bold(),
|
||||||
|
"c".bold(),
|
||||||
|
"q".bold()
|
||||||
|
)
|
||||||
|
.into_bytes();
|
||||||
|
|
||||||
|
Self {
|
||||||
|
writer,
|
||||||
|
rx,
|
||||||
|
exercises,
|
||||||
|
exercise,
|
||||||
|
current_exercise_ind,
|
||||||
|
stdout: None,
|
||||||
|
stderr: None,
|
||||||
|
message: None,
|
||||||
|
prompt,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub fn into_writer(self) -> StdoutLock<'a> {
|
||||||
|
self.writer
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn run_exercise(&mut self) -> Result<bool> {
|
||||||
|
let output = self.exercise.run()?;
|
||||||
|
|
||||||
|
if !output.status.success() {
|
||||||
|
self.stdout = Some(output.stdout);
|
||||||
|
self.stderr = Some(output.stderr);
|
||||||
|
return Ok(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
if let State::Pending(context) = self.exercise.state()? {
|
||||||
|
let mut message = format!(
|
||||||
|
"
|
||||||
|
You can keep working on this exercise or jump into the next one by removing the {} comment:
|
||||||
|
|
||||||
|
",
|
||||||
|
"`I AM NOT DONE`".bold(),
|
||||||
|
);
|
||||||
|
|
||||||
|
for context_line in context {
|
||||||
|
let formatted_line = if context_line.important {
|
||||||
|
context_line.line.bold()
|
||||||
|
} else {
|
||||||
|
context_line.line.stylize()
|
||||||
|
};
|
||||||
|
|
||||||
|
writeln!(
|
||||||
|
message,
|
||||||
|
"{:>2} {} {}",
|
||||||
|
ContentStyle {
|
||||||
|
foreground_color: Some(crossterm::style::Color::Blue),
|
||||||
|
background_color: None,
|
||||||
|
underline_color: None,
|
||||||
|
attributes: Attribute::Bold.into()
|
||||||
|
}
|
||||||
|
.apply(context_line.number),
|
||||||
|
"|".blue(),
|
||||||
|
formatted_line,
|
||||||
|
)?;
|
||||||
|
}
|
||||||
|
|
||||||
|
self.stdout = Some(output.stdout);
|
||||||
|
self.message = Some(message);
|
||||||
|
return Ok(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(true)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn try_recv_event(&mut self) -> Result<()> {
|
||||||
|
let Ok(events) = self.rx.recv_timeout(Duration::from_millis(100)) else {
|
||||||
|
return Ok(());
|
||||||
|
};
|
||||||
|
|
||||||
|
if let Some(current_exercise_ind) = events?
|
||||||
|
.iter()
|
||||||
|
.filter_map(|event| {
|
||||||
|
if event.kind != DebouncedEventKind::Any
|
||||||
|
|| !event.path.extension().is_some_and(|ext| ext == "rs")
|
||||||
|
{
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
|
||||||
|
self.exercises
|
||||||
|
.iter()
|
||||||
|
.position(|exercise| event.path.ends_with(&exercise.path))
|
||||||
|
})
|
||||||
|
.min()
|
||||||
|
{
|
||||||
|
self.current_exercise_ind = current_exercise_ind;
|
||||||
|
} else {
|
||||||
|
return Ok(());
|
||||||
|
};
|
||||||
|
|
||||||
|
while self.current_exercise_ind < self.exercises.len() {
|
||||||
|
self.exercise = &self.exercises[self.current_exercise_ind];
|
||||||
|
if !self.run_exercise()? {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
self.current_exercise_ind += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn show_prompt(&mut self) -> io::Result<()> {
|
||||||
|
self.writer.write_all(&self.prompt)?;
|
||||||
|
self.writer.flush()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn render(&mut self) -> io::Result<()> {
|
||||||
|
self.writer.execute(Clear(ClearType::All))?;
|
||||||
|
|
||||||
|
if let Some(stdout) = &self.stdout {
|
||||||
|
self.writer.write_all(stdout)?;
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(stderr) = &self.stderr {
|
||||||
|
self.writer.write_all(stderr)?;
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(message) = &self.message {
|
||||||
|
self.writer.write_all(message.as_bytes())?;
|
||||||
|
}
|
||||||
|
|
||||||
|
self.show_prompt()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn show_hint(&mut self) -> io::Result<()> {
|
||||||
|
self.writer.write_all(self.exercise.hint.as_bytes())?;
|
||||||
|
self.show_prompt()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn handle_invalid_cmd(&mut self) -> io::Result<()> {
|
||||||
|
self.writer.write_all(b"Invalid command")?;
|
||||||
|
self.show_prompt()
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,14 +1,7 @@
|
||||||
use assert_cmd::prelude::*;
|
use assert_cmd::prelude::*;
|
||||||
use glob::glob;
|
use glob::glob;
|
||||||
use predicates::boolean::PredicateBooleanExt;
|
use predicates::boolean::PredicateBooleanExt;
|
||||||
use std::fs::File;
|
use std::{fs::File, io::Read, process::Command};
|
||||||
use std::io::Read;
|
|
||||||
use std::process::Command;
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn runs_without_arguments() {
|
|
||||||
Command::cargo_bin("rustlings").unwrap().assert().success();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn fails_when_in_wrong_dir() {
|
fn fails_when_in_wrong_dir() {
|
||||||
|
@ -201,57 +194,3 @@ fn run_single_test_success_with_output() {
|
||||||
.code(0)
|
.code(0)
|
||||||
.stdout(predicates::str::contains("THIS TEST TOO SHALL PASS"));
|
.stdout(predicates::str::contains("THIS TEST TOO SHALL PASS"));
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn run_rustlings_list() {
|
|
||||||
Command::cargo_bin("rustlings")
|
|
||||||
.unwrap()
|
|
||||||
.args(["list"])
|
|
||||||
.current_dir("tests/fixture/success")
|
|
||||||
.assert()
|
|
||||||
.success();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn run_rustlings_list_no_pending() {
|
|
||||||
Command::cargo_bin("rustlings")
|
|
||||||
.unwrap()
|
|
||||||
.args(["list"])
|
|
||||||
.current_dir("tests/fixture/success")
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(predicates::str::contains("Pending").not());
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn run_rustlings_list_both_done_and_pending() {
|
|
||||||
Command::cargo_bin("rustlings")
|
|
||||||
.unwrap()
|
|
||||||
.args(["list"])
|
|
||||||
.current_dir("tests/fixture/state")
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(predicates::str::contains("Done").and(predicates::str::contains("Pending")));
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn run_rustlings_list_without_pending() {
|
|
||||||
Command::cargo_bin("rustlings")
|
|
||||||
.unwrap()
|
|
||||||
.args(["list", "--solved"])
|
|
||||||
.current_dir("tests/fixture/state")
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(predicates::str::contains("Pending").not());
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn run_rustlings_list_without_done() {
|
|
||||||
Command::cargo_bin("rustlings")
|
|
||||||
.unwrap()
|
|
||||||
.args(["list", "--unsolved"])
|
|
||||||
.current_dir("tests/fixture/state")
|
|
||||||
.assert()
|
|
||||||
.success()
|
|
||||||
.stdout(predicates::str::contains("Done").not());
|
|
||||||
}
|
|
||||||
|
|
Loading…
Reference in a new issue