4 releases (2 breaking)

0.3.0 Jul 30, 2024
0.2.0 Sep 21, 2023
0.1.1 Mar 10, 2023
0.1.0 Jan 10, 2023

#1065 in Parser implementations

Download history 32/week @ 2024-05-20 45/week @ 2024-05-27 30/week @ 2024-06-03 21/week @ 2024-06-10 17/week @ 2024-06-17 41/week @ 2024-06-24 18/week @ 2024-07-08 24/week @ 2024-07-15 25/week @ 2024-07-22 213/week @ 2024-07-29 20/week @ 2024-08-05 110/week @ 2024-08-12 38/week @ 2024-08-19 62/week @ 2024-08-26 34/week @ 2024-09-02

246 downloads per month
Used in 3 crates

MIT/Apache

21KB
296 lines

pulldown-cmark-frontmatter

pulldown-cmark-frontmatter forbids unsafe code crate version Live Build Status HTML Coverage Report for main branch Documentation

This crate was written by someone unaffiliated with the pulldown-cmark crate.

This crate makes it easy to parse frontmatter contained within Markdown documents when using the pulldown-cmark Markdown parser.

Unlike many other frontmatter styles, this crate enforces a basic document format:

  • Optional top-level (h1) heading
  • Optional code block
  • Remaining Markdown document

By utilizing a code block instead of other markers, most Markdown editing software can more intelligently handle syntax highlighting, errors, and more.

The FrontmatterExtractor type will detect and return a plain-text representation of a top-level heading, if it's the first element in the document. The heading will still be returned when iterating over the pulldown_cmark::Events.

After the optional top-level heading, if a code block is encountered, it will be returned as Frontmatter::code_block. Unlike the heading, the frontmatter code block will not appear in the iterated Events.

This repository includes frontmatter-example.md which both the HTML rendering example and the extractor example use.

HTML Rendering Example

This example shows how to use this crate with pulldown-cmark's html module. It is included in the repository at examples/html.rs.

// This example renders the example Markdown to html using
// `pulldown_cmark::html`, while also extracting the frontmatter from
// Markdown.
let mut extractor = FrontmatterExtractor::new(pulldown_cmark::Parser::new(include_str!(
    "../frontmatter-example.md"
)));

// The only difference from using the FrontmatterExtractor and the regular
// pulldown_cmark::Parser is that you must pass a mutable reference to the
// extractor to be able to read the Frontmatter it extracts.
let mut rendered = String::new();
pulldown_cmark::html::push_html(&mut rendered, &mut extractor);
assert_eq!(rendered, include_str!("../frontmatter-example.html"));

let frontmatter = extractor.frontmatter.expect("frontmatter not detected");
assert_eq!(
    frontmatter.title.expect("title not detected"),
    "Frontmatter Example Document"
);
let code_block = frontmatter.code_block.expect("code block not detected");
assert_eq!(code_block.language.as_deref(), Some("toml"));
let attrs: ExampleAttributes = toml::from_str(&code_block.source).expect("invalid toml");
assert_eq!(attrs.author, "https://fosstodon.org/@ecton");

The repository includes the rendered html output to see what the produced HTML looks like.

Extractor Example

This example extracts the frontmatter from a Markdown document without parsing the entire document. It is included in the repository at examples/extractor.rs.

// This example extracts the frontmatter from the Markdown,
// `FrontmatterExtractor::extract()` which stops parsing the Markdown
// document after the frontmatter extraction is complete.
let extractor = FrontmatterExtractor::from_markdown(include_str!("../frontmatter-example.md"));
let frontmatter = extractor.extract().expect("frontmatter not detected");
assert_eq!(
    frontmatter.title.expect("title not detected"),
    "Frontmatter Example Document"
);
let code_block = frontmatter.code_block.expect("code block not detected");
assert_eq!(code_block.language.as_deref(), Some("toml"));
let attrs: ExampleAttributes = toml::from_str(&code_block.source).expect("invalid toml");
assert_eq!(attrs.author, "https://fosstodon.org/@ecton");

Open-source Licenses

This project, like all projects from Khonsu Labs, is open-source. This repository is available under the MIT License or the Apache License 2.0.

To learn more about contributing, please see CONTRIBUTING.md.

Dependencies

~1MB
~20K SLoC