|
@@ -4,19 +4,33 @@ use std::{
|
|
|
rc::{Rc, Weak},
|
|
|
};
|
|
|
|
|
|
-pub struct Event<Data: 'static> {
|
|
|
- pub data: Data,
|
|
|
- pub emptied: bool,
|
|
|
+pub enum SubscriptionFunction<Host: 'static, Context: 'static + ?Sized, Data: 'static> {
|
|
|
+ ByRef(Box<dyn Fn(&Host, &Context, &mut Data)>),
|
|
|
+ ByValue(Box<dyn Fn(&Host, &Context, Data) -> Option<Data>>),
|
|
|
}
|
|
|
|
|
|
-struct ConcreteEventSub<Host, Context: 'static + ?Sized, Data: 'static> {
|
|
|
+impl<Host: 'static, Context: 'static + ?Sized, Data: 'static> SubscriptionFunction<Host, Context, Data> {
|
|
|
+ fn invoke(&self, host: &Host, context: &Context, mut data: Data) -> Option<Data> {
|
|
|
+ match self {
|
|
|
+ Self::ByRef(f) => {
|
|
|
+ f(host, context, &mut data);
|
|
|
+ Some(data)
|
|
|
+ },
|
|
|
+ Self::ByValue(f) => {
|
|
|
+ f(host, context, data)
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+struct ConcreteEventSub<Host: 'static, Context: 'static + ?Sized, Data: 'static> {
|
|
|
host: Weak<Host>,
|
|
|
- callback: Box<dyn Fn(&Host, &Context, &mut Event<Data>)>,
|
|
|
+ callback: SubscriptionFunction<Host, Context, Data>,
|
|
|
}
|
|
|
|
|
|
trait EventSub<Context: 'static + ?Sized, Data: 'static> {
|
|
|
fn is_healthy(&self) -> bool;
|
|
|
- fn invoke(&self, context: &Context, data: &mut Event<Data>);
|
|
|
+ fn invoke(&self, context: &Context, data: Data) -> Option<Data>;
|
|
|
}
|
|
|
|
|
|
impl<Host: 'static, Context: 'static + ?Sized, Data: 'static> EventSub<Context, Data>
|
|
@@ -25,10 +39,11 @@ impl<Host: 'static, Context: 'static + ?Sized, Data: 'static> EventSub<Context,
|
|
|
fn is_healthy(&self) -> bool {
|
|
|
self.host.strong_count() > 0
|
|
|
}
|
|
|
- fn invoke(&self, context: &Context, data: &mut Event<Data>) {
|
|
|
+ fn invoke(&self, context: &Context, data: Data) -> Option<Data> {
|
|
|
self.host
|
|
|
.upgrade()
|
|
|
- .map(|h| (self.callback)(h.as_ref(), context, data));
|
|
|
+ .map(|h| self.callback.invoke(h.as_ref(), context, data))
|
|
|
+ .flatten()
|
|
|
}
|
|
|
}
|
|
|
|
|
@@ -54,45 +69,28 @@ impl<Tag: 'static, Context: 'static + ?Sized, Data: 'static, Priority: 'static>
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-/*pub struct PrioritySubscriptionBuilder<
|
|
|
- 'l,
|
|
|
- Tag: 'static,
|
|
|
- Context: 'static + ?Sized,
|
|
|
- Data: 'static,
|
|
|
- Priority: AbstractServicePriority,
|
|
|
-> {
|
|
|
- parent: &'l PriorityChannel<Tag, Context, Data>,
|
|
|
- _ghost: std::marker::PhantomData<Priority>,
|
|
|
-}
|
|
|
-
|
|
|
-impl<
|
|
|
- 'l,
|
|
|
- Tag: 'static,
|
|
|
- Context: 'static + ?Sized,
|
|
|
- Data: 'static,
|
|
|
- Priority: AbstractServicePriority,
|
|
|
- > PrioritySubscriptionBuilder<'l, Tag, Context, Data, Priority>
|
|
|
-{
|
|
|
- pub fn subscribe<
|
|
|
+impl<Tag: 'static, Context: 'static + ?Sized, Data: 'static> Channel<Tag, Context, Data, NoPriorityTag> {
|
|
|
+ pub fn sub_ref<
|
|
|
Host: 'static,
|
|
|
HC: crate::helper::IntoWeak<Host>,
|
|
|
- CB: Fn(&Host, &Context, &mut Event<Data>) + 'static,
|
|
|
+ CB: Fn(&Host, &Context, &mut Data) + 'static,
|
|
|
>(
|
|
|
&self,
|
|
|
who: HC,
|
|
|
cb: CB,
|
|
|
) {
|
|
|
- self.parent.subscribe::<Priority, Host, HC, CB>(who, cb);
|
|
|
- }
|
|
|
-}
|
|
|
-*/
|
|
|
+ let sub = Rc::new(ConcreteEventSub {
|
|
|
+ host: who.as_weak(),
|
|
|
+ callback: SubscriptionFunction::ByRef(Box::new(cb)),
|
|
|
+ });
|
|
|
|
|
|
+ self.subs.borrow_mut().push((NoPriorityTag {}, sub));
|
|
|
+ }
|
|
|
|
|
|
-impl<Tag: 'static, Context: 'static + ?Sized, Data: 'static> Channel<Tag, Context, Data, NoPriorityTag> {
|
|
|
- pub fn subscribe<
|
|
|
+ pub fn sub_opt<
|
|
|
Host: 'static,
|
|
|
HC: crate::helper::IntoWeak<Host>,
|
|
|
- CB: Fn(&Host, &Context, &mut Event<Data>) + 'static,
|
|
|
+ CB: Fn(&Host, &Context, Data) -> Option<Data> + 'static,
|
|
|
>(
|
|
|
&self,
|
|
|
who: HC,
|
|
@@ -100,7 +98,7 @@ impl<Tag: 'static, Context: 'static + ?Sized, Data: 'static> Channel<Tag, Contex
|
|
|
) {
|
|
|
let sub = Rc::new(ConcreteEventSub {
|
|
|
host: who.as_weak(),
|
|
|
- callback: Box::new(cb),
|
|
|
+ callback: SubscriptionFunction::ByValue(Box::new(cb)),
|
|
|
});
|
|
|
|
|
|
self.subs.borrow_mut().push((NoPriorityTag {}, sub));
|
|
@@ -108,10 +106,31 @@ impl<Tag: 'static, Context: 'static + ?Sized, Data: 'static> Channel<Tag, Contex
|
|
|
}
|
|
|
|
|
|
impl<Tag: 'static, Context: 'static + ?Sized, Data: 'static, Priority: PartialOrd> Channel<Tag, Context, Data, Priority> {
|
|
|
- pub fn subscribe<
|
|
|
+ pub fn sub_ref<
|
|
|
+ Host: 'static,
|
|
|
+ HC: crate::helper::IntoWeak<Host>,
|
|
|
+ CB: Fn(&Host, &Context, &mut Data) + 'static,
|
|
|
+ >(
|
|
|
+ &self,
|
|
|
+ p: Priority,
|
|
|
+ who: HC,
|
|
|
+ cb: CB,
|
|
|
+ ) {
|
|
|
+ let sub = Rc::new(ConcreteEventSub {
|
|
|
+ host: who.as_weak(),
|
|
|
+ callback: SubscriptionFunction::ByRef(Box::new(cb)),
|
|
|
+ });
|
|
|
+
|
|
|
+ let mut subs = self.subs.borrow_mut();
|
|
|
+ subs.push((p.into(), sub));
|
|
|
+ // XXX: what happens if we actually get an undefined order?...
|
|
|
+ subs.sort_by(|a,b| a.0.partial_cmp(&b.0).unwrap());
|
|
|
+ }
|
|
|
+
|
|
|
+ pub fn sub_opt<
|
|
|
Host: 'static,
|
|
|
HC: crate::helper::IntoWeak<Host>,
|
|
|
- CB: Fn(&Host, &Context, &mut Event<Data>) + 'static,
|
|
|
+ CB: Fn(&Host, &Context, Data) -> Option<Data> + 'static,
|
|
|
>(
|
|
|
&self,
|
|
|
p: Priority,
|
|
@@ -120,7 +139,7 @@ impl<Tag: 'static, Context: 'static + ?Sized, Data: 'static, Priority: PartialOr
|
|
|
) {
|
|
|
let sub = Rc::new(ConcreteEventSub {
|
|
|
host: who.as_weak(),
|
|
|
- callback: Box::new(cb),
|
|
|
+ callback: SubscriptionFunction::ByValue(Box::new(cb)),
|
|
|
});
|
|
|
|
|
|
let mut subs = self.subs.borrow_mut();
|
|
@@ -139,16 +158,11 @@ impl<Tag: 'static, Context: 'static + ?Sized, Data: 'static, Priority: 'static>
|
|
|
let mut count = 0;
|
|
|
|
|
|
while !self.queue.borrow().is_empty() {
|
|
|
- let mut event = Event {
|
|
|
- data: self.queue.borrow_mut().pop_front().unwrap(),
|
|
|
- emptied: false,
|
|
|
- };
|
|
|
+ let mut event = self.queue.borrow_mut().pop_front();
|
|
|
|
|
|
for sub in self.subs.borrow().iter() {
|
|
|
- sub.1.invoke(context, &mut event);
|
|
|
- if event.emptied {
|
|
|
- break;
|
|
|
- }
|
|
|
+ event = sub.1.invoke(context, event.unwrap());
|
|
|
+ if event.is_none() { break }
|
|
|
}
|
|
|
|
|
|
count += 1;
|
|
@@ -175,7 +189,7 @@ impl<Tag: 'static, Context: 'static + ?Sized, Data: 'static, Priority: 'static>
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-pub trait ChannelSpec {
|
|
|
+pub trait ChannelSpec: 'static {
|
|
|
type Tag: 'static;
|
|
|
type Data: 'static;
|
|
|
type Priority: 'static;
|
|
@@ -221,29 +235,6 @@ impl<Context: 'static + ?Sized> EventRoot<Context> {
|
|
|
self.channels.insert(tid, ch);
|
|
|
}
|
|
|
|
|
|
- /*pub fn create_priority_channel<CS: ChannelSpec>(&mut self) {
|
|
|
- let tid = std::any::TypeId::of::<PriorityChannel<CS::Tag, Context, CS::Data>>();
|
|
|
-
|
|
|
- let ch = Rc::new(PriorityChannel::<CS::Tag, Context, CS::Data>::default());
|
|
|
- self.metadata
|
|
|
- .insert(tid, (std::any::type_name::<CS>(), ch.clone()));
|
|
|
- self.channels.insert(tid, ch);
|
|
|
- }*/
|
|
|
-
|
|
|
- /*pub fn channel<CS: ChannelSpec>(&self) -> &EventChannel<CS::Tag, Context, CS::Data> {
|
|
|
- let tid = std::any::TypeId::of::<EventChannel<CS::Tag, Context, CS::Data>>();
|
|
|
-
|
|
|
- match self.channels.get(&tid) {
|
|
|
- Some(ch) => ch.downcast_ref().expect("internal inconsistency"),
|
|
|
- None => {
|
|
|
- panic!(
|
|
|
- "Asked for channel {} that has not been created!",
|
|
|
- std::any::type_name::<CS>()
|
|
|
- )
|
|
|
- }
|
|
|
- }
|
|
|
- }*/
|
|
|
-
|
|
|
pub fn channel<CS: ChannelSpec>(
|
|
|
&self,
|
|
|
) -> &Channel<CS::Tag, Context, CS::Data, CS::Priority> {
|
|
@@ -284,14 +275,14 @@ mod tests {
|
|
|
rc::Rc,
|
|
|
};
|
|
|
|
|
|
- use super::{Event, EventRoot};
|
|
|
+ use super::EventRoot;
|
|
|
|
|
|
struct Receiver {
|
|
|
int_count: Cell<usize>,
|
|
|
}
|
|
|
|
|
|
impl Receiver {
|
|
|
- fn receive_i32(&self, _ctx: &(), _val: &mut Event<i32>) {
|
|
|
+ fn receive_i32(&self, _ctx: &(), _val: &mut i32) {
|
|
|
self.int_count.set(self.int_count.get() + 1);
|
|
|
}
|
|
|
}
|
|
@@ -302,7 +293,7 @@ mod tests {
|
|
|
}
|
|
|
|
|
|
impl OrderedReceiver {
|
|
|
- fn receive_i32(&self, _ctx: &(), _val: &mut Event<i32>) {
|
|
|
+ fn receive_i32(&self, _ctx: &(), _val: &mut i32) {
|
|
|
self.order.borrow_mut().push(self.id)
|
|
|
}
|
|
|
}
|
|
@@ -329,7 +320,7 @@ mod tests {
|
|
|
});
|
|
|
|
|
|
root.channel::<IntChannel>()
|
|
|
- .subscribe(&recv, Receiver::receive_i32);
|
|
|
+ .sub_ref(&recv, Receiver::receive_i32);
|
|
|
root.channel::<IntChannel>().queue(0i32);
|
|
|
assert_eq!(recv.int_count.get(), 0);
|
|
|
root.channel::<IntChannel>().do_fire_all(&());
|
|
@@ -358,11 +349,11 @@ mod tests {
|
|
|
});
|
|
|
|
|
|
root.channel::<IntPriorityChannel>()
|
|
|
- .subscribe(IntPriority::Second, &recv2, OrderedReceiver::receive_i32);
|
|
|
+ .sub_ref(IntPriority::Second, &recv2, OrderedReceiver::receive_i32);
|
|
|
root.channel::<IntPriorityChannel>()
|
|
|
- .subscribe(IntPriority::First, &recv1, OrderedReceiver::receive_i32);
|
|
|
+ .sub_ref(IntPriority::First, &recv1, OrderedReceiver::receive_i32);
|
|
|
root.channel::<IntPriorityChannel>()
|
|
|
- .subscribe(IntPriority::Third, &recv3, OrderedReceiver::receive_i32);
|
|
|
+ .sub_ref(IntPriority::Third, &recv3, OrderedReceiver::receive_i32);
|
|
|
root.channel::<IntPriorityChannel>().queue(0i32);
|
|
|
assert_eq!(order.borrow().deref(), &vec![]);
|
|
|
root.channel::<IntPriorityChannel>().do_fire_all(&());
|