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
126
127
128
129
130
131
132
133
134
135
136
137
138
use crate::{pipe2::PipeSystem, Context, OckamMessage};
use ockam_core::{
compat::{boxed::Box, collections::VecDeque},
Address, Any, Result, Route, Routed, Worker,
};
pub enum PeerRoute {
Peer(Route),
Listener(Route, Address),
}
pub struct PipeSender {
system: PipeSystem,
out_buf: VecDeque<OckamMessage>,
peer: Option<PeerRoute>,
api_addr: Address,
fin_addr: Address,
}
#[crate::worker]
impl Worker for PipeSender {
type Context = Context;
type Message = Any;
async fn initialize(&mut self, ctx: &mut Context) -> Result<()> {
ctx.set_cluster(crate::pipe2::CLUSTER_NAME).await?;
if let Some(PeerRoute::Listener(ref route, ref addr)) = self.peer {
debug!("Sending pipe2 handshake request to listener: {}", route);
ctx.send_from_address(route.clone(), OckamMessage::new(Any)?, addr.clone())
.await?;
}
Ok(())
}
async fn handle_message(&mut self, ctx: &mut Context, msg: Routed<Any>) -> Result<()> {
debug!(
"PipeSender '{}': handling incoming message to {}",
ctx.address(),
msg.onward_route()
);
match (msg.msg_addr(), self.peer.as_ref()) {
(ref addr, Some(&PeerRoute::Listener(_, ref _self))) if addr == _self => {
let return_route = msg.return_route();
self.peer = Some(PeerRoute::Peer(return_route.clone()));
for msg in core::mem::replace(&mut self.out_buf, vec![].into()) {
ctx.send(return_route.clone(), msg).await?;
}
Ok(())
}
(addr, _) if addr == self.api_addr => self.handle_api_msg(ctx, msg).await,
(addr, _) if addr == self.fin_addr => {
self.handle_fin_msg(ctx, OckamMessage::from_any(msg)?).await
}
_ => self.system.handle_message(ctx, msg.cast()?).await,
}
}
}
impl PipeSender {
pub fn new(system: PipeSystem, peer: PeerRoute, api_addr: Address, fin_addr: Address) -> Self {
Self {
out_buf: VecDeque::default(),
peer: Some(peer),
system,
api_addr,
fin_addr,
}
}
async fn handle_api_msg(&mut self, ctx: &mut Context, msg: Routed<Any>) -> Result<()> {
trace!(
"PipeSender '{}' handling initial user message stage...",
ctx.address()
);
let msg_addr = msg.msg_addr().clone();
let onward_route = msg.onward_route();
let return_route = msg.return_route();
let mut inner = msg.into_transport_message();
inner.onward_route.modify().pop_front();
let ockam_msg = OckamMessage::new(inner)?;
if self.system.is_empty() {
self.handle_fin_msg(ctx, ockam_msg).await?;
} else {
let routed = ockam_msg.into_routed(msg_addr, onward_route, return_route)?;
self.system.dispatch_entry(ctx, routed).await?;
}
Ok(())
}
async fn handle_fin_msg(&mut self, ctx: &mut Context, msg: OckamMessage) -> Result<()> {
trace!(
"PipeSender '{}' handling final user message stage...",
ctx.address()
);
match self.peer {
Some(PeerRoute::Peer(ref peer)) => ctx.send(peer.clone(), msg).await?,
_ => self.out_buf.push_back(msg),
}
Ok(())
}
}