#![allow(clippy::type_complexity)]
use mpstthree::binary::struct_trait::{end::End, recv::Recv, send::Send, session::Session};
use mpstthree::role::broadcast::RoleBroadcast;
use mpstthree::role::end::RoleEnd;
use mpstthree::{
bundle_struct_fork_close_multi, choose_mpst_multi_to_all, create_multiple_normal_role,
create_recv_mpst_session_bundle, create_send_mpst_session_bundle, offer_mpst,
};
use rand::{thread_rng, Rng};
use std::error::Error;
use std::marker;
bundle_struct_fork_close_multi!(close_mpst_multi, fork_mpst, MeshedChannelsTwo, 2);
create_multiple_normal_role!(
RoleA, RoleADual |
RoleB, RoleBDual |
);
create_send_mpst_session_bundle!(
send_mpst_a_to_b, RoleB, 1 | =>
RoleA, MeshedChannelsTwo, 2
);
create_send_mpst_session_bundle!(
send_mpst_b_to_a, RoleA, 1 | =>
RoleB, MeshedChannelsTwo, 2
);
create_recv_mpst_session_bundle!(
recv_mpst_a_from_b, RoleB, 1 | =>
RoleA, MeshedChannelsTwo, 2
);
create_recv_mpst_session_bundle!(
recv_mpst_b_from_a, RoleA, 1 | =>
RoleB, MeshedChannelsTwo, 2
);
type NameA = RoleA<RoleEnd>;
type NameB = RoleB<RoleEnd>;
type Choose0fromAtoB<N> = <RecursBtoA<N> as Session>::Dual;
enum Branching0fromAtoB<N: marker::Send> {
More(MeshedChannelsTwo<RSRecursBtoA<N>, ThreeRoleA, NameB>),
Done(MeshedChannelsTwo<End, RoleEnd, NameB>),
}
type RSRecursBtoA<N> = Recv<N, Send<N, RecursBtoA<N>>>;
type ThreeRoleA = RoleA<RoleA<RoleA<RoleEnd>>>;
type RecursBtoA<N> = Recv<Branching0fromAtoB<N>, End>;
type EndpointA<N> = MeshedChannelsTwo<Choose0fromAtoB<N>, RoleBroadcast, NameA>;
type EndpointB<N> = MeshedChannelsTwo<RecursBtoA<N>, RoleA<RoleEnd>, NameB>;
fn endpoint_a(s: EndpointA<i64>) -> Result<(), Box<dyn Error>> {
recurs_a(s, thread_rng().gen_range(1..20), 1)
}
fn recurs_a(s: EndpointA<i64>, index: i64, old: i64) -> Result<(), Box<dyn Error>> {
match index {
0 => {
let s = choose_mpst_multi_to_all!(
s,
Branching0fromAtoB::<i64>::Done, =>
RoleB, =>
RoleA,
MeshedChannelsTwo,
1
);
close_mpst_multi(s)
}
i => {
let s = choose_mpst_multi_to_all!(
s,
Branching0fromAtoB::<i64>::More, =>
RoleB, =>
RoleA,
MeshedChannelsTwo,
1
);
let s = send_mpst_a_to_b(old, s);
let (new, s) = recv_mpst_a_from_b(s)?;
recurs_a(s, i - 1, new)
}
}
}
fn endpoint_b(s: EndpointB<i64>) -> Result<(), Box<dyn Error>> {
recurs_b(s, 0)
}
fn recurs_b(s: EndpointB<i64>, old: i64) -> Result<(), Box<dyn Error>> {
offer_mpst!(s, recv_mpst_b_from_a, {
Branching0fromAtoB::Done(s) => {
close_mpst_multi(s)
},
Branching0fromAtoB::More(s) => {
let (new, s) = recv_mpst_b_from_a(s)?;
let s = send_mpst_b_to_a(new + old, s);
recurs_b(s, new + old)
},
})
}
fn main() {
let (thread_a, thread_b) = fork_mpst(endpoint_a, endpoint_b);
assert!(thread_a.join().is_ok());
assert!(thread_b.join().is_ok());
}