summaryrefslogtreecommitdiff
path: root/src/main.rs
blob: 8a8e9d8e5da86247f7cb4091eb4fcdcb4d09ae6a (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
use axum::{
    body::HttpBody,
    extract::DefaultBodyLimit,
    http::Request,
    middleware::{self, Next},
    response::Response,
    RequestExt, Router,
};
use public::console;
use std::{fs, net::SocketAddr, process::exit};
use tower_cookies::CookieManagerLayer;
use tracing::{error, info, metadata::LevelFilter};
use tracing_subscriber::{
    filter::filter_fn, prelude::__tracing_subscriber_SubscriberExt, util::SubscriberInitExt, Layer,
};
use types::extract::RequestIp;

mod api;
mod database;
mod public;
mod types;

async fn log<B>(mut req: Request<B>, next: Next<B>) -> Response
where
    B: Send + Sync + 'static + HttpBody,
{
    let Ok(RequestIp(ip)) = req.extract_parts::<RequestIp>().await else {
        return next.run(req).await
    };

    console::log(ip, req.method().clone(), req.uri().clone(), None, None).await;

    next.run(req).await
}

async fn not_found() -> Response {
    public::serve("/404.html").await
}

#[tokio::main]
async fn main() {
    let fmt_layer = tracing_subscriber::fmt::layer();
    tracing_subscriber::registry()
        .with(
            fmt_layer
                .with_filter(LevelFilter::TRACE)
                .with_filter(filter_fn(|metadata| {
                    metadata.target().starts_with("xssbook")
                })),
        )
        .init();

    if database::init().is_err() {
        error!("Failed to connect to the sqlite database");
        exit(1)
    };

    fs::create_dir_all("./public/image/custom").expect("Coudn't make custom data directory");
    fs::create_dir_all("./public/image/custom/avatar")
        .expect("Coudn't make custom avatar directory");
    fs::create_dir_all("./public/image/custom/banner")
        .expect("Coudn't make custom banner directory");

    let app = Router::new()
        .fallback(not_found)
        .layer(middleware::from_fn(log))
        .nest("/", public::router())
        .nest("/api", api::router())
        .layer(CookieManagerLayer::new())
        .layer(DefaultBodyLimit::max(512_000));

    let Ok(addr) = "[::]:8080".parse::<std::net::SocketAddr>() else {
        error!("Failed to parse port binding");
        exit(1)
    };

    info!("listening on {}", addr);

    axum::Server::bind(&addr)
        .serve(app.into_make_service_with_connect_info::<SocketAddr>())
        .await
        .unwrap_or(());
}