forked from sourcedelica/manual
-
Notifications
You must be signed in to change notification settings - Fork 0
/
NetworkTransparency.tex
78 lines (62 loc) · 3.71 KB
/
NetworkTransparency.tex
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
\section{Middleman}
\label{middleman}
The middleman is the main component of the I/O module and enables distribution. It transparently manages proxy actor instances representing remote actors, maintains connections to other nodes, and takes care of serialization of messages. Applications install a middleman by loading \lstinline^caf::io::middleman^ as module~\see{system-config}. Users can include \lstinline^"caf/io/all.hpp"^ to get access to all public classes of the I/O module.
\subsection{Class \texttt{middleman}}
\begin{center}\small
\begin{tabular}{ll}
\textbf{Remoting} & ~ \\
\hline
\lstinline^expected<uint16> publish(T, uint16, const char*, bool)^ & See~\sref{remoting}. \\
\hline
\lstinline^expected<void> unpublish(T x, uint16)^ & See~\sref{remoting}. \\
\hline
\lstinline^expected<T> remote_actor<T = actor>(string, uint16)^ & See~\sref{remoting}. \\
\hline
\lstinline^expected<T> spawn_broker(F fun, ...)^ & See~\sref{broker}. \\
\hline
\lstinline^expected<T> spawn_client(F, string, uint16, ...)^ & See~\sref{broker}. \\
\hline
\lstinline^expected<T> spawn_server(F, uint16, ...)^ & See~\sref{broker}. \\
\hline
\end{tabular}
\end{center}
\subsection{Publishing and Connecting}
\label{remoting}
The member function \lstinline^publish^ binds an actor to a given port, thereby allowing other nodes to access it over the network.
\begin{lstlisting}
template <class T>
expected<uint16_t> middleman::publish(T x, uint16_t port,
const char* in = nullptr,
bool reuse_addr = false);
\end{lstlisting}
The first argument is a handle of type \lstinline^actor^ or \lstinline^typed_actor<...>^. The second argument denotes the TCP port. The OS will pick a random high-level port when passing 0. The third parameter configures the listening address. Passing null will accept all incoming connections (\lstinline^INADDR_ANY^). Finally, the flag \lstinline^reuse_addr^ controls the behavior when binding an IP address to a port, with the same semantics as the BSD socket flag \lstinline^SO_REUSEADDR^.
For example, with \lstinline^reuse_addr = false^, binding two sockets to 0.0.0.0:42 and 10.0.0.1:42 will fail with \texttt{EADDRINUSE} since 0.0.0.0 includes 10.0.0.1.
With \lstinline^reuse_addr = true^ binding would succeed because 10.0.0.1 and
0.0.0.0 are not literally equal addresses.
The member function returns the bound port on success. Otherwise, an \lstinline^error^ \see{error} is returned.
\begin{lstlisting}
template <class T>
expected<uint16_t> middleman::unpublish(T x, uint16_t port = 0);
\end{lstlisting}
The member function \lstinline^unpublish^ allows actors to close a port manually. This is performed automatically if the published actor terminates. Passing 0 as second argument closes all ports an actor is published to, otherwise only one specific port is closed.
The function returns an \lstinline^error^ \see{error} if the actor was not bound to given port.
\clearpage
\begin{lstlisting}
template<class T = actor>
expected<T> middleman::remote_actor(std::string host, uint16_t port);
\end{lstlisting}
After a server has published an actor with \lstinline^publish^, clients can connect to the published actor by calling \lstinline^remote_actor^:
\begin{lstlisting}
// node A
auto ping = spawn(ping);
system.middleman().publish(ping, 4242);
// node B
auto ping = system.middleman().remote_actor("node A", 4242);
if (! ping) {
cerr << "unable to connect to node A: "
<< system.render(ping.error()) << std::endl;
} else {
self->send(*ping, ping_atom::value);
}
\end{lstlisting}
There is no difference between server and client after the connection phase. Remote actors use the same handle types as local actors and are thus fully transparent.