Compare commits
2 Commits
fbaf386ebb
...
f28c9fee83
Author | SHA1 | Date | |
---|---|---|---|
f28c9fee83 | |||
41175e487e |
42
src/code_generator.rs
Normal file
42
src/code_generator.rs
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
use rand::RngCore;
|
||||||
|
use sha2::{Sha256, Digest};
|
||||||
|
|
||||||
|
pub struct CodeGenerator {
|
||||||
|
counter: u64,
|
||||||
|
salt: [u8; 32],
|
||||||
|
}
|
||||||
|
|
||||||
|
impl CodeGenerator {
|
||||||
|
pub fn generate(&mut self) -> String {
|
||||||
|
let count = self.counter;
|
||||||
|
self.counter += 1;
|
||||||
|
|
||||||
|
let mut hasher = Sha256::new();
|
||||||
|
hasher.update(self.salt);
|
||||||
|
hasher.update(count.to_be_bytes());
|
||||||
|
|
||||||
|
format!("{:x}", hasher.finalize())[..6].to_string()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for CodeGenerator {
|
||||||
|
fn default() -> Self {
|
||||||
|
let mut salt = [0; 32];
|
||||||
|
rand::thread_rng().fill_bytes(&mut salt);
|
||||||
|
|
||||||
|
CodeGenerator {
|
||||||
|
counter: 0,
|
||||||
|
salt,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod test {
|
||||||
|
use super::*;
|
||||||
|
#[test]
|
||||||
|
fn test_generate() {
|
||||||
|
let code = CodeGenerator::default().generate();
|
||||||
|
assert_eq!(code.len(), 6);
|
||||||
|
}
|
||||||
|
}
|
65
src/main.rs
65
src/main.rs
@ -5,12 +5,10 @@ use std::collections::HashMap;
|
|||||||
|
|
||||||
mod message;
|
mod message;
|
||||||
use message::Message;
|
use message::Message;
|
||||||
|
mod code_generator;
|
||||||
use rand::RngCore;
|
use code_generator::CodeGenerator;
|
||||||
|
mod websocket;
|
||||||
use tungstenite::protocol::Message as WsMessage;
|
use websocket::WebsocketWrapper;
|
||||||
|
|
||||||
use sha2::{Sha256, Digest};
|
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
let code_generator = Arc::new(Mutex::new(CodeGenerator::default()));
|
let code_generator = Arc::new(Mutex::new(CodeGenerator::default()));
|
||||||
@ -24,30 +22,21 @@ fn main() {
|
|||||||
thread::spawn (move || {
|
thread::spawn (move || {
|
||||||
let mut ws = tungstenite::accept(stream.unwrap()).unwrap();
|
let mut ws = tungstenite::accept(stream.unwrap()).unwrap();
|
||||||
println!("New client!");
|
println!("New client!");
|
||||||
ws.write_message(WsMessage::Text("HOSTJOIN:".to_string())).unwrap();
|
let mut ws = WebsocketWrapper::new(ws);
|
||||||
loop {
|
ws.send(msg!("HOSTJOIN"));
|
||||||
let message = ws.read_message();
|
|
||||||
println!("{:?}", message);
|
let msg = ws.recv();
|
||||||
match message {
|
|
||||||
Err(_) => break,
|
|
||||||
Ok(WsMessage::Close(_)) => break,
|
|
||||||
Ok(WsMessage::Text(msg)) => {
|
|
||||||
let msg = Message::parse(msg.as_str().trim());
|
|
||||||
match msg {
|
match msg {
|
||||||
Ok(msg) =>
|
None => (),
|
||||||
match msg.command {
|
|
||||||
|
Some(msg) => match msg.command.as_str() {
|
||||||
"HOST" => {
|
"HOST" => {
|
||||||
let code = code_generator.lock().unwrap().generate();
|
let code = code_generator.lock().unwrap().generate();
|
||||||
let mut room = Room::default();
|
let mut room = Room::default();
|
||||||
let player = Player { name: "Guest".to_string() };
|
let player = Player { name: "Guest".to_string() };
|
||||||
room.players.push(player);
|
room.players.push(player);
|
||||||
rooms.lock().unwrap().insert(code.clone(), room);
|
rooms.lock().unwrap().insert(code.clone(), room);
|
||||||
ws.write_message(WsMessage::Text(code)).unwrap();
|
ws.send(msg!("CODE", code));
|
||||||
}
|
|
||||||
_ => todo!(),
|
|
||||||
}
|
|
||||||
Err(_) => todo!(),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
_ => unimplemented!(),
|
_ => unimplemented!(),
|
||||||
}
|
}
|
||||||
@ -63,33 +52,3 @@ struct Room {
|
|||||||
struct Player {
|
struct Player {
|
||||||
name: String,
|
name: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
struct CodeGenerator {
|
|
||||||
counter: u64,
|
|
||||||
salt: [u8; 32],
|
|
||||||
}
|
|
||||||
|
|
||||||
impl CodeGenerator {
|
|
||||||
fn generate(&mut self) -> String {
|
|
||||||
let count = self.counter;
|
|
||||||
self.counter += 1;
|
|
||||||
|
|
||||||
let mut hasher = Sha256::new();
|
|
||||||
hasher.update(self.salt);
|
|
||||||
hasher.update(count.to_be_bytes());
|
|
||||||
|
|
||||||
format!("{:x}", hasher.finalize())[..6].to_string()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Default for CodeGenerator {
|
|
||||||
fn default() -> Self {
|
|
||||||
let mut salt = [0; 32];
|
|
||||||
rand::thread_rng().fill_bytes(&mut salt);
|
|
||||||
|
|
||||||
CodeGenerator {
|
|
||||||
counter: 0,
|
|
||||||
salt,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
#[derive(PartialEq, Debug)]
|
#[derive(PartialEq, Debug)]
|
||||||
pub struct Message<'a> {
|
pub struct Message {
|
||||||
pub command: &'a str,
|
pub command: String,
|
||||||
pub args: Vec<&'a str>,
|
pub args: Vec<String>,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub type Result<T> = std::result::Result<T, Error>;
|
pub type Result<T> = std::result::Result<T, Error>;
|
||||||
@ -13,18 +13,21 @@ pub enum Error {
|
|||||||
UnknownCommand,
|
UnknownCommand,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a> Message<'a> {
|
impl Message {
|
||||||
pub fn parse(text: &'a str) -> Result<Message<'a>> {
|
pub fn new(command: String, args: Vec<String>) -> Self {
|
||||||
|
Message { command, args }
|
||||||
|
}
|
||||||
|
pub fn parse(text: String) -> Result<Message> {
|
||||||
let re = regex::Regex::new(r"^([A-Z_]+):\s*(.*)").unwrap();
|
let re = regex::Regex::new(r"^([A-Z_]+):\s*(.*)").unwrap();
|
||||||
match re.captures(text) {
|
match re.captures(text.as_str()) {
|
||||||
Some(captures) => {
|
Some(captures) => {
|
||||||
if captures.len() < 3 {
|
if captures.len() < 3 {
|
||||||
Err(Error::BadParse)
|
Err(Error::BadParse)
|
||||||
} else {
|
} else {
|
||||||
let command = captures.get(1).unwrap().as_str();
|
let command = captures.get(1).unwrap().as_str().to_string();
|
||||||
let args = captures.get(2).unwrap().as_str()
|
let args = captures.get(2).unwrap().as_str()
|
||||||
.split(',')
|
.split(',')
|
||||||
.map(|s| s.trim())
|
.map(|s| s.trim().to_string())
|
||||||
.collect();
|
.collect();
|
||||||
Ok(Message { command, args })
|
Ok(Message { command, args })
|
||||||
}
|
}
|
||||||
@ -34,6 +37,35 @@ impl<'a> Message<'a> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[macro_export]
|
||||||
|
macro_rules! msg {
|
||||||
|
( $command:expr) => {
|
||||||
|
{
|
||||||
|
let command = $command.to_string();
|
||||||
|
let args = Vec::new();
|
||||||
|
Message { command, args }
|
||||||
|
}
|
||||||
|
};
|
||||||
|
( $command:expr, $( $arg:expr ),*) => {
|
||||||
|
{
|
||||||
|
let command = $command.to_string();
|
||||||
|
let mut args = Vec::new();
|
||||||
|
$(
|
||||||
|
args.push($arg.to_string());
|
||||||
|
)*
|
||||||
|
Message { command, args }
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
impl std::fmt::Display for Message {
|
||||||
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::result::Result<(), std::fmt::Error> {
|
||||||
|
write!(f, "{}: {}", self.command, self.args.as_slice().join(", "))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod test {
|
mod test {
|
||||||
use super::*;
|
use super::*;
|
||||||
@ -45,7 +77,14 @@ mod test {
|
|||||||
command: "COMMAND",
|
command: "COMMAND",
|
||||||
args: vec!["arg1", "arg2"],
|
args: vec!["arg1", "arg2"],
|
||||||
}, msg);
|
}, msg);
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
#[test]
|
||||||
|
fn test_to_string() {
|
||||||
|
let msg = Message {
|
||||||
|
command: "COMMAND",
|
||||||
|
args: vec!["arg1", "arg2"],
|
||||||
|
};
|
||||||
|
assert_eq!(msg.to_string(), "COMMAND: arg1, arg2".to_string());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
36
src/websocket.rs
Normal file
36
src/websocket.rs
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
use std::net::TcpStream;
|
||||||
|
|
||||||
|
use tungstenite::protocol::{ WebSocket, Message as WsMessage };
|
||||||
|
|
||||||
|
use crate::message::Message;
|
||||||
|
|
||||||
|
pub struct WebsocketWrapper {
|
||||||
|
websocket: WebSocket<TcpStream>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl WebsocketWrapper {
|
||||||
|
pub fn new(websocket: WebSocket<TcpStream>) -> Self {
|
||||||
|
WebsocketWrapper { websocket }
|
||||||
|
}
|
||||||
|
pub fn send(&mut self, msg: Message) {
|
||||||
|
self.websocket.write_message(WsMessage::Text(msg.to_string())).unwrap();
|
||||||
|
}
|
||||||
|
pub fn recv(&mut self) -> Option<Message> {
|
||||||
|
match self.websocket.read_message() {
|
||||||
|
Err(_) | Ok(WsMessage::Close(_)) => None,
|
||||||
|
Ok(WsMessage::Text(text)) => match Message::parse(text) {
|
||||||
|
Ok(msg) => Some(msg),
|
||||||
|
Err(_) => {
|
||||||
|
self.websocket.write_message(WsMessage::Text("ERROR: bad_format".to_string()));
|
||||||
|
self.recv()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
self.websocket.write_message(WsMessage::Text("ERROR: bad_command".to_string()));
|
||||||
|
self.recv()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue
Block a user