improving the experience brick by brick

This commit is contained in:
slonkazoid 2024-12-02 20:07:54 +03:00
parent ff2eae0ae1
commit 589de5b9da
Signed by: slonk
SSH key fingerprint: SHA256:tbZfJX4IOvZ0LGWOWu5Ijo8jfMPi78TU7x1VoEeCIjM
8 changed files with 45 additions and 92 deletions

View file

@ -173,10 +173,9 @@ async fn rss(
let posts = posts let posts = posts
.get_all_posts(|metadata, _| { .get_all_posts(|metadata, _| {
!query query
.tag .tag
.as_ref() .as_ref().is_none_or(|tag| metadata.tags.contains(tag))
.is_some_and(|tag| !metadata.tags.contains(tag))
}) })
.await?; .await?;

View file

@ -4,6 +4,7 @@ use askama_axum::Template;
use axum::http::StatusCode; use axum::http::StatusCode;
use axum::response::{IntoResponse, Response}; use axum::response::{IntoResponse, Response};
use thiserror::Error; use thiserror::Error;
use tracing::error;
#[derive(Debug)] #[derive(Debug)]
#[repr(transparent)] #[repr(transparent)]
@ -76,17 +77,14 @@ struct ErrorTemplate {
impl IntoResponse for AppError { impl IntoResponse for AppError {
fn into_response(self) -> Response { fn into_response(self) -> Response {
let error = self.to_string();
error!("error while handling request: {error}");
let status_code = match &self { let status_code = match &self {
AppError::PostError(PostError::NotFound(_)) => StatusCode::NOT_FOUND, AppError::PostError(PostError::NotFound(_)) => StatusCode::NOT_FOUND,
AppError::RssDisabled => StatusCode::FORBIDDEN, AppError::RssDisabled => StatusCode::FORBIDDEN,
_ => StatusCode::INTERNAL_SERVER_ERROR, _ => StatusCode::INTERNAL_SERVER_ERROR,
}; };
( (status_code, ErrorTemplate { error }).into_response()
status_code,
ErrorTemplate {
error: self.to_string(),
},
)
.into_response()
} }
} }

View file

@ -1,37 +0,0 @@
use std::hash::{DefaultHasher, Hash, Hasher};
use std::marker::PhantomData;
use std::sync::Arc;
pub struct HashArcStore<T, Lookup>
where
Lookup: Hash,
{
inner: Option<Arc<T>>,
hash: Option<u64>,
_phantom: PhantomData<Lookup>,
}
impl<T, Lookup> HashArcStore<T, Lookup>
where
Lookup: Hash,
{
pub fn new() -> Self {
Self {
inner: None,
hash: None,
_phantom: PhantomData,
}
}
pub fn get_or_init(&mut self, key: &Lookup, init: impl Fn(&Lookup) -> Arc<T>) -> Arc<T> {
let mut h = DefaultHasher::new();
key.hash(&mut h);
let hash = h.finish();
if !self.hash.is_some_and(|inner_hash| inner_hash == hash) {
self.inner = Some(init(key));
self.hash = Some(hash);
}
// safety: please.
unsafe { self.inner.as_ref().unwrap_unchecked().clone() }
}
}

View file

@ -3,7 +3,6 @@
mod app; mod app;
mod config; mod config;
mod error; mod error;
mod hash_arc_store;
mod helpers; mod helpers;
mod markdown_render; mod markdown_render;
mod platform; mod platform;
@ -77,25 +76,19 @@ async fn main() -> eyre::Result<()> {
let reg = Arc::new(RwLock::new(reg)); let reg = Arc::new(RwLock::new(reg));
let watcher_token = cancellation_token.child_token();
let posts = Arc::new(MarkdownPosts::new(Arc::clone(&config)).await?);
let state = AppState {
config: Arc::clone(&config),
posts: Arc::clone(&posts),
reg: Arc::clone(&reg),
};
debug!("setting up watcher"); debug!("setting up watcher");
let watcher_token = cancellation_token.child_token();
tasks.spawn( tasks.spawn(
watch_templates( watch_templates(
config.dirs.custom_templates.clone(), config.dirs.custom_templates.clone(),
watcher_token.clone(), watcher_token.clone(),
reg, reg.clone(),
) )
.instrument(info_span!("custom_template_watcher")), .instrument(info_span!("custom_template_watcher")),
); );
let posts = Arc::new(MarkdownPosts::new(Arc::clone(&config)).await?);
if config.cache.enable && config.cache.cleanup { if config.cache.enable && config.cache.cleanup {
if let Some(millis) = config.cache.cleanup_interval { if let Some(millis) = config.cache.cleanup_interval {
let posts = Arc::clone(&posts); let posts = Arc::clone(&posts);
@ -117,6 +110,11 @@ async fn main() -> eyre::Result<()> {
} }
} }
let state = AppState {
config: Arc::clone(&config),
posts: Arc::clone(&posts),
reg: Arc::clone(&reg),
};
let app = app::new(&config).with_state(state.clone()); let app = app::new(&config).with_state(state.clone());
let listener = TcpListener::bind(socket_addr) let listener = TcpListener::bind(socket_addr)

View file

@ -1,5 +1,5 @@
use std::sync::{Arc, OnceLock, RwLock}; use color_eyre::eyre::{self, Context};
use comrak::adapters::SyntaxHighlighterAdapter;
use comrak::markdown_to_html_with_plugins; use comrak::markdown_to_html_with_plugins;
use comrak::plugins::syntect::{SyntectAdapter, SyntectAdapterBuilder}; use comrak::plugins::syntect::{SyntectAdapter, SyntectAdapterBuilder};
use comrak::ComrakOptions; use comrak::ComrakOptions;
@ -7,32 +7,26 @@ use comrak::RenderPlugins;
use syntect::highlighting::ThemeSet; use syntect::highlighting::ThemeSet;
use crate::config::RenderConfig; use crate::config::RenderConfig;
use crate::hash_arc_store::HashArcStore;
fn syntect_adapter(config: &RenderConfig) -> Arc<SyntectAdapter> { pub fn build_syntect(config: &RenderConfig) -> eyre::Result<SyntectAdapter> {
static STATE: OnceLock<RwLock<HashArcStore<SyntectAdapter, RenderConfig>>> = OnceLock::new();
let lock = STATE.get_or_init(|| RwLock::new(HashArcStore::new()));
let mut guard = lock.write().unwrap();
guard.get_or_init(config, build_syntect)
}
fn build_syntect(config: &RenderConfig) -> Arc<SyntectAdapter> {
let mut theme_set = if config.syntect.load_defaults { let mut theme_set = if config.syntect.load_defaults {
ThemeSet::load_defaults() ThemeSet::load_defaults()
} else { } else {
ThemeSet::new() ThemeSet::new()
}; };
if let Some(path) = config.syntect.themes_dir.as_ref() { if let Some(path) = config.syntect.themes_dir.as_ref() {
theme_set.add_from_folder(path).unwrap(); theme_set
.add_from_folder(path)
.with_context(|| format!("failed to add themes from {path:?}"))?;
} }
let mut builder = SyntectAdapterBuilder::new().theme_set(theme_set); let mut builder = SyntectAdapterBuilder::new().theme_set(theme_set);
if let Some(theme) = config.syntect.theme.as_ref() { if let Some(theme) = config.syntect.theme.as_ref() {
builder = builder.theme(theme); builder = builder.theme(theme);
} }
Arc::new(builder.build()) Ok(builder.build())
} }
pub fn render(markdown: &str, config: &RenderConfig) -> String { pub fn render(markdown: &str, syntect: Option<&dyn SyntaxHighlighterAdapter>) -> String {
let mut options = ComrakOptions::default(); let mut options = ComrakOptions::default();
options.extension.table = true; options.extension.table = true;
options.extension.autolink = true; options.extension.autolink = true;
@ -43,8 +37,7 @@ pub fn render(markdown: &str, config: &RenderConfig) -> String {
options.extension.header_ids = Some(String::new()); options.extension.header_ids = Some(String::new());
let mut render_plugins = RenderPlugins::default(); let mut render_plugins = RenderPlugins::default();
let syntect = syntect_adapter(config); render_plugins.codefence_syntax_highlighter = syntect;
render_plugins.codefence_syntax_highlighter = Some(syntect.as_ref());
let plugins = comrak::PluginsBuilder::default() let plugins = comrak::PluginsBuilder::default()
.render(render_plugins) .render(render_plugins)

View file

@ -9,6 +9,7 @@ use std::time::SystemTime;
use axum::http::HeaderValue; use axum::http::HeaderValue;
use chrono::{DateTime, Utc}; use chrono::{DateTime, Utc};
use color_eyre::eyre::{self, Context}; use color_eyre::eyre::{self, Context};
use comrak::plugins::syntect::SyntectAdapter;
use fronma::parser::{parse, ParsedData}; use fronma::parser::{parse, ParsedData};
use serde::Deserialize; use serde::Deserialize;
use tokio::fs; use tokio::fs;
@ -16,7 +17,7 @@ use tokio::io::AsyncReadExt;
use tracing::{error, info, warn}; use tracing::{error, info, warn};
use crate::config::Config; use crate::config::Config;
use crate::markdown_render::render; use crate::markdown_render::{build_syntect, render};
use crate::post::cache::{load_cache, Cache, CACHE_VERSION}; use crate::post::cache::{load_cache, Cache, CACHE_VERSION};
use crate::post::{PostError, PostManager, PostMetadata, RenderStats, ReturnedPost}; use crate::post::{PostError, PostManager, PostMetadata, RenderStats, ReturnedPost};
use crate::systemtime_as_secs::as_secs; use crate::systemtime_as_secs::as_secs;
@ -62,6 +63,7 @@ where
{ {
cache: Option<Cache>, cache: Option<Cache>,
config: C, config: C,
syntect: SyntectAdapter,
} }
impl<C> MarkdownPosts<C> impl<C> MarkdownPosts<C>
@ -69,7 +71,10 @@ where
C: Deref<Target = Config>, C: Deref<Target = Config>,
{ {
pub async fn new(config: C) -> eyre::Result<MarkdownPosts<C>> { pub async fn new(config: C) -> eyre::Result<MarkdownPosts<C>> {
if config.cache.enable { let syntect =
build_syntect(&config.render).context("failed to create syntax highlighting engine")?;
let cache = if config.cache.enable {
if config.cache.persistence && tokio::fs::try_exists(&config.cache.file).await? { if config.cache.persistence && tokio::fs::try_exists(&config.cache.file).await? {
info!("loading cache from file"); info!("loading cache from file");
let mut cache = load_cache(&config).await.unwrap_or_else(|err| { let mut cache = load_cache(&config).await.unwrap_or_else(|err| {
@ -83,22 +88,19 @@ where
cache = Default::default(); cache = Default::default();
}; };
Ok(Self { Some(cache)
cache: Some(cache),
config,
})
} else { } else {
Ok(Self { Some(Default::default())
cache: Some(Default::default()),
config,
})
} }
} else { } else {
Ok(Self { None
cache: None, };
config,
}) Ok(Self {
} cache,
config,
syntect,
})
} }
async fn parse_and_render( async fn parse_and_render(
@ -126,7 +128,7 @@ where
let parsing = parsing_start.elapsed(); let parsing = parsing_start.elapsed();
let before_render = Instant::now(); let before_render = Instant::now();
let post = render(body, &self.config.render); let post = render(body, Some(&self.syntect));
let rendering = before_render.elapsed(); let rendering = before_render.elapsed();
if let Some(cache) = self.cache.as_ref() { if let Some(cache) = self.cache.as_ref() {

View file

@ -58,7 +58,7 @@ pub trait PostManager {
tag: Option<&String>, tag: Option<&String>,
) -> Result<Vec<PostMetadata>, PostError> { ) -> Result<Vec<PostMetadata>, PostError> {
let mut posts = self let mut posts = self
.get_all_post_metadata(|metadata| !tag.is_some_and(|tag| !metadata.tags.contains(tag))) .get_all_post_metadata(|metadata| tag.is_none_or(|tag| metadata.tags.contains(tag)))
.await?; .await?;
// we still want some semblance of order if created_at is None so sort by mtime as well // we still want some semblance of order if created_at is None so sort by mtime as well
posts.sort_unstable_by_key(|metadata| metadata.modified_at.unwrap_or_default()); posts.sort_unstable_by_key(|metadata| metadata.modified_at.unwrap_or_default());

View file

@ -1,5 +1,5 @@
pub struct RangedI128Visitor<const START: i128, const END: i128>; pub struct RangedI128Visitor<const START: i128, const END: i128>;
impl<'de, const START: i128, const END: i128> serde::de::Visitor<'de> impl<const START: i128, const END: i128> serde::de::Visitor<'_>
for RangedI128Visitor<START, END> for RangedI128Visitor<START, END>
{ {
type Value = i128; type Value = i128;