summaryrefslogtreecommitdiff
path: root/src/main.rs
blob: a72ec5f3a8efd4ee218db191a3141eb3c2ebf2f0 (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
84
85
86
87
88
89
90
91
use axum::{
    body::HttpBody,
    http::{Request, StatusCode},
    middleware::{self, Next},
    response::Response,
    RequestExt, Router,
};
use axum_client_ip::ClientIp;
use std::{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::http::ResponseCode;

use crate::api::pages;

mod admin;
mod api;
mod console;
mod database;
mod types;

async fn serve<B>(req: Request<B>, next: Next<B>) -> Response
where
    B: Send + Sync + 'static + HttpBody,
{
    let uri = req.uri();
    let file = ResponseCode::Success.file(&uri.to_string()).await;
    if file.status() != StatusCode::OK {
        return next.run(req).await;
    }
    file
}

async fn log<B>(mut req: Request<B>, next: Next<B>) -> Response
where
    B: Send + Sync + 'static + HttpBody,
{
    let Ok(ClientIp(ip)) = req.extract_parts::<ClientIp>().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 {
    ResponseCode::NotFound.file("/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)
    };

    let app = Router::new()
        .fallback(not_found)
        .layer(middleware::from_fn(log))
        .layer(middleware::from_fn(serve))
        .nest("/", pages::router())
        .nest("/api", api::router())
        .layer(CookieManagerLayer::new());

    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(());
}