2025-03-29 09:22:53 -04:00
|
|
|
use crate::field::Field;
|
|
|
|
use std::{
|
|
|
|
collections::HashMap,
|
2025-03-30 11:38:41 -04:00
|
|
|
sync::{
|
|
|
|
mpsc::{channel, Receiver, Sender},
|
|
|
|
Arc, RwLock,
|
|
|
|
},
|
2025-03-29 09:22:53 -04:00
|
|
|
thread::spawn,
|
|
|
|
};
|
|
|
|
use uuid::Uuid;
|
|
|
|
|
|
|
|
enum MsgType {
|
2025-03-30 11:38:41 -04:00
|
|
|
ClientMessage,
|
|
|
|
NewClientMessage,
|
2025-03-29 09:22:53 -04:00
|
|
|
NoOp,
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct Message {
|
|
|
|
id: Uuid,
|
|
|
|
class: MsgType,
|
|
|
|
data: HashMap<String, Field>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Message {
|
|
|
|
pub fn new() -> Self {
|
|
|
|
Self {
|
|
|
|
id: Uuid::nil(),
|
|
|
|
class: MsgType::NoOp,
|
|
|
|
data: HashMap::new(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn reply(&self, data: MsgType) -> Message {
|
|
|
|
Self {
|
|
|
|
id: self.id.clone(),
|
|
|
|
class: data,
|
|
|
|
data: self.data.clone(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_class(&self) -> &MsgType {
|
|
|
|
&self.class
|
|
|
|
}
|
|
|
|
|
2025-03-30 11:38:41 -04:00
|
|
|
fn add_data<S, F>(&mut self, name: S, data: F)
|
|
|
|
where
|
|
|
|
S: Into<String>,
|
|
|
|
F: Into<Field>,
|
|
|
|
{
|
2025-03-29 09:22:53 -04:00
|
|
|
self.data.insert(name.into(), data.into());
|
|
|
|
}
|
|
|
|
|
|
|
|
fn get_data(&self) -> &HashMap<String, Field> {
|
|
|
|
&self.data
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod messages {
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn new_message() {
|
|
|
|
let msg = Message::new();
|
|
|
|
assert_eq!(msg.id, Uuid::nil());
|
|
|
|
match msg.class {
|
|
|
|
MsgType::NoOp => (),
|
|
|
|
_ => unreachable!("new defaults to noop"),
|
|
|
|
}
|
|
|
|
assert!(msg.data.is_empty());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn create_reply() {
|
|
|
|
let id = Uuid::new_v4();
|
|
|
|
let mut msg = Message::new();
|
|
|
|
msg.id = id.clone();
|
2025-03-30 11:38:41 -04:00
|
|
|
let data = MsgType::NewClientMessage;
|
2025-03-29 09:22:53 -04:00
|
|
|
let result = msg.reply(data);
|
|
|
|
assert_eq!(result.id, id);
|
|
|
|
match result.class {
|
2025-03-30 11:38:41 -04:00
|
|
|
MsgType::NewClientMessage => {}
|
2025-03-29 09:22:53 -04:00
|
|
|
_ => unreachable!("should have been a registration request"),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn get_message_type() {
|
|
|
|
let msg = Message::new();
|
|
|
|
match msg.get_class() {
|
2025-03-30 11:38:41 -04:00
|
|
|
MsgType::NoOp => {}
|
2025-03-29 09:22:53 -04:00
|
|
|
_ => unreachable!("should have bneen noopn"),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn add_data() {
|
|
|
|
let mut msg = Message::new();
|
|
|
|
let one = "one";
|
|
|
|
let two = "two".to_string();
|
|
|
|
msg.add_data(one, one);
|
|
|
|
msg.add_data(two.clone(), two.clone());
|
|
|
|
let result = msg.get_data();
|
|
|
|
assert_eq!(result.get(one).unwrap().to_string(), one);
|
|
|
|
assert_eq!(result.get(&two).unwrap().to_string(), two);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct Queue {
|
2025-03-30 11:38:41 -04:00
|
|
|
registry: Arc<RwLock<Vec<Sender<Message>>>>,
|
2025-03-29 09:22:53 -04:00
|
|
|
rx: Receiver<Message>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Queue {
|
2025-03-30 11:38:41 -04:00
|
|
|
fn new(rx: Receiver<Message>, registry: Arc<RwLock<Vec<Sender<Message>>>>) -> Self {
|
2025-03-29 09:22:53 -04:00
|
|
|
Self {
|
2025-03-30 11:38:41 -04:00
|
|
|
registry: registry,
|
2025-03-29 09:22:53 -04:00
|
|
|
rx: rx,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-03-30 11:38:41 -04:00
|
|
|
fn start(registry: Arc<RwLock<Vec<Sender<Message>>>>) -> Sender<Message> {
|
2025-03-29 09:22:53 -04:00
|
|
|
let (tx, rx) = channel();
|
|
|
|
spawn(move || {
|
2025-03-30 11:38:41 -04:00
|
|
|
let mut queue = Queue::new(rx, registry);
|
2025-03-29 09:22:53 -04:00
|
|
|
queue.listen();
|
|
|
|
});
|
|
|
|
tx
|
|
|
|
}
|
|
|
|
|
|
|
|
fn listen(&mut self) {
|
|
|
|
loop {
|
|
|
|
let msg = self.rx.recv().unwrap();
|
2025-03-30 11:38:41 -04:00
|
|
|
let senders = self.registry.read().unwrap();
|
|
|
|
for sender in senders.iter() {
|
|
|
|
sender.send(Message::new()).unwrap();
|
2025-03-29 09:22:53 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod queues {
|
|
|
|
use super::*;
|
2025-03-30 11:38:41 -04:00
|
|
|
use std::time::Duration;
|
|
|
|
|
|
|
|
static TIMEOUT: Duration = Duration::from_millis(500);
|
2025-03-29 09:22:53 -04:00
|
|
|
|
2025-03-30 11:38:41 -04:00
|
|
|
fn start_queue() -> (
|
|
|
|
Sender<Message>,
|
|
|
|
Receiver<Message>,
|
|
|
|
) {
|
|
|
|
let reg: Arc<RwLock<Vec<Sender<Message>>>> = Arc::new(RwLock::new(Vec::new()));
|
2025-03-29 09:22:53 -04:00
|
|
|
let (tx, rx) = channel::<Message>();
|
2025-03-30 11:38:41 -04:00
|
|
|
let mut data = reg.write().unwrap();
|
|
|
|
data.push(tx.clone());
|
|
|
|
drop(data);
|
|
|
|
let queue_tx = Queue::start(Arc::clone(®));
|
2025-03-29 09:22:53 -04:00
|
|
|
(queue_tx, rx)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn get_new_client_message() {
|
|
|
|
let (tx, rx) = start_queue();
|
2025-03-30 11:38:41 -04:00
|
|
|
let initial = Message::new();
|
|
|
|
let msg = initial.reply(MsgType::NewClientMessage);
|
|
|
|
tx.send(msg).unwrap();
|
|
|
|
rx.recv_timeout(TIMEOUT).unwrap();
|
2025-03-29 09:22:53 -04:00
|
|
|
}
|
|
|
|
}
|