diff --git a/Cargo.lock b/Cargo.lock index a52c576..ff79ebb 100755 --- a/Cargo.lock +++ b/Cargo.lock @@ -187,6 +187,27 @@ dependencies = [ "windows-targets", ] +[[package]] +name = "base64" +version = "0.22.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "72b3254f16251a8381aa12e40e3c4d2f0199f8c6508fbecb9d91f575e0fbb8c6" + +[[package]] +name = "bincode" +version = "1.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b1f45e9417d87227c7a56d22e471c6206462cba514c7590c09aff4cf6d1ddcad" +dependencies = [ + "serde", +] + +[[package]] +name = "bitflags" +version = "1.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a" + [[package]] name = "blog-thing" version = "0.1.0" @@ -196,7 +217,9 @@ dependencies = [ "chrono", "clap", "markdown", + "syntect", "tokio", + "two-face", ] [[package]] @@ -292,6 +315,40 @@ version = "0.8.7" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "773648b94d0e5d620f64f280777445740e61fe701025087ec8b57f45c791888b" +[[package]] +name = "crc32fast" +version = "1.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a97769d94ddab943e4510d138150169a2758b5ef3eb191a9ee688de3e23ef7b3" +dependencies = [ + "cfg-if", +] + +[[package]] +name = "deranged" +version = "0.3.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b42b6fa04a440b495c8b04d0e71b707c585f83cb9cb28cf8cd0d976c315e31b4" +dependencies = [ + "powerfmt", +] + +[[package]] +name = "equivalent" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5443807d6dff69373d433ab9ef5378ad8df50ca6298caf15de6e52e24aaf54d5" + +[[package]] +name = "flate2" +version = "1.0.34" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a1b589b4dc103969ad3cf85c950899926ec64300a1a46d76c03a6072957036f0" +dependencies = [ + "crc32fast", + "miniz_oxide", +] + [[package]] name = "fnv" version = "1.0.7" @@ -346,6 +403,12 @@ version = "0.31.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "32085ea23f3234fc7846555e85283ba4de91e21016dc0455a16286d87a292d64" +[[package]] +name = "hashbrown" +version = "0.15.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e087f84d4f86bf4b218b927129862374b72199ae7d8657835f1e89000eea4fb" + [[package]] name = "heck" version = "0.5.0" @@ -462,6 +525,16 @@ dependencies = [ "cc", ] +[[package]] +name = "indexmap" +version = "2.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "707907fe3c25f5424cce2cb7e1cbcafee6bdbe735ca90ef77c29e84591e5b9da" +dependencies = [ + "equivalent", + "hashbrown", +] + [[package]] name = "is_terminal_polyfill" version = "1.70.1" @@ -489,6 +562,12 @@ version = "0.2.159" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "561d97a539a36e26a9a5fad1ea11a3039a67714694aaa379433e580854bc3dc5" +[[package]] +name = "linked-hash-map" +version = "0.5.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0717cef1bc8b636c6e1c1bbdefc09e6322da8a9321966e8928ef80d20f7f770f" + [[package]] name = "log" version = "0.4.22" @@ -543,6 +622,12 @@ dependencies = [ "windows-sys", ] +[[package]] +name = "num-conv" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "51d515d32fb182ee37cda2ccdcb92950d6a3c2893aa280e540671c2cd0f3b1d9" + [[package]] name = "num-traits" version = "0.2.19" @@ -570,6 +655,28 @@ dependencies = [ "portable-atomic", ] +[[package]] +name = "onig" +version = "6.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8c4b31c8722ad9171c6d77d3557db078cab2bd50afcc9d09c8b315c59df8ca4f" +dependencies = [ + "bitflags", + "libc", + "once_cell", + "onig_sys", +] + +[[package]] +name = "onig_sys" +version = "69.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7b829e3d7e9cc74c7e315ee8edb185bf4190da5acde74afd7fc59c35b1f086e7" +dependencies = [ + "cc", + "pkg-config", +] + [[package]] name = "percent-encoding" version = "2.3.1" @@ -588,12 +695,37 @@ version = "0.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184" +[[package]] +name = "pkg-config" +version = "0.3.31" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "953ec861398dccce10c670dfeaf3ec4911ca479e9c02154b3a215178c5f566f2" + +[[package]] +name = "plist" +version = "1.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "42cf17e9a1800f5f396bc67d193dc9411b59012a5876445ef450d449881e1016" +dependencies = [ + "base64", + "indexmap", + "quick-xml", + "serde", + "time", +] + [[package]] name = "portable-atomic" version = "1.9.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "cc9c68a3f6da06753e9335d63e27f6b9754dd1920d941135b7ea8224f141adb2" +[[package]] +name = "powerfmt" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "439ee305def115ba05938db6eb1644ff94165c5ab5e9420d1c1bcedbba909391" + [[package]] name = "proc-macro2" version = "1.0.86" @@ -603,6 +735,15 @@ dependencies = [ "unicode-ident", ] +[[package]] +name = "quick-xml" +version = "0.32.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1d3a6e5838b60e0e8fa7a43f22ade549a37d61f8bdbe636d0d7816191de969c2" +dependencies = [ + "memchr", +] + [[package]] name = "quote" version = "1.0.37" @@ -659,6 +800,15 @@ version = "1.0.18" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "f3cb5ba0dc43242ce17de99c180e96db90b235b8a9fdc9543c96d2209116bd9f" +[[package]] +name = "same-file" +version = "1.0.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "93fc1dc3aaa9bfed95e02e6eadabb4baf7e3078b0bd1b4d7b6b0b68378900502" +dependencies = [ + "winapi-util", +] + [[package]] name = "serde" version = "1.0.210" @@ -764,6 +914,28 @@ version = "1.0.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a7065abeca94b6a8a577f9bd45aa0867a2238b74e8eb67cf10d492bc39351394" +[[package]] +name = "syntect" +version = "5.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "874dcfa363995604333cf947ae9f751ca3af4522c60886774c4963943b4746b1" +dependencies = [ + "bincode", + "bitflags", + "flate2", + "fnv", + "once_cell", + "onig", + "plist", + "regex-syntax", + "serde", + "serde_derive", + "serde_json", + "thiserror", + "walkdir", + "yaml-rust", +] + [[package]] name = "thiserror" version = "1.0.64" @@ -784,6 +956,37 @@ dependencies = [ "syn", ] +[[package]] +name = "time" +version = "0.3.36" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5dfd88e563464686c916c7e46e623e520ddc6d79fa6641390f2e3fa86e83e885" +dependencies = [ + "deranged", + "itoa", + "num-conv", + "powerfmt", + "serde", + "time-core", + "time-macros", +] + +[[package]] +name = "time-core" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ef927ca75afb808a4d64dd374f00a2adf8d0fcff8e7b184af886c3c87ec4a3f3" + +[[package]] +name = "time-macros" +version = "0.2.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3f252a68540fde3a3877aeea552b832b40ab9a69e318efd078774a01ddee1ccf" +dependencies = [ + "num-conv", + "time-core", +] + [[package]] name = "tokio" version = "1.40.0" @@ -858,6 +1061,17 @@ dependencies = [ "once_cell", ] +[[package]] +name = "two-face" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0ccd4843ea031c609fe9c16cae00e9657bad8a9f735a3cc2e420955d802b4268" +dependencies = [ + "once_cell", + "serde", + "syntect", +] + [[package]] name = "unicode-id" version = "0.3.5" @@ -876,6 +1090,16 @@ version = "0.2.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "06abde3611657adf66d383f00b093d7faecc7fa57071cce2578660c9f1010821" +[[package]] +name = "walkdir" +version = "2.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "29790946404f91d9c5d06f9874efddea1dc06c5efe94541a7d6863108e3a5e4b" +dependencies = [ + "same-file", + "winapi-util", +] + [[package]] name = "wasi" version = "0.11.0+wasi-snapshot-preview1" @@ -937,6 +1161,15 @@ version = "0.2.93" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "c62a0a307cb4a311d3a07867860911ca130c3494e8c2719593806c08bc5d0484" +[[package]] +name = "winapi-util" +version = "0.1.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cf221c93e13a30d793f7645a0e7762c55d169dbb0a49671918a2319d289b10bb" +dependencies = [ + "windows-sys", +] + [[package]] name = "windows-core" version = "0.52.0" @@ -1018,3 +1251,12 @@ name = "windows_x86_64_msvc" version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "589f6da84c646204747d1270a2a5661ea66ed1cced2631d546fdfb155959f9ec" + +[[package]] +name = "yaml-rust" +version = "0.4.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "56c1936c4cc7a1c9ab21a1ebb602eb942ba868cbd44a99cb7cdc5892335e1c85" +dependencies = [ + "linked-hash-map", +] diff --git a/Cargo.toml b/Cargo.toml index 94bc374..e397d41 100755 --- a/Cargo.toml +++ b/Cargo.toml @@ -9,4 +9,6 @@ axum = "0.7.7" chrono = "0.4.38" clap = { version = "4.5.18", features = ["derive", "env"] } markdown = "1.0.0-alpha.21" +syntect = "5.2.0" tokio = { version = "1.40.0", features = [ "rt-multi-thread" ] } +two-face = "0.4.0" diff --git a/src/main.rs b/src/main.rs index 50821bf..a4da543 100755 --- a/src/main.rs +++ b/src/main.rs @@ -8,12 +8,21 @@ use axum::{ use clap::Parser; use markdown::ParseOptions; use std::{collections::HashMap, path::PathBuf, sync::Arc}; +use syntect::{ + highlighting::{Theme, ThemeSet}, + html::highlighted_html_for_string, + parsing::{SyntaxSet, SyntaxSetBuilder}, +}; +use two_face::theme::EmbeddedLazyThemeSet; -#[derive(Debug)] +// #[derive(Debug)] struct BlogState { templates: HashMap>, posts: HashMap>, + ps: SyntaxSet, + ts: EmbeddedLazyThemeSet, + config: Config, } type AppState = State>; @@ -80,12 +89,12 @@ struct Post { trait ToHtml { fn is_directive(&self) -> bool; - fn to_html(&self) -> String; + fn to_html(&self, ps: &SyntaxSet, theme: &Theme) -> String; fn template(&self) -> Option; } impl ToHtml for markdown::mdast::Node { - fn to_html(&self) -> String { + fn to_html(&self, ps: &SyntaxSet, theme: &Theme) -> String { let mut s = String::new(); if self.is_directive() { @@ -95,13 +104,13 @@ impl ToHtml for markdown::mdast::Node { match self { markdown::mdast::Node::Root(root) => { for child in &root.children { - s += &child.to_html(); + s += &child.to_html(ps, theme); } } markdown::mdast::Node::Blockquote(block_quote) => { s += "
"; for child in &block_quote.children { - s += &child.to_html(); + s += &child.to_html(ps, theme); } s += "
"; } @@ -110,7 +119,7 @@ impl ToHtml for markdown::mdast::Node { markdown::mdast::Node::List(list) => { s += "
    "; for child in &list.children { - s += &child.to_html(); + s += &child.to_html(ps, theme); } s += "
"; } @@ -129,7 +138,7 @@ impl ToHtml for markdown::mdast::Node { markdown::mdast::Node::Emphasis(emphasis) => { s += ""; for child in &emphasis.children { - s += &child.to_html(); + s += &child.to_html(ps, theme); } s += ""; } @@ -148,7 +157,7 @@ impl ToHtml for markdown::mdast::Node { markdown::mdast::Node::Link(link) => { s += &format!("", link.url); for child in &link.children { - s += &child.to_html(); + s += &child.to_html(ps, theme); } s += ""; } @@ -156,23 +165,34 @@ impl ToHtml for markdown::mdast::Node { markdown::mdast::Node::Strong(strong) => { s += ""; for child in &strong.children { - s += &child.to_html(); + s += &child.to_html(ps, theme); } s += ""; } markdown::mdast::Node::Text(text) => s += &ansi_to_html::convert(&text.value).unwrap(), markdown::mdast::Node::Code(code) => { - s += &format!( - "
{}
", - &ansi_to_html::convert(&code.value).unwrap() - ); + if let Some(syntax) = code + .lang + .as_ref() + .and_then(|lang| ps.find_syntax_by_extension(&lang)) + { + println!("{:?}", code.lang); + let escaped = highlighted_html_for_string(&code.value, &ps, syntax, theme) + .expect("generate html"); + + s += &escaped; + } else { + s += "
";
+                    s += &ansi_to_html::convert(&code.value).unwrap();
+                    s += "
"; + } } markdown::mdast::Node::Math(_) => {} markdown::mdast::Node::MdxFlowExpression(_) => {} markdown::mdast::Node::Heading(heading) => { s += &format!("", heading.depth); for child in &heading.children { - s += &child.to_html(); + s += &child.to_html(ps, theme); } s += &format!("", heading.depth); } @@ -185,7 +205,7 @@ impl ToHtml for markdown::mdast::Node { markdown::mdast::Node::ListItem(item) => { s += "
  • "; for child in &item.children { - s += &child.to_html(); + s += &child.to_html(ps, theme); } s += "
  • "; } @@ -193,7 +213,7 @@ impl ToHtml for markdown::mdast::Node { markdown::mdast::Node::Paragraph(paragraph) => { s += "

    "; for child in ¶graph.children { - s += &child.to_html(); + s += &child.to_html(ps, theme); } s += "

    "; } @@ -214,8 +234,8 @@ impl ToHtml for markdown::mdast::Node { } } _ => false, - }).is_some_and(|x| x) - + }) + .is_some_and(|x| x) } fn template(&self) -> Option { @@ -236,7 +256,12 @@ impl ToHtml for markdown::mdast::Node { } impl Post { - fn to_html(&self, templates: &HashMap>) -> String { + fn to_html( + &self, + templates: &HashMap>, + ps: &SyntaxSet, + ts: &Theme, + ) -> String { let mut s = String::with_capacity(self.content.len()); let ast = markdown::to_mdast(&self.content, &ParseOptions::default()).unwrap(); @@ -247,13 +272,13 @@ impl Post { s += &template.data.content[0..template.data.index]; } - s += &ast.to_html(); + s += &ast.to_html(ps, ts); if template.data.index > 0 { s += &template.data.content[template.data.index + 3..] } } else { - s += &ast.to_html(); + s += &ast.to_html(ps, ts); } s @@ -291,9 +316,18 @@ struct Config { #[tokio::main] async fn main() { let config = Config::parse(); + + let mut builder = two_face::syntax::extra_newlines().into_builder(); + builder + .add_from_folder(format!("{}/syntaxes", &config.blog_dir), true) + .expect("load syntaxes"); + let ps = builder.build(); + let mut blog_state = BlogState { templates: HashMap::new(), posts: HashMap::new(), + ps, + ts: two_face::theme::extra(), config, }; @@ -341,7 +375,15 @@ async fn main() { async fn home_page(State(state): AppState) -> Result, Html> { match state.posts.get("home") { Some(post) => match post.data() { - Ok(content) => Ok(Html(content.to_html(&state.templates))), + Ok(content) => Ok(Html( + content.to_html( + &state.templates, + &state.ps, + state + .ts + .get(two_face::theme::EmbeddedThemeName::GruvboxDark), + ), + )), Err(e) => Err(Html(e.to_string())), }, None => Err(Html("

    404 - Not Found

    ".to_string())), @@ -354,7 +396,15 @@ async fn blog_post( ) -> Result, Html> { match state.posts.get(&post) { Some(post) => match post.data() { - Ok(content) => Ok(Html(content.to_html(&state.templates))), + Ok(content) => Ok(Html( + content.to_html( + &state.templates, + &state.ps, + state + .ts + .get(two_face::theme::EmbeddedThemeName::GruvboxDark), + ), + )), Err(e) => Err(Html(e.to_string())), }, None => Err(not_found_page(state)), @@ -374,7 +424,15 @@ async fn image(State(state): AppState, Path(image_path): Path) -> impl I fn not_found_page(state: Arc) -> Html { match state.posts.get("404") { Some(post) => match post.data() { - Ok(content) => Html(content.to_html(&state.templates)), + Ok(content) => Html( + content.to_html( + &state.templates, + &state.ps, + state + .ts + .get(two_face::theme::EmbeddedThemeName::GruvboxDark), + ), + ), Err(e) => Html(e.to_string()), }, None => Html("

    404 - Not Found

    ".to_string()),