morethantext/src/queue.rs

241 lines
5.8 KiB
Rust
Raw Normal View History

2025-04-02 14:26:09 -04:00
use crate::{client::Request, field::Field};
2025-03-29 09:22:53 -04:00
use std::{
collections::HashMap,
2025-03-30 11:38:41 -04:00
sync::{
mpsc::{channel, Receiver, Sender},
2025-04-03 08:24:08 -04:00
Arc, Mutex, RwLock,
2025-03-30 11:38:41 -04:00
},
2025-03-29 09:22:53 -04:00
thread::spawn,
};
use uuid::Uuid;
2025-04-03 08:24:08 -04:00
#[derive(Clone)]
2025-04-02 22:10:16 -04:00
pub enum MsgType {
2025-03-30 11:38:41 -04:00
ClientMessage,
2025-04-02 14:26:09 -04:00
ClientRequest,
2025-03-30 11:38:41 -04:00
NewClientMessage,
2025-03-29 09:22:53 -04:00
NoOp,
}
2025-04-03 08:24:08 -04:00
#[derive(Clone)]
2025-03-29 09:22:53 -04:00
pub struct Message {
id: Uuid,
class: MsgType,
data: HashMap<String, Field>,
}
impl Message {
2025-04-02 22:10:16 -04:00
pub fn new(msg_type: MsgType) -> Self {
2025-03-29 09:22:53 -04:00
Self {
id: Uuid::nil(),
2025-04-02 22:10:16 -04:00
class: msg_type,
2025-03-29 09:22:53 -04:00
data: HashMap::new(),
}
}
fn reply(&self, data: MsgType) -> Message {
Self {
id: self.id.clone(),
class: data,
data: self.data.clone(),
}
}
2025-04-02 22:10:16 -04:00
pub fn get_class(&self) -> &MsgType {
2025-03-29 09:22:53 -04:00
&self.class
}
2025-04-02 22:10:16 -04:00
pub fn add_data<S, F>(&mut self, name: S, data: F)
2025-03-30 11:38:41 -04:00
where
S: Into<String>,
F: Into<Field>,
{
2025-03-29 09:22:53 -04:00
self.data.insert(name.into(), data.into());
}
2025-04-02 22:10:16 -04:00
pub fn get_data(&self) -> &HashMap<String, Field> {
2025-03-29 09:22:53 -04:00
&self.data
}
}
2025-04-02 14:26:09 -04:00
impl From<Request> for Message {
fn from(value: Request) -> Self {
2025-04-02 22:10:16 -04:00
let msg = Message::new(MsgType::ClientRequest);
2025-04-02 14:26:09 -04:00
msg.reply(MsgType::ClientRequest)
}
}
2025-03-29 09:22:53 -04:00
#[cfg(test)]
mod messages {
use super::*;
#[test]
fn new_message() {
2025-04-02 22:10:16 -04:00
let msg = Message::new(MsgType::NoOp);
2025-03-29 09:22:53 -04:00
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();
2025-04-02 22:10:16 -04:00
let mut msg = Message::new(MsgType::NoOp);
2025-03-29 09:22:53 -04:00
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() {
2025-04-02 22:10:16 -04:00
let msg = Message::new(MsgType::NoOp);
2025-03-29 09:22:53 -04:00
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() {
2025-04-02 22:10:16 -04:00
let mut msg = Message::new(MsgType::NoOp);
2025-03-29 09:22:53 -04:00
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);
}
}
2025-04-03 08:24:08 -04:00
struct ServiceRegistry {
store: Arc<Mutex<Vec<Sender<Message>>>>,
}
impl ServiceRegistry {
fn new() -> Self {
Self {
store: Arc::new(Mutex::new(Vec::new())),
}
}
fn add(&self, tx: Sender<Message>) {
let mut store = self.store.lock().unwrap();
store.push(tx);
}
fn send(&self, msg: Message) {
let mut store = self.store.lock().unwrap();
for sender in store.iter() {
sender.send(msg.clone()).unwrap();
}
}
}
#[cfg(test)]
mod serviceredistries {
use super::*;
#[test]
fn create_registry() {
let reg = ServiceRegistry::new();
let (tx1, rx1) = channel();
let (tx2, rx2) = channel();
reg.add(tx1);
reg.add(tx2);
reg.send(Message::new(MsgType::NoOp));
rx1.recv().unwrap();
rx2.recv().unwrap();
}
}
2025-03-29 09:22:53 -04:00
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 {
2025-03-30 16:24:58 -04:00
let mut msg = self.rx.recv().unwrap();
2025-03-30 17:04:01 -04:00
msg.id = Uuid::new_v4();
2025-03-30 11:38:41 -04:00
let senders = self.registry.read().unwrap();
for sender in senders.iter() {
2025-03-30 16:24:58 -04:00
sender.send(msg.reply(MsgType::ClientMessage)).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 16:24:58 -04:00
fn start_queue() -> (Sender<Message>, Receiver<Message>) {
2025-03-30 11:38:41 -04:00
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();
2025-03-30 16:24:58 -04:00
data.push(tx);
2025-03-30 11:38:41 -04:00
drop(data);
let queue_tx = Queue::start(Arc::clone(&reg));
2025-03-29 09:22:53 -04:00
(queue_tx, rx)
}
#[test]
fn get_new_client_message() {
let (tx, rx) = start_queue();
2025-04-02 22:10:16 -04:00
let initial = Message::new(MsgType::NoOp);
2025-03-30 11:38:41 -04:00
let msg = initial.reply(MsgType::NewClientMessage);
tx.send(msg).unwrap();
2025-03-30 16:24:58 -04:00
let msg = rx.recv_timeout(TIMEOUT).unwrap();
match msg.class {
MsgType::ClientMessage => assert_ne!(msg.id, initial.id),
_ => unreachable!("should have been a client message"),
}
2025-03-29 09:22:53 -04:00
}
2025-03-30 17:04:01 -04:00
#[test]
fn new_client_messages_are_unique() {
let (tx, rx) = start_queue();
2025-04-02 22:10:16 -04:00
let msg = Message::new(MsgType::NoOp);
2025-03-30 17:04:01 -04:00
let mut ids: Vec<Uuid> = Vec::new();
for _ in 0..10 {
tx.send(msg.reply(MsgType::NewClientMessage)).unwrap();
let result = rx.recv().unwrap();
2025-04-02 14:26:09 -04:00
assert!(
!ids.contains(&result.id.clone()),
"{} is a duplicate",
&result.id
);
2025-03-30 17:04:01 -04:00
ids.push(result.id);
}
}
2025-03-29 09:22:53 -04:00
}