mmm big sloppy commit
This commit is contained in:
parent
f9b6e57503
commit
10a72d3e4b
5 changed files with 42 additions and 116 deletions
4
Cargo.lock
generated
4
Cargo.lock
generated
|
@ -39,9 +39,9 @@ dependencies = [
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "autocfg"
|
name = "autocfg"
|
||||||
version = "1.2.0"
|
version = "1.3.0"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "f1fdabc7756949593fe60f30ec81974b613357de856987752631dea1e3394c80"
|
checksum = "0c4b4d0bd25bd0b74681c0ad21497610ce1b7c91b1022cd21c80c6fbdd9476b0"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "axum"
|
name = "axum"
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
use crate::api::NewGameRequest; // change this
|
||||||
use rand::prelude::IteratorRandom;
|
use rand::prelude::IteratorRandom;
|
||||||
use rand::thread_rng;
|
use rand::thread_rng;
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
@ -85,28 +86,6 @@ pub struct CAHGame {
|
||||||
pub current_black: Option<CAHCardBlack>,
|
pub current_black: Option<CAHCardBlack>,
|
||||||
}
|
}
|
||||||
|
|
||||||
/// New game request structure
|
|
||||||
#[derive(Debug, Deserialize)]
|
|
||||||
pub struct NewGameRequest {
|
|
||||||
/// Game name
|
|
||||||
pub name: String,
|
|
||||||
/// Game host
|
|
||||||
pub host: CAHPlayer,
|
|
||||||
/// Chosen packs
|
|
||||||
pub packs: Vec<u8>,
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Game join request structure
|
|
||||||
pub struct GameJoinRequest {
|
|
||||||
/// Game id
|
|
||||||
pub id: u8, // increase later
|
|
||||||
/// Game password
|
|
||||||
pub password: String,
|
|
||||||
/// Player info
|
|
||||||
pub player: CAHPlayer,
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
impl CAHGame {
|
impl CAHGame {
|
||||||
/// Build game decks from input data for game start.
|
/// Build game decks from input data for game start.
|
||||||
/// This should only run once and at startup.
|
/// This should only run once and at startup.
|
||||||
|
@ -188,10 +167,7 @@ impl CAHGame {
|
||||||
|
|
||||||
/// Create a new player and add them to the game.
|
/// Create a new player and add them to the game.
|
||||||
pub fn create_player(&mut self, mut player: CAHPlayer) -> Result<()> {
|
pub fn create_player(&mut self, mut player: CAHPlayer) -> Result<()> {
|
||||||
println!(
|
println!("Creating player {} as {:?}", &player.name, &player.role);
|
||||||
"Creating player {} as {:?}",
|
|
||||||
&player.name, &player.role
|
|
||||||
);
|
|
||||||
|
|
||||||
let mut hand_buf = vec![];
|
let mut hand_buf = vec![];
|
||||||
for _ in 0..10 {
|
for _ in 0..10 {
|
||||||
|
|
114
src/api.rs
114
src/api.rs
|
@ -8,8 +8,30 @@ use axum::{
|
||||||
response::IntoResponse,
|
response::IntoResponse,
|
||||||
};
|
};
|
||||||
use futures::{sink::SinkExt, stream::StreamExt};
|
use futures::{sink::SinkExt, stream::StreamExt};
|
||||||
|
use serde::Deserialize;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
|
|
||||||
|
/// New game request structure
|
||||||
|
#[derive(Debug, Deserialize)]
|
||||||
|
pub struct NewGameRequest {
|
||||||
|
/// Game name
|
||||||
|
pub name: String,
|
||||||
|
/// Game host
|
||||||
|
pub host: CAHPlayer,
|
||||||
|
/// Chosen packs
|
||||||
|
pub packs: Vec<u8>,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Game join request structure
|
||||||
|
pub struct GameJoinRequest {
|
||||||
|
/// Game id
|
||||||
|
pub id: u8, // increase later
|
||||||
|
/// Game password
|
||||||
|
pub password: String,
|
||||||
|
/// Player info
|
||||||
|
pub player: CAHPlayer,
|
||||||
|
}
|
||||||
|
|
||||||
pub async fn websocket_handler(
|
pub async fn websocket_handler(
|
||||||
ws: WebSocketUpgrade,
|
ws: WebSocketUpgrade,
|
||||||
State(state): State<Arc<AppState>>,
|
State(state): State<Arc<AppState>>,
|
||||||
|
@ -21,14 +43,21 @@ pub async fn websocket(stream: WebSocket, state: Arc<AppState>) {
|
||||||
// By splitting, we can send and receive at the same time.
|
// By splitting, we can send and receive at the same time.
|
||||||
let (mut sender, mut receiver) = stream.split();
|
let (mut sender, mut receiver) = stream.split();
|
||||||
|
|
||||||
// Username gets set in the receive loop, if it's valid.
|
let _greeting = sender
|
||||||
let mut newplayer = String::new();
|
.send(Message::Text(format!(
|
||||||
|
"Greetings! \n\
|
||||||
|
{:#?} Card packs loaded\n\
|
||||||
|
{:#?} Current active games",
|
||||||
|
state.all_cards.lock().unwrap().len(),
|
||||||
|
state.games.lock().unwrap().len(),
|
||||||
|
)))
|
||||||
|
.await;
|
||||||
|
|
||||||
// Loop until a text message is found.
|
// Loop until a text message is found.
|
||||||
while let Some(Ok(message)) = receiver.next().await {
|
while let Some(Ok(message)) = receiver.next().await {
|
||||||
match message {
|
match message {
|
||||||
Message::Text(text) => {
|
Message::Text(text) => {
|
||||||
tracing::debug!("Text: {}", text);
|
tracing::debug!("Text: {}", text);
|
||||||
// let message_str: &str = message.to_text().unwrap();
|
|
||||||
if let Ok(new_game) = serde_json::from_str::<NewGameRequest>(&text) {
|
if let Ok(new_game) = serde_json::from_str::<NewGameRequest>(&text) {
|
||||||
tracing::debug!("{:#?}", new_game);
|
tracing::debug!("{:#?}", new_game);
|
||||||
} else {
|
} else {
|
||||||
|
@ -46,88 +75,15 @@ pub async fn websocket(stream: WebSocket, state: Arc<AppState>) {
|
||||||
cf.reason
|
cf.reason
|
||||||
)
|
)
|
||||||
} else {
|
} else {
|
||||||
tracing::debug!("close sent without close frame")
|
tracing::debug!("close received without close frame")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Message::Pong(ping) => {
|
Message::Pong(ping) => {
|
||||||
tracing::debug!("Pong sent with {:?}", ping);
|
tracing::debug!("Pong received with: {:?}", ping);
|
||||||
}
|
}
|
||||||
Message::Ping(pong) => {
|
Message::Ping(pong) => {
|
||||||
tracing::debug!("Pong sent with {:?}", pong);
|
tracing::debug!("Pong received with: {:?}", pong);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// if let Message::Text(name) = message {
|
|
||||||
// // If newplayer that is sent by client is not taken, fill newplayer string.
|
|
||||||
// check_username(&state, &mut newplayer, &name);
|
|
||||||
//
|
|
||||||
// // If not empty we want to quit the loop else we want to quit function.
|
|
||||||
// if !newplayer.is_empty() {
|
|
||||||
// break;
|
|
||||||
// } else {
|
|
||||||
// // Only send our client that newplayer is taken.
|
|
||||||
// let _ = sender
|
|
||||||
// .send(Message::Text(String::from("Username already taken.")))
|
|
||||||
// .await;
|
|
||||||
//
|
|
||||||
// return;
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
}
|
|
||||||
|
|
||||||
// We subscribe *before* sending the "joined" message, so that we will also
|
|
||||||
// display it to our client.
|
|
||||||
let mut rx = state.tx.subscribe();
|
|
||||||
|
|
||||||
// Now send the "joined" message to all subscribers.
|
|
||||||
let msg = format!("{newplayer} joined.");
|
|
||||||
tracing::debug!("{msg}");
|
|
||||||
let _ = state.tx.send(msg);
|
|
||||||
|
|
||||||
// Spawn the first task that will receive broadcast messages and send text
|
|
||||||
// messages over the websocket to our client.
|
|
||||||
let mut send_task = tokio::spawn(async move {
|
|
||||||
while let Ok(msg) = rx.recv().await {
|
|
||||||
// In any websocket error, break loop.
|
|
||||||
if sender.send(Message::Text(msg)).await.is_err() {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
// Clone things we want to pass (move) to the receiving task.
|
|
||||||
let tx = state.tx.clone();
|
|
||||||
let name = newplayer.clone();
|
|
||||||
|
|
||||||
// Spawn a task that takes messages from the websocket, prepends the user
|
|
||||||
// name, and sends them to all broadcast subscribers.
|
|
||||||
let mut recv_task = tokio::spawn(async move {
|
|
||||||
while let Some(Ok(Message::Text(text))) = receiver.next().await {
|
|
||||||
// Add newplayer before message.
|
|
||||||
let _ = tx.send(format!("{name}: {text}"));
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
// If any one of the tasks run to completion, we abort the other.
|
|
||||||
tokio::select! {
|
|
||||||
_ = (&mut send_task) => recv_task.abort(),
|
|
||||||
_ = (&mut recv_task) => send_task.abort(),
|
|
||||||
};
|
|
||||||
|
|
||||||
// Send "user left" message (similar to "joined" above).
|
|
||||||
let msg = format!("{newplayer} left.");
|
|
||||||
tracing::debug!("{msg}");
|
|
||||||
let _ = state.tx.send(msg);
|
|
||||||
|
|
||||||
// Remove newplayer from map so new clients can take it again.
|
|
||||||
state.user_set.lock().unwrap().remove(&newplayer);
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn check_username(state: &AppState, string: &mut String, name: &str) {
|
|
||||||
let mut user_set = state.user_set.lock().unwrap();
|
|
||||||
|
|
||||||
if !user_set.contains(name) {
|
|
||||||
user_set.insert(name.to_owned());
|
|
||||||
|
|
||||||
string.push_str(name);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -113,13 +113,6 @@ async fn main() -> Result<(), Box<dyn Error>> {
|
||||||
games,
|
games,
|
||||||
});
|
});
|
||||||
|
|
||||||
tracing::debug!(
|
|
||||||
"Loaded {} Card packs!",
|
|
||||||
&app_state.all_cards.lock().unwrap().len()
|
|
||||||
);
|
|
||||||
|
|
||||||
tracing::debug!("{} active games", &app_state.games.lock().unwrap().len());
|
|
||||||
|
|
||||||
let app = Router::new()
|
let app = Router::new()
|
||||||
.route("/", get(index))
|
.route("/", get(index))
|
||||||
.route("/test", get(spawnclients))
|
.route("/test", get(spawnclients))
|
||||||
|
|
|
@ -67,7 +67,7 @@
|
||||||
<div id="chat">
|
<div id="chat">
|
||||||
<h3>Chat</h3>
|
<h3>Chat</h3>
|
||||||
<form id="chat" onsubmit="chatSubmit();return false">
|
<form id="chat" onsubmit="chatSubmit();return false">
|
||||||
<textarea id="chat-history" readonly="true" wrap="soft" style="display:block; width:30rem; height:10rem; box-sizing: border-box" cols="30" rows="10">Not in game</textarea>
|
<textarea id="chat-history" readonly="true" wrap="soft" style="display:block; width:30rem; height:10rem; box-sizing: border-box" cols="30" rows="10"></textarea>
|
||||||
<input id="chat-input" type="text" style="width: 30rem;" placeholder="chat" />
|
<input id="chat-input" type="text" style="width: 30rem;" placeholder="chat" />
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
|
@ -110,6 +110,7 @@
|
||||||
socket.onclose = function() {
|
socket.onclose = function() {
|
||||||
console.log("connection closed");
|
console.log("connection closed");
|
||||||
document.getElementById("status").innerHTML = '<p><em>Disconnected...</em></p>';
|
document.getElementById("status").innerHTML = '<p><em>Disconnected...</em></p>';
|
||||||
|
document.getElementById("chat-history").value = "";
|
||||||
}
|
}
|
||||||
|
|
||||||
socket.onmessage = function(e) {
|
socket.onmessage = function(e) {
|
||||||
|
|
Loading…
Add table
Reference in a new issue