|
@@ -1,6 +1,6 @@
|
|
|
use crate::runtime::{ActorRuntime, Runtime};
|
|
|
use flume::{SendError, Sender};
|
|
|
-use message::{Envelope, Message, MessagePacker, MessageRequest};
|
|
|
+use message::{Envelope, Enveloper, Message, MessageRequest};
|
|
|
use std::fmt::Debug;
|
|
|
use tokio::sync::oneshot;
|
|
|
pub mod debug;
|
|
@@ -8,6 +8,8 @@ pub mod message;
|
|
|
pub mod runtime;
|
|
|
pub mod ws;
|
|
|
|
|
|
+const DEFAULT_CHANNEL_CAPACITY: usize = 16;
|
|
|
+
|
|
|
pub trait Actor {
|
|
|
fn start(self) -> ActorHandle<Self>
|
|
|
where
|
|
@@ -18,6 +20,17 @@ pub trait Actor {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
+/// The main trait to implement on an [Actor] to enable it to handle messages.
|
|
|
+pub trait Handler<M>: Actor
|
|
|
+where
|
|
|
+ M: Message,
|
|
|
+{
|
|
|
+ fn handle(&mut self, message: M) -> Result<M::Response, Error>;
|
|
|
+}
|
|
|
+
|
|
|
+/// A handle to a spawned actor. Obtained when calling `start` on an [Actor] and is used to send messages
|
|
|
+/// to it.
|
|
|
+#[derive(Debug, Clone)]
|
|
|
pub struct ActorHandle<A>
|
|
|
where
|
|
|
A: Actor,
|
|
@@ -26,55 +39,155 @@ where
|
|
|
command_tx: Sender<ActorCommand>,
|
|
|
}
|
|
|
|
|
|
-impl<A> Clone for ActorHandle<A>
|
|
|
+impl<A> ActorHandle<A>
|
|
|
where
|
|
|
A: Actor,
|
|
|
{
|
|
|
- fn clone(&self) -> Self {
|
|
|
+ pub fn new(message_tx: Sender<Envelope<A>>, command_tx: Sender<ActorCommand>) -> Self {
|
|
|
Self {
|
|
|
- message_tx: self.message_tx.clone(),
|
|
|
- command_tx: self.command_tx.clone(),
|
|
|
+ message_tx,
|
|
|
+ command_tx,
|
|
|
}
|
|
|
}
|
|
|
-}
|
|
|
|
|
|
-impl<A> ActorHandle<A>
|
|
|
-where
|
|
|
- A: Actor + 'static,
|
|
|
-{
|
|
|
- pub async fn send_sync<M>(&self, message: M) -> Result<M::Response, Error>
|
|
|
+ /// Sends a message to the actor and returns a [MessageRequest] that can
|
|
|
+ /// be awaited. This method should be used when one needs a response from the
|
|
|
+ /// actor.
|
|
|
+ pub fn send_wait<M>(&self, message: M) -> Result<MessageRequest<M::Response>, SendError<M>>
|
|
|
where
|
|
|
- M: Message + Send + 'static,
|
|
|
- A: Handler<M> + MessagePacker<A, M>,
|
|
|
+ M: Message + Send,
|
|
|
+ A: Handler<M> + Enveloper<A, M>,
|
|
|
{
|
|
|
+ if self.message_tx.is_full() {
|
|
|
+ return Err(SendError(message));
|
|
|
+ }
|
|
|
let (tx, rx) = oneshot::channel();
|
|
|
- let packed = A::pack(message, Some(tx));
|
|
|
- self.message_tx
|
|
|
- .send(packed)
|
|
|
- .map_err(Error::send_err_boxed)?;
|
|
|
- MessageRequest { response_rx: rx }.await
|
|
|
+ let _ = self.message_tx.send(A::pack(message, Some(tx)));
|
|
|
+ Ok(MessageRequest { response_rx: rx })
|
|
|
}
|
|
|
|
|
|
- pub async fn send<M>(&self, message: M) -> Result<(), Error>
|
|
|
+ /// Send a message to the actor without waiting for any response, but still returning an
|
|
|
+ /// error if the channel is full.
|
|
|
+ pub fn send<M>(&self, message: M) -> Result<(), SendError<M>>
|
|
|
where
|
|
|
M: Message + Send + 'static,
|
|
|
- A: Handler<M> + MessagePacker<A, M> + 'static,
|
|
|
+ A: Handler<M> + Enveloper<A, M> + 'static,
|
|
|
{
|
|
|
- let packed = A::pack(message, None);
|
|
|
- self.message_tx.send(packed).map_err(Error::send_err_boxed)
|
|
|
+ if self.message_tx.is_full() {
|
|
|
+ return Err(SendError(message));
|
|
|
+ }
|
|
|
+ let _ = self.message_tx.send(A::pack(message, None));
|
|
|
+ Ok(())
|
|
|
}
|
|
|
|
|
|
- pub async fn send_cmd(&self, cmd: ActorCommand) -> Result<(), Error> {
|
|
|
- self.command_tx.send(cmd).unwrap();
|
|
|
- Ok(())
|
|
|
+ /// Send a message ignoring any errors in the process. The true YOLO way to send messages.
|
|
|
+ pub fn send_forget<M>(&self, message: M)
|
|
|
+ where
|
|
|
+ M: Message + Send + 'static,
|
|
|
+ A: Handler<M> + Enveloper<A, M> + 'static,
|
|
|
+ {
|
|
|
+ let _ = self.message_tx.send(A::pack(message, None));
|
|
|
+ }
|
|
|
+
|
|
|
+ pub fn send_cmd(&self, cmd: ActorCommand) -> Result<(), SendError<ActorCommand>> {
|
|
|
+ self.command_tx.send(cmd)
|
|
|
+ }
|
|
|
+
|
|
|
+ pub fn recipient<M>(&self) -> Recipient<M>
|
|
|
+ where
|
|
|
+ M: Message + Send + 'static,
|
|
|
+ M::Response: Send,
|
|
|
+ A: Handler<M> + 'static,
|
|
|
+ {
|
|
|
+ Recipient {
|
|
|
+ message_tx: Box::new(self.message_tx.clone()),
|
|
|
+ command_tx: self.command_tx.clone(),
|
|
|
+ }
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-pub trait Handler<M>: Actor
|
|
|
+/// The same as an [ActorHandle], but instead of being tied to a specific actor, it is only
|
|
|
+/// tied to the message type. Can be obtained from an [ActorHandle].
|
|
|
+///
|
|
|
+/// Useful for grouping different types of actors that can handle the same message.
|
|
|
+pub struct Recipient<M>
|
|
|
where
|
|
|
M: Message,
|
|
|
{
|
|
|
- fn handle(&mut self, message: M) -> Result<M::Response, Error>;
|
|
|
+ message_tx: Box<dyn DynamicSender<M>>,
|
|
|
+ command_tx: Sender<ActorCommand>,
|
|
|
+}
|
|
|
+
|
|
|
+impl<M> Recipient<M>
|
|
|
+where
|
|
|
+ M: Message + Send,
|
|
|
+{
|
|
|
+ pub fn send_wait(&self, message: M) -> Result<MessageRequest<M::Response>, SendError<M>> {
|
|
|
+ self.message_tx.send_sync(message)
|
|
|
+ }
|
|
|
+
|
|
|
+ pub fn send(&self, message: M) -> Result<(), SendError<M>> {
|
|
|
+ self.message_tx.send(message)
|
|
|
+ }
|
|
|
+
|
|
|
+ pub fn send_forget(&self, message: M) {
|
|
|
+ let _ = self.message_tx.send(message);
|
|
|
+ }
|
|
|
+
|
|
|
+ pub fn send_cmd(&self, cmd: ActorCommand) -> Result<(), SendError<ActorCommand>> {
|
|
|
+ self.command_tx.send(cmd)
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+/// A helper trait used solely by [Recipient]'s message channel to erase the actor type.
|
|
|
+/// This is achieved by implementing it on [Sender<Envelope<A>].
|
|
|
+trait DynamicSender<M>
|
|
|
+where
|
|
|
+ M: Message + Send,
|
|
|
+{
|
|
|
+ fn send_sync(&self, message: M) -> Result<MessageRequest<M::Response>, SendError<M>>;
|
|
|
+
|
|
|
+ fn send(&self, message: M) -> Result<(), SendError<M>>;
|
|
|
+}
|
|
|
+
|
|
|
+impl<A, M> DynamicSender<M> for Sender<Envelope<A>>
|
|
|
+where
|
|
|
+ M: Message + Send + 'static,
|
|
|
+ M::Response: Send,
|
|
|
+ A: Actor + Handler<M> + Enveloper<A, M>,
|
|
|
+{
|
|
|
+ fn send(&self, message: M) -> Result<(), SendError<M>> {
|
|
|
+ if self.is_full() {
|
|
|
+ return Err(SendError(message));
|
|
|
+ }
|
|
|
+ let _ = self.send(A::pack(message, None));
|
|
|
+ Ok(())
|
|
|
+ }
|
|
|
+
|
|
|
+ fn send_sync(
|
|
|
+ &self,
|
|
|
+ message: M,
|
|
|
+ ) -> Result<MessageRequest<<M as Message>::Response>, SendError<M>> {
|
|
|
+ if self.is_full() {
|
|
|
+ return Err(SendError(message));
|
|
|
+ }
|
|
|
+ let (tx, rx) = oneshot::channel();
|
|
|
+ let _ = self.send(A::pack(message, Some(tx)));
|
|
|
+ Ok(MessageRequest { response_rx: rx })
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+impl<A, M> From<ActorHandle<A>> for Recipient<M>
|
|
|
+where
|
|
|
+ M: Message + Send + 'static,
|
|
|
+ M::Response: Send,
|
|
|
+ A: Actor + Handler<M> + Enveloper<A, M> + 'static,
|
|
|
+{
|
|
|
+ /// Just calls `ActorHandler::recipient`, i.e. clones the underlying channels
|
|
|
+ /// into the recipient.
|
|
|
+ fn from(handle: ActorHandle<A>) -> Self {
|
|
|
+ handle.recipient()
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
#[derive(Debug, thiserror::Error)]
|
|
@@ -83,18 +196,10 @@ pub enum Error {
|
|
|
ActorChannelClosed,
|
|
|
#[error("Channel closed: {0}")]
|
|
|
ChannelClosed(#[from] oneshot::error::TryRecvError),
|
|
|
- #[error("Send error: {0}")]
|
|
|
- Send(Box<dyn std::error::Error + Send + 'static>),
|
|
|
#[error("Warp error: {0}")]
|
|
|
Warp(#[from] warp::Error),
|
|
|
}
|
|
|
|
|
|
-impl Error {
|
|
|
- fn send_err_boxed<T: Send + 'static>(error: SendError<T>) -> Self {
|
|
|
- Self::Send(Box::new(error))
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
#[derive(Debug)]
|
|
|
pub enum ActorCommand {
|
|
|
Stop,
|
|
@@ -145,13 +250,15 @@ mod tests {
|
|
|
let mut res = 0;
|
|
|
let mut res2 = 0;
|
|
|
for _ in 0..100 {
|
|
|
- res += handle.send_sync(Foo {}).await.unwrap();
|
|
|
- res2 += handle.send_sync(Bar {}).await.unwrap();
|
|
|
+ res += handle.send_wait(Foo {}).unwrap().await.unwrap();
|
|
|
+ res2 += handle.send_wait(Bar {}).unwrap().await.unwrap();
|
|
|
}
|
|
|
|
|
|
- handle.send_cmd(ActorCommand::Stop).await.unwrap();
|
|
|
+ let rec: Recipient<Foo> = handle.recipient();
|
|
|
+ res += rec.send_wait(Foo {}).unwrap().await.unwrap();
|
|
|
+ handle.send_cmd(ActorCommand::Stop).unwrap();
|
|
|
|
|
|
- assert_eq!(res, 1000);
|
|
|
+ assert_eq!(res, 1010);
|
|
|
assert_eq!(res2, 1000);
|
|
|
}
|
|
|
}
|