This adds a bunch of tests to better exercise the HTML rendering and to be able to track any changes in its behavior. This includes a new `check_all_main_files` to more conveniently check the HTML content of every chapter in a book.
227 lines
6.2 KiB
Rust
227 lines
6.2 KiB
Rust
//! Tests for HTML rendering.
|
|
//!
|
|
//! Note that markdown-specific rendering tests are in the `markdown` module.
|
|
|
|
use crate::prelude::*;
|
|
|
|
// Checks that edit-url-template works.
|
|
#[test]
|
|
fn edit_url_template() {
|
|
BookTest::from_dir("rendering/edit_url_template").check_file_contains(
|
|
"book/index.html",
|
|
"<a href=\"https://github.com/rust-lang/mdBook/edit/master/guide/src/README.md\" \
|
|
title=\"Suggest an edit\" aria-label=\"Suggest an edit\" rel=\"edit\">",
|
|
);
|
|
}
|
|
|
|
// Checks that an alternate `src` setting works with the edit url template.
|
|
#[test]
|
|
fn edit_url_template_explicit_src() {
|
|
BookTest::from_dir("rendering/edit_url_template_explicit_src").check_file_contains(
|
|
"book/index.html",
|
|
"<a href=\"https://github.com/rust-lang/mdBook/edit/master/guide/src2/README.md\" \
|
|
title=\"Suggest an edit\" aria-label=\"Suggest an edit\" rel=\"edit\">",
|
|
);
|
|
}
|
|
|
|
// Checks that index.html is generated correctly, even when the first few
|
|
// chapters are drafts.
|
|
#[test]
|
|
fn first_chapter_is_copied_as_index_even_if_not_first_elem() {
|
|
BookTest::from_dir("rendering/first_chapter_is_copied_as_index_even_if_not_first_elem")
|
|
// These two files should be equal.
|
|
.check_main_file(
|
|
"book/chapter_1.html",
|
|
str![[
|
|
r##"<h1 id="chapter-1"><a class="header" href="#chapter-1">Chapter 1</a></h1>"##
|
|
]],
|
|
)
|
|
.check_main_file(
|
|
"book/index.html",
|
|
str![[
|
|
r##"<h1 id="chapter-1"><a class="header" href="#chapter-1">Chapter 1</a></h1>"##
|
|
]],
|
|
);
|
|
}
|
|
|
|
// Fontawesome `<i>` tag support.
|
|
#[test]
|
|
fn fontawesome() {
|
|
BookTest::from_dir("rendering/fontawesome").check_all_main_files();
|
|
}
|
|
|
|
// Tests the rendering when setting the default rust edition.
|
|
#[test]
|
|
fn default_rust_edition() {
|
|
BookTest::from_dir("rendering/default_rust_edition").check_all_main_files();
|
|
}
|
|
|
|
// Tests the rendering for editable code blocks.
|
|
#[test]
|
|
fn editable_rust_block() {
|
|
BookTest::from_dir("rendering/editable_rust_block").check_all_main_files();
|
|
}
|
|
|
|
// Tests for custom hide lines.
|
|
#[test]
|
|
fn hidelines() {
|
|
BookTest::from_dir("rendering/hidelines").check_all_main_files();
|
|
}
|
|
|
|
// Tests for code blocks of basic rust code.
|
|
#[test]
|
|
fn language_rust_playground() {
|
|
fn expect(input: &str, info: &str, expected: impl snapbox::IntoData) {
|
|
BookTest::init(|_| {})
|
|
.change_file("book.toml", "output.html.playground.editable = true")
|
|
.change_file("src/chapter_1.md", &format!("```rust {info}\n{input}\n```"))
|
|
.check_main_file("book/chapter_1.html", expected);
|
|
}
|
|
// No-main should be wrapped in `fn main` boring lines.
|
|
expect(
|
|
"x()",
|
|
"",
|
|
str![[r#"
|
|
<pre><pre class="playground"><code class="language-rust"><span class="boring">#![allow(unused)]
|
|
</span><span class="boring">fn main() {
|
|
</span>x()
|
|
<span class="boring">}</span></code></pre></pre>
|
|
"#]],
|
|
);
|
|
// `fn main` should not be wrapped, not boring.
|
|
expect(
|
|
"fn main() {}",
|
|
"",
|
|
str![[
|
|
r#"<pre><pre class="playground"><code class="language-rust">fn main() {}</code></pre></pre>"#
|
|
]],
|
|
);
|
|
// Lines starting with `#` are boring.
|
|
expect(
|
|
"let s = \"foo\n # bar\n\";",
|
|
"editable",
|
|
str![[r#"
|
|
<pre><pre class="playground"><code class="language-rust editable">let s = "foo
|
|
<span class="boring"> bar
|
|
</span>";</code></pre></pre>
|
|
"#]],
|
|
);
|
|
// `##` is not boring and is used as an escape.
|
|
expect(
|
|
"let s = \"foo\n ## bar\n\";",
|
|
"editable",
|
|
str![[r#"
|
|
<pre><pre class="playground"><code class="language-rust editable">let s = "foo
|
|
# bar
|
|
";</code></pre></pre>
|
|
"#]],
|
|
);
|
|
// `#` on a line by itself is boring.
|
|
expect(
|
|
"let s = \"foo\n # bar\n#\n\";",
|
|
"editable",
|
|
str![[r#"
|
|
<pre><pre class="playground"><code class="language-rust editable">let s = "foo
|
|
<span class="boring"> bar
|
|
</span><span class="boring">
|
|
</span>";</code></pre></pre>
|
|
"#]],
|
|
);
|
|
// `#` must be followed by a space to be boring.
|
|
expect(
|
|
"#x;",
|
|
"",
|
|
str![[r#"
|
|
<pre><pre class="playground"><code class="language-rust"><span class="boring">#![allow(unused)]
|
|
</span><span class="boring">fn main() {
|
|
</span>#x;
|
|
<span class="boring">}</span></code></pre></pre>
|
|
"#]],
|
|
);
|
|
|
|
// Other classes like "ignore" should not change things, and the class is
|
|
// included in the code tag.
|
|
expect(
|
|
"let s = \"foo\n # bar\n\";",
|
|
"ignore",
|
|
str![[r#"
|
|
<pre><code class="language-rust ignore">let s = "foo
|
|
<span class="boring"> bar
|
|
</span>";</code></pre>
|
|
"#]],
|
|
);
|
|
// Inner attributes and normal attributes are not boring.
|
|
expect(
|
|
"#![no_std]\nlet s = \"foo\";\n #[some_attr]",
|
|
"editable",
|
|
str![[r#"
|
|
<pre><pre class="playground"><code class="language-rust editable">#![no_std]
|
|
let s = "foo";
|
|
#[some_attr]</code></pre></pre>
|
|
"#]],
|
|
);
|
|
}
|
|
|
|
// Rust code block in a list.
|
|
#[test]
|
|
fn code_block_in_list() {
|
|
BookTest::init(|_| {})
|
|
.change_file(
|
|
"src/chapter_1.md",
|
|
r#"- inside list
|
|
|
|
```rust
|
|
fn foo() {
|
|
let x = 1;
|
|
}
|
|
```
|
|
"#,
|
|
)
|
|
.check_main_file(
|
|
"book/chapter_1.html",
|
|
str![[r#"
|
|
<ul>
|
|
<li>
|
|
<p>inside list</p>
|
|
<pre><pre class="playground"><code class="language-rust"><span class="boring">#![allow(unused)]
|
|
</span><span class="boring">fn main() {
|
|
</span>fn foo() {
|
|
let x = 1;
|
|
}
|
|
<span class="boring">}</span></code></pre></pre>
|
|
</li>
|
|
</ul>
|
|
"#]],
|
|
);
|
|
}
|
|
|
|
// Checks the rendering of links added to headers.
|
|
#[test]
|
|
fn header_links() {
|
|
BookTest::from_dir("rendering/header_links").check_all_main_files();
|
|
}
|
|
|
|
// A corrupted HTML end tag.
|
|
#[test]
|
|
fn busted_end_tag() {
|
|
BookTest::init(|_| {})
|
|
.change_file("src/chapter_1.md", "<div>x<span>foo</span/>y</div>")
|
|
.run("build", |cmd| {
|
|
cmd.expect_stderr(str![[r#"
|
|
INFO Book building has started
|
|
INFO Running the html backend
|
|
INFO HTML book written to `[ROOT]/book`
|
|
|
|
"#]]);
|
|
})
|
|
.check_main_file(
|
|
"book/chapter_1.html",
|
|
str!["<div>x<span>foo</span/>y</div>"],
|
|
);
|
|
}
|
|
|
|
// Various html blocks.
|
|
#[test]
|
|
fn html_blocks() {
|
|
BookTest::from_dir("rendering/html_blocks").check_all_main_files();
|
|
}
|