// Test for Macro, exact same as usecase-recursive use mpstthree::binary::struct_trait::{end::End, recv::Recv, send::Send, session::Session}; use mpstthree::meshedchannels::MeshedChannels; use mpstthree::role::broadcast::RoleBroadcast; use mpstthree::role::end::RoleEnd; use std::error::Error; use std::marker; use rand::{thread_rng, Rng}; use mpstthree::{ choose_mpst_to_all, close_mpst_interleaved, create_multiple_normal_name, create_multiple_normal_role, create_recv_mpst_session_1, create_recv_mpst_session_2, create_send_mpst_session_1, create_send_mpst_session_2, fork_mpst_multi_solo, offer_mpst_interleaved, }; // Create new roles create_multiple_normal_role!( RoleA, RoleADual | RoleB, RoleBDual | RoleC, RoleCDual | ); // Create new names create_multiple_normal_name!(NameA, NameB, NameC); // Create new send functions create_send_mpst_session_1!(send_mpst_c_to_a, RoleA, NameC); create_send_mpst_session_2!(send_mpst_a_to_c, RoleC, NameA); create_send_mpst_session_2!(send_mpst_c_to_b, RoleB, NameC); create_send_mpst_session_1!(send_mpst_b_to_a, RoleA, NameB); create_send_mpst_session_1!(send_mpst_a_to_b, RoleB, NameA); // Create new recv functions and related types create_recv_mpst_session_1!(recv_mpst_c_from_a, RoleA, NameC); create_recv_mpst_session_2!(recv_mpst_a_from_c, RoleC, NameA); create_recv_mpst_session_2!(recv_mpst_b_from_c, RoleC, NameB); create_recv_mpst_session_1!(recv_mpst_b_from_a, RoleA, NameB); create_recv_mpst_session_1!(recv_mpst_a_from_b, RoleB, NameA); close_mpst_interleaved!(close_mpst_multi, MeshedChannels, 3); fork_mpst_multi_solo!(fork_mpst, MeshedChannels, 3); // Types type AtoBVideo = Send>; type AtoCVideo = Recv>>; type BtoAVideo = as Session>::Dual; type RecursAtoC = Recv, End>; type RecursBtoC = Recv, End>; enum Branches0AtoC { End(MeshedChannels), Video(MeshedChannels, AtoCVideo, StackAVideo, NameA>), } enum Branches0BtoC { End(MeshedChannels), Video(MeshedChannels, RecursBtoC, StackBVideo, NameB>), } // Stacks type StackAVideo = RoleC>>>>; type StackBVideo = RoleA>>; // Creating the MP sessions // For C type EndpointCFour = MeshedChannels< Send, End>>>, Send, End>, RoleA>, NameC, >; type EndpointCThree = MeshedChannels, End>, Send, End>, RoleBroadcast, NameC>; type EndpointCOne = MeshedChannels< Send, End>>>, Send, End>, RoleA>, NameC, >; // For A type EndpointAFour = MeshedChannels< Send>, Recv>>, RoleC>>>>, NameA, >; type EndpointAThree = MeshedChannels, RoleC, NameA>; type EndpointAOne = MeshedChannels>>, RoleC>>, NameA>; // For B type EndpointBFour = MeshedChannels< Recv>, Recv, End>, RoleA>>, NameB, >; type EndpointBThree = MeshedChannels, End>, RoleC, NameB>; type EndpointBOne = MeshedChannels, End>, RoleC, NameB>; // Functions related to endpoints fn step_one( s_a: EndpointAOne, s_b: EndpointBOne, s_c: EndpointCOne, ) -> Result<(), Box> { let s_c = send_mpst_c_to_a(0, s_c); let (_, s_a) = recv_mpst_a_from_c(s_a)?; let s_a = send_mpst_a_to_c(1, s_a); let (_, s_c) = recv_mpst_c_from_a(s_c)?; let mut rng = thread_rng(); let xs: Vec = (1..100).map(|_| rng.gen()).collect(); step_two_recurs(s_a, s_b, s_c, xs, 1) } fn step_two_recurs( s_a: EndpointAThree, s_b: EndpointBThree, s_c: EndpointCThree, mut xs: Vec, index: i32, ) -> Result<(), Box> { match xs.pop() { Option::Some(_) => { let s_c = choose_mpst_to_all!( s_c, Branches0AtoC::Video, Branches0BtoC::Video, => NameC ); let (s_a, s_b) = offer_mpst_interleaved!( s_a, recv_mpst_a_from_c, Branches0AtoC::Video, s_b, recv_mpst_b_from_c, Branches0BtoC::Video ); step_three_recurs(s_a, s_b, s_c, xs, index) } Option::None => { let s_c = choose_mpst_to_all!( s_c, Branches0AtoC::End, Branches0BtoC::End, => NameC ); assert_eq!(index, 100); let (s_a, s_b) = offer_mpst_interleaved!( s_a, recv_mpst_a_from_c, Branches0AtoC::End, s_b, recv_mpst_b_from_c, Branches0BtoC::End ); close_mpst_multi(s_a, s_b, s_c) } } } fn step_three_recurs( s_a: EndpointAFour, s_b: EndpointBFour, s_c: EndpointCFour, xs: Vec, index: i32, ) -> Result<(), Box> { let s_c = send_mpst_c_to_a(1, s_c); let (request_a, s_a) = recv_mpst_a_from_c(s_a)?; let s_a = send_mpst_a_to_b(request_a + 1, s_a); let (request_b, s_b) = recv_mpst_b_from_a(s_b)?; let s_b = send_mpst_b_to_a(request_b + 1, s_b); let (video, s_a) = recv_mpst_a_from_b(s_a)?; let s_a = send_mpst_a_to_c(video + 1, s_a); let (_, s_c) = recv_mpst_c_from_a(s_c)?; step_two_recurs(s_a, s_b, s_c, xs, index + 1) } ///////////////////////////////////////// pub fn interleaved_main() { assert!(fork_mpst(step_one).is_ok()); }