1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
use ferrite_session::prelude::*;
type Queue = LinearToShared<ExternalChoice<QueueOps>>;
define_choice! { QueueOps;
Enqueue: ReceiveValue<String, Release>,
Dequeue: SendValue<Option<String>, Release>
}
fn shared_queue(mut queue: Vec<String>) -> SharedSession<Queue>
{
accept_shared_session(async move {
offer_choice! {
Enqueue => {
receive_value(move |val: String| {
queue.push(val);
detach_shared_session(shared_queue(queue))
})
}
Dequeue => {
send_value(queue.pop(),
detach_shared_session(shared_queue(queue)))
}
}
})
}
fn create_shared_queue() -> SharedChannel<Queue>
{
run_shared_session(shared_queue(vec![]))
}
async fn enqueue(
queue: SharedChannel<Queue>,
val: String,
)
{
run_session(acquire_shared_session(queue, move |chan| {
choose!(
chan,
Enqueue,
send_value_to(chan, val, release_shared_session(chan, terminate()))
)
}))
.await;
}
async fn dequeue_and_print(queue: SharedChannel<Queue>)
{
run_session(acquire_shared_session(queue, move |chan| {
choose!(
chan,
Dequeue,
receive_value_from(chan, move |val| {
match val {
Some(val) => {
println!("Gotten dequeue value: {}", val);
}
None => {
println!("Dequeue returns None");
}
}
release_shared_session(chan, terminate())
})
)
}))
.await
}
#[tokio::main]
pub async fn main()
{
env_logger::init();
let queue = create_shared_queue();
enqueue(queue.clone(), "Hello".to_string()).await;
enqueue(queue.clone(), "World".to_string()).await;
dequeue_and_print(queue.clone()).await;
dequeue_and_print(queue.clone()).await;
dequeue_and_print(queue.clone()).await;
}