2023-04-14 11:30:58 +02:00
|
|
|
use actix_web::{get, web, HttpResponse, Responder};
|
|
|
|
use cached::proc_macro::once;
|
|
|
|
use ramhorns::Content;
|
|
|
|
|
2023-04-19 20:17:03 +02:00
|
|
|
use crate::{
|
|
|
|
config::Config,
|
2023-04-20 14:41:36 +02:00
|
|
|
template::{get_md_asm, get_md_metadata, read_md_file, Date, File, FileMetadata, Infos},
|
2023-04-19 20:17:03 +02:00
|
|
|
};
|
2023-04-14 11:30:58 +02:00
|
|
|
|
|
|
|
#[get("/blog")]
|
|
|
|
pub async fn index(config: web::Data<Config>) -> impl Responder {
|
|
|
|
HttpResponse::Ok().body(get_index(config.get_ref().clone()))
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Content)]
|
2023-04-19 18:55:03 +02:00
|
|
|
struct BlogIndexTemplate {
|
2023-04-19 20:27:40 +02:00
|
|
|
posts: Option<Vec<Post>>,
|
|
|
|
}
|
|
|
|
|
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-19 18:55:03 +02:00
|
|
|
}
|
2023-04-14 11:30:58 +02:00
|
|
|
|
|
|
|
#[once(time = 60)]
|
2023-04-19 20:17:03 +02:00
|
|
|
pub fn get_index(config: Config) -> String {
|
2023-04-19 21:16:39 +02:00
|
|
|
let location = "data/blog";
|
2023-04-20 11:57:06 +02:00
|
|
|
|
|
|
|
let mut entries = std::fs::read_dir(location)
|
2023-04-19 20:54:05 +02:00
|
|
|
.unwrap()
|
2023-04-20 11:57:06 +02:00
|
|
|
.flatten()
|
|
|
|
.filter(|f| f.path().extension().unwrap() == "md")
|
|
|
|
.collect::<Vec<std::fs::DirEntry>>();
|
|
|
|
|
2023-04-20 14:41:36 +02:00
|
|
|
// Sort by latest modification/ date of creation
|
2023-04-20 11:57:06 +02:00
|
|
|
entries.sort_by_cached_key(|f| {
|
|
|
|
f.metadata()
|
|
|
|
.unwrap()
|
|
|
|
.modified()
|
|
|
|
.unwrap_or(f.metadata().unwrap().created().unwrap())
|
|
|
|
});
|
|
|
|
entries.reverse();
|
|
|
|
|
|
|
|
let paths = entries
|
|
|
|
.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}")) {
|
|
|
|
Ok(text) => {
|
|
|
|
let md_tree = get_md_asm(&text);
|
|
|
|
let md_nodes = md_tree.children().unwrap();
|
2023-04-20 11:57:06 +02:00
|
|
|
let mut metadata = get_md_metadata(md_nodes);
|
|
|
|
|
|
|
|
metadata.title = match metadata.title {
|
|
|
|
Some(t) => Some(t),
|
|
|
|
None => Some(file_without_ext.to_string()),
|
|
|
|
};
|
|
|
|
|
|
|
|
metadata
|
2023-04-19 21:16:39 +02:00
|
|
|
}
|
2023-04-20 11:57:06 +02:00
|
|
|
Err(_) => FileMetadata {
|
|
|
|
title: Some(file_without_ext.to_string()),
|
|
|
|
link: None,
|
|
|
|
date: None,
|
|
|
|
},
|
2023-04-19 21:16:39 +02:00
|
|
|
};
|
|
|
|
|
2023-04-19 20:54:05 +02:00
|
|
|
Post {
|
|
|
|
url: file_without_ext.to_string(),
|
2023-04-20 11:57:06 +02:00
|
|
|
title: file_metadata.title.unwrap(),
|
|
|
|
date: file_metadata.date.unwrap_or_default(),
|
2023-04-19 20:54:05 +02:00
|
|
|
}
|
|
|
|
})
|
|
|
|
.collect::<Vec<Post>>();
|
|
|
|
|
2023-04-20 14:41:36 +02:00
|
|
|
// TODO sort vec by post date metadata, removing the sort above
|
|
|
|
|
2023-04-19 20:17:03 +02:00
|
|
|
config.tmpl.render(
|
|
|
|
"blog/index.html",
|
2023-04-19 20:54:05 +02:00
|
|
|
BlogIndexTemplate {
|
|
|
|
posts: if paths.is_empty() { None } else { Some(paths) },
|
|
|
|
},
|
2023-04-19 20:17:03 +02:00
|
|
|
Infos {
|
2023-04-19 20:32:04 +02:00
|
|
|
page_title: Some("Blog".to_string()),
|
|
|
|
page_desc: Some("Liste des posts d'Anri".to_string()),
|
|
|
|
page_kw: Some(["blog", "blogging"].join(", ")),
|
2023-04-19 20:17:03 +02:00
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Content)]
|
|
|
|
struct BlogPostTemplate {
|
|
|
|
post: Option<File>,
|
2023-04-14 11:30:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[get("/blog/{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-19 20:17:03 +02:00
|
|
|
HttpResponse::Ok().body(get_post(path.into_inner().0, config.get_ref().clone()))
|
2023-04-14 11:30:58 +02:00
|
|
|
}
|
|
|
|
|
2023-04-19 20:17:03 +02:00
|
|
|
pub fn get_post(file: String, config: Config) -> String {
|
2023-04-19 20:08:15 +02:00
|
|
|
let mut post = None;
|
2023-04-19 20:17:03 +02:00
|
|
|
let infos = _get_post(&mut post, file);
|
2023-04-19 18:55:03 +02:00
|
|
|
|
|
|
|
config
|
|
|
|
.tmpl
|
2023-04-19 20:17:03 +02:00
|
|
|
.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_md_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".to_string()),
|
|
|
|
page_kw: Some(["blog", "blogging", "write", "writing"].join(", ")),
|
|
|
|
}
|
2023-04-14 11:30:58 +02:00
|
|
|
}
|