2023-04-02 15:26:20 +02:00
|
|
|
use std::{collections::HashMap, sync::Arc, time::Duration};
|
2022-09-05 00:05:24 +02:00
|
|
|
|
2023-04-02 15:26:20 +02:00
|
|
|
use axum::extract::MatchedPath;
|
2023-03-26 13:05:39 +02:00
|
|
|
use color_eyre::eyre::{Error, Result};
|
2023-03-29 18:20:51 +02:00
|
|
|
|
2023-04-02 15:26:20 +02:00
|
|
|
use hyper::{Body, Request, Response};
|
2023-03-25 12:23:11 +01:00
|
|
|
use post::Post;
|
2023-03-29 18:20:51 +02:00
|
|
|
|
2023-03-29 21:48:27 +02:00
|
|
|
use tag::Tag;
|
2022-08-31 23:20:59 +02:00
|
|
|
use tera::Tera;
|
2023-04-02 15:26:20 +02:00
|
|
|
use tower_http::{compression::CompressionLayer, cors::CorsLayer};
|
|
|
|
use tracing::{info_span, log::*, Span};
|
|
|
|
use tracing_subscriber::{prelude::*, EnvFilter};
|
2022-06-16 23:44:37 +02:00
|
|
|
|
2022-08-31 23:20:59 +02:00
|
|
|
mod handlers;
|
2023-03-25 12:23:11 +01:00
|
|
|
mod post;
|
2023-03-29 21:48:27 +02:00
|
|
|
mod tag;
|
2023-04-03 23:33:25 +02:00
|
|
|
mod feed;
|
2022-08-31 23:20:59 +02:00
|
|
|
|
2023-03-29 18:03:54 +02:00
|
|
|
pub struct AppState {
|
2023-04-03 23:33:25 +02:00
|
|
|
base_url: String,
|
2023-03-25 16:14:53 +01:00
|
|
|
posts: HashMap<String, Post>,
|
2023-03-29 21:48:27 +02:00
|
|
|
tags: HashMap<String, Tag>,
|
2022-08-31 23:20:59 +02:00
|
|
|
tera: Tera,
|
|
|
|
}
|
|
|
|
|
2022-06-16 23:44:37 +02:00
|
|
|
#[tokio::main]
|
|
|
|
async fn main() -> Result<()> {
|
|
|
|
color_eyre::install()?;
|
2023-04-02 15:26:20 +02:00
|
|
|
init_tracing();
|
|
|
|
|
2022-06-16 23:44:37 +02:00
|
|
|
info!("Starting server...");
|
|
|
|
|
2023-04-03 23:33:25 +02:00
|
|
|
let base_url = option_env!("SITE_BASE_URL").unwrap_or("http://localhost:8180").to_string();
|
2023-03-29 18:03:54 +02:00
|
|
|
let tera = Tera::new("templates/**/*")?;
|
|
|
|
let posts = post::load_all().await?;
|
2023-03-29 21:48:27 +02:00
|
|
|
let tags = tag::get_tags(posts.values());
|
2023-04-03 23:33:25 +02:00
|
|
|
let state = Arc::new(AppState { base_url, tera, posts, tags });
|
2023-03-29 18:03:54 +02:00
|
|
|
|
2023-03-29 18:19:39 +02:00
|
|
|
let app = handlers::routes(&state)
|
2023-03-29 21:48:27 +02:00
|
|
|
.layer(CorsLayer::permissive())
|
2023-03-29 18:03:54 +02:00
|
|
|
.layer(CompressionLayer::new())
|
2023-04-02 15:27:06 +02:00
|
|
|
.layer(
|
|
|
|
tower_http::trace::TraceLayer::new_for_http()
|
|
|
|
.make_span_with(make_span)
|
|
|
|
.on_response(on_response),
|
|
|
|
)
|
2023-03-29 18:03:54 +02:00
|
|
|
.with_state(state);
|
2022-09-05 00:02:58 +02:00
|
|
|
|
2023-03-25 16:14:53 +01:00
|
|
|
info!("Now listening at http://localhost:8180");
|
|
|
|
|
|
|
|
axum::Server::bind(&"0.0.0.0:8180".parse().unwrap())
|
|
|
|
.serve(app.into_make_service())
|
|
|
|
.await?;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2023-04-02 15:26:20 +02:00
|
|
|
fn init_tracing() {
|
|
|
|
let filter = EnvFilter::builder()
|
|
|
|
.with_default_directive("into".parse().unwrap())
|
2023-04-03 23:33:25 +02:00
|
|
|
.from_env_lossy();
|
2023-04-02 15:26:20 +02:00
|
|
|
|
|
|
|
tracing_subscriber::registry()
|
|
|
|
.with(filter)
|
|
|
|
.with(tracing_subscriber::fmt::layer())
|
|
|
|
.init();
|
|
|
|
}
|
|
|
|
|
|
|
|
fn make_span(request: &Request<Body>) -> Span {
|
|
|
|
let uri = request.uri();
|
|
|
|
let route = request
|
|
|
|
.extensions()
|
|
|
|
.get::<MatchedPath>()
|
|
|
|
.map(|mp| mp.as_str())
|
|
|
|
.unwrap_or_default();
|
|
|
|
let method = request.method().as_str();
|
2023-04-02 15:27:06 +02:00
|
|
|
let target = uri.path_and_query().map(|p| p.as_str()).unwrap_or_default();
|
2023-04-02 15:26:20 +02:00
|
|
|
|
|
|
|
let name = format!("{method} {route}");
|
|
|
|
|
|
|
|
use tracing::field::Empty;
|
|
|
|
info_span!(
|
|
|
|
"request",
|
|
|
|
otel.name = %name,
|
|
|
|
http.route = %route,
|
|
|
|
http.method = %method,
|
|
|
|
http.target = %target,
|
|
|
|
http.status_code = Empty
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn on_response<B>(response: &Response<B>, _latency: Duration, span: &Span) {
|
|
|
|
span.record("http.status_code", response.status().as_str());
|
|
|
|
}
|
|
|
|
|
2023-03-25 21:38:16 +01:00
|
|
|
#[derive(Debug)]
|
|
|
|
pub enum WebsiteError {
|
|
|
|
NotFound,
|
2023-03-26 12:40:25 +02:00
|
|
|
InternalError(Error),
|
2023-03-25 21:38:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<E> From<E> for WebsiteError
|
|
|
|
where
|
2023-03-26 12:40:25 +02:00
|
|
|
E: Into<Error>,
|
2023-03-25 21:38:16 +01:00
|
|
|
{
|
|
|
|
fn from(value: E) -> Self {
|
|
|
|
WebsiteError::InternalError(value.into())
|
|
|
|
}
|
|
|
|
}
|