mylloon.fr/src/routes/blog.rs

260 lines
7.7 KiB
Rust
Raw Normal View History

2023-04-26 14:19:02 +02:00
use std::{
collections::hash_map::DefaultHasher,
hash::{Hash, Hasher},
};
use ::rss::{Category, Channel, Guid, Image, Item};
2023-04-26 12:50:08 +02:00
use actix_web::{dev::ConnectionInfo, get, web, HttpRequest, HttpResponse, Responder};
2023-04-14 11:30:58 +02:00
use cached::proc_macro::once;
2023-04-26 13:49:05 +02:00
use chrono::{DateTime, Datelike, Local, NaiveDateTime, Utc};
use chrono_tz::Europe;
2023-04-21 16:48:31 +02:00
use comrak::{parse_document, Arena};
2023-04-14 11:30:58 +02:00
use ramhorns::Content;
use crate::{
config::Config,
misc::{
date::Date,
2023-04-21 16:48:31 +02:00
markdown::{get_metadata, get_options, read_file, File, FileMetadata},
},
template::Infos,
};
2023-04-14 11:30:58 +02:00
#[get("/blog")]
pub async fn index(config: web::Data<Config>) -> impl Responder {
2023-04-24 12:18:21 +02:00
HttpResponse::Ok().body(build_index(config.get_ref().to_owned()))
2023-04-14 11:30:58 +02:00
}
#[derive(Content)]
2023-04-19 18:55:03 +02:00
struct BlogIndexTemplate {
2023-04-24 18:01:38 +02:00
posts: Vec<Post>,
no_posts: bool,
2023-04-19 20:27:40 +02:00
}
2023-04-24 19:15:28 +02:00
#[once(time = 120)]
pub fn build_index(config: Config) -> String {
let mut posts = get_posts("data/blog");
// Sort from newest to oldest
posts.sort_by_cached_key(|p| (p.date.year, p.date.month, p.date.day));
posts.reverse();
config.tmpl.render(
"blog/index.html",
BlogIndexTemplate {
2023-04-24 18:01:38 +02:00
no_posts: posts.is_empty(),
posts,
},
Infos {
page_title: Some("Blog".into()),
page_desc: Some("Liste des posts d'Anri".into()),
page_kw: Some(["blog", "blogging"].join(", ")),
},
)
}
2023-04-20 14:41:36 +02:00
#[derive(Content)]
2023-04-19 20:27:40 +02:00
struct Post {
title: String,
2023-04-20 14:41:36 +02:00
date: Date,
2023-04-19 20:27:40 +02:00
url: String,
2023-04-24 18:41:40 +02:00
desc: Option<String>,
2023-04-19 18:55:03 +02:00
}
2023-04-14 11:30:58 +02:00
2023-04-26 14:19:02 +02:00
impl Hash for Post {
fn hash<H: Hasher>(&self, state: &mut H) {
let blog_dir = "data/blog";
let ext = ".md";
if let Some(file) = read_file(&format!("{blog_dir}/{}{ext}", self.url)) {
file.content.hash(state)
}
}
}
fn get_posts(location: &str) -> Vec<Post> {
let entries = match std::fs::read_dir(location) {
Ok(res) => res
.flatten()
.filter(|f| f.path().extension().unwrap() == "md")
.collect::<Vec<std::fs::DirEntry>>(),
Err(_) => vec![],
};
2023-04-20 11:57:06 +02:00
entries
2023-04-20 11:57:06 +02:00
.iter()
2023-04-19 20:54:05 +02:00
.map(|f| {
2023-04-20 11:57:06 +02:00
let _filename = f.file_name();
let filename = _filename.to_string_lossy();
2023-04-19 20:54:05 +02:00
let file_without_ext = filename.split_at(filename.len() - 3).0;
2023-04-19 21:16:39 +02:00
let file_metadata = match std::fs::read_to_string(format!("{location}/{filename}")) {
2023-04-21 16:48:31 +02:00
Ok(text) => {
let arena = Arena::new();
let options = get_options();
let root = parse_document(&arena, &text, &options);
let mut metadata = get_metadata(root);
2023-04-20 11:57:06 +02:00
metadata.title = match metadata.title {
2023-04-21 16:48:31 +02:00
Some(title) => Some(title),
None => Some(file_without_ext.into()),
2023-04-20 11:57:06 +02:00
};
2023-04-21 16:48:31 +02:00
metadata
2023-04-19 21:16:39 +02:00
}
2023-04-20 11:57:06 +02:00
Err(_) => FileMetadata {
title: Some(file_without_ext.into()),
2023-04-24 18:41:40 +02:00
..FileMetadata::default()
2023-04-20 11:57:06 +02:00
},
2023-04-19 21:16:39 +02:00
};
2023-04-19 20:54:05 +02:00
Post {
url: file_without_ext.into(),
2023-04-20 11:57:06 +02:00
title: file_metadata.title.unwrap(),
2023-04-20 15:08:09 +02:00
date: file_metadata.date.unwrap_or({
let m = f.metadata().unwrap();
let date = std::convert::Into::<DateTime<Utc>>::into(
m.modified().unwrap_or(m.created().unwrap()),
)
.date_naive();
Date {
day: date.day(),
month: date.month(),
year: date.year(),
}
}),
2023-04-24 18:41:40 +02:00
desc: file_metadata.description,
2023-04-19 20:54:05 +02:00
}
})
.collect::<Vec<Post>>()
}
#[derive(Content)]
struct BlogPostTemplate {
post: Option<File>,
2023-04-14 11:30:58 +02:00
}
2023-04-26 10:41:49 +02:00
#[get("/blog/p/{id}")]
2023-04-19 18:55:03 +02:00
pub async fn page(path: web::Path<(String,)>, config: web::Data<Config>) -> impl Responder {
2023-04-24 12:18:21 +02:00
HttpResponse::Ok().body(build_post(path.into_inner().0, config.get_ref().to_owned()))
2023-04-14 11:30:58 +02:00
}
2023-04-26 10:41:49 +02:00
fn build_post(file: String, config: Config) -> String {
2023-04-19 20:08:15 +02:00
let mut post = None;
let infos = get_post(&mut post, file);
2023-04-19 18:55:03 +02:00
config
.tmpl
.render("blog/post.html", BlogPostTemplate { post }, infos)
}
fn get_post(post: &mut Option<File>, filename: String) -> Infos {
let blog_dir = "data/blog";
let ext = ".md";
*post = read_file(&format!("{blog_dir}/{filename}{ext}"));
let title = match post {
Some(data) => match &data.metadata.info.title {
Some(text) => text,
None => &filename,
},
None => &filename,
};
Infos {
page_title: Some(format!("Post: {}", title)),
page_desc: Some("Blog d'Anri".into()),
page_kw: Some(["blog", "blogging", "write", "writing"].join(", ")),
}
2023-04-14 11:30:58 +02:00
}
2023-04-26 12:50:08 +02:00
#[get("/blog/rss")]
pub async fn rss(req: HttpRequest, config: web::Data<Config>) -> impl Responder {
2023-04-26 15:16:57 +02:00
HttpResponse::Ok()
.append_header(("content-type", "application/rss+xml"))
.body(build_rss(
config.get_ref().to_owned(),
req.connection_info().to_owned(),
))
2023-04-26 12:50:08 +02:00
}
2023-04-26 16:22:54 +02:00
#[once(time = 10800)] // 3h
2023-04-26 12:50:08 +02:00
fn build_rss(config: Config, info: ConnectionInfo) -> String {
let mut posts = get_posts("data/blog");
// Sort from newest to oldest
posts.sort_by_cached_key(|p| (p.date.year, p.date.month, p.date.day));
posts.reverse();
// Only the 20 newest
let max = 20;
if posts.len() > max {
posts.drain(max..);
}
let link_to_site = format!("{}://{}", info.scheme(), info.host());
2023-04-26 16:21:57 +02:00
let author = if let (Some(mail), Some(name)) = (config.fc.mail, config.fc.name) {
Some(format!("{mail} ({name})"))
} else {
None
};
2023-04-26 12:50:08 +02:00
let channel = Channel {
title: "Blog d'Anri".into(),
link: link_to_site.to_owned(),
2023-04-26 13:49:05 +02:00
description: "Un fil qui parle d'informatique notamment".into(),
2023-04-26 12:50:08 +02:00
language: Some("fr".into()),
2023-04-26 16:21:57 +02:00
managing_editor: author.to_owned(),
webmaster: author,
2023-04-26 13:49:05 +02:00
pub_date: Some(Local::now().to_rfc2822()),
2023-04-26 12:50:08 +02:00
categories: ["blog", "blogging", "write", "writing"]
.iter()
.map(|&c| Category {
name: c.into(),
..Category::default()
})
.collect(),
generator: Some("ewp with rss crate".into()),
docs: Some("https://www.rssboard.org/rss-specification".into()),
image: Some(Image {
2023-04-26 13:49:05 +02:00
url: format!("{}/icons/favicon-32x32.png", link_to_site),
link: link_to_site.to_owned(),
2023-04-26 12:50:08 +02:00
..Image::default()
}),
2023-04-26 13:49:05 +02:00
items: posts
.iter()
.map(|p| Item {
title: Some(p.title.to_owned()),
link: Some(format!("{}/blog/p/{}", link_to_site, p.url)),
description: p.desc.to_owned(),
2023-04-26 14:19:02 +02:00
guid: Some(Guid {
value: format!("urn:hash:{}", {
let mut hasher = DefaultHasher::new();
p.hash(&mut hasher);
hasher.finish()
}),
permalink: false,
}),
2023-04-26 13:49:05 +02:00
pub_date: Some(
NaiveDateTime::parse_from_str(
&format!("{}-{}-{} 13:12:00", p.date.day, p.date.month, p.date.year),
"%d-%m-%Y %H:%M:%S",
)
.unwrap()
.and_local_timezone(Europe::Paris)
.unwrap()
.to_rfc2822(),
),
..Item::default()
})
.collect(),
2023-04-26 12:50:08 +02:00
..Channel::default()
};
std::str::from_utf8(&channel.write_to(Vec::new()).unwrap())
.unwrap()
.into()
}