limbusart/src/main.rs

207 lines
6.7 KiB
Rust
Raw Normal View History

use axum::{
extract::State,
response::{Html, IntoResponse},
routing::get,
Router,
};
2023-07-17 09:01:10 +03:00
use dashmap::DashMap;
use data::{Art, ArtKind, Data};
use error::AppResult;
use futures_util::TryFutureExt;
2023-07-17 09:01:10 +03:00
use http::Uri;
use maud::PreEscaped;
2023-07-17 09:38:54 +03:00
use std::{
ops::Deref,
sync::{Arc, Mutex},
2023-07-17 09:38:54 +03:00
};
2023-07-17 09:01:10 +03:00
mod data;
2023-07-17 09:01:10 +03:00
mod error;
#[tokio::main]
async fn main() {
let arts_file_path = get_conf("ARTS_PATH");
let arts = std::fs::read_to_string(&arts_file_path).unwrap();
2023-07-17 09:38:54 +03:00
let state = AppState::new(Data::parse(&arts).unwrap());
std::thread::spawn({
use signal_hook::{consts::SIGUSR2, iterator::Signals};
let state = state.clone();
move || {
let mut signals = Signals::new(&[SIGUSR2]).unwrap();
for _ in signals.forever() {
let data = std::fs::read_to_string(&arts_file_path).unwrap();
2023-07-17 09:38:54 +03:00
state.data.lock().unwrap().reload(&data).unwrap();
}
}
});
let app = Router::new().route("/", get(show_art)).with_state(state);
2023-07-17 09:01:10 +03:00
let listener = tokio::net::TcpListener::bind("127.0.0.1:3000")
.await
.unwrap();
println!("listening on {}", listener.local_addr().unwrap());
axum::serve(listener, app).await.unwrap();
}
#[axum::debug_handler]
2023-07-17 11:03:42 +03:00
async fn show_art(state: State<AppState>) -> AppResult<axum::response::Response> {
2023-07-17 09:38:54 +03:00
let art = state.data.lock().unwrap().pick_random_art().clone();
let image_link = if let Some(image_link) = state.direct_links.get(&art.url) {
image_link.to_string()
2023-07-17 09:01:10 +03:00
} else {
let image_link = match art.kind {
ArtKind::Twitter => fetch_twitter_image_link(&state.http, &art.url).await?,
2023-07-17 13:58:47 +03:00
ArtKind::Safebooru => fetch_safebooru_image_link(&state.http, &art.url).await?,
2023-07-17 09:01:10 +03:00
};
state
.direct_links
.insert(art.url.clone(), image_link.clone());
image_link
};
let page = render_page(&art, &image_link);
Ok(page.into_response())
2023-07-17 09:01:10 +03:00
}
const BODY_STYLE: &str =
2024-06-11 01:07:55 +03:00
"margin: 0px; background: #0e0e0e; height: 100vh; width: 100vw; display: flex; font-family: \"PT Mono\", monospace; font-weight: 400; font-style: normal; font-optical-sizing: auto;";
const IMG_STYLE: &str = "display: block; margin: auto; max-height: 100vh; max-width: 100vw;";
const ABOUT_STYLE: &str = "position: absolute; bottom: 0; font-size: 0.75em; color: #ffffff; background-color: #0e0e0eaa;";
fn get_page_head_common() -> PreEscaped<String> {
let title = get_conf("SITE_TITLE");
maud::html! {
meta charset="utf8";
link rel="preconnect" href="https://fonts.googleapis.com";
link rel="preconnect" href="https://fonts.gstatic.com" crossorigin;
2024-06-11 01:07:55 +03:00
link rel="stylesheet" href="https://fonts.googleapis.com/css2?family=PT+Mono&display=swap";
title { (title) }
}
}
fn render_page(art: &Art, image_link: &str) -> Html<String> {
let embed_title = get_conf("EMBED_TITLE");
let embed_content = get_conf("EMBED_DESC");
let embed_color = get_conf("EMBED_COLOR");
2023-07-17 09:01:10 +03:00
let content = maud::html! {
(maud::DOCTYPE)
head {
(get_page_head_common())
meta property="og:title" content=(embed_title);
meta property="og:description" content=(embed_content);
meta name="theme-color" content=(embed_color);
2023-07-17 09:01:10 +03:00
}
body style=(BODY_STYLE) {
img style=(IMG_STYLE) src=(image_link);
a style=(format!("{ABOUT_STYLE} left: 0;")) href=(art.url) target="_blank" {
2023-07-17 09:01:10 +03:00
"source: " (art.url)
}
a style=(format!("{ABOUT_STYLE} right: 0;")) href="https://gaze.systems" target="_blank" {
2023-07-17 09:01:10 +03:00
"website made by dusk"
2023-07-18 01:00:16 +03:00
br;
"report problems / feedback @ yusdacra on Discord"
2023-07-17 09:01:10 +03:00
}
}
};
Html(content.into_string())
}
2023-07-17 13:58:47 +03:00
async fn fetch_safebooru_image_link(http: &reqwest::Client, url: &Uri) -> AppResult<String> {
let mut id = String::new();
for (name, value) in form_urlencoded::parse(url.query().unwrap().as_bytes()) {
if name == "id" {
id = value.into_owned();
}
}
if id.is_empty() {
return Err("no id?".into());
}
let url = format!("https://safebooru.org/index.php?page=dapi&s=post&q=index&json=1&id={id}");
type Data = Vec<serde_json::Map<String, serde_json::Value>>;
async fn try_request(count: usize, url: &str, http: &reqwest::Client) -> AppResult<Data> {
println!("[safebooru] trying to fetch url (count {count}): {url}");
let req = http.get(url).build()?;
let resp = http.execute(req).await?.error_for_status()?;
let data = resp.json::<Data>().await?;
AppResult::Ok(data)
}
let data = try_request(0, &url, http)
.or_else(|_| try_request(1, &url, http))
.or_else(|_| try_request(2, &url, http))
.or_else(|_| try_request(3, &url, http))
.or_else(|_| try_request(4, &url, http))
.await?;
2023-07-17 13:58:47 +03:00
let image_filename = data[0].get("image").unwrap().as_str().unwrap();
let image_directory = data[0].get("directory").unwrap().as_str().unwrap();
Ok(format!(
"http://safebooru.org/images/{image_directory}/{image_filename}"
))
}
2023-07-17 09:01:10 +03:00
async fn fetch_twitter_image_link(http: &reqwest::Client, url: &Uri) -> AppResult<String> {
let fxurl = Uri::builder()
.scheme("https")
.authority("d.fxtwitter.com")
.path_and_query(url.path_and_query().unwrap().clone())
.build()?
.to_string();
2024-06-11 03:17:19 +03:00
println!("[fxtwitter] trying to fetch url: {fxurl}");
let req = http.get(&fxurl).build()?;
2023-07-17 09:01:10 +03:00
let resp = http.execute(req).await?.error_for_status()?;
let link = resp
.headers()
.get(http::header::LOCATION)
.ok_or_else(|| format!("twitter link {fxurl} did not return an image location"))?
.to_str()?;
// use webp format for direct twitter links since webp is cheaper
Ok(format!("{link}?format=webp"))
2023-07-17 02:42:55 +03:00
}
fn get_conf(name: &str) -> String {
std::env::var(name).unwrap()
}
struct InternalAppState {
// cached direct links to images
direct_links: DashMap<Uri, String>,
data: Mutex<Data>,
http: reqwest::Client,
}
#[derive(Clone)]
struct AppState {
internal: Arc<InternalAppState>,
}
impl AppState {
fn new(data: Data) -> Self {
Self {
internal: Arc::new(InternalAppState {
data: Mutex::new(data),
direct_links: Default::default(),
http: reqwest::ClientBuilder::new()
.redirect(reqwest::redirect::Policy::none())
2024-06-11 04:22:09 +03:00
.user_agent("limbusart 0.1.0")
.build()
.unwrap(),
}),
}
}
}
impl Deref for AppState {
type Target = InternalAppState;
fn deref(&self) -> &Self::Target {
&self.internal
}
}