/* * Copyright (c) 2022-2024 Niko Bonnieure, Par le Peuple, NextGraph.org developers * All rights reserved. * Licensed under the Apache License, Version 2.0 * * or the MIT license , * at your option. All files in the project carrying such * notice may not be copied, modified, or distributed except * according to those terms. */ //! Actor handles messages in the Protocol. common types are here use std::any::TypeId; use std::marker::PhantomData; use std::sync::Arc; use async_std::stream::StreamExt; use async_std::sync::Mutex; use futures::{channel::mpsc, SinkExt}; use ng_repo::errors::{NgError, ProtocolError, ServerError}; use ng_repo::log::*; use crate::utils::{spawn_and_log_error, Receiver, ResultSend, Sender}; use crate::{connection::*, types::ProtocolMessage}; impl TryFrom for () { type Error = ProtocolError; fn try_from(_msg: ProtocolMessage) -> Result { Ok(()) } } #[doc(hidden)] #[async_trait::async_trait] pub trait EActor: Send + Sync + std::fmt::Debug { async fn respond( &mut self, msg: ProtocolMessage, fsm: Arc>, ) -> Result<(), ProtocolError>; fn set_id(&mut self, _id: i64) {} } #[derive(Debug)] pub(crate) struct Actor< 'a, A: Into + std::fmt::Debug, B: TryFrom + std::fmt::Debug + Sync, > { id: i64, phantom_a: PhantomData<&'a A>, phantom_b: PhantomData<&'a B>, receiver: Option>, receiver_tx: Sender, //initiator: bool, } pub enum SoS { Single(B), Stream(Receiver), } impl SoS { pub fn is_single(&self) -> bool { if let Self::Single(_b) = self { true } else { false } } pub fn is_stream(&self) -> bool { !self.is_single() } pub fn unwrap_single(self) -> B { match self { Self::Single(s) => s, Self::Stream(_s) => { panic!("called `unwrap_single()` on a `Stream` value") } } } pub fn unwrap_stream(self) -> Receiver { match self { Self::Stream(s) => s, Self::Single(_s) => { panic!("called `unwrap_stream()` on a `Single` value") } } } } impl< A: Into + std::fmt::Debug + 'static, B: TryFrom + Sync + Send + std::fmt::Debug + 'static, > Actor<'_, A, B> { pub fn new(id: i64, _initiator: bool) -> Self { let (receiver_tx, receiver) = mpsc::unbounded::(); Self { id, receiver: Some(receiver), receiver_tx, phantom_a: PhantomData, phantom_b: PhantomData, //initiator, } } // pub fn verify(&self, msg: ProtocolMessage) -> bool { // self.initiator && msg.type_id() == TypeId::of::() // || !self.initiator && msg.type_id() == TypeId::of::() // } pub fn detach_receiver(&mut self) -> Receiver { self.receiver.take().unwrap() } pub async fn request( &mut self, msg: ProtocolMessage, fsm: Arc>, ) -> Result, NgError> { fsm.lock().await.send(msg).await?; let mut receiver = self.receiver.take().unwrap(); match receiver.next().await { Some(ConnectionCommand::Msg(msg)) => { if let Some(bm) = msg.is_streamable() { if bm.result() == Into::::into(ServerError::PartialContent) && TypeId::of::() != TypeId::of::<()>() { let (mut b_sender, b_receiver) = mpsc::unbounded::(); let response = msg.try_into().map_err(|e| { log_err!("msg.try_into {}", e); ProtocolError::ActorError })?; b_sender .send(response) .await .map_err(|_err| ProtocolError::IoError)?; async fn pump_stream>( mut actor_receiver: Receiver, mut sos_sender: Sender, fsm: Arc>, id: i64, ) -> ResultSend<()> { async move { while let Some(ConnectionCommand::Msg(msg)) = actor_receiver.next().await { if let Some(bm) = msg.is_streamable() { if bm.result() == Into::::into(ServerError::EndOfStream) { break; } let response = msg.try_into(); if response.is_err() { // TODO deal with errors. break; } if sos_sender.send(response.unwrap()).await.is_err() { break; } } else { // todo deal with error (not a ClientMessage) break; } } fsm.lock().await.remove_actor(id).await; } .await; Ok(()) } spawn_and_log_error(pump_stream::( receiver, b_sender, Arc::clone(&fsm), self.id, )); return Ok(SoS::::Stream(b_receiver)); } } fsm.lock().await.remove_actor(self.id).await; let server_error: Result = (&msg).try_into(); let response: B = match msg.try_into() { Ok(b) => b, Err(ProtocolError::ServerError) => { return Err(NgError::ServerError(server_error?)); } Err(e) => return Err(NgError::ProtocolError(e)), }; Ok(SoS::::Single(response)) } Some(ConnectionCommand::ProtocolError(e)) => Err(e.into()), Some(ConnectionCommand::Error(e)) => Err(ProtocolError::from(e).into()), Some(ConnectionCommand::Close) => Err(ProtocolError::Closing.into()), _ => Err(ProtocolError::ActorError.into()), } } pub fn new_responder(id: i64) -> Box { Box::new(Self::new(id, false)) } pub fn get_receiver_tx(&self) -> Sender { self.receiver_tx.clone() } pub fn id(&self) -> i64 { self.id } } #[cfg(test)] mod test { use crate::actor::*; use crate::actors::*; #[async_std::test] pub async fn test_actor() { let _a = Actor::::new(1, true); // a.handle(ProtocolMessage::Start(StartProtocol::Client( // ClientHello::Noise3(Noise::V0(NoiseV0 { data: vec![] })), // ))) // .await; // a.handle(ProtocolMessage::Noise(Noise::V0(NoiseV0 { data: vec![] }))) // .await; } }