support videos
ci/woodpecker/tag/woodpecker Pipeline was successful Details
ci/woodpecker/push/woodpecker Pipeline failed Details

main v0.1.0_2
Patrick Cleavelin 2024-10-02 20:07:41 -05:00
parent 0437d2b2e5
commit d496efe835
3 changed files with 246 additions and 12 deletions

69
Cargo.lock generated
View File

@ -17,6 +17,15 @@ version = "2.0.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "512761e0bb2578dd7380c6baaa0f4ce03e84f95e960231d1dec8bf4d7d6e2627"
[[package]]
name = "aho-corasick"
version = "1.1.3"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "8e60d3430d3a69478ad0993f19238d2df97c507009a52b3c10addcd7f6bcb916"
dependencies = [
"memchr",
]
[[package]]
name = "android-tzdata"
version = "0.1.1"
@ -32,6 +41,16 @@ dependencies = [
"libc",
]
[[package]]
name = "ansi-to-html"
version = "0.2.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d73c455ae09fa2223a75114789f30ad605e9e297f79537953523366c05995f5f"
dependencies = [
"regex",
"thiserror",
]
[[package]]
name = "anstream"
version = "0.6.15"
@ -172,6 +191,7 @@ dependencies = [
name = "blog-thing"
version = "0.1.0"
dependencies = [
"ansi-to-html",
"axum",
"chrono",
"clap",
@ -592,6 +612,35 @@ dependencies = [
"proc-macro2",
]
[[package]]
name = "regex"
version = "1.11.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "38200e5ee88914975b69f657f0801b6f6dccafd44fd9326302a4aaeecfacb1d8"
dependencies = [
"aho-corasick",
"memchr",
"regex-automata",
"regex-syntax",
]
[[package]]
name = "regex-automata"
version = "0.4.8"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "368758f23274712b504848e9d5a6f010445cc8b87a7cdb4d7cbee666c1288da3"
dependencies = [
"aho-corasick",
"memchr",
"regex-syntax",
]
[[package]]
name = "regex-syntax"
version = "0.8.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "2b15c43186be67a4fd63bee50d0303afffcef381492ebe2c5d87f324e1b8815c"
[[package]]
name = "rustc-demangle"
version = "0.1.24"
@ -715,6 +764,26 @@ version = "1.0.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a7065abeca94b6a8a577f9bd45aa0867a2238b74e8eb67cf10d492bc39351394"
[[package]]
name = "thiserror"
version = "1.0.64"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d50af8abc119fb8bb6dbabcfa89656f46f84aa0ac7688088608076ad2b459a84"
dependencies = [
"thiserror-impl",
]
[[package]]
name = "thiserror-impl"
version = "1.0.64"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "08904e7672f5eb876eaaf87e0ce17857500934f4981c4a0ab2b4aa98baac7fc3"
dependencies = [
"proc-macro2",
"quote",
"syn",
]
[[package]]
name = "tokio"
version = "1.40.0"

View File

@ -4,6 +4,7 @@ version = "0.1.0"
edition = "2021"
[dependencies]
ansi-to-html = "0.2.1"
axum = "0.7.7"
chrono = "0.4.38"
clap = { version = "4.5.18", features = ["derive", "env"] }

View File

@ -5,8 +5,9 @@ use axum::{
routing::get,
Router,
};
use std::{collections::HashMap, path::PathBuf, sync::Arc};
use clap::Parser;
use markdown::ParseOptions;
use std::{collections::HashMap, path::PathBuf, sync::Arc};
#[derive(Debug)]
struct BlogState {
@ -77,25 +78,182 @@ struct Post {
embed_template_name: Option<String>,
}
trait ToHtml {
fn is_directive(&self) -> bool;
fn to_html(&self) -> String;
fn template(&self) -> Option<String>;
}
impl ToHtml for markdown::mdast::Node {
fn to_html(&self) -> String {
let mut s = String::new();
if self.is_directive() {
return s;
}
match self {
markdown::mdast::Node::Root(root) => {
for child in &root.children {
s += &child.to_html();
}
}
markdown::mdast::Node::Blockquote(block_quote) => {
s += "<blockquote>";
for child in &block_quote.children {
s += &child.to_html();
}
s += "</blockquote>";
}
markdown::mdast::Node::FootnoteDefinition(_) => {}
markdown::mdast::Node::MdxJsxFlowElement(_) => {}
markdown::mdast::Node::List(list) => {
s += "<ul>";
for child in &list.children {
s += &child.to_html();
}
s += "</ul>";
}
markdown::mdast::Node::MdxjsEsm(_) => {}
markdown::mdast::Node::Toml(_) => {}
markdown::mdast::Node::Yaml(_) => {}
markdown::mdast::Node::Break(_) => {}
markdown::mdast::Node::InlineCode(inline_code) => {
s += &format!(
"<code>{}</code>",
&ansi_to_html::convert(&inline_code.value).unwrap()
);
}
markdown::mdast::Node::InlineMath(_) => {}
markdown::mdast::Node::Delete(_) => {}
markdown::mdast::Node::Emphasis(emphasis) => {
s += "<em>";
for child in &emphasis.children {
s += &child.to_html();
}
s += "</em>";
}
markdown::mdast::Node::MdxTextExpression(_) => {}
markdown::mdast::Node::FootnoteReference(_) => {}
markdown::mdast::Node::Html(_) => {}
markdown::mdast::Node::Image(image) => {
if image.url.ends_with(".mp4") {
s += &format!("<video controls src={}>", image.url);
} else {
s += &format!("<img src={}>", image.url);
}
}
markdown::mdast::Node::ImageReference(_) => {}
markdown::mdast::Node::MdxJsxTextElement(_) => {}
markdown::mdast::Node::Link(link) => {
s += &format!("<a href={}>", link.url);
for child in &link.children {
s += &child.to_html();
}
s += "</a>";
}
markdown::mdast::Node::LinkReference(_) => {}
markdown::mdast::Node::Strong(strong) => {
s += "<b>";
for child in &strong.children {
s += &child.to_html();
}
s += "</b>";
}
markdown::mdast::Node::Text(text) => s += &ansi_to_html::convert(&text.value).unwrap(),
markdown::mdast::Node::Code(code) => {
s += &format!(
"<pre><code>{}</code></pre>",
&ansi_to_html::convert(&code.value).unwrap()
);
}
markdown::mdast::Node::Math(_) => {}
markdown::mdast::Node::MdxFlowExpression(_) => {}
markdown::mdast::Node::Heading(heading) => {
s += &format!("<h{}>", heading.depth);
for child in &heading.children {
s += &child.to_html();
}
s += &format!("</h{}>", heading.depth);
}
markdown::mdast::Node::Table(_) => {}
markdown::mdast::Node::ThematicBreak(_) => {
s += "<hr>";
}
markdown::mdast::Node::TableRow(_) => {}
markdown::mdast::Node::TableCell(_) => {}
markdown::mdast::Node::ListItem(item) => {
s += "<li>";
for child in &item.children {
s += &child.to_html();
}
s += "</li>";
}
markdown::mdast::Node::Definition(_) => {}
markdown::mdast::Node::Paragraph(paragraph) => {
s += "<p>";
for child in &paragraph.children {
s += &child.to_html();
}
s += "</p>";
}
}
s
}
fn is_directive(&self) -> bool {
self.children()
.and_then(|children| children.first())
.map(|first| match first {
markdown::mdast::Node::Text(text) => {
if text.value.starts_with("%{") && text.value.ends_with("}") {
text.value.chars().filter(|x| *x == '}').count() == 1
} else {
false
}
}
_ => false,
}).is_some_and(|x| x)
}
fn template(&self) -> Option<String> {
self.children()
.and_then(|children| children.first())
.and_then(|first| match first {
markdown::mdast::Node::Paragraph(_) => first.template(),
markdown::mdast::Node::Text(text) => {
if text.value.starts_with("%{") {
text.value[2..].split('}').next().map(String::from)
} else {
None
}
}
_ => None,
})
}
}
impl Post {
fn to_html(&self, templates: &HashMap<String, Cached<Template>>) -> String {
let mut s = String::with_capacity(self.content.len());
if let Some(template) = self
.embed_template_name
.as_ref()
.and_then(|name| templates.get(name))
{
let ast = markdown::to_mdast(&self.content, &ParseOptions::default()).unwrap();
let template_name = ast.template();
if let Some(template) = template_name.and_then(|name| templates.get(&name)) {
if template.data.index > 0 {
s += &template.data.content[0..template.data.index];
}
s += &markdown::to_html(&self.content);
s += &ast.to_html();
if template.data.index > 0 {
s += &template.data.content[template.data.index + 3..]
}
} else {
s += &markdown::to_html(&self.content);
s += &ast.to_html();
}
s
@ -105,7 +263,8 @@ impl Post {
impl From<String> for Post {
fn from(value: String) -> Self {
let template_name = if value.starts_with("%{") {
value[2..].split('}').next().map(Into::into)
// value[2..].split('}').next().map(Into::into)
None
} else {
None
};
@ -138,7 +297,10 @@ async fn main() {
config,
};
for dir in std::fs::read_dir(&blog_state.config.blog_dir).unwrap().flatten() {
for dir in std::fs::read_dir(&blog_state.config.blog_dir)
.unwrap()
.flatten()
{
if dir.file_type().unwrap().is_file() {
let file_name = dir.file_name().into_string().unwrap();
let Some((file_name, extension)) = file_name.split_once('.') else {
@ -169,7 +331,9 @@ async fn main() {
.route("/:post", get(blog_post))
.route("/images/:image", get(image))
.with_state(Arc::new(blog_state));
let listener = tokio::net::TcpListener::bind(format!("0.0.0.0:{}", port)).await.unwrap();
let listener = tokio::net::TcpListener::bind(format!("0.0.0.0:{}", port))
.await
.unwrap();
axum::serve(listener, app).await.unwrap();
}
@ -199,7 +363,7 @@ async fn blog_post(
async fn image(State(state): AppState, Path(image_path): Path<String>) -> impl IntoResponse {
match std::fs::read(format!("{}/images/{image_path}", state.config.blog_dir)) {
Ok(contents) => Ok(([(header::CONTENT_TYPE, "image/png")], contents)),
Ok(contents) => Ok(([(header::CONTENT_TYPE, "image/png;video/mp4")], contents)),
Err(e) => {
eprintln!("{e:#?}");
Err(StatusCode::NOT_FOUND)