morethantext/src/main.rs

239 lines
7.2 KiB
Rust
Raw Normal View History

use axum::{
2025-04-26 10:29:58 -04:00
extract::{Extension, FromRequestParts, Path, State},
http::{request::Parts, Method, StatusCode},
response::IntoResponse,
2025-04-23 10:46:33 -04:00
routing::{get, post},
RequestPartsExt, Router,
};
use clap::Parser;
2025-04-25 14:02:40 -04:00
use morethantext::{ActionType, MoreThanText};
2025-04-26 10:29:58 -04:00
use std::{collections::HashMap, convert::Infallible};
use tokio::{spawn, sync::mpsc::channel};
use tower_cookies::{Cookie, 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();
}
async fn create_app(state: MoreThanText) -> Router {
Router::new()
.route("/", get(mtt_conn))
2025-04-24 12:00:17 -04:00
.route("/{document}", get(mtt_conn))
2025-04-23 11:12:15 -04:00
.route("/api/{document}", post(mtt_conn))
.layer(CookieManagerLayer::new())
.layer(Extension(state.clone()))
.with_state(state)
}
#[derive(Clone)]
struct SessionID(Uuid);
impl<S> FromRequestParts<S> for SessionID
where
S: Send + Sync,
{
type Rejection = Infallible;
2025-04-21 22:29:15 -04:00
async fn from_request_parts(parts: &mut Parts, _state: &S) -> Result<Self, Self::Rejection> {
let Extension(cookies) = parts.extract::<Extension<Cookies>>().await.unwrap();
let Extension(mut state) = parts.extract::<Extension<MoreThanText>>().await.unwrap();
let req_id = match cookies.get(SESSION_KEY) {
Some(cookie) => Some(cookie.value().to_string()),
None => None,
};
let requested = req_id.clone();
let (tx, mut rx) = channel(1);
spawn(async move {
tx.send(state.validate_session(requested)).await.unwrap();
});
let id = rx.recv().await.unwrap();
if !req_id.is_some_and(|x| x == id.to_string()) {
cookies.add(Cookie::new(SESSION_KEY, id.to_string()));
}
Ok(SessionID(id))
}
}
2025-04-26 10:29:58 -04:00
async fn mtt_conn(
sess_id: SessionID,
method: Method,
path: Path<HashMap<String, String>>,
state: State<MoreThanText>,
2025-05-02 18:04:17 -04:00
body: String,
2025-04-26 10:29:58 -04:00
) -> impl IntoResponse {
2025-04-22 08:31:25 -04:00
let (tx, mut rx) = channel(1);
2025-04-26 10:29:58 -04:00
let action = match method {
Method::GET => ActionType::Get,
Method::POST => ActionType::Add,
_ => unreachable!("reouter should prevent this"),
};
let doc = match path.get("document") {
Some(result) => result.clone(),
None => "root".to_string(),
};
2025-04-22 08:31:25 -04:00
spawn(async move {
2025-05-02 18:04:17 -04:00
tx.send(state.get_document(sess_id.0, action, doc, body))
2025-04-26 10:29:58 -04:00
.await
.unwrap();
2025-04-22 08:31:25 -04:00
});
2025-04-25 14:02:40 -04:00
let reply = rx.recv().await.unwrap();
let status = match reply.get_error() {
Some(_) => StatusCode::NOT_FOUND,
None => StatusCode::OK,
};
(status, reply.get_document())
}
#[cfg(test)]
mod servers {
use super::*;
use axum::{
body::Body,
http::{
header::{COOKIE, SET_COOKIE},
2025-04-26 10:29:58 -04:00
Method, Request,
},
};
2025-05-02 18:04:17 -04:00
use http_body_util::BodyExt;
use serde_json::json;
use std::time::Duration;
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 cookie_only_issued_once() {
let app = create_app(MoreThanText::new()).await;
let initial = app
.clone()
.oneshot(Request::builder().uri("/").body(Body::empty()).unwrap())
.await
.unwrap();
assert_eq!(initial.status(), StatusCode::OK);
let sessid = initial.headers().get(SET_COOKIE).unwrap();
2025-04-21 22:29:15 -04:00
let request = Request::builder()
.uri("/")
.header(COOKIE, sessid.clone())
.body(Body::empty())
.unwrap();
let response = app.clone().oneshot(request).await.unwrap();
assert_eq!(response.status(), StatusCode::OK);
match response.headers().get(SET_COOKIE) {
Some(info) => assert!(false, "first pass: {:?}, second pass: {:?}", sessid, info),
None => {}
}
}
2025-04-26 10:29:58 -04:00
#[tokio::test]
2025-04-16 10:16:41 -04:00
async fn receive_file_not_found() {
2025-04-24 12:00:17 -04:00
let uri = "/something";
2025-04-16 10:16:41 -04:00
let app = create_app(MoreThanText::new()).await;
let response = app
2025-04-26 10:29:58 -04:00
.oneshot(Request::builder().uri(uri).body(Body::empty()).unwrap())
2025-04-16 10:16:41 -04:00
.await
.unwrap();
2025-04-26 10:29:58 -04:00
assert_eq!(
response.status(),
StatusCode::NOT_FOUND,
"'{}' should not exist",
uri
);
2025-04-16 10:16:41 -04:00
}
2025-04-23 10:46:33 -04:00
2025-05-02 08:09:05 -04:00
#[tokio::test]
2025-04-23 10:46:33 -04:00
async fn add_new_page() {
2025-04-24 12:00:17 -04:00
let base = "/something".to_string();
let api = format!("/api{}", &base);
2025-05-02 18:04:17 -04:00
let content = format!("content-{}", Uuid::new_v4());
let document = json!({
"template": content.clone()
});
2025-04-23 10:46:33 -04:00
let app = create_app(MoreThanText::new()).await;
let response = app
2025-04-24 12:00:17 -04:00
.clone()
2025-04-23 10:46:33 -04:00
.oneshot(
Request::builder()
.method(Method::POST)
.uri(&api)
2025-05-02 18:04:17 -04:00
.body(document.to_string())
2025-04-23 10:46:33 -04:00
.unwrap(),
)
.await
.unwrap();
2025-04-26 10:29:58 -04:00
assert_eq!(
response.status(),
StatusCode::OK,
"failed to post ro {:?}",
api
);
2025-04-24 12:00:17 -04:00
let response = app
2025-04-26 10:29:58 -04:00
.oneshot(Request::builder().uri(&base).body(Body::empty()).unwrap())
2025-04-24 12:00:17 -04:00
.await
.unwrap();
2025-04-26 10:29:58 -04:00
assert_eq!(
response.status(),
StatusCode::OK,
"failed to get ro {:?}",
base
);
2025-05-02 18:04:17 -04:00
let body = response.into_body().collect().await.unwrap().to_bytes();
assert_eq!(body, content);
2025-04-23 10:46:33 -04:00
}
2024-02-26 08:41:24 -05:00
}