morethantext/src/main.rs

142 lines
3.9 KiB
Rust
Raw Normal View History

use axum::{
async_trait,
extract::{Extension, FromRequestParts, State},
http::request::Parts,
response::IntoResponse,
routing::get,
RequestPartsExt,
Router,
};
2024-03-17 15:40:00 -04:00
use axum_extra::extract::cookie::{Cookie, CookieJar};
use clap::Parser;
2024-05-12 14:10:36 -04:00
use morethantext::MoreThanText;
use std::convert::Infallible;
use tokio::{spawn, sync::mpsc::channel};
use tower_cookies::{CookieManagerLayer, Cookies};
use uuid::Uuid;
2024-02-29 18:46:01 -05:00
const LOCALHOST: &str = "127.0.0.1";
2024-03-11 10:45:20 -04:00
const SESSION_KEY: &str = "sessionid";
2024-02-29 18:46:01 -05:00
#[derive(Parser, Debug)]
#[command(version, about, long_about = None)]
struct Args {
/// Post used
#[arg(short, long, default_value_t = 3000)]
2024-02-29 18:46:01 -05:00
port: u16,
/// IP used
#[arg(short, long, default_value_t = LOCALHOST.to_string())]
address: String,
2024-03-29 07:51:14 -04:00
/// cluster host
#[arg(short, long, num_args(0..))]
2024-07-30 15:11:31 -04:00
node: Vec<String>,
}
#[tokio::main]
async fn main() {
let args = Args::parse();
2024-03-11 10:45:20 -04:00
let addr = format!("{}:{}", args.address, args.port);
2024-11-06 21:05:52 -05:00
let state = MoreThanText::new();
2025-04-16 10:16:41 -04:00
let app = create_app(state).await;
2024-03-11 10:45:20 -04:00
let listener = tokio::net::TcpListener::bind(&addr).await.unwrap();
axum::serve(listener, app.into_make_service())
.await
.unwrap();
}
struct SessionID;
#[async_trait]
impl<S> FromRequestParts<S> for SessionID
where
S: Send + Sync,
{
type Rejection = Infallible;
async fn from_request_parts(parts: &mut Parts, state: &S) -> Result<Self, Self::Rejection> {
let cookies = parts.extract::<Extension<Cookies>>();
Ok(Self {})
}
}
2025-04-16 10:16:41 -04:00
async fn create_app(state: MoreThanText) -> Router {
Router::new().route("/", get(mtt_conn)).with_state(state)
}
async fn mtt_conn(
jar: CookieJar,
sess_id: SessionID,
state: State<MoreThanText>,
) -> impl IntoResponse {
let sid = match jar.get(SESSION_KEY) {
Some(cookie) => Some(cookie.value().to_string()),
None => None,
};
2025-04-08 13:13:06 -04:00
let sess_info = sid.clone();
let (tx, mut rx) = channel(5);
spawn(async move {
2025-04-08 13:13:06 -04:00
tx.send(state.clone().request(sess_info)).await.unwrap();
});
let reply = rx.recv().await.unwrap();
let cookie = Cookie::build((SESSION_KEY, reply.get_data("sess_id").unwrap().to_string()));
let cookies = jar.add(cookie);
(cookies, reply.get_data("dov").unwrap().to_string())
}
#[cfg(test)]
mod servers {
use super::*;
use axum::{
body::Body,
http::{Request, StatusCode},
};
2025-04-08 13:13:06 -04:00
use tower::ServiceExt;
#[tokio::test]
async fn get_home_page() {
2025-04-16 10:16:41 -04:00
let app = create_app(MoreThanText::new()).await;
let response = app
.oneshot(Request::builder().uri("/").body(Body::empty()).unwrap())
.await
.unwrap();
assert_eq!(response.status(), StatusCode::OK);
let sessid = format!("{:?}", response.headers().get("set-cookie").unwrap());
assert!(sessid.contains(SESSION_KEY), "did not set session id");
}
#[tokio::test]
async fn session_ids_are_unique() {
2025-04-16 10:16:41 -04:00
let app = create_app(MoreThanText::new()).await;
let mut holder: Vec<String> = Vec::new();
for _ in 0..5 {
let response = app
.clone()
.oneshot(Request::builder().uri("/").body(Body::empty()).unwrap())
.await
.unwrap();
let sessid = format!("{:?}", response.headers().get("set-cookie").unwrap());
assert!(
!holder.contains(&sessid),
"found duplicate entry: {:?}",
holder
);
holder.push(sessid);
}
}
2025-04-16 10:16:41 -04:00
#[tokio::test]
async fn receive_file_not_found() {
let app = create_app(MoreThanText::new()).await;
let response = app
.oneshot(
Request::builder()
.uri("/isomething")
.body(Body::empty())
.unwrap(),
)
.await
.unwrap();
assert_eq!(response.status(), StatusCode::NOT_FOUND);
}
2024-02-26 08:41:24 -05:00
}