2019-05-26 02:50:41 +08:00
|
|
|
use crate::errors::*;
|
2018-07-24 01:45:01 +08:00
|
|
|
use memchr::{self, Memchr};
|
2021-12-25 20:46:27 +08:00
|
|
|
use pulldown_cmark::{self, Event, HeadingLevel, Tag};
|
2017-12-11 09:26:11 +08:00
|
|
|
use std::fmt::{self, Display, Formatter};
|
2017-12-11 12:17:20 +08:00
|
|
|
use std::iter::FromIterator;
|
2017-11-18 19:50:47 +08:00
|
|
|
use std::ops::{Deref, DerefMut};
|
|
|
|
use std::path::{Path, PathBuf};
|
|
|
|
|
|
|
|
/// Parse the text from a `SUMMARY.md` file into a sort of "recipe" to be
|
|
|
|
/// used when loading a book from disk.
|
|
|
|
///
|
|
|
|
/// # Summary Format
|
|
|
|
///
|
|
|
|
/// **Title:** It's common practice to begin with a title, generally
|
2017-12-11 15:50:31 +08:00
|
|
|
/// "# Summary". It's not mandatory and the parser (currently) ignores it, so
|
|
|
|
/// you can too if you feel like it.
|
2017-11-18 19:50:47 +08:00
|
|
|
///
|
|
|
|
/// **Prefix Chapter:** Before the main numbered chapters you can add a couple
|
|
|
|
/// of elements that will not be numbered. This is useful for forewords,
|
|
|
|
/// introductions, etc. There are however some constraints. You can not nest
|
|
|
|
/// prefix chapters, they should all be on the root level. And you can not add
|
|
|
|
/// prefix chapters once you have added numbered chapters.
|
|
|
|
///
|
|
|
|
/// ```markdown
|
|
|
|
/// [Title of prefix element](relative/path/to/markdown.md)
|
|
|
|
/// ```
|
|
|
|
///
|
2020-03-21 10:18:07 +08:00
|
|
|
/// **Part Title:** An optional title for the next collect of numbered chapters. The numbered
|
|
|
|
/// chapters can be broken into as many parts as desired.
|
|
|
|
///
|
2017-11-18 19:50:47 +08:00
|
|
|
/// **Numbered Chapter:** Numbered chapters are the main content of the book,
|
|
|
|
/// they
|
|
|
|
/// will be numbered and can be nested, resulting in a nice hierarchy (chapters,
|
|
|
|
/// sub-chapters, etc.)
|
|
|
|
///
|
|
|
|
/// ```markdown
|
2020-03-21 10:18:07 +08:00
|
|
|
/// # Title of Part
|
|
|
|
///
|
2017-11-18 19:50:47 +08:00
|
|
|
/// - [Title of the Chapter](relative/path/to/markdown.md)
|
|
|
|
/// ```
|
|
|
|
///
|
2017-12-11 15:50:31 +08:00
|
|
|
/// You can either use - or * to indicate a numbered chapter, the parser doesn't
|
|
|
|
/// care but you'll probably want to stay consistent.
|
2017-11-18 19:50:47 +08:00
|
|
|
///
|
|
|
|
/// **Suffix Chapter:** After the numbered chapters you can add a couple of
|
|
|
|
/// non-numbered chapters. They are the same as prefix chapters but come after
|
|
|
|
/// the numbered chapters instead of before.
|
|
|
|
///
|
|
|
|
/// All other elements are unsupported and will be ignored at best or result in
|
|
|
|
/// an error.
|
|
|
|
pub fn parse_summary(summary: &str) -> Result<Summary> {
|
|
|
|
let parser = SummaryParser::new(summary);
|
|
|
|
parser.parse()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// The parsed `SUMMARY.md`, specifying how the book should be laid out.
|
|
|
|
#[derive(Debug, Clone, Default, PartialEq, Serialize, Deserialize)]
|
|
|
|
pub struct Summary {
|
|
|
|
/// An optional title for the `SUMMARY.md`, currently just ignored.
|
|
|
|
pub title: Option<String>,
|
|
|
|
/// Chapters before the main text (e.g. an introduction).
|
|
|
|
pub prefix_chapters: Vec<SummaryItem>,
|
2020-03-21 10:18:07 +08:00
|
|
|
/// The main numbered chapters of the book, broken into one or more possibly named parts.
|
2020-05-19 00:18:14 +08:00
|
|
|
pub numbered_chapters: Vec<SummaryItem>,
|
2017-11-18 19:50:47 +08:00
|
|
|
/// Items which come after the main document (e.g. a conclusion).
|
|
|
|
pub suffix_chapters: Vec<SummaryItem>,
|
|
|
|
}
|
|
|
|
|
|
|
|
/// A struct representing an entry in the `SUMMARY.md`, possibly with nested
|
|
|
|
/// entries.
|
|
|
|
///
|
|
|
|
/// This is roughly the equivalent of `[Some section](./path/to/file.md)`.
|
|
|
|
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
|
|
|
|
pub struct Link {
|
|
|
|
/// The name of the chapter.
|
|
|
|
pub name: String,
|
|
|
|
/// The location of the chapter's source file, taking the book's `src`
|
|
|
|
/// directory as the root.
|
2020-03-01 00:55:45 +08:00
|
|
|
pub location: Option<PathBuf>,
|
2017-11-18 19:50:47 +08:00
|
|
|
/// The section number, if this chapter is in the numbered section.
|
|
|
|
pub number: Option<SectionNumber>,
|
|
|
|
/// Any nested items this chapter may contain.
|
|
|
|
pub nested_items: Vec<SummaryItem>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Link {
|
|
|
|
/// Create a new link with no nested items.
|
2020-03-25 06:52:24 +08:00
|
|
|
pub fn new<S: Into<String>, P: AsRef<Path>>(name: S, location: P) -> Link {
|
2017-11-18 19:50:47 +08:00
|
|
|
Link {
|
|
|
|
name: name.into(),
|
2020-03-01 00:55:45 +08:00
|
|
|
location: Some(location.as_ref().to_path_buf()),
|
2017-11-18 19:50:47 +08:00
|
|
|
number: None,
|
|
|
|
nested_items: Vec::new(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for Link {
|
|
|
|
fn default() -> Self {
|
|
|
|
Link {
|
|
|
|
name: String::new(),
|
2020-03-01 00:55:45 +08:00
|
|
|
location: Some(PathBuf::new()),
|
2017-11-18 19:50:47 +08:00
|
|
|
number: None,
|
|
|
|
nested_items: Vec::new(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// An item in `SUMMARY.md` which could be either a separator or a `Link`.
|
|
|
|
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
|
|
|
|
pub enum SummaryItem {
|
|
|
|
/// A link to a chapter.
|
|
|
|
Link(Link),
|
|
|
|
/// A separator (`---`).
|
|
|
|
Separator,
|
2020-05-19 00:18:14 +08:00
|
|
|
/// A part title.
|
|
|
|
PartTitle(String),
|
2017-11-18 19:50:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
impl SummaryItem {
|
|
|
|
fn maybe_link_mut(&mut self) -> Option<&mut Link> {
|
|
|
|
match *self {
|
|
|
|
SummaryItem::Link(ref mut l) => Some(l),
|
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Link> for SummaryItem {
|
|
|
|
fn from(other: Link) -> SummaryItem {
|
|
|
|
SummaryItem::Link(other)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-11 15:50:31 +08:00
|
|
|
/// A recursive descent (-ish) parser for a `SUMMARY.md`.
|
2017-11-18 19:50:47 +08:00
|
|
|
///
|
|
|
|
///
|
|
|
|
/// # Grammar
|
|
|
|
///
|
|
|
|
/// The `SUMMARY.md` file has a grammar which looks something like this:
|
|
|
|
///
|
|
|
|
/// ```text
|
|
|
|
/// summary ::= title prefix_chapters numbered_chapters
|
2020-03-21 10:18:07 +08:00
|
|
|
/// suffix_chapters
|
2017-11-18 19:50:47 +08:00
|
|
|
/// title ::= "# " TEXT
|
|
|
|
/// | EPSILON
|
|
|
|
/// prefix_chapters ::= item*
|
|
|
|
/// suffix_chapters ::= item*
|
2020-03-21 10:18:07 +08:00
|
|
|
/// numbered_chapters ::= part+
|
|
|
|
/// part ::= title dotted_item+
|
2017-11-18 19:50:47 +08:00
|
|
|
/// dotted_item ::= INDENT* DOT_POINT item
|
|
|
|
/// item ::= link
|
|
|
|
/// | separator
|
|
|
|
/// separator ::= "---"
|
|
|
|
/// link ::= "[" TEXT "]" "(" TEXT ")"
|
|
|
|
/// DOT_POINT ::= "-"
|
|
|
|
/// | "*"
|
|
|
|
/// ```
|
|
|
|
///
|
|
|
|
/// > **Note:** the `TEXT` terminal is "normal" text, and should (roughly)
|
|
|
|
/// > match the following regex: "[^<>\n[]]+".
|
|
|
|
struct SummaryParser<'a> {
|
2017-12-11 12:17:20 +08:00
|
|
|
src: &'a str,
|
2021-12-25 20:46:27 +08:00
|
|
|
stream: pulldown_cmark::OffsetIter<'a, 'a>,
|
2019-11-12 03:25:38 +08:00
|
|
|
offset: usize,
|
2020-03-21 10:18:07 +08:00
|
|
|
|
|
|
|
/// We can't actually put an event back into the `OffsetIter` stream, so instead we store it
|
|
|
|
/// here until somebody calls `next_event` again.
|
|
|
|
back: Option<Event<'a>>,
|
2017-11-18 19:50:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Reads `Events` from the provided stream until the corresponding
|
|
|
|
/// `Event::End` is encountered which matches the `$delimiter` pattern.
|
|
|
|
///
|
|
|
|
/// This is the equivalent of doing
|
2021-05-21 18:56:32 +08:00
|
|
|
/// `$stream.take_while(|e| e != $delimiter).collect()` but it allows you to
|
2017-11-18 19:50:47 +08:00
|
|
|
/// use pattern matching and you won't get errors because `take_while()`
|
|
|
|
/// moves `$stream` out of self.
|
|
|
|
macro_rules! collect_events {
|
2018-07-24 01:45:01 +08:00
|
|
|
($stream:expr,start $delimiter:pat) => {
|
2017-12-11 12:17:20 +08:00
|
|
|
collect_events!($stream, Event::Start($delimiter))
|
|
|
|
};
|
2018-07-24 01:45:01 +08:00
|
|
|
($stream:expr,end $delimiter:pat) => {
|
2017-12-11 12:17:20 +08:00
|
|
|
collect_events!($stream, Event::End($delimiter))
|
|
|
|
};
|
2018-07-24 01:45:01 +08:00
|
|
|
($stream:expr, $delimiter:pat) => {{
|
|
|
|
let mut events = Vec::new();
|
|
|
|
|
|
|
|
loop {
|
2019-11-12 03:25:38 +08:00
|
|
|
let event = $stream.next().map(|(ev, _range)| ev);
|
2018-07-24 01:45:01 +08:00
|
|
|
trace!("Next event: {:?}", event);
|
|
|
|
|
|
|
|
match event {
|
|
|
|
Some($delimiter) => break,
|
|
|
|
Some(other) => events.push(other),
|
|
|
|
None => {
|
|
|
|
debug!(
|
|
|
|
"Reached end of stream without finding the closing pattern, {}",
|
|
|
|
stringify!($delimiter)
|
|
|
|
);
|
|
|
|
break;
|
2017-11-18 19:50:47 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-07-24 01:45:01 +08:00
|
|
|
|
|
|
|
events
|
|
|
|
}};
|
2017-11-18 19:50:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> SummaryParser<'a> {
|
2019-05-07 04:50:34 +08:00
|
|
|
fn new(text: &str) -> SummaryParser<'_> {
|
2019-11-12 03:25:38 +08:00
|
|
|
let pulldown_parser = pulldown_cmark::Parser::new(text).into_offset_iter();
|
2017-11-18 19:50:47 +08:00
|
|
|
|
|
|
|
SummaryParser {
|
2017-12-11 12:17:20 +08:00
|
|
|
src: text,
|
2017-11-18 19:50:47 +08:00
|
|
|
stream: pulldown_parser,
|
2019-11-12 03:25:38 +08:00
|
|
|
offset: 0,
|
2020-03-21 10:18:07 +08:00
|
|
|
back: None,
|
2017-11-18 19:50:47 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-22 20:47:29 +08:00
|
|
|
/// Get the current line and column to give the user more useful error
|
2017-12-11 15:50:31 +08:00
|
|
|
/// messages.
|
2017-12-11 12:17:20 +08:00
|
|
|
fn current_location(&self) -> (usize, usize) {
|
2019-11-12 03:25:38 +08:00
|
|
|
let previous_text = self.src[..self.offset].as_bytes();
|
2017-12-11 14:20:05 +08:00
|
|
|
let line = Memchr::new(b'\n', previous_text).count() + 1;
|
|
|
|
let start_of_line = memchr::memrchr(b'\n', previous_text).unwrap_or(0);
|
2019-11-12 03:25:38 +08:00
|
|
|
let col = self.src[start_of_line..self.offset].chars().count();
|
2017-12-11 14:20:05 +08:00
|
|
|
|
2017-12-11 12:17:20 +08:00
|
|
|
(line, col)
|
|
|
|
}
|
|
|
|
|
2017-11-18 19:50:47 +08:00
|
|
|
/// Parse the text the `SummaryParser` was created with.
|
|
|
|
fn parse(mut self) -> Result<Summary> {
|
2017-12-11 12:17:20 +08:00
|
|
|
let title = self.parse_title();
|
|
|
|
|
2018-08-03 09:22:49 +08:00
|
|
|
let prefix_chapters = self
|
|
|
|
.parse_affix(true)
|
2020-05-21 05:32:00 +08:00
|
|
|
.with_context(|| "There was an error parsing the prefix chapters")?;
|
2020-05-19 00:18:14 +08:00
|
|
|
let numbered_chapters = self
|
2020-03-21 10:18:07 +08:00
|
|
|
.parse_parts()
|
2020-05-21 05:32:00 +08:00
|
|
|
.with_context(|| "There was an error parsing the numbered chapters")?;
|
2018-08-03 09:22:49 +08:00
|
|
|
let suffix_chapters = self
|
|
|
|
.parse_affix(false)
|
2020-05-21 05:32:00 +08:00
|
|
|
.with_context(|| "There was an error parsing the suffix chapters")?;
|
2017-12-11 12:17:20 +08:00
|
|
|
|
|
|
|
Ok(Summary {
|
|
|
|
title,
|
|
|
|
prefix_chapters,
|
2020-05-19 00:18:14 +08:00
|
|
|
numbered_chapters,
|
2017-12-11 12:17:20 +08:00
|
|
|
suffix_chapters,
|
|
|
|
})
|
|
|
|
}
|
2017-11-18 19:50:47 +08:00
|
|
|
|
2020-03-21 10:18:07 +08:00
|
|
|
/// Parse the affix chapters.
|
2017-12-11 12:17:20 +08:00
|
|
|
fn parse_affix(&mut self, is_prefix: bool) -> Result<Vec<SummaryItem>> {
|
|
|
|
let mut items = Vec::new();
|
|
|
|
debug!(
|
2018-01-23 01:28:37 +08:00
|
|
|
"Parsing {} items",
|
2017-12-11 12:17:20 +08:00
|
|
|
if is_prefix { "prefix" } else { "suffix" }
|
|
|
|
);
|
|
|
|
|
|
|
|
loop {
|
2017-12-11 14:20:05 +08:00
|
|
|
match self.next_event() {
|
2020-03-21 10:18:07 +08:00
|
|
|
Some(ev @ Event::Start(Tag::List(..)))
|
2021-12-25 20:46:27 +08:00
|
|
|
| Some(ev @ Event::Start(Tag::Heading(HeadingLevel::H1, ..))) => {
|
2017-12-11 12:17:20 +08:00
|
|
|
if is_prefix {
|
|
|
|
// we've finished prefix chapters and are at the start
|
|
|
|
// of the numbered section.
|
2020-03-21 10:18:07 +08:00
|
|
|
self.back(ev);
|
2017-12-11 12:17:20 +08:00
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
bail!(self.parse_error("Suffix chapters cannot be followed by a list"));
|
|
|
|
}
|
|
|
|
}
|
2019-06-12 00:26:24 +08:00
|
|
|
Some(Event::Start(Tag::Link(_type, href, _title))) => {
|
2020-03-01 00:55:45 +08:00
|
|
|
let link = self.parse_link(href.to_string());
|
2017-12-11 12:17:20 +08:00
|
|
|
items.push(SummaryItem::Link(link));
|
|
|
|
}
|
2019-11-12 03:25:38 +08:00
|
|
|
Some(Event::Rule) => items.push(SummaryItem::Separator),
|
2017-12-11 12:17:20 +08:00
|
|
|
Some(_) => {}
|
|
|
|
None => break,
|
|
|
|
}
|
2017-11-18 19:50:47 +08:00
|
|
|
}
|
|
|
|
|
2017-12-11 12:17:20 +08:00
|
|
|
Ok(items)
|
|
|
|
}
|
2017-11-18 19:50:47 +08:00
|
|
|
|
2020-05-19 00:18:14 +08:00
|
|
|
fn parse_parts(&mut self) -> Result<Vec<SummaryItem>> {
|
2020-03-21 10:18:07 +08:00
|
|
|
let mut parts = vec![];
|
|
|
|
|
|
|
|
// We want the section numbers to be continues through all parts.
|
|
|
|
let mut root_number = SectionNumber::default();
|
|
|
|
let mut root_items = 0;
|
|
|
|
|
|
|
|
loop {
|
|
|
|
// Possibly match a title or the end of the "numbered chapters part".
|
|
|
|
let title = match self.next_event() {
|
|
|
|
Some(ev @ Event::Start(Tag::Paragraph)) => {
|
|
|
|
// we're starting the suffix chapters
|
|
|
|
self.back(ev);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2021-12-25 20:46:27 +08:00
|
|
|
Some(Event::Start(Tag::Heading(HeadingLevel::H1, ..))) => {
|
2020-03-21 10:18:07 +08:00
|
|
|
debug!("Found a h1 in the SUMMARY");
|
|
|
|
|
2021-12-25 20:46:27 +08:00
|
|
|
let tags = collect_events!(self.stream, end Tag::Heading(HeadingLevel::H1, ..));
|
2020-03-21 10:18:07 +08:00
|
|
|
Some(stringify_events(tags))
|
|
|
|
}
|
|
|
|
|
|
|
|
Some(ev) => {
|
|
|
|
self.back(ev);
|
|
|
|
None
|
|
|
|
}
|
|
|
|
|
|
|
|
None => break, // EOF, bail...
|
|
|
|
};
|
|
|
|
|
|
|
|
// Parse the rest of the part.
|
|
|
|
let numbered_chapters = self
|
|
|
|
.parse_numbered(&mut root_items, &mut root_number)
|
2020-05-21 05:32:00 +08:00
|
|
|
.with_context(|| "There was an error parsing the numbered chapters")?;
|
2020-03-21 10:18:07 +08:00
|
|
|
|
2020-05-19 00:18:14 +08:00
|
|
|
if let Some(title) = title {
|
|
|
|
parts.push(SummaryItem::PartTitle(title));
|
|
|
|
}
|
|
|
|
parts.extend(numbered_chapters);
|
2020-03-21 10:18:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
Ok(parts)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Finishes parsing a link once the `Event::Start(Tag::Link(..))` has been opened.
|
2020-03-01 00:55:45 +08:00
|
|
|
fn parse_link(&mut self, href: String) -> Link {
|
2020-08-08 15:13:22 +08:00
|
|
|
let href = href.replace("%20", " ");
|
2017-12-11 12:17:20 +08:00
|
|
|
let link_content = collect_events!(self.stream, end Tag::Link(..));
|
|
|
|
let name = stringify_events(link_content);
|
|
|
|
|
2020-03-01 00:55:45 +08:00
|
|
|
let path = if href.is_empty() {
|
|
|
|
None
|
2020-03-25 06:52:24 +08:00
|
|
|
} else {
|
2020-03-01 00:55:45 +08:00
|
|
|
Some(PathBuf::from(href))
|
|
|
|
};
|
|
|
|
|
|
|
|
Link {
|
|
|
|
name,
|
|
|
|
location: path,
|
|
|
|
number: None,
|
|
|
|
nested_items: Vec::new(),
|
2018-01-22 20:47:29 +08:00
|
|
|
}
|
2017-11-18 19:50:47 +08:00
|
|
|
}
|
|
|
|
|
2020-03-21 10:18:07 +08:00
|
|
|
/// Parse the numbered chapters.
|
|
|
|
fn parse_numbered(
|
|
|
|
&mut self,
|
|
|
|
root_items: &mut u32,
|
|
|
|
root_number: &mut SectionNumber,
|
|
|
|
) -> Result<Vec<SummaryItem>> {
|
2017-12-11 12:17:20 +08:00
|
|
|
let mut items = Vec::new();
|
2017-11-18 19:50:47 +08:00
|
|
|
|
2020-03-21 10:18:07 +08:00
|
|
|
// For the first iteration, we want to just skip any opening paragraph tags, as that just
|
|
|
|
// marks the start of the list. But after that, another opening paragraph indicates that we
|
|
|
|
// have started a new part or the suffix chapters.
|
|
|
|
let mut first = true;
|
2017-11-18 19:50:47 +08:00
|
|
|
|
2017-12-11 12:17:20 +08:00
|
|
|
loop {
|
2017-12-11 14:20:05 +08:00
|
|
|
match self.next_event() {
|
2020-03-21 10:40:40 +08:00
|
|
|
Some(ev @ Event::Start(Tag::Paragraph)) => {
|
|
|
|
if !first {
|
|
|
|
// we're starting the suffix chapters
|
|
|
|
self.back(ev);
|
|
|
|
break;
|
|
|
|
}
|
2017-12-11 14:20:05 +08:00
|
|
|
}
|
2020-03-21 10:18:07 +08:00
|
|
|
// The expectation is that pulldown cmark will terminate a paragraph before a new
|
|
|
|
// heading, so we can always count on this to return without skipping headings.
|
2021-12-25 20:46:27 +08:00
|
|
|
Some(ev @ Event::Start(Tag::Heading(HeadingLevel::H1, ..))) => {
|
2020-03-21 10:18:07 +08:00
|
|
|
// we're starting a new part
|
|
|
|
self.back(ev);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
Some(ev @ Event::Start(Tag::List(..))) => {
|
|
|
|
self.back(ev);
|
2021-08-24 15:45:06 +08:00
|
|
|
let mut bunch_of_items = self.parse_nested_numbered(root_number)?;
|
2020-03-21 10:18:07 +08:00
|
|
|
|
|
|
|
// if we've resumed after something like a rule the root sections
|
|
|
|
// will be numbered from 1. We need to manually go back and update
|
|
|
|
// them
|
|
|
|
update_section_numbers(&mut bunch_of_items, 0, *root_items);
|
|
|
|
*root_items += bunch_of_items.len() as u32;
|
|
|
|
items.extend(bunch_of_items);
|
|
|
|
}
|
2017-12-11 14:20:05 +08:00
|
|
|
Some(Event::Start(other_tag)) => {
|
|
|
|
trace!("Skipping contents of {:?}", other_tag);
|
|
|
|
|
|
|
|
// Skip over the contents of this tag
|
2018-01-19 01:21:04 +08:00
|
|
|
while let Some(event) = self.next_event() {
|
2018-03-11 22:17:38 +08:00
|
|
|
if event == Event::End(other_tag.clone()) {
|
|
|
|
break;
|
2017-12-11 14:20:05 +08:00
|
|
|
}
|
|
|
|
}
|
2017-12-11 12:17:20 +08:00
|
|
|
}
|
2019-11-12 03:25:38 +08:00
|
|
|
Some(Event::Rule) => {
|
|
|
|
items.push(SummaryItem::Separator);
|
2017-12-11 14:20:05 +08:00
|
|
|
}
|
2020-03-21 10:18:07 +08:00
|
|
|
|
|
|
|
// something else... ignore
|
|
|
|
Some(_) => {}
|
|
|
|
|
|
|
|
// EOF, bail...
|
2017-12-11 14:20:05 +08:00
|
|
|
None => {
|
2017-12-11 12:17:20 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2020-03-21 10:18:07 +08:00
|
|
|
|
2021-04-05 18:31:11 +08:00
|
|
|
// From now on, we cannot accept any new paragraph opening tags.
|
|
|
|
first = false;
|
2017-11-18 19:50:47 +08:00
|
|
|
}
|
|
|
|
|
2017-12-11 12:17:20 +08:00
|
|
|
Ok(items)
|
2017-11-18 19:50:47 +08:00
|
|
|
}
|
|
|
|
|
2020-03-21 10:18:07 +08:00
|
|
|
/// Push an event back to the tail of the stream.
|
|
|
|
fn back(&mut self, ev: Event<'a>) {
|
|
|
|
assert!(self.back.is_none());
|
|
|
|
trace!("Back: {:?}", ev);
|
|
|
|
self.back = Some(ev);
|
|
|
|
}
|
|
|
|
|
2017-12-11 14:20:05 +08:00
|
|
|
fn next_event(&mut self) -> Option<Event<'a>> {
|
2020-03-21 10:18:07 +08:00
|
|
|
let next = self.back.take().or_else(|| {
|
|
|
|
self.stream.next().map(|(ev, range)| {
|
|
|
|
self.offset = range.start;
|
|
|
|
ev
|
|
|
|
})
|
2019-11-12 03:25:38 +08:00
|
|
|
});
|
2020-03-21 10:18:07 +08:00
|
|
|
|
2017-12-11 14:20:05 +08:00
|
|
|
trace!("Next event: {:?}", next);
|
|
|
|
|
|
|
|
next
|
|
|
|
}
|
|
|
|
|
2017-12-11 12:17:20 +08:00
|
|
|
fn parse_nested_numbered(&mut self, parent: &SectionNumber) -> Result<Vec<SummaryItem>> {
|
2018-01-23 01:28:37 +08:00
|
|
|
debug!("Parsing numbered chapters at level {}", parent);
|
2017-12-11 12:17:20 +08:00
|
|
|
let mut items = Vec::new();
|
2017-11-18 19:50:47 +08:00
|
|
|
|
2017-12-11 12:17:20 +08:00
|
|
|
loop {
|
2017-12-11 14:20:05 +08:00
|
|
|
match self.next_event() {
|
|
|
|
Some(Event::Start(Tag::Item)) => {
|
|
|
|
let item = self.parse_nested_item(parent, items.len())?;
|
|
|
|
items.push(item);
|
|
|
|
}
|
2017-12-11 12:17:20 +08:00
|
|
|
Some(Event::Start(Tag::List(..))) => {
|
2020-03-17 16:11:12 +08:00
|
|
|
// Skip this tag after comment bacause it is not nested.
|
|
|
|
if items.is_empty() {
|
|
|
|
continue;
|
|
|
|
}
|
2017-12-11 14:20:05 +08:00
|
|
|
// recurse to parse the nested list
|
2017-12-11 12:17:20 +08:00
|
|
|
let (_, last_item) = get_last_link(&mut items)?;
|
|
|
|
let last_item_number = last_item
|
|
|
|
.number
|
|
|
|
.as_ref()
|
|
|
|
.expect("All numbered chapters have numbers");
|
2017-11-18 19:50:47 +08:00
|
|
|
|
2017-12-11 12:17:20 +08:00
|
|
|
let sub_items = self.parse_nested_numbered(last_item_number)?;
|
2017-11-18 19:50:47 +08:00
|
|
|
|
2017-12-11 12:17:20 +08:00
|
|
|
last_item.nested_items = sub_items;
|
2017-11-18 19:50:47 +08:00
|
|
|
}
|
2017-12-11 12:17:20 +08:00
|
|
|
Some(Event::End(Tag::List(..))) => break,
|
|
|
|
Some(_) => {}
|
|
|
|
None => break,
|
|
|
|
}
|
2017-11-18 19:50:47 +08:00
|
|
|
}
|
|
|
|
|
2017-12-11 12:17:20 +08:00
|
|
|
Ok(items)
|
2017-11-18 19:50:47 +08:00
|
|
|
}
|
|
|
|
|
2017-12-11 14:20:05 +08:00
|
|
|
fn parse_nested_item(
|
|
|
|
&mut self,
|
|
|
|
parent: &SectionNumber,
|
|
|
|
num_existing_items: usize,
|
|
|
|
) -> Result<SummaryItem> {
|
|
|
|
loop {
|
|
|
|
match self.next_event() {
|
|
|
|
Some(Event::Start(Tag::Paragraph)) => continue,
|
2019-06-12 00:26:24 +08:00
|
|
|
Some(Event::Start(Tag::Link(_type, href, _title))) => {
|
2020-03-01 00:55:45 +08:00
|
|
|
let mut link = self.parse_link(href.to_string());
|
2017-12-11 14:20:05 +08:00
|
|
|
|
|
|
|
let mut number = parent.clone();
|
|
|
|
number.0.push(num_existing_items as u32 + 1);
|
|
|
|
trace!(
|
2018-01-23 01:28:37 +08:00
|
|
|
"Found chapter: {} {} ({})",
|
2017-12-11 14:20:05 +08:00
|
|
|
number,
|
|
|
|
link.name,
|
2020-03-01 00:55:45 +08:00
|
|
|
link.location
|
|
|
|
.as_ref()
|
|
|
|
.map(|p| p.to_str().unwrap_or(""))
|
|
|
|
.unwrap_or("[draft]")
|
2017-12-11 14:20:05 +08:00
|
|
|
);
|
|
|
|
|
|
|
|
link.number = Some(number);
|
|
|
|
|
|
|
|
return Ok(SummaryItem::Link(link));
|
|
|
|
}
|
|
|
|
other => {
|
|
|
|
warn!("Expected a start of a link, actually got {:?}", other);
|
|
|
|
bail!(self.parse_error(
|
|
|
|
"The link items for nested chapters must only contain a hyperlink"
|
|
|
|
));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-11 12:17:20 +08:00
|
|
|
fn parse_error<D: Display>(&self, msg: D) -> Error {
|
|
|
|
let (line, col) = self.current_location();
|
2020-05-21 05:32:00 +08:00
|
|
|
anyhow::anyhow!(
|
|
|
|
"failed to parse SUMMARY.md line {}, column {}: {}",
|
|
|
|
line,
|
|
|
|
col,
|
|
|
|
msg
|
|
|
|
)
|
2017-11-18 19:50:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Try to parse the title line.
|
|
|
|
fn parse_title(&mut self) -> Option<String> {
|
2021-01-06 22:24:02 +08:00
|
|
|
loop {
|
|
|
|
match self.next_event() {
|
2021-12-25 20:46:27 +08:00
|
|
|
Some(Event::Start(Tag::Heading(HeadingLevel::H1, ..))) => {
|
2021-01-06 22:24:02 +08:00
|
|
|
debug!("Found a h1 in the SUMMARY");
|
2017-11-18 19:50:47 +08:00
|
|
|
|
2021-12-25 20:46:27 +08:00
|
|
|
let tags = collect_events!(self.stream, end Tag::Heading(HeadingLevel::H1, ..));
|
2021-01-06 22:24:02 +08:00
|
|
|
return Some(stringify_events(tags));
|
|
|
|
}
|
|
|
|
// Skip a HTML element such as a comment line.
|
|
|
|
Some(Event::Html(_)) => {}
|
|
|
|
// Otherwise, no title.
|
|
|
|
_ => return None,
|
2020-03-21 10:18:07 +08:00
|
|
|
}
|
2017-11-18 19:50:47 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-11 14:20:05 +08:00
|
|
|
fn update_section_numbers(sections: &mut [SummaryItem], level: usize, by: u32) {
|
|
|
|
for section in sections {
|
|
|
|
if let SummaryItem::Link(ref mut link) = *section {
|
|
|
|
if let Some(ref mut number) = link.number {
|
|
|
|
number.0[level] += by;
|
|
|
|
}
|
|
|
|
|
|
|
|
update_section_numbers(&mut link.nested_items, level, by);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-18 19:50:47 +08:00
|
|
|
/// Gets a pointer to the last `Link` in a list of `SummaryItem`s, and its
|
|
|
|
/// index.
|
|
|
|
fn get_last_link(links: &mut [SummaryItem]) -> Result<(usize, &mut Link)> {
|
|
|
|
links
|
|
|
|
.iter_mut()
|
|
|
|
.enumerate()
|
|
|
|
.filter_map(|(i, item)| item.maybe_link_mut().map(|l| (i, l)))
|
|
|
|
.rev()
|
|
|
|
.next()
|
2020-05-21 05:32:00 +08:00
|
|
|
.ok_or_else(||
|
|
|
|
anyhow::anyhow!("Unable to get last link because the list of SummaryItems doesn't contain any Links")
|
|
|
|
)
|
2017-11-18 19:50:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Removes the styling from a list of Markdown events and returns just the
|
|
|
|
/// plain text.
|
2019-05-07 04:50:34 +08:00
|
|
|
fn stringify_events(events: Vec<Event<'_>>) -> String {
|
2017-11-18 19:50:47 +08:00
|
|
|
events
|
|
|
|
.into_iter()
|
|
|
|
.filter_map(|t| match t {
|
2019-06-12 00:26:24 +08:00
|
|
|
Event::Text(text) | Event::Code(text) => Some(text.into_string()),
|
2020-07-29 09:54:49 +08:00
|
|
|
Event::SoftBreak => Some(String::from(" ")),
|
2017-11-18 19:50:47 +08:00
|
|
|
_ => None,
|
2019-05-05 22:57:43 +08:00
|
|
|
})
|
|
|
|
.collect()
|
2017-11-18 19:50:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// A section number like "1.2.3", basically just a newtype'd `Vec<u32>` with
|
|
|
|
/// a pretty `Display` impl.
|
|
|
|
#[derive(Debug, PartialEq, Clone, Default, Serialize, Deserialize)]
|
|
|
|
pub struct SectionNumber(pub Vec<u32>);
|
|
|
|
|
|
|
|
impl Display for SectionNumber {
|
2019-05-07 04:50:34 +08:00
|
|
|
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
|
2017-12-11 12:17:20 +08:00
|
|
|
if self.0.is_empty() {
|
|
|
|
write!(f, "0")
|
|
|
|
} else {
|
|
|
|
for item in &self.0 {
|
|
|
|
write!(f, "{}.", item)?;
|
|
|
|
}
|
|
|
|
Ok(())
|
2017-11-18 19:50:47 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Deref for SectionNumber {
|
|
|
|
type Target = Vec<u32>;
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
|
|
&self.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl DerefMut for SectionNumber {
|
|
|
|
fn deref_mut(&mut self) -> &mut Self::Target {
|
|
|
|
&mut self.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-11 12:17:20 +08:00
|
|
|
impl FromIterator<u32> for SectionNumber {
|
|
|
|
fn from_iter<I: IntoIterator<Item = u32>>(it: I) -> Self {
|
|
|
|
SectionNumber(it.into_iter().collect())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-18 19:50:47 +08:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn section_number_has_correct_dotted_representation() {
|
|
|
|
let inputs = vec![
|
|
|
|
(vec![0], "0."),
|
|
|
|
(vec![1, 3], "1.3."),
|
|
|
|
(vec![1, 2, 3], "1.2.3."),
|
|
|
|
];
|
|
|
|
|
|
|
|
for (input, should_be) in inputs {
|
|
|
|
let section_number = SectionNumber(input).to_string();
|
|
|
|
assert_eq!(section_number, should_be);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parse_initial_title() {
|
|
|
|
let src = "# Summary";
|
|
|
|
let should_be = String::from("Summary");
|
|
|
|
|
|
|
|
let mut parser = SummaryParser::new(src);
|
|
|
|
let got = parser.parse_title().unwrap();
|
|
|
|
|
|
|
|
assert_eq!(got, should_be);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parse_title_with_styling() {
|
|
|
|
let src = "# My **Awesome** Summary";
|
|
|
|
let should_be = String::from("My Awesome Summary");
|
|
|
|
|
|
|
|
let mut parser = SummaryParser::new(src);
|
|
|
|
let got = parser.parse_title().unwrap();
|
|
|
|
|
|
|
|
assert_eq!(got, should_be);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn convert_markdown_events_to_a_string() {
|
|
|
|
let src = "Hello *World*, `this` is some text [and a link](./path/to/link)";
|
|
|
|
let should_be = "Hello World, this is some text and a link";
|
|
|
|
|
|
|
|
let events = pulldown_cmark::Parser::new(src).collect();
|
|
|
|
let got = stringify_events(events);
|
|
|
|
|
|
|
|
assert_eq!(got, should_be);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2017-12-11 12:17:20 +08:00
|
|
|
fn parse_some_prefix_items() {
|
|
|
|
let src = "[First](./first.md)\n[Second](./second.md)\n";
|
2017-11-18 19:50:47 +08:00
|
|
|
let mut parser = SummaryParser::new(src);
|
|
|
|
|
2017-12-11 12:17:20 +08:00
|
|
|
let should_be = vec![
|
|
|
|
SummaryItem::Link(Link {
|
|
|
|
name: String::from("First"),
|
2020-03-01 00:55:45 +08:00
|
|
|
location: Some(PathBuf::from("./first.md")),
|
2017-12-11 12:17:20 +08:00
|
|
|
..Default::default()
|
|
|
|
}),
|
|
|
|
SummaryItem::Link(Link {
|
|
|
|
name: String::from("Second"),
|
2020-03-01 00:55:45 +08:00
|
|
|
location: Some(PathBuf::from("./second.md")),
|
2017-12-11 12:17:20 +08:00
|
|
|
..Default::default()
|
|
|
|
}),
|
|
|
|
];
|
2017-11-18 19:50:47 +08:00
|
|
|
|
2017-12-11 12:17:20 +08:00
|
|
|
let got = parser.parse_affix(true).unwrap();
|
|
|
|
|
|
|
|
assert_eq!(got, should_be);
|
2017-11-18 19:50:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2017-12-11 12:17:20 +08:00
|
|
|
fn parse_prefix_items_with_a_separator() {
|
|
|
|
let src = "[First](./first.md)\n\n---\n\n[Second](./second.md)\n";
|
2017-11-18 19:50:47 +08:00
|
|
|
let mut parser = SummaryParser::new(src);
|
|
|
|
|
2017-12-11 12:17:20 +08:00
|
|
|
let got = parser.parse_affix(true).unwrap();
|
2017-11-18 19:50:47 +08:00
|
|
|
|
2017-12-11 12:17:20 +08:00
|
|
|
assert_eq!(got.len(), 3);
|
|
|
|
assert_eq!(got[1], SummaryItem::Separator);
|
2017-11-18 19:50:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2017-12-11 12:17:20 +08:00
|
|
|
fn suffix_items_cannot_be_followed_by_a_list() {
|
|
|
|
let src = "[First](./first.md)\n- [Second](./second.md)\n";
|
2017-11-18 19:50:47 +08:00
|
|
|
let mut parser = SummaryParser::new(src);
|
|
|
|
|
2017-12-11 12:17:20 +08:00
|
|
|
let got = parser.parse_affix(false);
|
2017-11-18 19:50:47 +08:00
|
|
|
|
2017-12-11 12:17:20 +08:00
|
|
|
assert!(got.is_err());
|
2017-11-18 19:50:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2017-12-11 12:17:20 +08:00
|
|
|
fn parse_a_link() {
|
|
|
|
let src = "[First](./first.md)";
|
|
|
|
let should_be = Link {
|
|
|
|
name: String::from("First"),
|
2020-03-01 00:55:45 +08:00
|
|
|
location: Some(PathBuf::from("./first.md")),
|
2017-12-11 12:17:20 +08:00
|
|
|
..Default::default()
|
|
|
|
};
|
2017-11-18 19:50:47 +08:00
|
|
|
|
|
|
|
let mut parser = SummaryParser::new(src);
|
2020-03-21 10:18:07 +08:00
|
|
|
let _ = parser.stream.next(); // Discard opening paragraph
|
2017-11-18 19:50:47 +08:00
|
|
|
|
2017-12-11 12:17:20 +08:00
|
|
|
let href = match parser.stream.next() {
|
2019-11-12 03:25:38 +08:00
|
|
|
Some((Event::Start(Tag::Link(_type, href, _title)), _range)) => href.to_string(),
|
2017-12-11 12:17:20 +08:00
|
|
|
other => panic!("Unreachable, {:?}", other),
|
|
|
|
};
|
2017-11-18 19:50:47 +08:00
|
|
|
|
2020-03-01 00:55:45 +08:00
|
|
|
let got = parser.parse_link(href);
|
2017-12-11 12:17:20 +08:00
|
|
|
assert_eq!(got, should_be);
|
2017-11-18 19:50:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2017-12-11 12:17:20 +08:00
|
|
|
fn parse_a_numbered_chapter() {
|
|
|
|
let src = "- [First](./first.md)\n";
|
|
|
|
let link = Link {
|
|
|
|
name: String::from("First"),
|
2020-03-01 00:55:45 +08:00
|
|
|
location: Some(PathBuf::from("./first.md")),
|
2017-12-11 12:17:20 +08:00
|
|
|
number: Some(SectionNumber(vec![1])),
|
|
|
|
..Default::default()
|
|
|
|
};
|
|
|
|
let should_be = vec![SummaryItem::Link(link)];
|
2017-11-18 19:50:47 +08:00
|
|
|
|
2017-12-11 12:17:20 +08:00
|
|
|
let mut parser = SummaryParser::new(src);
|
2020-03-21 10:18:07 +08:00
|
|
|
let got = parser
|
|
|
|
.parse_numbered(&mut 0, &mut SectionNumber::default())
|
|
|
|
.unwrap();
|
2017-12-11 12:17:20 +08:00
|
|
|
|
|
|
|
assert_eq!(got, should_be);
|
2017-11-18 19:50:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2017-12-11 12:17:20 +08:00
|
|
|
fn parse_nested_numbered_chapters() {
|
|
|
|
let src = "- [First](./first.md)\n - [Nested](./nested.md)\n- [Second](./second.md)";
|
|
|
|
|
|
|
|
let should_be = vec![
|
|
|
|
SummaryItem::Link(Link {
|
|
|
|
name: String::from("First"),
|
2020-03-01 00:55:45 +08:00
|
|
|
location: Some(PathBuf::from("./first.md")),
|
2017-12-11 12:17:20 +08:00
|
|
|
number: Some(SectionNumber(vec![1])),
|
2018-07-24 01:45:01 +08:00
|
|
|
nested_items: vec![SummaryItem::Link(Link {
|
|
|
|
name: String::from("Nested"),
|
2020-03-01 00:55:45 +08:00
|
|
|
location: Some(PathBuf::from("./nested.md")),
|
2018-07-24 01:45:01 +08:00
|
|
|
number: Some(SectionNumber(vec![1, 1])),
|
|
|
|
nested_items: Vec::new(),
|
|
|
|
})],
|
2017-12-11 12:17:20 +08:00
|
|
|
}),
|
|
|
|
SummaryItem::Link(Link {
|
|
|
|
name: String::from("Second"),
|
2020-03-01 00:55:45 +08:00
|
|
|
location: Some(PathBuf::from("./second.md")),
|
2017-12-11 12:17:20 +08:00
|
|
|
number: Some(SectionNumber(vec![2])),
|
2020-03-17 16:11:12 +08:00
|
|
|
nested_items: Vec::new(),
|
|
|
|
}),
|
|
|
|
];
|
|
|
|
|
|
|
|
let mut parser = SummaryParser::new(src);
|
2020-03-21 10:18:07 +08:00
|
|
|
let got = parser
|
|
|
|
.parse_numbered(&mut 0, &mut SectionNumber::default())
|
|
|
|
.unwrap();
|
2020-03-17 16:11:12 +08:00
|
|
|
|
|
|
|
assert_eq!(got, should_be);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parse_numbered_chapters_separated_by_comment() {
|
|
|
|
let src = "- [First](./first.md)\n<!-- this is a comment -->\n- [Second](./second.md)";
|
|
|
|
|
|
|
|
let should_be = vec![
|
|
|
|
SummaryItem::Link(Link {
|
|
|
|
name: String::from("First"),
|
2020-03-01 00:55:45 +08:00
|
|
|
location: Some(PathBuf::from("./first.md")),
|
2020-03-17 16:11:12 +08:00
|
|
|
number: Some(SectionNumber(vec![1])),
|
|
|
|
nested_items: Vec::new(),
|
|
|
|
}),
|
|
|
|
SummaryItem::Link(Link {
|
|
|
|
name: String::from("Second"),
|
2020-03-01 00:55:45 +08:00
|
|
|
location: Some(PathBuf::from("./second.md")),
|
2020-03-17 16:11:12 +08:00
|
|
|
number: Some(SectionNumber(vec![2])),
|
2017-12-11 12:17:20 +08:00
|
|
|
nested_items: Vec::new(),
|
|
|
|
}),
|
|
|
|
];
|
|
|
|
|
2017-11-18 19:50:47 +08:00
|
|
|
let mut parser = SummaryParser::new(src);
|
2020-03-21 10:18:07 +08:00
|
|
|
let got = parser
|
|
|
|
.parse_numbered(&mut 0, &mut SectionNumber::default())
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
assert_eq!(got, should_be);
|
|
|
|
}
|
2017-11-18 19:50:47 +08:00
|
|
|
|
2020-03-21 10:18:07 +08:00
|
|
|
#[test]
|
|
|
|
fn parse_titled_parts() {
|
|
|
|
let src = "- [First](./first.md)\n- [Second](./second.md)\n\
|
|
|
|
# Title 2\n- [Third](./third.md)\n\t- [Fourth](./fourth.md)";
|
|
|
|
|
|
|
|
let should_be = vec![
|
2020-05-19 00:18:14 +08:00
|
|
|
SummaryItem::Link(Link {
|
|
|
|
name: String::from("First"),
|
|
|
|
location: Some(PathBuf::from("./first.md")),
|
|
|
|
number: Some(SectionNumber(vec![1])),
|
|
|
|
nested_items: Vec::new(),
|
|
|
|
}),
|
|
|
|
SummaryItem::Link(Link {
|
|
|
|
name: String::from("Second"),
|
|
|
|
location: Some(PathBuf::from("./second.md")),
|
|
|
|
number: Some(SectionNumber(vec![2])),
|
|
|
|
nested_items: Vec::new(),
|
|
|
|
}),
|
|
|
|
SummaryItem::PartTitle(String::from("Title 2")),
|
|
|
|
SummaryItem::Link(Link {
|
|
|
|
name: String::from("Third"),
|
|
|
|
location: Some(PathBuf::from("./third.md")),
|
|
|
|
number: Some(SectionNumber(vec![3])),
|
|
|
|
nested_items: vec![SummaryItem::Link(Link {
|
|
|
|
name: String::from("Fourth"),
|
|
|
|
location: Some(PathBuf::from("./fourth.md")),
|
|
|
|
number: Some(SectionNumber(vec![3, 1])),
|
|
|
|
nested_items: Vec::new(),
|
2020-03-21 10:18:07 +08:00
|
|
|
})],
|
2020-05-19 00:18:14 +08:00
|
|
|
}),
|
2020-03-21 10:18:07 +08:00
|
|
|
];
|
|
|
|
|
|
|
|
let mut parser = SummaryParser::new(src);
|
|
|
|
let got = parser.parse_parts().unwrap();
|
2017-11-18 19:50:47 +08:00
|
|
|
|
2017-12-11 12:17:20 +08:00
|
|
|
assert_eq!(got, should_be);
|
2017-11-18 19:50:47 +08:00
|
|
|
}
|
2017-12-11 14:20:05 +08:00
|
|
|
|
|
|
|
/// This test ensures the book will continue to pass because it breaks the
|
|
|
|
/// `SUMMARY.md` up using level 2 headers ([example]).
|
|
|
|
///
|
|
|
|
/// [example]: https://github.com/rust-lang/book/blob/2c942dc094f4ddcdc7aba7564f80782801197c99/second-edition/src/SUMMARY.md#basic-rust-literacy
|
|
|
|
#[test]
|
|
|
|
fn can_have_a_subheader_between_nested_items() {
|
|
|
|
let src = "- [First](./first.md)\n\n## Subheading\n\n- [Second](./second.md)\n";
|
|
|
|
let should_be = vec![
|
|
|
|
SummaryItem::Link(Link {
|
|
|
|
name: String::from("First"),
|
2020-03-01 00:55:45 +08:00
|
|
|
location: Some(PathBuf::from("./first.md")),
|
2017-12-11 14:20:05 +08:00
|
|
|
number: Some(SectionNumber(vec![1])),
|
|
|
|
nested_items: Vec::new(),
|
|
|
|
}),
|
|
|
|
SummaryItem::Link(Link {
|
|
|
|
name: String::from("Second"),
|
2020-03-01 00:55:45 +08:00
|
|
|
location: Some(PathBuf::from("./second.md")),
|
2017-12-11 14:20:05 +08:00
|
|
|
number: Some(SectionNumber(vec![2])),
|
|
|
|
nested_items: Vec::new(),
|
|
|
|
}),
|
|
|
|
];
|
|
|
|
|
|
|
|
let mut parser = SummaryParser::new(src);
|
2020-03-21 10:18:07 +08:00
|
|
|
let got = parser
|
|
|
|
.parse_numbered(&mut 0, &mut SectionNumber::default())
|
|
|
|
.unwrap();
|
2017-12-11 14:20:05 +08:00
|
|
|
|
|
|
|
assert_eq!(got, should_be);
|
|
|
|
}
|
2018-01-22 20:47:29 +08:00
|
|
|
|
|
|
|
#[test]
|
2020-03-01 00:55:45 +08:00
|
|
|
fn an_empty_link_location_is_a_draft_chapter() {
|
2020-03-25 06:52:24 +08:00
|
|
|
let src = "- [Empty]()\n";
|
|
|
|
let mut parser = SummaryParser::new(src);
|
2018-01-22 20:47:29 +08:00
|
|
|
|
2020-03-21 10:18:07 +08:00
|
|
|
let got = parser.parse_numbered(&mut 0, &mut SectionNumber::default());
|
2020-03-01 00:55:45 +08:00
|
|
|
let should_be = vec![SummaryItem::Link(Link {
|
|
|
|
name: String::from("Empty"),
|
|
|
|
location: None,
|
|
|
|
number: Some(SectionNumber(vec![1])),
|
|
|
|
nested_items: Vec::new(),
|
|
|
|
})];
|
|
|
|
|
|
|
|
assert!(got.is_ok());
|
|
|
|
assert_eq!(got.unwrap(), should_be);
|
2018-01-22 20:47:29 +08:00
|
|
|
}
|
2018-09-19 23:33:28 +08:00
|
|
|
|
2019-10-29 21:04:16 +08:00
|
|
|
/// Regression test for https://github.com/rust-lang/mdBook/issues/779
|
2018-09-19 23:33:28 +08:00
|
|
|
/// Ensure section numbers are correctly incremented after a horizontal separator.
|
|
|
|
#[test]
|
|
|
|
fn keep_numbering_after_separator() {
|
2018-12-04 07:10:09 +08:00
|
|
|
let src =
|
|
|
|
"- [First](./first.md)\n---\n- [Second](./second.md)\n---\n- [Third](./third.md)\n";
|
2018-09-19 23:33:28 +08:00
|
|
|
let should_be = vec![
|
|
|
|
SummaryItem::Link(Link {
|
|
|
|
name: String::from("First"),
|
2020-03-01 00:55:45 +08:00
|
|
|
location: Some(PathBuf::from("./first.md")),
|
2018-09-19 23:33:28 +08:00
|
|
|
number: Some(SectionNumber(vec![1])),
|
|
|
|
nested_items: Vec::new(),
|
|
|
|
}),
|
|
|
|
SummaryItem::Separator,
|
|
|
|
SummaryItem::Link(Link {
|
|
|
|
name: String::from("Second"),
|
2020-03-01 00:55:45 +08:00
|
|
|
location: Some(PathBuf::from("./second.md")),
|
2018-09-19 23:33:28 +08:00
|
|
|
number: Some(SectionNumber(vec![2])),
|
|
|
|
nested_items: Vec::new(),
|
|
|
|
}),
|
|
|
|
SummaryItem::Separator,
|
|
|
|
SummaryItem::Link(Link {
|
|
|
|
name: String::from("Third"),
|
2020-03-01 00:55:45 +08:00
|
|
|
location: Some(PathBuf::from("./third.md")),
|
2018-09-19 23:33:28 +08:00
|
|
|
number: Some(SectionNumber(vec![3])),
|
|
|
|
nested_items: Vec::new(),
|
|
|
|
}),
|
|
|
|
];
|
|
|
|
|
|
|
|
let mut parser = SummaryParser::new(src);
|
2020-03-21 10:18:07 +08:00
|
|
|
let got = parser
|
|
|
|
.parse_numbered(&mut 0, &mut SectionNumber::default())
|
|
|
|
.unwrap();
|
2018-09-19 23:33:28 +08:00
|
|
|
|
|
|
|
assert_eq!(got, should_be);
|
|
|
|
}
|
2020-07-29 09:54:49 +08:00
|
|
|
|
|
|
|
/// Regression test for https://github.com/rust-lang/mdBook/issues/1218
|
|
|
|
/// Ensure chapter names spread across multiple lines have spaces between all the words.
|
|
|
|
#[test]
|
|
|
|
fn add_space_for_multi_line_chapter_names() {
|
|
|
|
let src = "- [Chapter\ntitle](./chapter.md)";
|
|
|
|
let should_be = vec![SummaryItem::Link(Link {
|
|
|
|
name: String::from("Chapter title"),
|
|
|
|
location: Some(PathBuf::from("./chapter.md")),
|
|
|
|
number: Some(SectionNumber(vec![1])),
|
|
|
|
nested_items: Vec::new(),
|
|
|
|
})];
|
|
|
|
|
|
|
|
let mut parser = SummaryParser::new(src);
|
|
|
|
let got = parser
|
|
|
|
.parse_numbered(&mut 0, &mut SectionNumber::default())
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
assert_eq!(got, should_be);
|
|
|
|
}
|
2020-08-08 15:13:22 +08:00
|
|
|
|
2020-08-03 09:49:01 +08:00
|
|
|
#[test]
|
|
|
|
fn allow_space_in_link_destination() {
|
2020-08-08 15:13:22 +08:00
|
|
|
let src = "- [test1](./test%20link1.md)\n- [test2](<./test link2.md>)";
|
2020-08-03 09:49:01 +08:00
|
|
|
let should_be = vec![
|
|
|
|
SummaryItem::Link(Link {
|
|
|
|
name: String::from("test1"),
|
|
|
|
location: Some(PathBuf::from("./test link1.md")),
|
|
|
|
number: Some(SectionNumber(vec![1])),
|
|
|
|
nested_items: Vec::new(),
|
|
|
|
}),
|
|
|
|
SummaryItem::Link(Link {
|
|
|
|
name: String::from("test2"),
|
|
|
|
location: Some(PathBuf::from("./test link2.md")),
|
|
|
|
number: Some(SectionNumber(vec![2])),
|
|
|
|
nested_items: Vec::new(),
|
|
|
|
}),
|
|
|
|
];
|
|
|
|
let mut parser = SummaryParser::new(src);
|
|
|
|
let got = parser
|
|
|
|
.parse_numbered(&mut 0, &mut SectionNumber::default())
|
|
|
|
.unwrap();
|
2020-08-08 15:13:22 +08:00
|
|
|
|
2020-08-03 09:49:01 +08:00
|
|
|
assert_eq!(got, should_be);
|
|
|
|
}
|
2021-01-06 22:24:02 +08:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn skip_html_comments() {
|
|
|
|
let src = r#"<!--
|
|
|
|
# Title - En
|
|
|
|
-->
|
|
|
|
# Title - Local
|
|
|
|
|
|
|
|
<!--
|
|
|
|
[Prefix 00-01 - En](ch00-01.md)
|
|
|
|
[Prefix 00-02 - En](ch00-02.md)
|
|
|
|
-->
|
|
|
|
[Prefix 00-01 - Local](ch00-01.md)
|
|
|
|
[Prefix 00-02 - Local](ch00-02.md)
|
|
|
|
|
|
|
|
<!--
|
|
|
|
## Section Title - En
|
|
|
|
-->
|
|
|
|
## Section Title - Localized
|
|
|
|
|
|
|
|
<!--
|
|
|
|
- [Ch 01-00 - En](ch01-00.md)
|
|
|
|
- [Ch 01-01 - En](ch01-01.md)
|
|
|
|
- [Ch 01-02 - En](ch01-02.md)
|
|
|
|
-->
|
|
|
|
- [Ch 01-00 - Local](ch01-00.md)
|
|
|
|
- [Ch 01-01 - Local](ch01-01.md)
|
|
|
|
- [Ch 01-02 - Local](ch01-02.md)
|
|
|
|
|
|
|
|
<!--
|
|
|
|
- [Ch 02-00 - En](ch02-00.md)
|
|
|
|
-->
|
|
|
|
- [Ch 02-00 - Local](ch02-00.md)
|
|
|
|
|
|
|
|
<!--
|
|
|
|
[Appendix A - En](appendix-01.md)
|
|
|
|
[Appendix B - En](appendix-02.md)
|
|
|
|
-->`
|
|
|
|
[Appendix A - Local](appendix-01.md)
|
|
|
|
[Appendix B - Local](appendix-02.md)
|
|
|
|
"#;
|
|
|
|
|
|
|
|
let mut parser = SummaryParser::new(src);
|
|
|
|
|
|
|
|
// ---- Title ----
|
|
|
|
let title = parser.parse_title();
|
|
|
|
assert_eq!(title, Some(String::from("Title - Local")));
|
|
|
|
|
|
|
|
// ---- Prefix Chapters ----
|
|
|
|
|
|
|
|
let new_affix_item = |name, location| {
|
|
|
|
SummaryItem::Link(Link {
|
|
|
|
name: String::from(name),
|
|
|
|
location: Some(PathBuf::from(location)),
|
|
|
|
..Default::default()
|
|
|
|
})
|
|
|
|
};
|
|
|
|
|
|
|
|
let should_be = vec![
|
|
|
|
new_affix_item("Prefix 00-01 - Local", "ch00-01.md"),
|
|
|
|
new_affix_item("Prefix 00-02 - Local", "ch00-02.md"),
|
|
|
|
];
|
|
|
|
|
|
|
|
let got = parser.parse_affix(true).unwrap();
|
|
|
|
assert_eq!(got, should_be);
|
|
|
|
|
|
|
|
// ---- Numbered Chapters ----
|
|
|
|
|
|
|
|
let new_numbered_item = |name, location, numbers: &[u32], nested_items| {
|
|
|
|
SummaryItem::Link(Link {
|
|
|
|
name: String::from(name),
|
|
|
|
location: Some(PathBuf::from(location)),
|
|
|
|
number: Some(SectionNumber(numbers.to_vec())),
|
|
|
|
nested_items,
|
|
|
|
})
|
|
|
|
};
|
|
|
|
|
|
|
|
let ch01_nested = vec![
|
|
|
|
new_numbered_item("Ch 01-01 - Local", "ch01-01.md", &[1, 1], vec![]),
|
|
|
|
new_numbered_item("Ch 01-02 - Local", "ch01-02.md", &[1, 2], vec![]),
|
|
|
|
];
|
|
|
|
|
|
|
|
let should_be = vec![
|
|
|
|
new_numbered_item("Ch 01-00 - Local", "ch01-00.md", &[1], ch01_nested),
|
|
|
|
new_numbered_item("Ch 02-00 - Local", "ch02-00.md", &[2], vec![]),
|
|
|
|
];
|
|
|
|
let got = parser.parse_parts().unwrap();
|
|
|
|
assert_eq!(got, should_be);
|
|
|
|
|
|
|
|
// ---- Suffix Chapters ----
|
|
|
|
|
|
|
|
let should_be = vec![
|
|
|
|
new_affix_item("Appendix A - Local", "appendix-01.md"),
|
|
|
|
new_affix_item("Appendix B - Local", "appendix-02.md"),
|
|
|
|
];
|
|
|
|
|
|
|
|
let got = parser.parse_affix(false).unwrap();
|
|
|
|
assert_eq!(got, should_be);
|
|
|
|
}
|
2018-01-22 20:47:29 +08:00
|
|
|
}
|