diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 9b3e119..5fbaae0 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -26,33 +26,14 @@ jobs: override: true components: clippy + # Do *not* use `--all-features` here, as the optional dependencies take a + # long time to build, and will be tested in the "examples" job anyway - name: Run tests run: cargo test --workspace --all-targets - name: Check Clippy run: cargo clippy --workspace --all-targets -- -D warnings - # Optional features (i.e. web framework integrations) take a long time to - # build and rarely break. Speed up CI by checking them separately. - all-features: - name: All features - runs-on: ubuntu-latest - - steps: - - - name: Check out repository - uses: actions/checkout@v2 - - - name: Install Rust - uses: actions-rs/toolchain@v1 - with: - toolchain: nightly - profile: minimal - override: true - - - name: Check build - run: cargo check --workspace --all-features --all-targets - # Please keep this in sync with `publish-docs.yml` documentation: name: Documentation @@ -76,6 +57,25 @@ jobs: - name: Build documentation run: cd docs && make -j$(nproc) + examples: + name: Examples + runs-on: ubuntu-latest + + steps: + + - name: Check out repository + uses: actions/checkout@v2 + + - name: Install Rust + uses: actions-rs/toolchain@v1 + with: + toolchain: nightly + profile: minimal + override: true + + - name: Doctest + run: cd doctest && cargo test + rustfmt: name: Rustfmt runs-on: ubuntu-latest diff --git a/.gitignore b/.gitignore index 8cdc97e..e1508c1 100644 --- a/.gitignore +++ b/.gitignore @@ -1,3 +1,5 @@ target /Cargo.lock docs/site +doctest/Cargo.lock +doctest/lib.rs diff --git a/Cargo.toml b/Cargo.toml index eec1379..1631727 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -6,4 +6,5 @@ members = [ ] exclude = [ "docs", + "doctest", ] diff --git a/docs/content/control-structures.md b/docs/content/control-structures.md index d66af3a..48dcf16 100644 --- a/docs/content/control-structures.md +++ b/docs/content/control-structures.md @@ -12,6 +12,7 @@ enum Princess { Celestia, Luna, Cadance, TwilightSparkle } let user = Princess::Celestia; +# let _ = maud:: html! { @if user == Princess::Luna { h1 { "Super secret woona to-do list" } @@ -26,12 +27,14 @@ html! { p { "Nothing to see here; move along." } } } +# ; ``` `@if let` is supported as well: ```rust let user = Some("Pinkie Pie"); +# let _ = maud:: html! { p { "Hello, " @@ -43,6 +46,7 @@ html! { "!" } } +# ; ``` ## Looping with `@for` @@ -51,6 +55,7 @@ Use `@for .. in ..` to loop over the elements of an iterator. ```rust let names = ["Applejack", "Rarity", "Fluttershy"]; +# let _ = maud:: html! { p { "My favorite ponies are:" } ol { @@ -59,6 +64,7 @@ html! { } } } +# ; ``` ## Declaring variables with `@let` @@ -67,6 +73,7 @@ Declare a new variable within a template using `@let`. This can be useful when w ```rust let names = ["Applejack", "Rarity", "Fluttershy"]; +# let _ = maud:: html! { @for name in &names { @let first_letter = name.chars().next().unwrap(); @@ -79,6 +86,7 @@ html! { } } } +# ; ``` ## Matching with `@match` @@ -90,6 +98,7 @@ enum Princess { Celestia, Luna, Cadance, TwilightSparkle } let user = Princess::Celestia; +# let _ = maud:: html! { @match user { Princess::Luna => { @@ -106,4 +115,5 @@ html! { _ => p { "Nothing to see here; move along." } } } +# ; ``` diff --git a/docs/content/elements-attributes.md b/docs/content/elements-attributes.md index 9c2518c..d650e10 100644 --- a/docs/content/elements-attributes.md +++ b/docs/content/elements-attributes.md @@ -5,6 +5,7 @@ Write an element using curly braces: ```rust +# let _ = maud:: html! { h1 { "Poem" } p { @@ -12,6 +13,7 @@ html! { " you are a rock." } } +# ; ``` Before version 0.18, Maud allowed the curly braces to be omitted. This syntax was [removed][#137] and now causes an error instead. @@ -23,6 +25,7 @@ Before version 0.18, Maud allowed the curly braces to be omitted. This syntax wa Terminate a void element using a semicolon: ```rust +# let _ = maud:: html! { link rel="stylesheet" href="poetry.css"; p { @@ -35,6 +38,7 @@ html! { "Rock." } } +# ; ``` The result will be rendered with HTML syntax – `<br>` not `<br />`. @@ -48,12 +52,14 @@ Maud also supports ending a void element with a slash: `br /`. This syntax is [d Maud also supports elements and attributes with hyphens in them. This includes [custom elements], [data attributes], and [ARIA annotations]. ```rust +# let _ = maud:: html! { article data-index="12345" { h1 { "My blog" } tag-cloud { "pinkie pie pony cute" } } } +# ; ``` [custom elements]: https://developer.mozilla.org/en-US/docs/Web/Web_Components/Using_custom_elements @@ -65,6 +71,7 @@ html! { Add attributes using the syntax: `attr="value"`. You can attach any number of attributes to an element. The values must be quoted: they are parsed as string literals. ```rust +# let _ = maud:: html! { ul { li { @@ -79,6 +86,7 @@ html! { } } } +# ; ``` ## Empty attributes: `checked` @@ -86,6 +94,7 @@ html! { Declare an empty attribute by omitting the value. ```rust +# let _ = maud:: html! { form { input type="checkbox" name="cupcakes" checked; @@ -93,6 +102,7 @@ html! { label for="cupcakes" { "Do you like cupcakes?" } } } +# ; ``` Before version 0.22.2, Maud required a `?` suffix on empty attributes: `checked?`. This is no longer necessary ([#238]), but still supported for backward compatibility. @@ -104,17 +114,21 @@ Before version 0.22.2, Maud required a `?` suffix on empty attributes: `checked? Add classes and IDs to an element using `.foo` and `#bar` syntax. You can chain multiple classes and IDs together, and mix and match them with other attributes: ```rust +# let _ = maud:: html! { input#cannon.big.scary.bright-red type="button" value="Launch Party Cannon"; } +# ; ``` The classes and IDs can be quoted. This is useful for names with numbers or symbols which otherwise wouldn't parse: ```rust +# let _ = maud:: html! { div."col-sm-2" { "Bootstrap column!" } } +# ; ``` ## Implicit `div` elements @@ -122,10 +136,12 @@ html! { If the element name is omitted, but there is a class or ID, then it is assumed to be a `div`. ```rust +# let _ = maud:: html! { #main { "Main content!" .tip { "Storing food in a refrigerator can make it 20% cooler." } } } +# ; ``` diff --git a/docs/content/getting-started.md b/docs/content/getting-started.md index b24fdd3..cdb19bd 100644 --- a/docs/content/getting-started.md +++ b/docs/content/getting-started.md @@ -37,7 +37,7 @@ fn main() { Run this program with `cargo run`, and you should get the following: -``` +```html <p>Hi, Lyra!</p> ``` diff --git a/docs/content/index.md b/docs/content/index.md index 414f4d1..38e9302 100644 --- a/docs/content/index.md +++ b/docs/content/index.md @@ -3,6 +3,7 @@ # A macro for writing HTML ```rust +# let _ = maud:: html! { h1 { "Hello, world!" } p.intro { @@ -11,6 +12,7 @@ html! { " template language." } } +# ; ``` Maud is an HTML [template engine] for Rust. It's implemented as a macro, `html!`, which compiles your markup to specialized Rust code. This unique approach makes Maud templates blazing fast, super type-safe, and easy to deploy. diff --git a/docs/content/partials.md b/docs/content/partials.md index 520e13c..2b3b9e2 100644 --- a/docs/content/partials.md +++ b/docs/content/partials.md @@ -44,6 +44,8 @@ Using the `page` function will return the markup for the whole page. Here's an example: ```rust +# use maud::{html, Markup}; +# fn page(title: &str, greeting_box: Markup) -> Markup { greeting_box } page("Hello!", html! { div { "Greetings, Maud." } }); diff --git a/docs/content/render-trait.md b/docs/content/render-trait.md index b90211c..65604e7 100644 --- a/docs/content/render-trait.md +++ b/docs/content/render-trait.md @@ -11,7 +11,7 @@ Below are some examples of using `Render`. Feel free to use these snippets in yo When writing a web page, it can be annoying to write `link rel="stylesheet"` over and over again. This example provides a shorthand for linking to CSS stylesheets. ```rust -use maud::{Markup, Render}; +use maud::{html, Markup, Render}; /// Links to a CSS stylesheet at the given path. struct Css(&'static str); @@ -32,8 +32,9 @@ When debugging an application, it can be useful to see its internal state. But t To avoid extra allocation, we override the `.render_to()` method instead of `.render()`. This doesn't do any escaping by default, so we wrap the output in an `Escaper` as well. ```rust -use maud::{Render, Escaper}; +use maud::{Escaper, html, Render}; use std::fmt; +use std::fmt::Write as _; /// Renders the given value using its `Debug` implementation. struct Debug<T: fmt::Debug>(T); @@ -53,9 +54,7 @@ impl<T: fmt::Debug> Render for Debug<T> { We also use the [`ammonia`][ammonia] library, which sanitizes the resulting markup. ```rust -extern crate ammonia; -extern crate pulldown_cmark; - +use ammonia; use maud::{Markup, PreEscaped, Render}; use pulldown_cmark::{Parser, html}; diff --git a/docs/content/splices-toggles.md b/docs/content/splices-toggles.md index ba4942a..ea396c3 100644 --- a/docs/content/splices-toggles.md +++ b/docs/content/splices-toggles.md @@ -7,6 +7,7 @@ Use `(foo)` syntax to insert the value of `foo` at runtime. Any HTML special cha ```rust let best_pony = "Pinkie Pie"; let numbers = [1, 2, 3, 4]; +# let _ = maud:: html! { p { "Hi, " (best_pony) "!" } p { @@ -14,11 +15,19 @@ html! { "and the first one is " (numbers[0]) } } +# ; ``` Arbitrary Rust code can be included in a splice by using a [block](https://doc.rust-lang.org/reference.html#block-expressions). This can be helpful for complex expressions that would be difficult to read otherwise. ```rust +# struct Foo; +# impl Foo { fn time(self) -> Bar { Bar } } +# struct Bar; +# impl Bar { fn format(self, _: &str) -> &str { "" } } +# fn something_convertible_to_foo() -> Option<Foo> { Some(Foo) } +# fn test() -> Option<()> { +# let _ = maud:: html! { p { ({ @@ -27,6 +36,9 @@ html! { }) } } +# ; +# Some(()) +# } ``` ### Splices in attributes @@ -35,22 +47,26 @@ Splices work in attributes as well. ```rust let secret_message = "Surprise!"; +# let _ = maud:: html! { p title=(secret_message) { "Nothing to see here, move along." } } +# ; ``` To concatenate multiple values within an attribute, wrap the whole thing in braces. This syntax is useful for building URLs. ```rust const GITHUB: &'static str = "https://github.com"; +# let _ = maud:: html! { a href={ (GITHUB) "/lambda-fairy/maud" } { "Fork me on GitHub" } } +# ; ``` ### Splices in classes and IDs @@ -60,11 +76,13 @@ Splices can also be used in classes and IDs. ```rust let name = "rarity"; let severity = "critical"; +# let _ = maud:: html! { aside#(name) { p.{ "color-" (severity) } { "This is the worst! Possible! Thing!" } } } +# ; ``` ### What can be spliced? @@ -76,10 +94,12 @@ To change this behavior for some type, you can implement the [`Render`][Render] ```rust use maud::PreEscaped; let post = "<p>Pre-escaped</p>"; +# let _ = maud:: html! { h1 { "My super duper blog post" } (PreEscaped(post)) } +# ; ``` [Display]: http://doc.rust-lang.org/std/fmt/trait.Display.html @@ -94,6 +114,7 @@ This works on empty attributes: ```rust let allow_editing = true; +# let _ = maud:: html! { p contenteditable[allow_editing] { "Edit me, I " @@ -101,13 +122,16 @@ html! { " you." } } +# ; ``` And classes: ```rust let cuteness = 95; +# let _ = maud:: html! { p.cute[cuteness > 50] { "Squee!" } } +# ; ``` diff --git a/docs/content/text-escaping.md b/docs/content/text-escaping.md index 600f962..ee05c67 100644 --- a/docs/content/text-escaping.md +++ b/docs/content/text-escaping.md @@ -5,9 +5,11 @@ Literal strings use the same syntax as Rust. Wrap them in double quotes, and use a backslash for escapes. ```rust +# let _ = maud:: html! { "Oatmeal, are you crazy?" } +# ; ``` ## Raw strings @@ -15,6 +17,7 @@ html! { If the string is long, or contains many special characters, then it may be worth using [raw strings] instead: ```rust +# let _ = maud:: html! { pre { r#" @@ -27,6 +30,7 @@ html! { "# } } +# ; ``` [raw strings]: https://doc.rust-lang.org/reference/tokens.html#raw-string-literals @@ -37,10 +41,12 @@ By default, HTML special characters are escaped automatically. Wrap the string i ```rust use maud::PreEscaped; +# let _ = maud:: html! { "<script>alert(\"XSS\")</script>" // <script>... (PreEscaped("<script>alert(\"XSS\")</script>")) // <script>... } +# ; ``` ## The `DOCTYPE` constant @@ -49,7 +55,9 @@ If you want to add a `<!DOCTYPE html>` declaration to your page, you may use the ```rust use maud::DOCTYPE; +# let _ = maud:: html! { (DOCTYPE) // <!DOCTYPE html> } +# ; ``` diff --git a/docs/content/web-frameworks.md b/docs/content/web-frameworks.md index 1b1e0f3..1c06cf7 100644 --- a/docs/content/web-frameworks.md +++ b/docs/content/web-frameworks.md @@ -20,23 +20,28 @@ maud = { version = "*", features = ["actix-web"] } Actix request handlers can use a `Markup` that implements the `actix_web::Responder` trait. -```rust +```rust,no_run +use actix_web::{get, App, HttpServer, Result as AwResult}; use maud::{html, Markup}; -use actix_web::{web, App, HttpServer}; +use std::io; -fn index(params: web::Path<(String, u32)>) -> Markup { - html! { - h1 { "Hello " (params.0) " with id " (params.1) "!" } - } +#[get("/")] +async fn index() -> AwResult<Markup> { + Ok(html! { + html { + body { + h1 { "Hello World!" } + } + } + }) } -fn main() -> std::io::Result<()> { - HttpServer::new(|| { - App::new() - .route("/user/{name}/{id}", web::get().to(index)) - }) - .bind("127.0.0.1:8080")? - .run() +#[actix_web::main] +async fn main() -> io::Result<()> { + HttpServer::new(|| App::new().service(index)) + .bind(("127.0.0.1", 8080))? + .run() + .await } ``` @@ -53,7 +58,7 @@ maud = { version = "*", features = ["iron"] } With this feature enabled, you can then build a `Response` from a `Markup` object directly. Here's an example application using Iron and Maud: -```rust +```rust,no_run use iron::prelude::*; use iron::status; use maud::html; @@ -86,7 +91,9 @@ maud = { version = "*", features = ["rocket"] } This adds a `Responder` implementation for the `Markup` type, so you can return the result directly: -```rust +```rust,no_run +#![feature(decl_macro)] + use maud::{html, Markup}; use rocket::{get, routes}; use std::borrow::Cow; @@ -108,7 +115,7 @@ fn main() { Unlike with the other frameworks, Rouille doesn't need any extra features at all! Calling `Response::html` on the rendered `Markup` will Just Work®. -```rust +```rust,no_run use maud::html; use rouille::{Response, router}; @@ -116,10 +123,10 @@ fn main() { rouille::start_server("localhost:8000", move |request| { router!(request, (GET) (/{name: String}) => { - html! { + Response::html(html! { h1 { "Hello, " (name) "!" } p { "Nice to meet you!" } - } + }) }, _ => Response::empty_404() ) diff --git a/docs/src/main.rs b/docs/src/main.rs index ffa67a9..e7f03f7 100644 --- a/docs/src/main.rs +++ b/docs/src/main.rs @@ -7,6 +7,7 @@ use std::fs::{self, File}; use std::io::{self, BufReader}; use std::mem; use std::path::Path; +use std::str::{self, Utf8Error}; use std::string::FromUtf8Error; use syntect::highlighting::{Color, ThemeSet}; use syntect::html::highlighted_html_for_string; @@ -102,13 +103,13 @@ fn load_page<'a>( arena: &'a Arena<AstNode<'a>>, options: &ComrakOptions, path: impl AsRef<Path>, -) -> io::Result<Page<'a>> { +) -> Result<Page<'a>, Box<dyn Error>> { let page = load_page_raw(arena, options, path)?; lower_headings(page.content); - rewrite_md_links(page.content) - .map_err(|err| io::Error::new(io::ErrorKind::InvalidData, err))?; - highlight_code(page.content).map_err(|err| io::Error::new(io::ErrorKind::InvalidData, err))?; + rewrite_md_links(page.content)?; + strip_hidden_code(page.content)?; + highlight_code(page.content)?; Ok(page) } @@ -173,7 +174,32 @@ fn rewrite_md_links<'a>(root: &'a AstNode<'a>) -> Result<(), FromUtf8Error> { Ok(()) } -fn highlight_code<'a>(root: &'a AstNode<'a>) -> Result<(), FromUtf8Error> { +fn strip_hidden_code<'a>(root: &'a AstNode<'a>) -> Result<(), Box<dyn Error>> { + for node in root.descendants() { + let mut data = node.data.borrow_mut(); + if let NodeValue::CodeBlock(NodeCodeBlock { info, literal, .. }) = &mut data.value { + let info = parse_code_block_info(info)?; + if !info.contains(&"rust") { + continue; + } + *literal = strip_hidden_code_inner(str::from_utf8(literal)?).into_bytes(); + } + } + Ok(()) +} + +fn strip_hidden_code_inner(literal: &str) -> String { + let lines = literal + .split("\n") + .filter(|line| { + let line = line.trim(); + line != "#" && !line.starts_with("# ") + }) + .collect::<Vec<_>>(); + lines.join("\n") +} + +fn highlight_code<'a>(root: &'a AstNode<'a>) -> Result<(), Box<dyn Error>> { let ss = SyntaxSet::load_defaults_newlines(); let ts = ThemeSet::load_defaults(); let mut theme = ts.themes["InspiredGitHub"].clone(); @@ -186,9 +212,11 @@ fn highlight_code<'a>(root: &'a AstNode<'a>) -> Result<(), FromUtf8Error> { for node in root.descendants() { let mut data = node.data.borrow_mut(); if let NodeValue::CodeBlock(NodeCodeBlock { info, literal, .. }) = &mut data.value { - let info = String::from_utf8(mem::replace(info, Vec::new()))?; - let syntax = ss - .find_syntax_by_token(&info) + let info = parse_code_block_info(info)?; + let syntax = info + .into_iter() + .filter_map(|token| ss.find_syntax_by_token(&token)) + .next() .unwrap_or_else(|| ss.find_syntax_plain_text()); let mut literal = String::from_utf8(mem::replace(literal, Vec::new()))?; if !literal.ends_with('\n') { @@ -204,6 +232,10 @@ fn highlight_code<'a>(root: &'a AstNode<'a>) -> Result<(), FromUtf8Error> { Ok(()) } +fn parse_code_block_info(info: &[u8]) -> Result<Vec<&str>, Utf8Error> { + str::from_utf8(info).map(|info| info.split(",").map(str::trim).collect()) +} + fn comrak_options() -> ComrakOptions { let mut options = ComrakOptions::default(); options.extension.header_ids = Some("".to_string()); diff --git a/doctest/Cargo.toml b/doctest/Cargo.toml new file mode 100644 index 0000000..078fbe7 --- /dev/null +++ b/doctest/Cargo.toml @@ -0,0 +1,17 @@ +[package] +name = "doctest" +version = "0.1.0" +authors = ["Chris Wong <lambda.fairy@gmail.com>"] +edition = "2018" + +[dependencies] +actix-web = "3" +ammonia = "3" +iron = "0.6" +maud = { path = "../maud", features = ["actix-web", "iron", "rocket"] } +pulldown-cmark = "0.8" +rocket = "0.4" +rouille = "3" + +[lib] +path = "lib.rs" diff --git a/doctest/README.md b/doctest/README.md new file mode 100644 index 0000000..052cda8 --- /dev/null +++ b/doctest/README.md @@ -0,0 +1,4 @@ +This is a placeholder package that imports the entire [book] as a doc comment. +This allows for testing the book's code snippets via `cargo test`. + +[book]: ../docs/content diff --git a/doctest/build.rs b/doctest/build.rs new file mode 100644 index 0000000..7002dae --- /dev/null +++ b/doctest/build.rs @@ -0,0 +1,31 @@ +use std::ffi::OsStr; +use std::fmt::Write as _; +use std::fs; + +fn main() { + const DOCS_DIR: &str = "../docs/content"; + + // Rebuild if a chapter is added or removed + println!("cargo:rerun-if-changed={}", DOCS_DIR); + + let mut buffer = r#"// Automatically @generated – do not edit + +#![feature(extended_key_value_attributes)] +"#.to_string(); + + for entry in fs::read_dir(DOCS_DIR).unwrap() { + let entry = entry.unwrap(); + assert!(entry.file_type().unwrap().is_file()); + + let path = entry.path(); + assert_eq!(path.extension(), Some(OsStr::new("md"))); + + let path_str = path.to_str().unwrap(); + let slug_str = path.file_stem().unwrap().to_str().unwrap().replace("-", "_"); + + writeln!(buffer, r#"#[doc = include_str!("{}")]"#, path_str).unwrap(); + writeln!(buffer, r#"mod {} {{ }}"#, slug_str).unwrap(); + } + + fs::write("lib.rs", buffer).unwrap(); +} diff --git a/maud/Cargo.toml b/maud/Cargo.toml index 6d5fc3d..3fa34fa 100644 --- a/maud/Cargo.toml +++ b/maud/Cargo.toml @@ -30,7 +30,3 @@ trybuild = { version = "1.0.33", features = ["diff"] } [package.metadata.docs.rs] all-features = true - -[[example]] -name = "actix" -required-features = ["actix-web"] diff --git a/maud/examples/actix.rs b/maud/examples/actix.rs deleted file mode 100644 index a4832b4..0000000 --- a/maud/examples/actix.rs +++ /dev/null @@ -1,25 +0,0 @@ -// do not use this line in your application -use actix_web_dep as actix_web; - -use actix_web::{get, App, HttpServer, Result as AwResult}; -use maud::{html, Markup}; -use std::io; - -#[get("/")] -async fn index() -> AwResult<Markup> { - Ok(html! { - html { - body { - h1 { "Hello World!" } - } - } - }) -} - -#[actix_web::main] -async fn main() -> io::Result<()> { - HttpServer::new(|| App::new().service(index)) - .bind(("127.0.0.1", 8080))? - .run() - .await -}