runtime_tests.rs 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821
  1. #![feature(impl_trait_in_assoc_type)]
  2. use btrun::model::*;
  3. use btrun::test_setup;
  4. use btrun::*;
  5. use btlib::Result;
  6. use btproto::protocol;
  7. use log;
  8. use once_cell::sync::Lazy;
  9. use serde::{Deserialize, Serialize};
  10. use std::{
  11. future::{ready, Future, Ready},
  12. sync::{
  13. atomic::{AtomicU8, Ordering},
  14. Arc,
  15. },
  16. };
  17. test_setup!();
  18. mod ping_pong {
  19. use super::*;
  20. use btlib::bterr;
  21. // The following code is a proof-of-concept for what types should be generated for a
  22. // simple ping-pong protocol:
  23. protocol! {
  24. named PingProtocol;
  25. let server = [Server];
  26. let client = [Client];
  27. Client -> End, >service(Server)!Ping;
  28. Server?Ping -> End, >Client!Ping::Reply;
  29. }
  30. //
  31. // In words, the protocol is described as follows.
  32. // 1. When the Listening state receives the Ping message it returns the End state and a
  33. // Ping::Reply message to be sent to the SentPing state.
  34. // 2. When the SentPing state receives the Ping::Reply message it returns the End state.
  35. //
  36. // The End state represents an end to the session described by the protocol. When an actor
  37. // transitions to the End state its function returns.
  38. // When a state is expecting a Reply message, an error occurs if the message is not received
  39. // in a timely manner.
  40. enum PingClientState<T: Client> {
  41. Client(T),
  42. End(End),
  43. }
  44. impl<T: Client> PingClientState<T> {
  45. const fn name(&self) -> &'static str {
  46. match self {
  47. Self::Client(_) => "Client",
  48. Self::End(_) => "End",
  49. }
  50. }
  51. }
  52. struct ClientHandleManual<T: Client> {
  53. state: Option<PingClientState<T>>,
  54. client_name: ActorName,
  55. runtime: &'static Runtime,
  56. }
  57. impl<T: Client> ClientHandleManual<T> {
  58. async fn send_ping(
  59. mut self,
  60. msg: Ping,
  61. service: ServiceAddr,
  62. ) -> TransResult<Self, ClientHandleManual<T>> {
  63. let state = if let Some(state) = self.state.take() {
  64. state
  65. } else {
  66. return TransResult::Abort {
  67. from: self,
  68. err: bterr!("The shared state was not returned."),
  69. };
  70. };
  71. match state {
  72. PingClientState::Client(state) => {
  73. let result = self
  74. .runtime
  75. .call_service(
  76. service,
  77. self.client_name.clone(),
  78. PingProtocolMsgs::Ping(msg),
  79. )
  80. .await;
  81. let reply_enum = match result {
  82. Ok(reply_enum) => reply_enum,
  83. Err(err) => {
  84. self.state = Some(PingClientState::Client(state));
  85. return TransResult::Abort { from: self, err };
  86. }
  87. };
  88. if let PingProtocolMsgs::PingReply(reply) = reply_enum {
  89. match state.on_send_ping(reply).await {
  90. TransResult::Ok(new_state) => {
  91. self.state = Some(PingClientState::End(new_state));
  92. TransResult::Ok(self)
  93. }
  94. TransResult::Abort { from, err } => {
  95. self.state = Some(PingClientState::Client(from));
  96. TransResult::Abort { from: self, err }
  97. }
  98. TransResult::Fatal { err } => return TransResult::Fatal { err },
  99. }
  100. } else {
  101. TransResult::Abort {
  102. from: self,
  103. err: bterr!("Unexpected reply type."),
  104. }
  105. }
  106. }
  107. state => {
  108. let err = bterr!("Can't send Ping in state {}.", state.name());
  109. self.state = Some(state);
  110. TransResult::Abort { from: self, err }
  111. }
  112. }
  113. }
  114. }
  115. async fn spawn_client_manual<T: Client>(
  116. init: T,
  117. runtime: &'static Runtime,
  118. ) -> ClientHandleManual<T> {
  119. let state = Some(PingClientState::Client(init));
  120. let client_name = runtime.spawn(None, do_nothing_actor).await.unwrap();
  121. ClientHandleManual {
  122. state,
  123. client_name,
  124. runtime,
  125. }
  126. }
  127. async fn register_server_manual<Init, F>(
  128. make_init: F,
  129. rt: &'static Runtime,
  130. id: ServiceId,
  131. ) -> Result<ServiceName>
  132. where
  133. Init: 'static + Server,
  134. F: 'static + Send + Sync + Clone + Fn() -> Init,
  135. {
  136. enum ServerState<S> {
  137. Server(S),
  138. End(End),
  139. }
  140. impl<S> Named for ServerState<S> {
  141. fn name(&self) -> Arc<String> {
  142. static SERVER_NAME: Lazy<Arc<String>> = Lazy::new(|| Arc::new("Server".into()));
  143. static END_NAME: Lazy<Arc<String>> = Lazy::new(|| Arc::new("End".into()));
  144. match self {
  145. Self::Server(_) => SERVER_NAME.clone(),
  146. Self::End(_) => END_NAME.clone(),
  147. }
  148. }
  149. }
  150. async fn server_loop<Init, F>(
  151. _runtime: &'static Runtime,
  152. make_init: F,
  153. mut mailbox: Mailbox<PingProtocolMsgs>,
  154. actor_id: ActorId,
  155. ) -> ActorResult
  156. where
  157. Init: 'static + Server,
  158. F: 'static + Send + Sync + FnOnce() -> Init,
  159. {
  160. let mut state = ServerState::Server(make_init());
  161. while let Some(envelope) = mailbox.recv().await {
  162. state = match envelope {
  163. Envelope::Call {
  164. msg,
  165. reply: replier,
  166. ..
  167. } => match (state, msg) {
  168. (ServerState::Server(listening_state), PingProtocolMsgs::Ping(msg)) => {
  169. match listening_state.handle_ping(msg).await {
  170. TransResult::Ok((new_state, reply)) => {
  171. let replier = replier
  172. .ok_or_else(|| bterr!("Reply has already been sent."))
  173. .unwrap();
  174. if let Err(_) = replier.send(PingProtocolMsgs::PingReply(reply))
  175. {
  176. return Err(ActorError::new(
  177. bterr!("Failed to send Ping reply."),
  178. ActorErrorPayload {
  179. actor_id,
  180. actor_impl: Init::actor_impl(),
  181. state: Init::state_name(),
  182. message: PingProtocolMsgKinds::Ping.name(),
  183. kind: TransKind::Receive,
  184. },
  185. ));
  186. }
  187. ServerState::End(new_state)
  188. }
  189. TransResult::Abort { from, err } => {
  190. log::warn!("Aborted transition from the {} while handling the {} message: {}", "Server", "Ping", err);
  191. ServerState::Server(from)
  192. }
  193. TransResult::Fatal { err } => {
  194. return Err(ActorError::new(
  195. err,
  196. ActorErrorPayload {
  197. actor_id,
  198. actor_impl: Init::actor_impl(),
  199. state: Init::state_name(),
  200. message: PingProtocolMsgKinds::Ping.name(),
  201. kind: TransKind::Receive,
  202. },
  203. ));
  204. }
  205. }
  206. }
  207. (state, _) => state,
  208. },
  209. envelope => {
  210. return Err(ActorError::new(
  211. bterr!("Unexpected envelope type: {}", envelope.name()),
  212. ActorErrorPayload {
  213. actor_id,
  214. actor_impl: Init::actor_impl(),
  215. state: state.name(),
  216. message: envelope.msg_name(),
  217. kind: TransKind::Receive,
  218. },
  219. ))
  220. }
  221. };
  222. if let ServerState::End(_) = state {
  223. break;
  224. }
  225. }
  226. Ok(actor_id)
  227. }
  228. rt.register::<PingProtocolMsgs, _>(id, move |runtime| {
  229. let make_init = make_init.clone();
  230. let fut = async move {
  231. let actor_impl = runtime
  232. .spawn(None, move |mailbox, act_id, runtime| {
  233. server_loop(runtime, make_init, mailbox, act_id)
  234. })
  235. .await
  236. .unwrap();
  237. Ok(actor_impl)
  238. };
  239. Box::pin(fut)
  240. })
  241. .await
  242. }
  243. #[derive(Serialize, Deserialize)]
  244. pub struct Ping;
  245. impl CallMsg for Ping {
  246. type Reply = PingReply;
  247. }
  248. #[derive(Serialize, Deserialize)]
  249. pub struct PingReply;
  250. struct ClientImpl {
  251. counter: Arc<AtomicU8>,
  252. }
  253. impl ClientImpl {
  254. fn new(counter: Arc<AtomicU8>) -> Self {
  255. counter.fetch_add(1, Ordering::SeqCst);
  256. Self { counter }
  257. }
  258. }
  259. impl Client for ClientImpl {
  260. actor_name!("ping_client");
  261. type OnSendPingFut = impl Future<Output = TransResult<Self, End>>;
  262. fn on_send_ping(self, _msg: PingReply) -> Self::OnSendPingFut {
  263. self.counter.fetch_sub(1, Ordering::SeqCst);
  264. ready(TransResult::Ok(End))
  265. }
  266. }
  267. struct ServerState {
  268. counter: Arc<AtomicU8>,
  269. }
  270. impl ServerState {
  271. fn new(counter: Arc<AtomicU8>) -> Self {
  272. counter.fetch_add(1, Ordering::SeqCst);
  273. Self { counter }
  274. }
  275. }
  276. impl Server for ServerState {
  277. actor_name!("ping_server");
  278. type HandlePingFut = impl Future<Output = TransResult<Self, (End, PingReply)>>;
  279. fn handle_ping(self, _msg: Ping) -> Self::HandlePingFut {
  280. self.counter.fetch_sub(1, Ordering::SeqCst);
  281. ready(TransResult::Ok((End, PingReply)))
  282. }
  283. }
  284. #[test]
  285. fn ping_pong_test() {
  286. ASYNC_RT.block_on(async {
  287. const SERVICE_ID: &str = "PingPongProtocolServer";
  288. let service_id = ServiceId::from(SERVICE_ID);
  289. let counter = Arc::new(AtomicU8::new(0));
  290. let service_name = {
  291. let service_counter = counter.clone();
  292. let make_init = move || {
  293. let server_counter = service_counter.clone();
  294. ServerState::new(server_counter)
  295. };
  296. register_server_manual(make_init, &RUNTIME, service_id.clone())
  297. .await
  298. .unwrap()
  299. };
  300. let client_handle =
  301. spawn_client_manual(ClientImpl::new(counter.clone()), &RUNTIME).await;
  302. let service_addr = ServiceAddr::new(service_name, true);
  303. client_handle.send_ping(Ping, service_addr).await.unwrap();
  304. assert_eq!(0, counter.load(Ordering::SeqCst));
  305. });
  306. }
  307. }
  308. mod travel_agency {
  309. use super::*;
  310. // Here's another protocol example. This is the Customer and Travel Agency protocol used as an
  311. // example in the survey paper "Behavioral Types in Programming Languages."
  312. // Note that the Choosing state can send messages at any time, not just in response to another
  313. // message because there is a transition from Choosing that doesn't use the receive operator
  314. // (`?`).
  315. protocol! {
  316. named TravelAgency;
  317. let agency = [Listening];
  318. let customer = [Choosing];
  319. Choosing -> Choosing, >service(Listening)!Query;
  320. Choosing -> Choosing, >service(Listening)!Accept;
  321. Choosing -> Choosing, >service(Listening)!Reject;
  322. Listening?Query -> Listening, >Choosing!Query::Reply;
  323. Listening?Accept -> End, >Choosing!Accept::Reply;
  324. Listening?Reject -> End, >Choosing!Reject::Reply;
  325. }
  326. #[derive(Serialize, Deserialize)]
  327. pub struct Query;
  328. impl CallMsg for Query {
  329. type Reply = ();
  330. }
  331. #[derive(Serialize, Deserialize)]
  332. pub struct Reject;
  333. impl CallMsg for Reject {
  334. type Reply = ();
  335. }
  336. #[derive(Serialize, Deserialize)]
  337. pub struct Accept;
  338. impl CallMsg for Accept {
  339. type Reply = ();
  340. }
  341. }
  342. #[allow(dead_code)]
  343. mod client_callback {
  344. use super::*;
  345. use btlib::bterr;
  346. use once_cell::sync::Lazy;
  347. use std::{marker::PhantomData, panic::panic_any, time::Duration};
  348. use tokio::{sync::oneshot, time::timeout};
  349. #[derive(Serialize, Deserialize)]
  350. pub struct Register {
  351. factor: usize,
  352. }
  353. #[derive(Serialize, Deserialize)]
  354. pub struct Completed {
  355. value: usize,
  356. }
  357. protocol! {
  358. named ClientCallback;
  359. let server = [Listening];
  360. let worker = [Working];
  361. let client = [Unregistered, Registered];
  362. Unregistered -> Registered, >service(Listening)!Register[Registered];
  363. Listening?Register[Registered] -> Listening, Working[Registered];
  364. Working[Registered] -> End, >Registered!Completed;
  365. Registered?Completed -> End;
  366. }
  367. struct UnregisteredState {
  368. sender: oneshot::Sender<usize>,
  369. }
  370. impl Unregistered for UnregisteredState {
  371. actor_name!("callback_client");
  372. type OnSendRegisterRegistered = RegisteredState;
  373. type OnSendRegisterFut = Ready<TransResult<Self, Self::OnSendRegisterRegistered>>;
  374. fn on_send_register(self) -> Self::OnSendRegisterFut {
  375. ready(TransResult::Ok(RegisteredState {
  376. sender: self.sender,
  377. }))
  378. }
  379. }
  380. struct RegisteredState {
  381. sender: oneshot::Sender<usize>,
  382. }
  383. impl Registered for RegisteredState {
  384. type HandleCompletedFut = Ready<TransResult<Self, End>>;
  385. fn handle_completed(self, arg: Completed) -> Self::HandleCompletedFut {
  386. self.sender.send(arg.value).unwrap();
  387. ready(TransResult::Ok(End))
  388. }
  389. }
  390. struct ListeningState {
  391. multiple: usize,
  392. }
  393. impl Listening for ListeningState {
  394. actor_name!("callback_server");
  395. type HandleRegisterListening = ListeningState;
  396. type HandleRegisterWorking = WorkingState;
  397. type HandleRegisterFut = Ready<TransResult<Self, (ListeningState, WorkingState)>>;
  398. fn handle_register(self, arg: Register) -> Self::HandleRegisterFut {
  399. let multiple = self.multiple;
  400. ready(TransResult::Ok((
  401. self,
  402. WorkingState {
  403. factor: arg.factor,
  404. multiple,
  405. },
  406. )))
  407. }
  408. }
  409. struct WorkingState {
  410. factor: usize,
  411. multiple: usize,
  412. }
  413. impl Working for WorkingState {
  414. actor_name!("callback_worker");
  415. type OnSendCompletedFut = Ready<TransResult<Self, (End, Completed)>>;
  416. fn on_send_completed(self) -> Self::OnSendCompletedFut {
  417. let value = self.multiple * self.factor;
  418. ready(TransResult::Ok((End, Completed { value })))
  419. }
  420. }
  421. use ::tokio::sync::Mutex;
  422. enum ClientStateManual<Init: Unregistered> {
  423. Unregistered(Init),
  424. Registered(Init::OnSendRegisterRegistered),
  425. End(End),
  426. }
  427. impl<Init: Unregistered> Named for ClientStateManual<Init> {
  428. fn name(&self) -> Arc<String> {
  429. static UNREGISTERED_NAME: Lazy<Arc<String>> =
  430. Lazy::new(|| Arc::new("Unregistered".into()));
  431. static REGISTERED_NAME: Lazy<Arc<String>> = Lazy::new(|| Arc::new("Registered".into()));
  432. static END_NAME: Lazy<Arc<String>> = Lazy::new(|| Arc::new("End".into()));
  433. match self {
  434. Self::Unregistered(_) => UNREGISTERED_NAME.clone(),
  435. Self::Registered(_) => REGISTERED_NAME.clone(),
  436. Self::End(_) => END_NAME.clone(),
  437. }
  438. }
  439. }
  440. struct ClientHandleManual<Init: Unregistered, State> {
  441. runtime: &'static Runtime,
  442. state: Arc<Mutex<Option<ClientStateManual<Init>>>>,
  443. name: ActorName,
  444. type_state: PhantomData<State>,
  445. }
  446. impl<Init: Unregistered, State> ClientHandleManual<Init, State> {
  447. fn new_type<NewState>(self) -> ClientHandleManual<Init, NewState> {
  448. ClientHandleManual {
  449. runtime: self.runtime,
  450. state: self.state,
  451. name: self.name,
  452. type_state: PhantomData,
  453. }
  454. }
  455. }
  456. impl<
  457. Init: Unregistered,
  458. State: Unregistered<OnSendRegisterRegistered = NewState>,
  459. NewState: Registered,
  460. > ClientHandleManual<Init, State>
  461. {
  462. async fn send_register(
  463. self,
  464. to: ServiceAddr,
  465. msg: Register,
  466. ) -> Result<ClientHandleManual<Init, NewState>> {
  467. {
  468. let mut guard = self.state.lock().await;
  469. let state = guard
  470. .take()
  471. .unwrap_or_else(|| panic!("Logic error. The state was not returned."));
  472. let new_state = match state {
  473. ClientStateManual::Unregistered(state) => {
  474. match state.on_send_register().await {
  475. TransResult::Ok(new_state) => {
  476. let msg = ClientCallbackMsgs::Register(msg);
  477. self.runtime
  478. .send_service(to, self.name.clone(), msg)
  479. .await?;
  480. ClientStateManual::Registered(new_state)
  481. }
  482. TransResult::Abort { from, err } => {
  483. log::warn!(
  484. "Aborted transition from the {} state: {}",
  485. "Unregistered",
  486. err
  487. );
  488. ClientStateManual::Unregistered(from)
  489. }
  490. TransResult::Fatal { err } => {
  491. return Err(err);
  492. }
  493. }
  494. }
  495. state => state,
  496. };
  497. *guard = Some(new_state);
  498. }
  499. Ok(self.new_type())
  500. }
  501. }
  502. async fn spawn_client_manual<Init>(
  503. init: Init,
  504. runtime: &'static Runtime,
  505. ) -> ClientHandleManual<Init, Init>
  506. where
  507. Init: 'static + Unregistered,
  508. {
  509. let state = Arc::new(Mutex::new(Some(ClientStateManual::Unregistered(init))));
  510. let name = {
  511. let state = state.clone();
  512. runtime.spawn(None, move |mut mailbox, actor_id, _| async move {
  513. while let Some(envelope) = mailbox.recv().await {
  514. let mut guard = state.lock().await;
  515. let state = guard.take()
  516. .unwrap_or_else(|| panic!("Logic error. The state was not returned."));
  517. let new_state = match envelope {
  518. Envelope::Send { msg, .. } => {
  519. match (state, msg) {
  520. (ClientStateManual::Registered(curr_state), ClientCallbackMsgs::Completed(msg)) => {
  521. match curr_state.handle_completed(msg).await {
  522. TransResult::Ok(next) => ClientStateManual::<Init>::End(next),
  523. TransResult::Abort { from, err } => {
  524. log::warn!("Aborted transition from the {} state while handling the {} message: {}", "Registered", "Completed", err);
  525. ClientStateManual::Registered(from)
  526. }
  527. TransResult::Fatal { err } => {
  528. panic_any(ActorError::new(
  529. err,
  530. ActorErrorPayload {
  531. actor_id,
  532. actor_impl: Init::actor_impl(),
  533. state: Init::OnSendRegisterRegistered::state_name(),
  534. message: ClientCallbackMsgKinds::Completed.name(),
  535. kind: TransKind::Receive,
  536. }));
  537. }
  538. }
  539. }
  540. (state, msg) => {
  541. log::error!("Unexpected message {} in state {}.", msg.name(), state.name());
  542. state
  543. }
  544. }
  545. }
  546. envelope => return Err(ActorError::new(
  547. bterr!("Unexpected envelope type: {}", envelope.name()),
  548. ActorErrorPayload {
  549. actor_id,
  550. actor_impl: Init::actor_impl(),
  551. state: state.name(),
  552. message: envelope.msg_name(),
  553. kind: TransKind::Receive,
  554. }))
  555. };
  556. *guard = Some(new_state);
  557. if let Some(state) = &*guard {
  558. if let ClientStateManual::End(_) = state {
  559. break;
  560. }
  561. }
  562. }
  563. Ok(actor_id)
  564. }).await.unwrap()
  565. };
  566. ClientHandleManual {
  567. runtime,
  568. state,
  569. name,
  570. type_state: PhantomData,
  571. }
  572. }
  573. async fn register_server_manual<Init, F>(
  574. make_init: F,
  575. runtime: &'static Runtime,
  576. service_id: ServiceId,
  577. ) -> Result<ServiceName>
  578. where
  579. Init: 'static + Listening<HandleRegisterListening = Init>,
  580. F: 'static + Send + Sync + Clone + Fn() -> Init,
  581. {
  582. enum ServerState<Init: Listening> {
  583. Listening(Init),
  584. }
  585. impl<S: Listening> Named for ServerState<S> {
  586. fn name(&self) -> Arc<String> {
  587. static LISTENING_NAME: Lazy<Arc<String>> =
  588. Lazy::new(|| Arc::new("Listening".into()));
  589. match self {
  590. Self::Listening(_) => LISTENING_NAME.clone(),
  591. }
  592. }
  593. }
  594. async fn server_loop<Init, F>(
  595. runtime: &'static Runtime,
  596. make_init: F,
  597. mut mailbox: Mailbox<ClientCallbackMsgs>,
  598. actor_id: ActorId,
  599. ) -> ActorResult
  600. where
  601. Init: 'static + Listening<HandleRegisterListening = Init>,
  602. F: 'static + Send + Sync + Fn() -> Init,
  603. {
  604. let mut state = ServerState::Listening(make_init());
  605. while let Some(envelope) = mailbox.recv().await {
  606. let new_state = match envelope {
  607. Envelope::Send { msg, from, .. } => match (state, msg) {
  608. (ServerState::Listening(curr_state), ClientCallbackMsgs::Register(msg)) => {
  609. match curr_state.handle_register(msg).await {
  610. TransResult::Ok((new_state, working_state)) => {
  611. start_worker_manual(working_state, from, runtime).await;
  612. ServerState::Listening(new_state)
  613. }
  614. TransResult::Abort { from, err } => {
  615. log::warn!("Aborted transition from the {} state while handling the {} message: {}", "Listening", "Register", err);
  616. ServerState::Listening(from)
  617. }
  618. TransResult::Fatal { err } => {
  619. let err = ActorError::new(
  620. err,
  621. ActorErrorPayload {
  622. actor_id,
  623. actor_impl: Init::actor_impl(),
  624. state: Init::state_name(),
  625. message: ClientCallbackMsgKinds::Register.name(),
  626. kind: TransKind::Receive,
  627. },
  628. );
  629. panic_any(format!("{err}"));
  630. }
  631. }
  632. }
  633. (state, msg) => {
  634. log::error!(
  635. "Unexpected message {} in state {}.",
  636. msg.name(),
  637. state.name()
  638. );
  639. state
  640. }
  641. },
  642. envelope => {
  643. return Err(ActorError::new(
  644. bterr!("Unexpected envelope type: {}", envelope.name()),
  645. ActorErrorPayload {
  646. actor_id,
  647. actor_impl: Init::actor_impl(),
  648. state: state.name(),
  649. message: envelope.msg_name(),
  650. kind: TransKind::Receive,
  651. },
  652. ))
  653. }
  654. };
  655. state = new_state;
  656. }
  657. Ok(actor_id)
  658. }
  659. runtime
  660. .register::<ClientCallbackMsgs, _>(service_id, move |runtime: &'static Runtime| {
  661. let make_init = make_init.clone();
  662. let fut = async move {
  663. let make_init = make_init.clone();
  664. let actor_impl = runtime
  665. .spawn(None, move |mailbox, act_id, runtime| {
  666. server_loop(runtime, make_init, mailbox, act_id)
  667. })
  668. .await
  669. .unwrap();
  670. Ok(actor_impl)
  671. };
  672. Box::pin(fut)
  673. })
  674. .await
  675. }
  676. async fn start_worker_manual<Init>(
  677. init: Init,
  678. owned: ActorName,
  679. runtime: &'static Runtime,
  680. ) -> ActorName
  681. where
  682. Init: 'static + Working,
  683. {
  684. enum WorkerState<S: Working> {
  685. Working(S),
  686. }
  687. runtime
  688. .spawn::<ClientCallbackMsgs, _, _>(
  689. Some(owned.clone()),
  690. move |_, actor_id, _| async move {
  691. let msg = match init.on_send_completed().await {
  692. TransResult::Ok((End, msg)) => msg,
  693. TransResult::Abort { err, .. } | TransResult::Fatal { err } => {
  694. let err = ActorError::new(
  695. err,
  696. ActorErrorPayload {
  697. actor_id,
  698. actor_impl: Init::actor_impl(),
  699. state: Init::state_name(),
  700. message: ClientCallbackMsgKinds::Completed.name(),
  701. kind: TransKind::Send,
  702. },
  703. );
  704. panic_any(format!("{err}"))
  705. }
  706. };
  707. let from = runtime.actor_name(actor_id);
  708. let msg = ClientCallbackMsgs::Completed(msg);
  709. runtime.send(owned, from, msg).await.unwrap_or_else(|err| {
  710. let err = ActorError::new(
  711. err,
  712. ActorErrorPayload {
  713. actor_id,
  714. actor_impl: Init::actor_impl(),
  715. state: Init::state_name(),
  716. message: ClientCallbackMsgKinds::Completed.name(),
  717. kind: TransKind::Send,
  718. },
  719. );
  720. panic_any(format!("{err}"));
  721. });
  722. Ok(actor_id)
  723. },
  724. )
  725. .await
  726. .unwrap()
  727. }
  728. #[test]
  729. fn client_callback_protocol() {
  730. ASYNC_RT.block_on(async {
  731. const SERVICE_ID: &str = "ClientCallbackProtocolListening";
  732. let service_id = ServiceId::from(SERVICE_ID);
  733. let service_name = {
  734. let make_init = move || ListeningState { multiple: 2 };
  735. register_server_manual(make_init, &RUNTIME, service_id.clone())
  736. .await
  737. .unwrap()
  738. };
  739. let (sender, receiver) = oneshot::channel();
  740. let client_handle = spawn_client_manual(UnregisteredState { sender }, &RUNTIME).await;
  741. let service_addr = ServiceAddr::new(service_name, false);
  742. client_handle
  743. .send_register(service_addr, Register { factor: 21 })
  744. .await
  745. .unwrap();
  746. let value = timeout(Duration::from_millis(500), receiver)
  747. .await
  748. .unwrap()
  749. .unwrap();
  750. assert_eq!(42, value);
  751. });
  752. }
  753. }