From b272b177b646eef6bfafaf229bc3658c51d827ca Mon Sep 17 00:00:00 2001 From: Michael Bryan Date: Sun, 21 Jan 2018 22:21:19 +0800 Subject: [PATCH] Looks like we forgot to recursively apply replace_all() in #532 --- src/book/book.rs | 79 +++- src/preprocess/links.rs | 537 +++++++++++++------------ tests/dummy_book/src/SUMMARY.md | 9 +- tests/dummy_book/src/first/includes.md | 3 + tests/rendered_output.rs | 20 +- 5 files changed, 384 insertions(+), 264 deletions(-) create mode 100644 tests/dummy_book/src/first/includes.md diff --git a/src/book/book.rs b/src/book/book.rs index 7a03eae7d1..c531a891c7 100644 --- a/src/book/book.rs +++ b/src/book/book.rs @@ -8,7 +8,6 @@ use super::summary::{parse_summary, Link, SectionNumber, Summary, SummaryItem}; use config::BuildConfig; use errors::*; - /// Load a book into memory from its `src/` directory. pub fn load_book>(src_dir: P, cfg: &BuildConfig) -> Result { let src_dir = src_dir.as_ref(); @@ -60,14 +59,19 @@ fn create_missing(src_dir: &Path, summary: &Summary) -> Result<()> { Ok(()) } - /// A dumb tree structure representing a book. /// -/// For the moment a book is just a collection of `BookItems`. +/// For the moment a book is just a collection of `BookItems` which are +/// accessible by either iterating (immutably) over the book with [`iter()`], or +/// recursively applying a closure to each section to mutate the chapters, using +/// [`for_each_mut()`]. +/// +/// [`iter()`]: #method.iter +/// [`for_each_mut()`]: #method.for_each_mut #[derive(Debug, Clone, Default, PartialEq, Serialize, Deserialize)] pub struct Book { /// The sections in this book. - pub sections: Vec, + sections: Vec, } impl Book { @@ -82,6 +86,35 @@ impl Book { items: self.sections.iter().collect(), } } + + /// Recursively apply a closure to each item in the book, allowing you to + /// mutate them. + /// + /// # Note + /// + /// Unlike the `iter()` method, this requires a closure instead of returning + /// an iterator. This is because using iterators can possibly allow you + /// to have iterator invalidation errors. + pub fn for_each_mut(&mut self, mut func: F) + where + F: FnMut(&mut BookItem), + { + for_each_mut(&mut func, &mut self.sections); + } +} + +pub fn for_each_mut<'a, F, I>(func: &mut F, items: I) +where + F: FnMut(&mut BookItem), + I: IntoIterator, +{ + for item in items { + if let &mut BookItem::Chapter(ref mut ch) = item { + for_each_mut(func, &mut ch.sub_items); + } + + func(item); + } } /// Enum representing any type of item which can be added to a book. @@ -224,7 +257,6 @@ impl Display for Chapter { } } - #[cfg(test)] mod tests { use super::*; @@ -266,7 +298,6 @@ And here is some \ .write_all("Hello World!".as_bytes()) .unwrap(); - let mut second = Link::new("Nested Chapter 1", &second_path); second.number = Some(SectionNumber(vec![1, 2])); @@ -391,7 +422,6 @@ And here is some \ ], }; - let got: Vec<_> = book.iter().collect(); assert_eq!(got.len(), 5); @@ -411,4 +441,39 @@ And here is some \ assert_eq!(chapter_names, should_be); } + + #[test] + fn for_each_mut_visits_all_items() { + let mut book = Book { + sections: vec![ + BookItem::Chapter(Chapter { + name: String::from("Chapter 1"), + content: String::from(DUMMY_SRC), + number: None, + path: PathBuf::from("Chapter_1/index.md"), + sub_items: vec![ + BookItem::Chapter(Chapter::new( + "Hello World", + String::new(), + "Chapter_1/hello.md", + )), + BookItem::Separator, + BookItem::Chapter(Chapter::new( + "Goodbye World", + String::new(), + "Chapter_1/goodbye.md", + )), + ], + }), + BookItem::Separator, + ], + }; + + let num_items = book.iter().count(); + let mut visited = 0; + + book.for_each_mut(|_| visited += 1); + + assert_eq!(visited, num_items); + } } diff --git a/src/preprocess/links.rs b/src/preprocess/links.rs index c3cab8bff6..b46711da78 100644 --- a/src/preprocess/links.rs +++ b/src/preprocess/links.rs @@ -1,4 +1,4 @@ -use std::ops::{Range, RangeFrom, RangeTo, RangeFull}; +use std::ops::{Range, RangeFrom, RangeFull, RangeTo}; use std::path::{Path, PathBuf}; use regex::{CaptureMatches, Captures, Regex}; use utils::fs::file_to_string; @@ -26,38 +26,50 @@ impl Preprocessor for LinkPreprocessor { fn run(&self, ctx: &PreprocessorContext, book: &mut Book) -> Result<()> { let src_dir = ctx.root.join(&ctx.config.book.src); - for section in &mut book.sections { - match *section { - BookItem::Chapter(ref mut ch) => { - let base = ch.path.parent() - .map(|dir| src_dir.join(dir)) - .ok_or_else(|| String::from("Invalid bookitem path!"))?; - let content = replace_all(&ch.content, base)?; - ch.content = content - } - _ => {} + book.for_each_mut(|section: &mut BookItem| { + if let BookItem::Chapter(ref mut ch) = *section { + println!("Checking {}", ch); + let base = ch.path + .parent() + .map(|dir| src_dir.join(dir)) + .expect("All book items have a parent"); + + let content = replace_all(&ch.content, base); + ch.content = content; } - } + }); Ok(()) } } -fn replace_all>(s: &str, path: P) -> Result { +fn replace_all>(s: &str, path: P) -> String { // When replacing one thing in a string by something with a different length, // the indices after that will not correspond, // we therefore have to store the difference to correct this + let path = path.as_ref(); let mut previous_end_index = 0; let mut replaced = String::new(); for playpen in find_links(s) { + println!("{} {:?}", path.display(), playpen); replaced.push_str(&s[previous_end_index..playpen.start_index]); - replaced.push_str(&playpen.render_with_path(&path)?); - previous_end_index = playpen.end_index; + + match playpen.render_with_path(&path) { + Ok(new_content) => { + replaced.push_str(&new_content); + } + Err(e) => { + error!("Error updating \"{}\", {}", playpen.link_text, e); + // This should make sure we include the raw `{{# ... }}` snippet + // in the page content if there are any errors. + previous_end_index = playpen.start_index; + } + } } replaced.push_str(&s[previous_end_index..]); - Ok(replaced) + replaced } #[derive(PartialEq, Debug, Clone)] @@ -76,18 +88,20 @@ fn parse_include_path(path: &str) -> LinkType<'static> { let start = parts.next().and_then(|s| s.parse::().ok()); let end = parts.next().and_then(|s| s.parse::().ok()); match start { - Some(start) => { - match end { - Some(end) => LinkType::IncludeRange(path, Range{ start: start, end: end}), - None => LinkType::IncludeRangeFrom(path, RangeFrom{ start: start }), - } - } - None => { - match end { - Some(end) => LinkType::IncludeRangeTo(path, RangeTo{ end: end }), - None => LinkType::IncludeRangeFull(path, RangeFull), - } - } + Some(start) => match end { + Some(end) => LinkType::IncludeRange( + path, + Range { + start: start, + end: end, + }, + ), + None => LinkType::IncludeRangeFrom(path, RangeFrom { start: start }), + }, + None => match end { + Some(end) => LinkType::IncludeRangeTo(path, RangeTo { end: end }), + None => LinkType::IncludeRangeFull(path, RangeFull), + }, } } @@ -113,20 +127,18 @@ impl<'a> Link<'a> { _ => None, } } - (Some(mat), None, None) if mat.as_str().starts_with(ESCAPE_CHAR) => Some( - LinkType::Escaped, - ), + (Some(mat), None, None) if mat.as_str().starts_with(ESCAPE_CHAR) => { + Some(LinkType::Escaped) + } _ => None, }; link_type.and_then(|lnk| { - cap.get(0).map(|mat| { - Link { - start_index: mat.start(), - end_index: mat.end(), - link: lnk, - link_text: mat.as_str(), - } + cap.get(0).map(|mat| Link { + start_index: mat.start(), + end_index: mat.end(), + link: lnk, + link_text: mat.as_str(), }) }) } @@ -136,37 +148,20 @@ impl<'a> Link<'a> { match self.link { // omit the escape char LinkType::Escaped => Ok((&self.link_text[1..]).to_owned()), - LinkType::IncludeRange(ref pat, ref range) => { - file_to_string(base.join(pat)) - .map(|s| take_lines(&s, range.clone())) - .chain_err(|| { - format!("Could not read file for link {}", self.link_text) - }) - } - LinkType::IncludeRangeFrom(ref pat, ref range) => { - file_to_string(base.join(pat)) - .map(|s| take_lines(&s, range.clone())) - .chain_err(|| { - format!("Could not read file for link {}", self.link_text) - }) - } - LinkType::IncludeRangeTo(ref pat, ref range) => { - file_to_string(base.join(pat)) - .map(|s| take_lines(&s, range.clone())) - .chain_err(|| { - format!("Could not read file for link {}", self.link_text) - }) - } - LinkType::IncludeRangeFull(ref pat, _) => { - file_to_string(base.join(pat)) - .chain_err(|| { - format!("Could not read file for link {}", self.link_text) - }) - } + LinkType::IncludeRange(ref pat, ref range) => file_to_string(base.join(pat)) + .map(|s| take_lines(&s, range.clone())) + .chain_err(|| format!("Could not read file for link {}", self.link_text)), + LinkType::IncludeRangeFrom(ref pat, ref range) => file_to_string(base.join(pat)) + .map(|s| take_lines(&s, range.clone())) + .chain_err(|| format!("Could not read file for link {}", self.link_text)), + LinkType::IncludeRangeTo(ref pat, ref range) => file_to_string(base.join(pat)) + .map(|s| take_lines(&s, range.clone())) + .chain_err(|| format!("Could not read file for link {}", self.link_text)), + LinkType::IncludeRangeFull(ref pat, _) => file_to_string(base.join(pat)) + .chain_err(|| format!("Could not read file for link {}", self.link_text)), LinkType::Playpen(ref pat, ref attrs) => { - let contents = file_to_string(base.join(pat)).chain_err(|| { - format!("Could not read file for link {}", self.link_text) - })?; + let contents = file_to_string(base.join(pat)) + .chain_err(|| format!("Could not read file for link {}", self.link_text))?; let ftype = if !attrs.is_empty() { "rust," } else { "rust" }; Ok(format!( "```{}{}\n{}\n```\n", @@ -210,200 +205,242 @@ fn find_links(contents: &str) -> LinkIter { LinkIter(RE.captures_iter(contents)) } -// --------------------------------------------------------------------------------- -// Tests -// +#[cfg(test)] +mod tests { + use super::*; -#[test] -fn test_find_links_no_link() { - let s = "Some random text without link..."; - assert!(find_links(s).collect::>() == vec![]); -} + #[test] + fn test_find_links_no_link() { + let s = "Some random text without link..."; + assert!(find_links(s).collect::>() == vec![]); + } -#[test] -fn test_find_links_partial_link() { - let s = "Some random text with {{#playpen..."; - assert!(find_links(s).collect::>() == vec![]); - let s = "Some random text with {{#include..."; - assert!(find_links(s).collect::>() == vec![]); - let s = "Some random text with \\{{#include..."; - assert!(find_links(s).collect::>() == vec![]); -} + #[test] + fn test_find_links_partial_link() { + let s = "Some random text with {{#playpen..."; + assert!(find_links(s).collect::>() == vec![]); + let s = "Some random text with {{#include..."; + assert!(find_links(s).collect::>() == vec![]); + let s = "Some random text with \\{{#include..."; + assert!(find_links(s).collect::>() == vec![]); + } -#[test] -fn test_find_links_empty_link() { - let s = "Some random text with {{#playpen}} and {{#playpen }} {{}} {{#}}..."; - assert!(find_links(s).collect::>() == vec![]); -} + #[test] + fn test_find_links_empty_link() { + let s = "Some random text with {{#playpen}} and {{#playpen }} {{}} {{#}}..."; + assert!(find_links(s).collect::>() == vec![]); + } -#[test] -fn test_find_links_unknown_link_type() { - let s = "Some random text with {{#playpenz ar.rs}} and {{#incn}} {{baz}} {{#bar}}..."; - assert!(find_links(s).collect::>() == vec![]); -} + #[test] + fn test_find_links_unknown_link_type() { + let s = "Some random text with {{#playpenz ar.rs}} and {{#incn}} {{baz}} {{#bar}}..."; + assert!(find_links(s).collect::>() == vec![]); + } -#[test] -fn test_find_links_simple_link() { - let s = "Some random text with {{#playpen file.rs}} and {{#playpen test.rs }}..."; - - let res = find_links(s).collect::>(); - println!("\nOUTPUT: {:?}\n", res); - - assert_eq!(res, - vec![Link { - start_index: 22, - end_index: 42, - link: LinkType::Playpen(PathBuf::from("file.rs"), vec![]), - link_text: "{{#playpen file.rs}}", - }, - Link { - start_index: 47, - end_index: 68, - link: LinkType::Playpen(PathBuf::from("test.rs"), vec![]), - link_text: "{{#playpen test.rs }}", - }]); -} + #[test] + fn test_find_links_simple_link() { + let s = "Some random text with {{#playpen file.rs}} and {{#playpen test.rs }}..."; -#[test] -fn test_find_links_with_range() { - let s = "Some random text with {{#include file.rs:10:20}}..."; - let res = find_links(s).collect::>(); - println!("\nOUTPUT: {:?}\n", res); - assert_eq!( - res, - vec![ - Link { - start_index: 22, - end_index: 48, - link: LinkType::IncludeRange(PathBuf::from("file.rs"), 10..20), - link_text: "{{#include file.rs:10:20}}", - }, - ] - ); -} + let res = find_links(s).collect::>(); + println!("\nOUTPUT: {:?}\n", res); -#[test] -fn test_find_links_with_from_range() { - let s = "Some random text with {{#include file.rs:10:}}..."; - let res = find_links(s).collect::>(); - println!("\nOUTPUT: {:?}\n", res); - assert_eq!( - res, - vec![ - Link { - start_index: 22, - end_index: 46, - link: LinkType::IncludeRangeFrom(PathBuf::from("file.rs"), 10..), - link_text: "{{#include file.rs:10:}}", - }, - ] - ); -} + assert_eq!( + res, + vec![ + Link { + start_index: 22, + end_index: 42, + link: LinkType::Playpen(PathBuf::from("file.rs"), vec![]), + link_text: "{{#playpen file.rs}}", + }, + Link { + start_index: 47, + end_index: 68, + link: LinkType::Playpen(PathBuf::from("test.rs"), vec![]), + link_text: "{{#playpen test.rs }}", + }, + ] + ); + } -#[test] -fn test_find_links_with_to_range() { - let s = "Some random text with {{#include file.rs::20}}..."; - let res = find_links(s).collect::>(); - println!("\nOUTPUT: {:?}\n", res); - assert_eq!( - res, - vec![ - Link { - start_index: 22, - end_index: 46, - link: LinkType::IncludeRangeTo(PathBuf::from("file.rs"), ..20), - link_text: "{{#include file.rs::20}}", - }, - ] - ); -} + #[test] + fn test_find_links_with_range() { + let s = "Some random text with {{#include file.rs:10:20}}..."; + let res = find_links(s).collect::>(); + println!("\nOUTPUT: {:?}\n", res); + assert_eq!( + res, + vec![ + Link { + start_index: 22, + end_index: 48, + link: LinkType::IncludeRange(PathBuf::from("file.rs"), 10..20), + link_text: "{{#include file.rs:10:20}}", + }, + ] + ); + } -#[test] -fn test_find_links_with_full_range() { - let s = "Some random text with {{#include file.rs::}}..."; - let res = find_links(s).collect::>(); - println!("\nOUTPUT: {:?}\n", res); - assert_eq!( - res, - vec![ - Link { - start_index: 22, - end_index: 44, - link: LinkType::IncludeRangeFull(PathBuf::from("file.rs"), ..), - link_text: "{{#include file.rs::}}", - }, - ] - ); -} + #[test] + fn test_find_links_with_from_range() { + let s = "Some random text with {{#include file.rs:10:}}..."; + let res = find_links(s).collect::>(); + println!("\nOUTPUT: {:?}\n", res); + assert_eq!( + res, + vec![ + Link { + start_index: 22, + end_index: 46, + link: LinkType::IncludeRangeFrom(PathBuf::from("file.rs"), 10..), + link_text: "{{#include file.rs:10:}}", + }, + ] + ); + } -#[test] -fn test_find_links_escaped_link() { - let s = "Some random text with escaped playpen \\{{#playpen file.rs editable}} ..."; + #[test] + fn test_find_links_with_to_range() { + let s = "Some random text with {{#include file.rs::20}}..."; + let res = find_links(s).collect::>(); + println!("\nOUTPUT: {:?}\n", res); + assert_eq!( + res, + vec![ + Link { + start_index: 22, + end_index: 46, + link: LinkType::IncludeRangeTo(PathBuf::from("file.rs"), ..20), + link_text: "{{#include file.rs::20}}", + }, + ] + ); + } - let res = find_links(s).collect::>(); - println!("\nOUTPUT: {:?}\n", res); + #[test] + fn test_find_links_with_full_range() { + let s = "Some random text with {{#include file.rs::}}..."; + let res = find_links(s).collect::>(); + println!("\nOUTPUT: {:?}\n", res); + assert_eq!( + res, + vec![ + Link { + start_index: 22, + end_index: 44, + link: LinkType::IncludeRangeFull(PathBuf::from("file.rs"), ..), + link_text: "{{#include file.rs::}}", + }, + ] + ); + } - assert_eq!(res, - vec![Link { - start_index: 38, - end_index: 68, - link: LinkType::Escaped, - link_text: "\\{{#playpen file.rs editable}}", - }]); -} + #[test] + fn test_find_links_with_no_range_specified() { + let s = "Some random text with {{#include file.rs}}..."; + let res = find_links(s).collect::>(); + println!("\nOUTPUT: {:?}\n", res); + assert_eq!( + res, + vec![ + Link { + start_index: 22, + end_index: 42, + link: LinkType::IncludeRangeFull(PathBuf::from("file.rs"), ..), + link_text: "{{#include file.rs}}", + }, + ] + ); + } -#[test] -fn test_find_playpens_with_properties() { - let s = "Some random text with escaped playpen {{#playpen file.rs editable }} and some more\n \ - text {{#playpen my.rs editable no_run should_panic}} ..."; - - let res = find_links(s).collect::>(); - println!("\nOUTPUT: {:?}\n", res); - assert_eq!(res, - vec![Link { - start_index: 38, - end_index: 68, - link: LinkType::Playpen(PathBuf::from("file.rs"), vec!["editable"]), - link_text: "{{#playpen file.rs editable }}", - }, - Link { - start_index: 89, - end_index: 136, - link: LinkType::Playpen(PathBuf::from("my.rs"), - vec!["editable", "no_run", "should_panic"]), - link_text: "{{#playpen my.rs editable no_run should_panic}}", - }]); -} + #[test] + fn test_find_links_escaped_link() { + let s = "Some random text with escaped playpen \\{{#playpen file.rs editable}} ..."; + + let res = find_links(s).collect::>(); + println!("\nOUTPUT: {:?}\n", res); + + assert_eq!( + res, + vec![ + Link { + start_index: 38, + end_index: 68, + link: LinkType::Escaped, + link_text: "\\{{#playpen file.rs editable}}", + }, + ] + ); + } + + #[test] + fn test_find_playpens_with_properties() { + let s = "Some random text with escaped playpen {{#playpen file.rs editable }} and some \ + more\n text {{#playpen my.rs editable no_run should_panic}} ..."; + + let res = find_links(s).collect::>(); + println!("\nOUTPUT: {:?}\n", res); + assert_eq!( + res, + vec![ + Link { + start_index: 38, + end_index: 68, + link: LinkType::Playpen(PathBuf::from("file.rs"), vec!["editable"]), + link_text: "{{#playpen file.rs editable }}", + }, + Link { + start_index: 89, + end_index: 136, + link: LinkType::Playpen( + PathBuf::from("my.rs"), + vec!["editable", "no_run", "should_panic"], + ), + link_text: "{{#playpen my.rs editable no_run should_panic}}", + }, + ] + ); + } + + #[test] + fn test_find_all_link_types() { + let s = "Some random text with escaped playpen {{#include file.rs}} and \\{{#contents are \ + insignifficant in escaped link}} some more\n text {{#playpen my.rs editable \ + no_run should_panic}} ..."; + + let res = find_links(s).collect::>(); + println!("\nOUTPUT: {:?}\n", res); + assert_eq!(res.len(), 3); + assert_eq!( + res[0], + Link { + start_index: 38, + end_index: 58, + link: LinkType::IncludeRangeFull(PathBuf::from("file.rs"), ..), + link_text: "{{#include file.rs}}", + } + ); + assert_eq!( + res[1], + Link { + start_index: 63, + end_index: 112, + link: LinkType::Escaped, + link_text: "\\{{#contents are insignifficant in escaped link}}", + } + ); + assert_eq!( + res[2], + Link { + start_index: 130, + end_index: 177, + link: LinkType::Playpen( + PathBuf::from("my.rs"), + vec!["editable", "no_run", "should_panic"] + ), + link_text: "{{#playpen my.rs editable no_run should_panic}}", + } + ); + } -#[test] -fn test_find_all_link_types() { - let s = "Some random text with escaped playpen {{#include file.rs}} and \\{{#contents are \ - insignifficant in escaped link}} some more\n text {{#playpen my.rs editable no_run \ - should_panic}} ..."; - - let res = find_links(s).collect::>(); - println!("\nOUTPUT: {:?}\n", res); - assert_eq!(res.len(), 3); - assert_eq!(res[0], - Link { - start_index: 38, - end_index: 58, - link: LinkType::IncludeRangeFull(PathBuf::from("file.rs"), ..), - link_text: "{{#include file.rs}}", - }); - assert_eq!(res[1], - Link { - start_index: 63, - end_index: 112, - link: LinkType::Escaped, - link_text: "\\{{#contents are insignifficant in escaped link}}", - }); - assert_eq!(res[2], - Link { - start_index: 130, - end_index: 177, - link: LinkType::Playpen(PathBuf::from("my.rs"), - vec!["editable", "no_run", "should_panic"]), - link_text: "{{#playpen my.rs editable no_run should_panic}}", - }); } diff --git a/tests/dummy_book/src/SUMMARY.md b/tests/dummy_book/src/SUMMARY.md index 350b64402f..920885e625 100644 --- a/tests/dummy_book/src/SUMMARY.md +++ b/tests/dummy_book/src/SUMMARY.md @@ -2,10 +2,11 @@ [Introduction](intro.md) -- [First Chapter](./first/index.md) - - [Nested Chapter](./first/nested.md) -- [Second Chapter](./second.md) +- [First Chapter](first/index.md) + - [Nested Chapter](first/nested.md) + - [Includes](first/includes.md) +- [Second Chapter](second.md) --- -[Conclusion](./conclusion.md) +[Conclusion](conclusion.md) diff --git a/tests/dummy_book/src/first/includes.md b/tests/dummy_book/src/first/includes.md new file mode 100644 index 0000000000..a5a2fef1f3 --- /dev/null +++ b/tests/dummy_book/src/first/includes.md @@ -0,0 +1,3 @@ +# Includes + +{{#include ../SUMMARY.md::}} \ No newline at end of file diff --git a/tests/rendered_output.rs b/tests/rendered_output.rs index fe256fe455..9f5faa659e 100644 --- a/tests/rendered_output.rs +++ b/tests/rendered_output.rs @@ -29,7 +29,7 @@ const TOC_TOP_LEVEL: &[&'static str] = &[ "Conclusion", "Introduction", ]; -const TOC_SECOND_LEVEL: &[&'static str] = &["1.1. Nested Chapter"]; +const TOC_SECOND_LEVEL: &[&'static str] = &["1.1. Nested Chapter", "1.2. Includes"]; /// Make sure you can load the dummy book and build it without panicking. #[test] @@ -49,7 +49,8 @@ fn by_default_mdbook_generates_rendered_content_in_the_book_directory() { md.build().unwrap(); assert!(temp.path().join("book").exists()); - assert!(temp.path().join("book").join("index.html").exists()); + let index_file = md.build_dir_for("html").join("index.html"); + assert!(index_file.exists()); } #[test] @@ -281,7 +282,7 @@ fn create_missing_file_with_config() { /// This makes sure you can include a Rust file with `{{#playpen example.rs}}`. /// Specification is in `book-example/src/format/rust.md` #[test] -fn able_to_include_rust_files_in_chapters() { +fn able_to_include_playpen_files_in_chapters() { let temp = DummyBook::new().build().unwrap(); let md = MDBook::load(temp.path()).unwrap(); md.build().unwrap(); @@ -295,6 +296,19 @@ fn able_to_include_rust_files_in_chapters() { assert_contains_strings(second, playpen_strings); } +/// This makes sure you can include a Rust file with `{{#include ../SUMMARY.md}}`. +#[test] +fn able_to_include_files_in_chapters() { + let temp = DummyBook::new().build().unwrap(); + let md = MDBook::load(temp.path()).unwrap(); + md.build().unwrap(); + + let includes = temp.path().join("book/first/includes.html"); + + let summary_strings = &["

Summary

", ">First Chapter"]; + assert_contains_strings(includes, summary_strings); +} + #[test] fn example_book_can_build() { let example_book_dir = dummy_book::new_copy_of_example_book().unwrap();