2023-10-31 17:23:15 +01:00
|
|
|
use std::{sync::Arc, path::PathBuf};
|
|
|
|
|
use axum::{
|
|
|
|
|
extract::{ws::{WebSocketUpgrade, WebSocket, Message}, State},
|
|
|
|
|
response::Response,
|
|
|
|
|
routing::get,
|
|
|
|
|
Router,
|
|
|
|
|
};
|
2025-03-15 23:36:47 -04:00
|
|
|
use flate2::{write::GzEncoder, Compression};
|
|
|
|
|
use std::io::Write;
|
2023-10-31 17:23:15 +01:00
|
|
|
use tokio::sync::RwLock;
|
|
|
|
|
use tower_http::services::ServeDir;
|
|
|
|
|
|
2023-12-30 18:07:55 +01:00
|
|
|
use crate::comms::RadarData;
|
2023-10-31 17:23:15 +01:00
|
|
|
|
|
|
|
|
#[derive(Clone)]
|
|
|
|
|
struct AppState {
|
|
|
|
|
data_lock: Arc<RwLock<RadarData>>
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async fn ws_handler(ws: WebSocketUpgrade, State(state): State<AppState>) -> Response {
|
2024-01-08 00:22:24 +01:00
|
|
|
let clone = state.clone();
|
|
|
|
|
ws.on_upgrade(|socket| handle_socket(socket, clone))
|
2023-10-31 17:23:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async fn handle_socket(mut socket: WebSocket, state: AppState) {
|
2025-03-15 23:36:47 -04:00
|
|
|
let mut compression_buffer: Vec<u8> = Vec::with_capacity(65536);
|
|
|
|
|
|
2023-10-31 17:23:15 +01:00
|
|
|
while let Some(msg) = socket.recv().await {
|
|
|
|
|
if let Ok(msg) = msg {
|
2025-03-14 17:29:12 -04:00
|
|
|
if let Ok(text) = msg.to_text() {
|
|
|
|
|
if text == "requestInfo" {
|
2025-03-15 23:36:47 -04:00
|
|
|
let radar_data = state.data_lock.read().await;
|
2024-01-08 00:22:24 +01:00
|
|
|
|
2025-03-15 23:36:47 -04:00
|
|
|
if let Ok(json) = serde_json::to_string(&*radar_data) {
|
|
|
|
|
compression_buffer.clear();
|
|
|
|
|
|
|
|
|
|
let mut encoder = GzEncoder::new(Vec::new(), Compression::fast());
|
|
|
|
|
if encoder.write_all(json.as_bytes()).is_ok() {
|
|
|
|
|
match encoder.finish() {
|
|
|
|
|
Ok(compressed) => {
|
|
|
|
|
let mut message = vec![0x01];
|
|
|
|
|
message.extend_from_slice(&compressed);
|
2025-03-14 17:29:12 -04:00
|
|
|
|
2025-03-15 23:36:47 -04:00
|
|
|
let _ = socket.send(Message::Binary(message)).await;
|
|
|
|
|
},
|
|
|
|
|
Err(_) => {
|
|
|
|
|
let mut uncompressed = vec![0x00];
|
|
|
|
|
uncompressed.extend_from_slice(json.as_bytes());
|
|
|
|
|
let _ = socket.send(Message::Binary(uncompressed)).await;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
let mut uncompressed = vec![0x00];
|
|
|
|
|
uncompressed.extend_from_slice(json.as_bytes());
|
|
|
|
|
let _ = socket.send(Message::Binary(uncompressed)).await;
|
|
|
|
|
}
|
2023-10-31 17:23:15 +01:00
|
|
|
}
|
2025-03-14 17:29:12 -04:00
|
|
|
} else if text == "toggleMoneyReveal" {
|
|
|
|
|
let new_value = {
|
|
|
|
|
let mut data = state.data_lock.write().await;
|
|
|
|
|
data.money_reveal_enabled = !data.money_reveal_enabled;
|
|
|
|
|
data.money_reveal_enabled
|
|
|
|
|
};
|
2023-10-31 17:23:15 +01:00
|
|
|
|
2025-03-14 17:29:12 -04:00
|
|
|
let response = serde_json::json!({
|
|
|
|
|
"action": "toggleMoneyReveal",
|
|
|
|
|
"status": "success",
|
|
|
|
|
"enabled": new_value
|
|
|
|
|
});
|
2024-01-08 00:22:24 +01:00
|
|
|
|
2025-03-15 23:36:47 -04:00
|
|
|
let _ = socket.send(Message::Text(response.to_string())).await;
|
2023-10-31 17:23:15 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
2025-03-15 23:36:47 -04:00
|
|
|
break;
|
2023-10-31 17:23:15 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub async fn run(path: PathBuf, port: u16, data_lock: Arc<RwLock<RadarData>>) -> anyhow::Result<()> {
|
|
|
|
|
let app = Router::new()
|
|
|
|
|
.nest_service("/", ServeDir::new(path))
|
|
|
|
|
.route("/ws", get(ws_handler))
|
|
|
|
|
.with_state(AppState { data_lock });
|
|
|
|
|
|
|
|
|
|
let address = format!("0.0.0.0:{}", port);
|
2025-03-15 23:36:47 -04:00
|
|
|
log::info!("Starting WebSocket server on {}", address);
|
2024-04-03 17:59:53 +02:00
|
|
|
let listener = tokio::net::TcpListener::bind(address).await?;
|
|
|
|
|
axum::serve(listener, app.into_make_service())
|
2023-10-31 17:23:15 +01:00
|
|
|
.await?;
|
|
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|