Extract table logic

This commit is contained in:
Joel Wachsler 2022-07-17 12:19:22 +00:00
parent bebceb6581
commit b32311ae65

View File

@ -78,22 +78,44 @@ impl MdToken {
pub fn from(content: &str) -> Vec<MdToken> {
// to prevent infinite loops
let mut max_iterations = 10000;
let mut decrease_max_iterations = || {
max_iterations -= 1;
if max_iterations <= 0 {
panic!("Max iterations reached, missing termination?");
};
};
let mut max_iterator_checker = MaxIteratorChecker::default();
let mut output = Vec::new();
let mut iter = content.lines().peekable();
while let Some(line) = iter.next() {
decrease_max_iterations();
// assume this is a table
while let Some(line) = iter.next() {
max_iterator_checker.decrease();
// assume that lines starting with "|" are tables
if line.contains('|') {
let table = TableParser::new(&mut max_iterator_checker, &mut iter).parse(line);
output.push(MdToken::Content(table));
} else {
output.push(MdToken::parse_token(line));
}
}
output
}
}
struct TableParser<'a, 'b> {
max_iterator_checker: &'a mut MaxIteratorChecker,
iter: &'a mut std::iter::Peekable<std::str::Lines<'b>>,
}
impl<'a, 'b> TableParser<'a, 'b> {
fn new(
max_iterator_checker: &'a mut MaxIteratorChecker,
iter: &'a mut std::iter::Peekable<std::str::Lines<'b>>,
) -> Self {
Self {
max_iterator_checker,
iter,
}
}
fn parse(&mut self, line: &str) -> MdContent {
let to_columns = |column_line: &str| {
column_line
.replace('`', "")
@ -106,35 +128,50 @@ impl MdToken {
raw: line.into(),
columns: to_columns(line),
};
let table_split = iter.next().unwrap();
let table_split = self.iter.next().unwrap();
let mut table_rows = Vec::new();
while let Some(peeked_row_line) = iter.peek() {
decrease_max_iterations();
while let Some(peeked_row_line) = self.iter.peek() {
self.max_iterator_checker.decrease();
if !peeked_row_line.contains('|') {
// we've reached the end of the table, let's go back one step
break;
}
let next_row_line = iter.next().unwrap();
let table_row = TableRow {
let next_row_line = self.iter.next().unwrap();
table_rows.push(TableRow {
raw: next_row_line.to_string(),
columns: to_columns(next_row_line),
};
table_rows.push(table_row);
});
}
output.push(MdToken::Content(MdContent::Table(Table {
MdContent::Table(Table {
header: table_header,
split: table_split.to_string(),
rows: table_rows,
})));
} else {
output.push(MdToken::parse_token(line));
}
}
output
})
}
}
#[derive(Debug)]
struct MaxIteratorChecker {
max_iterations: i32,
}
impl MaxIteratorChecker {
fn decrease(&mut self) {
self.max_iterations -= 1;
if self.max_iterations <= 0 {
panic!("Max iterations reached, missing termination?");
}
}
}
impl Default for MaxIteratorChecker {
fn default() -> Self {
MaxIteratorChecker {
max_iterations: 10000,
}
}
}