Basic steam login and demo upload
This commit is contained in:
2
.gitignore
vendored
Normal file
2
.gitignore
vendored
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
target/
|
||||||
|
uploads/
|
||||||
2816
Cargo.lock
generated
Normal file
2816
Cargo.lock
generated
Normal file
File diff suppressed because it is too large
Load Diff
@@ -1,2 +1,3 @@
|
|||||||
[workspace]
|
[workspace]
|
||||||
members = []
|
members = ["backend", "frontend"]
|
||||||
|
resolver = "2"
|
||||||
|
|||||||
@@ -1,2 +1,11 @@
|
|||||||
# Knifer
|
# Knifer
|
||||||
A self-hosted demo analysis tool
|
A self-hosted demo analysis tool
|
||||||
|
|
||||||
|
## Development
|
||||||
|
### Frontend
|
||||||
|
1. Navigate to the frontend folder
|
||||||
|
2. Run `trunk watch`
|
||||||
|
|
||||||
|
### Backend
|
||||||
|
1. Navigate to the root folder
|
||||||
|
2. Run `cargo run --bin backend`
|
||||||
|
|||||||
17
backend/Cargo.toml
Normal file
17
backend/Cargo.toml
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
[package]
|
||||||
|
name = "backend"
|
||||||
|
version = "0.1.0"
|
||||||
|
edition = "2021"
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
async-trait = "0.1.82"
|
||||||
|
axum = { version = "0.7.5", features = ["multipart"] }
|
||||||
|
serde = { version = "1.0.210", features = ["derive"] }
|
||||||
|
steam-openid = "0.2.0"
|
||||||
|
time = "0.3.36"
|
||||||
|
tokio = { version = "1.40.0", features = ["rt", "macros", "net", "mio"] }
|
||||||
|
tower-sessions = "0.13.0"
|
||||||
|
tower-http = { version = "0.5", features = ["fs"] }
|
||||||
|
tracing = { version = "0.1.40", features = ["async-await"] }
|
||||||
|
tracing-subscriber = "0.3.18"
|
||||||
|
futures-util = "0.3"
|
||||||
76
backend/src/lib.rs
Normal file
76
backend/src/lib.rs
Normal file
@@ -0,0 +1,76 @@
|
|||||||
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, Deserialize, Serialize)]
|
||||||
|
pub struct UserSessionData {
|
||||||
|
pub steam_id: Option<u64>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for UserSessionData {
|
||||||
|
fn default() -> Self {
|
||||||
|
Self { steam_id: None }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct UserSession {
|
||||||
|
pub session: tower_sessions::Session,
|
||||||
|
data: UserSessionData,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl UserSession {
|
||||||
|
const KEY: &'static str = "user.data";
|
||||||
|
|
||||||
|
pub fn data(&self) -> &UserSessionData {
|
||||||
|
&self.data
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn modify_data<F>(&mut self, func: F)
|
||||||
|
where
|
||||||
|
F: FnOnce(&mut UserSessionData),
|
||||||
|
{
|
||||||
|
let mut entry = &mut self.data;
|
||||||
|
func(&mut entry);
|
||||||
|
|
||||||
|
self.session.insert(Self::KEY, entry).await.unwrap();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[async_trait::async_trait]
|
||||||
|
impl<S> axum::extract::FromRequestParts<S> for UserSession
|
||||||
|
where
|
||||||
|
S: Send + Sync,
|
||||||
|
{
|
||||||
|
type Rejection = (axum::http::StatusCode, &'static str);
|
||||||
|
|
||||||
|
async fn from_request_parts(
|
||||||
|
req: &mut axum::http::request::Parts,
|
||||||
|
state: &S,
|
||||||
|
) -> Result<Self, Self::Rejection> {
|
||||||
|
let session = tower_sessions::Session::from_request_parts(req, state).await?;
|
||||||
|
|
||||||
|
let guest_data: UserSessionData = session.get(Self::KEY).await.unwrap().unwrap_or_default();
|
||||||
|
|
||||||
|
Ok(Self {
|
||||||
|
session,
|
||||||
|
data: guest_data,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn get_demo_from_upload(name: &str, mut form: axum::extract::Multipart) -> Option<axum::body::Bytes> {
|
||||||
|
while let Ok(field) = form.next_field().await {
|
||||||
|
let field = match field {
|
||||||
|
Some(f) => f,
|
||||||
|
None => continue,
|
||||||
|
};
|
||||||
|
|
||||||
|
if field.name().map(|n| n != name).unwrap_or(false) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Ok(data) = field.bytes().await {
|
||||||
|
return Some(data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
None
|
||||||
|
}
|
||||||
105
backend/src/main.rs
Normal file
105
backend/src/main.rs
Normal file
@@ -0,0 +1,105 @@
|
|||||||
|
use tracing_subscriber::prelude::__tracing_subscriber_SubscriberExt;
|
||||||
|
|
||||||
|
static OPENID: std::sync::LazyLock<steam_openid::SteamOpenId> = std::sync::LazyLock::new(|| {
|
||||||
|
steam_openid::SteamOpenId::new("http://192.168.0.156:3000", "/api/steam/callback").unwrap()
|
||||||
|
});
|
||||||
|
static UPLOAD_FOLDER: &str = "uploads/";
|
||||||
|
|
||||||
|
#[tokio::main(flavor = "current_thread")]
|
||||||
|
async fn main() {
|
||||||
|
let registry = tracing_subscriber::Registry::default()
|
||||||
|
.with(tracing_subscriber::fmt::layer())
|
||||||
|
.with(tracing_subscriber::filter::filter_fn(|meta| {
|
||||||
|
meta.target().contains("backend")
|
||||||
|
}));
|
||||||
|
tracing::subscriber::set_global_default(registry).unwrap();
|
||||||
|
|
||||||
|
tracing::info!("Starting...");
|
||||||
|
|
||||||
|
let session_store = tower_sessions::MemoryStore::default();
|
||||||
|
let session_layer = tower_sessions::SessionManagerLayer::new(session_store)
|
||||||
|
.with_secure(false)
|
||||||
|
.with_expiry(tower_sessions::Expiry::OnInactivity(
|
||||||
|
time::Duration::minutes(15),
|
||||||
|
));
|
||||||
|
|
||||||
|
if !tokio::fs::try_exists(UPLOAD_FOLDER).await.unwrap_or(false) {
|
||||||
|
tokio::fs::create_dir_all(UPLOAD_FOLDER).await.unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
let router = axum::Router::new()
|
||||||
|
.nest_service(
|
||||||
|
"/api/",
|
||||||
|
axum::Router::new()
|
||||||
|
.route("/steam/callback", axum::routing::get(steam_callback))
|
||||||
|
.route("/steam/login", axum::routing::get(steam_login))
|
||||||
|
.route("/demos/upload", axum::routing::post(upload).layer(axum::extract::DefaultBodyLimit::max(1024*1024*500)))
|
||||||
|
.route("/demos/list", axum::routing::get(demos_list))
|
||||||
|
)
|
||||||
|
.layer(session_layer)
|
||||||
|
.nest_service("/", tower_http::services::ServeDir::new("frontend/dist/"));
|
||||||
|
|
||||||
|
let listener = tokio::net::TcpListener::bind("0.0.0.0:3000").await.unwrap();
|
||||||
|
axum::serve(listener, router).await.unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn upload(session: backend::UserSession, form: axum::extract::Multipart) -> Result<axum::response::Redirect, (axum::http::StatusCode, &'static str)> {
|
||||||
|
let steam_id = session.data().steam_id.ok_or_else(|| (axum::http::StatusCode::UNAUTHORIZED, "Not logged in"))?;
|
||||||
|
|
||||||
|
tracing::info!("Upload for Session: {:?}", steam_id);
|
||||||
|
|
||||||
|
let file_content = backend::get_demo_from_upload("demo", form).await.unwrap();
|
||||||
|
|
||||||
|
let user_folder = std::path::Path::new(UPLOAD_FOLDER).join(format!("{}/", steam_id));
|
||||||
|
if !tokio::fs::try_exists(&user_folder).await.unwrap_or(false) {
|
||||||
|
tokio::fs::create_dir_all(&user_folder).await.unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
let timestamp_secs = std::time::SystemTime::now().duration_since(std::time::SystemTime::UNIX_EPOCH).unwrap().as_secs();
|
||||||
|
let demo_file_path = user_folder.join(format!("{}.dem", timestamp_secs));
|
||||||
|
|
||||||
|
tokio::fs::write(demo_file_path, file_content).await.unwrap();
|
||||||
|
|
||||||
|
// TODO
|
||||||
|
// Insert Demo into users list of demos and possibly queue demo for analysis?
|
||||||
|
|
||||||
|
Ok(axum::response::Redirect::to("/"))
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn steam_login() -> Result<axum::response::Redirect, axum::http::StatusCode> {
|
||||||
|
let url = OPENID.get_redirect_url();
|
||||||
|
|
||||||
|
Ok(axum::response::Redirect::to(url))
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn steam_callback(
|
||||||
|
mut session: backend::UserSession,
|
||||||
|
request: axum::extract::Request,
|
||||||
|
) -> Result<axum::response::Redirect, axum::http::StatusCode> {
|
||||||
|
tracing::info!("Steam Callback");
|
||||||
|
|
||||||
|
let query = request.uri().query().ok_or_else(|| {
|
||||||
|
tracing::error!("Missing query in parameters");
|
||||||
|
axum::http::StatusCode::BAD_REQUEST
|
||||||
|
})?;
|
||||||
|
|
||||||
|
let id = OPENID.verify(query).await.map_err(|e| {
|
||||||
|
tracing::error!("Verifying OpenID: {:?}", e);
|
||||||
|
axum::http::StatusCode::BAD_REQUEST
|
||||||
|
})?;
|
||||||
|
|
||||||
|
session
|
||||||
|
.modify_data(|data| {
|
||||||
|
data.steam_id = Some(id);
|
||||||
|
})
|
||||||
|
.await;
|
||||||
|
|
||||||
|
Ok(axum::response::Redirect::to("/"))
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn demos_list(session: backend::UserSession) -> Result<(), axum::http::StatusCode> {
|
||||||
|
let steam_id = session.data().steam_id.ok_or_else(|| axum::http::StatusCode::UNAUTHORIZED)?;
|
||||||
|
tracing::info!("SteamID: {:?}", steam_id);
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
1
frontend/.gitignore
vendored
Normal file
1
frontend/.gitignore
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
dist/
|
||||||
8
frontend/Cargo.toml
Normal file
8
frontend/Cargo.toml
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
[package]
|
||||||
|
name = "frontend"
|
||||||
|
version = "0.1.0"
|
||||||
|
edition = "2021"
|
||||||
|
|
||||||
|
[dependencies]
|
||||||
|
leptos = { version = "0.6", features = ["csr", "nightly"] }
|
||||||
|
reqwasm = "0.5.0"
|
||||||
5
frontend/index.html
Normal file
5
frontend/index.html
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
<head></head>
|
||||||
|
<body></body>
|
||||||
|
</html>
|
||||||
29
frontend/src/main.rs
Normal file
29
frontend/src/main.rs
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
use leptos::*;
|
||||||
|
use leptos::prelude::*;
|
||||||
|
|
||||||
|
async fn load_demos() -> usize {
|
||||||
|
let res = reqwasm::http::Request::get("/api/demos/list").send().await.unwrap();
|
||||||
|
dbg!(res);
|
||||||
|
|
||||||
|
0
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
let async_data = create_resource(|| (), |_| async move {
|
||||||
|
load_demos().await
|
||||||
|
});
|
||||||
|
|
||||||
|
mount_to_body(move || view! {
|
||||||
|
<p>"Hello, world!"</p>
|
||||||
|
<a href="/api/steam/login">Steam Login</a> { move || match async_data.get() {
|
||||||
|
None => 123,
|
||||||
|
Some(v) => v,
|
||||||
|
} }
|
||||||
|
|
||||||
|
<form action="/api/demos/upload" method="post" enctype="multipart/form-data">
|
||||||
|
Select File to upload
|
||||||
|
<input type="file" name="demo" id="demo"></input>
|
||||||
|
<input type="submit" value="Upload Image" name="submit"></input>
|
||||||
|
</form>
|
||||||
|
})
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user