bingus-blog/src/config.rs

212 lines
5.9 KiB
Rust
Raw Normal View History

2024-05-01 18:25:01 +03:00
use std::env;
use std::net::{IpAddr, Ipv6Addr};
2024-05-01 18:25:01 +03:00
use std::path::PathBuf;
2024-04-18 04:05:38 +03:00
use color_eyre::eyre::{bail, Context, Result};
use serde::{Deserialize, Serialize};
use tokio::io::{AsyncReadExt, AsyncWriteExt};
use tracing::{error, info, instrument};
2024-05-02 19:23:20 +03:00
use url::Url;
2024-04-18 04:05:38 +03:00
use crate::ranged_i128_visitor::RangedI128Visitor;
#[derive(Serialize, Deserialize, Debug, Clone, PartialEq, Eq, Hash)]
2024-04-20 20:59:00 +03:00
#[serde(default)]
pub struct SyntectConfig {
pub load_defaults: bool,
pub themes_dir: Option<PathBuf>,
pub theme: Option<String>,
}
2024-04-20 20:59:00 +03:00
#[derive(Serialize, Deserialize, Debug, Clone, PartialEq, Eq, Hash, Default)]
2024-04-18 04:05:38 +03:00
#[serde(default)]
pub struct RenderConfig {
pub syntect: SyntectConfig,
2024-04-18 04:05:38 +03:00
}
2024-04-20 20:59:00 +03:00
#[derive(Serialize, Deserialize, Debug, Clone)]
#[serde(default)]
pub struct CacheConfig {
pub enable: bool,
pub cleanup: bool,
pub cleanup_interval: Option<u64>,
pub persistence: bool,
pub file: PathBuf,
pub compress: bool,
#[serde(deserialize_with = "check_zstd_level_bounds")]
pub compression_level: i32,
2024-04-20 20:59:00 +03:00
}
2024-04-18 04:05:38 +03:00
#[derive(Serialize, Deserialize, Debug, Clone)]
#[serde(default)]
2024-05-01 18:25:01 +03:00
pub struct HttpConfig {
2024-04-18 04:05:38 +03:00
pub host: IpAddr,
pub port: u16,
2024-05-01 18:25:01 +03:00
}
#[derive(Serialize, Deserialize, Debug, Clone)]
#[serde(default)]
pub struct DirsConfig {
pub posts: PathBuf,
pub media: PathBuf,
#[serde(rename = "static")]
pub _static: PathBuf,
2024-05-01 18:25:01 +03:00
}
2024-05-02 19:23:20 +03:00
#[derive(Serialize, Deserialize, Debug, Clone)]
pub struct RssConfig {
pub enable: bool,
pub link: Url,
}
#[derive(Serialize, Deserialize, Debug, Clone, Default)]
pub enum DateFormat {
#[default]
RFC3339,
#[serde(untagged)]
Strftime(String),
}
2024-05-01 18:25:01 +03:00
#[derive(Serialize, Deserialize, Debug, Clone)]
#[serde(default)]
pub struct Config {
2024-04-18 04:05:38 +03:00
pub title: String,
pub description: String,
pub markdown_access: bool,
pub date_format: DateFormat,
pub js_enable: bool,
2024-07-01 02:53:04 +03:00
pub default_color: Option<String>,
2024-05-02 19:23:20 +03:00
pub rss: RssConfig,
2024-05-01 18:25:01 +03:00
pub dirs: DirsConfig,
pub http: HttpConfig,
2024-04-18 04:05:38 +03:00
pub render: RenderConfig,
2024-04-20 20:59:00 +03:00
pub cache: CacheConfig,
2024-04-18 04:05:38 +03:00
}
impl Default for Config {
fn default() -> Self {
Self {
2024-04-18 04:16:12 +03:00
title: "bingus-blog".into(),
2024-04-18 04:05:38 +03:00
description: "blazingly fast markdown blog software written in rust memory safe".into(),
markdown_access: true,
date_format: Default::default(),
js_enable: true,
2024-07-01 02:53:04 +03:00
default_color: Some("#f5c2e7".into()),
2024-05-02 19:23:20 +03:00
// i have a love-hate relationship with serde
// it was engimatic at first, but then i started actually using it
// writing my own serialize and deserialize implementations.. spending
// a lot of time in the docs trying to understand each and every option..
// now with this knowledge i can do stuff like this! (see rss field)
// and i'm proud to say that it still makes 0 sense.
rss: RssConfig {
enable: false,
link: Url::parse("http://example.com").unwrap(),
},
2024-05-01 18:25:01 +03:00
dirs: Default::default(),
http: Default::default(),
2024-04-18 04:05:38 +03:00
render: Default::default(),
2024-04-20 20:59:00 +03:00
cache: Default::default(),
2024-05-01 18:25:01 +03:00
}
}
}
impl Default for DirsConfig {
fn default() -> Self {
Self {
posts: "posts".into(),
media: "media".into(),
_static: "static".into(),
2024-05-01 18:25:01 +03:00
}
}
}
impl Default for HttpConfig {
fn default() -> Self {
Self {
host: IpAddr::V6(Ipv6Addr::UNSPECIFIED),
2024-05-01 18:25:01 +03:00
port: 3000,
2024-04-18 04:05:38 +03:00
}
}
}
2024-04-20 20:59:00 +03:00
impl Default for SyntectConfig {
2024-04-18 04:05:38 +03:00
fn default() -> Self {
Self {
2024-04-20 20:59:00 +03:00
load_defaults: false,
themes_dir: Some("themes".into()),
theme: Some("Catppuccin Mocha".into()),
2024-04-18 04:05:38 +03:00
}
}
}
2024-04-20 20:59:00 +03:00
impl Default for CacheConfig {
fn default() -> Self {
Self {
enable: true,
cleanup: true,
cleanup_interval: None,
persistence: true,
file: "cache".into(),
compress: true,
compression_level: 3,
2024-04-20 20:59:00 +03:00
}
}
}
#[instrument(name = "config")]
2024-04-18 04:05:38 +03:00
pub async fn load() -> Result<Config> {
2024-05-04 00:18:40 +03:00
let config_file = env::var(format!(
"{}_CONFIG",
env!("CARGO_BIN_NAME").to_uppercase().replace('-', "_")
2024-05-04 00:18:40 +03:00
))
.unwrap_or(String::from("config.toml"));
2024-04-18 04:05:38 +03:00
match tokio::fs::OpenOptions::new()
.read(true)
.open(&config_file)
.await
{
Ok(mut file) => {
let mut buf = String::new();
file.read_to_string(&mut buf)
.await
2024-04-18 19:17:33 +03:00
.context("couldn't read configuration file")?;
toml::from_str(&buf).context("couldn't parse configuration")
2024-04-18 04:05:38 +03:00
}
Err(err) => match err.kind() {
std::io::ErrorKind::NotFound => {
let config = Config::default();
info!("configuration file doesn't exist, creating");
match tokio::fs::OpenOptions::new()
.write(true)
.create(true)
.truncate(true)
.open(&config_file)
.await
{
Ok(mut file) => file
.write_all(
toml::to_string_pretty(&config)
2024-04-18 19:17:33 +03:00
.context("couldn't serialize configuration")?
2024-04-18 04:05:38 +03:00
.as_bytes(),
)
.await
.unwrap_or_else(|err| error!("couldn't write configuration: {}", err)),
Err(err) => {
error!("couldn't open file {:?} for writing: {}", &config_file, err)
}
}
Ok(config)
}
_ => bail!("couldn't open config file: {}", err),
},
}
}
fn check_zstd_level_bounds<'de, D>(d: D) -> Result<i32, D::Error>
where
D: serde::Deserializer<'de>,
{
d.deserialize_i32(RangedI128Visitor::<1, 22>)
.map(|x| x as i32)
}