|
@@ -1,5 +1,9 @@
|
|
|
//! Broadcaster implementation
|
|
|
-use crate::{worker::Worker, Filter, Transaction};
|
|
|
+use crate::{
|
|
|
+ storage::{Cursor, PrimaryFilter},
|
|
|
+ worker::Worker,
|
|
|
+ Filter, Transaction,
|
|
|
+};
|
|
|
use async_trait::async_trait;
|
|
|
use std::{
|
|
|
collections::HashMap,
|
|
@@ -16,7 +20,7 @@ use tokio::sync::{
|
|
|
/// This structure broadcasts the transactions to all subscribers in a separated working thread.
|
|
|
pub struct Broadcaster {
|
|
|
subscribers: RwLock<HashMap<usize, Sender<Transaction>>>,
|
|
|
- subscriptions: RwLock<HashMap<Filter, Vec<usize>>>,
|
|
|
+ subscriptions: RwLock<HashMap<PrimaryFilter, Vec<(Filter, usize)>>>,
|
|
|
is_there_any_subscriber: AtomicBool,
|
|
|
index: AtomicUsize,
|
|
|
}
|
|
@@ -25,7 +29,7 @@ impl Default for Broadcaster {
|
|
|
fn default() -> Self {
|
|
|
Self {
|
|
|
subscribers: RwLock::new(HashMap::<usize, Sender<_>>::new()),
|
|
|
- subscriptions: RwLock::new(HashMap::<Filter, Vec<_>>::new()),
|
|
|
+ subscriptions: RwLock::new(HashMap::<PrimaryFilter, Vec<_>>::new()),
|
|
|
is_there_any_subscriber: false.into(),
|
|
|
index: 0.into(),
|
|
|
}
|
|
@@ -52,10 +56,14 @@ impl Broadcaster {
|
|
|
self.is_there_any_subscriber
|
|
|
.store(true, std::sync::atomic::Ordering::Release);
|
|
|
|
|
|
- if let Some(previous_values) = listeners.get_mut(&filter) {
|
|
|
- previous_values.push(sender_index);
|
|
|
- } else {
|
|
|
- listeners.insert(filter, vec![sender_index]);
|
|
|
+ let cursor: Cursor = filter.into();
|
|
|
+
|
|
|
+ for primary_filter in cursor.primary_filter.into_iter() {
|
|
|
+ if let Some(previous_values) = listeners.get_mut(&primary_filter) {
|
|
|
+ previous_values.push((cursor.filter.clone(), sender_index));
|
|
|
+ } else {
|
|
|
+ listeners.insert(primary_filter, vec![(cursor.filter.clone(), sender_index)]);
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
receiver
|
|
@@ -78,16 +86,22 @@ impl Worker for Broadcaster {
|
|
|
let mut subscriptions_to_reindex = vec![];
|
|
|
let mut senders_to_remove = vec![];
|
|
|
|
|
|
- for (filter, listeners) in listeners.iter() {
|
|
|
- if filter.matches(&transaction.transaction, &transaction.revision) {
|
|
|
- for sender_index in listeners.iter() {
|
|
|
- if let Some(Err(TrySendError::Closed(_))) = senders
|
|
|
- .get(sender_index)
|
|
|
- .map(|sender| sender.try_send(transaction.clone()))
|
|
|
- {
|
|
|
- senders_to_remove.push(*sender_index);
|
|
|
- subscriptions_to_reindex.push(filter.clone());
|
|
|
- }
|
|
|
+ for primary_filter in transaction.get_primary_filters().into_iter() {
|
|
|
+ let filters = if let Some(filters) = listeners.get(&primary_filter) {
|
|
|
+ filters
|
|
|
+ } else {
|
|
|
+ continue;
|
|
|
+ };
|
|
|
+ for (filter, sender_index) in filters.iter() {
|
|
|
+ if !filter.matches(&transaction.transaction, &transaction.revision) {
|
|
|
+ continue;
|
|
|
+ }
|
|
|
+ if let Some(Err(TrySendError::Closed(_))) = senders
|
|
|
+ .get(sender_index)
|
|
|
+ .map(|sender| sender.try_send(transaction.clone()))
|
|
|
+ {
|
|
|
+ senders_to_remove.push(*sender_index);
|
|
|
+ subscriptions_to_reindex.push((primary_filter.clone(), filter.clone()));
|
|
|
}
|
|
|
}
|
|
|
}
|
|
@@ -103,13 +117,15 @@ impl Worker for Broadcaster {
|
|
|
senders.remove(to_remove);
|
|
|
}
|
|
|
|
|
|
- for to_rebuild in &subscriptions_to_reindex {
|
|
|
- if let Some(list_of_senders) = listeners.get_mut(to_rebuild) {
|
|
|
- *list_of_senders = list_of_senders
|
|
|
- .iter()
|
|
|
- .filter(|x| senders.contains_key(*x))
|
|
|
- .copied()
|
|
|
- .collect::<Vec<_>>();
|
|
|
+ for to_rebuild in subscriptions_to_reindex {
|
|
|
+ if let Some(list_of_senders) = listeners.remove(&to_rebuild.0) {
|
|
|
+ listeners.insert(
|
|
|
+ to_rebuild.0,
|
|
|
+ list_of_senders
|
|
|
+ .into_iter()
|
|
|
+ .filter(|x| senders.contains_key(&x.1))
|
|
|
+ .collect::<Vec<_>>(),
|
|
|
+ );
|
|
|
}
|
|
|
}
|
|
|
|