Unlike Rust, C++ does not have a networking library yet (probably coming with C++2023). So MPL
implements a networking library with three main classes, TCPConnector, TCPResponder, and TCPSocket.
These take care of a lot of the low-level things that are hard to implement cleanly.
The MPL library:
-
Uses queued full-duplex buffered message transfers.
-
Each message has a fixed size header and a body consisting of an array of bytes.
-
For each incoming connection the TCPResponder requests a threadpool thread and processes
messages with an instance of ClientHandler.
-
For this demonstration ClientHandler instances echo back the incoming message, marked as
reply.
The long-term goal for MPL Comm is to serve as a lightweight, flexible, and performant messaging
middleware for systems that require simple and robust end to end messaging.
The TCPConnector and TCPResponder each provide an object-oriented interface that gives users
ability to define application specific message processing.
Additionally, The TCP socket library is an efficient and portable C++ wrapper around native
TCP socket APIs available on Windows and Linux. This library can be extracted and used on Windows
and Linux as an alternative the native (non-portable platform specific) APIs.
Current Design:
There are user-defined types Message, TCPConnector, TCPResponder, ClientHandler, and TCPSocket.
TCPConnector and ClientHandler have derived classes for specific message types, which in this
demonstration are: VariableSizeConnector and VariableSizeClientHandler.
TCPSocket has derived classes TCPClientSocket nad TCPServerSocket.
Messages consist of a header with mtype and content_len attributes and an array of bytes for the
body.
Message methods:
-
Message(usize sz, const u8 content_buf[], usize content_len, u8 mtype)
Create new Message from array of bytes
-
Message(usize sz, const std::string& str, u8 mtype)
Create new Message from string
-
Message(MessageType mtype=MessageType::DEFAULT)
Create new Message with no contents
-
Message(const Message &msg)
Copy constructor
-
Message &operator=(const Message &msg)
Copy assignment
-
Message(Message &&msg)
Move constructor
-
Message &operator=(Message &&msg)
Move assignment
-
~Message()
Destructor
-
u8 operator[](int index) const
Const index operator
-
void set_type(u8 mt)
Set MessageType to one of TEXT, BYTES, STRING
-
unsigned get_type() const
Returns instance of MessageType
-
usize get_content_len() const
Returns length of message body in bytes
-
void set_content_bytes(const u8 buff[], size_t len)
Copy byte array into message body
-
void set_content_str(const std::string &str)
Copy string to message body
-
std::string get_content_str()
Return message body as string
TCPConnector supports direct and queued messaging with a connected TCPResponder.
TCPConnector methods:
-
TCPConnector(TCPSocketOptions *sc = nullptr);
Create new TCPConnector
-
bool Close()
Shutdown TCPConnector and signal server-side ClientHandler to shutdown.
-
bool IsConnected() const;
Has valid connection?
-
bool IsSending() const;
Is send thread running?
-
bool IsReceiving() const;
Is receive thread running?
-
void UseSendReceiveQueues(bool use_qs);
Starts dedicated send and receive threads.
-
void UseSendQueue(bool use_q);
Start send thread on connection established.
-
void UseReceiveQueue(bool use_q);
Start receive thread on connection established.
-
void PostMessage(const Message &m);
Enqueues message for sending to associated TCPResponder.
-
void SendMessage(const Message &m);
Sends message directly instead of posting to send queue.
-
Message GetMessage()
Dequeue received message if available, else block.
-
Message ReceiveMessage()
Read message from socket if available, else block.
TCPResponder uses threadpool threads to support concurrent communication sessions.
TCPResponder methods:
-
TCPResponder(const EndPoint& ep, TCPSocketOptions* sc = nullptr)
Create new TCPResponder
-
void RegisterClientHandler(ClientHandler* ch);
Register ClientHandler prototype - defines server-side message processing.
-
void Start(int backlog=20)
Start dedicated server listening thread.
-
void Stop()
Stop listening service.
-
void UseClientSendReceiveQueues(bool use_qs);
Start dedicated send and receive threads when extablishing a connection.
-
void UseClientSendQueue(bool use_q);
Start dedicated send thread when establishing a connection.
-
void UseClientReceiveQueue(bool use_q);
Start dedicated receive thread when establishing a connection.
ClientHandlers communicate directly with associated TCPConnectors. Derived ClientHandler
classes define application specific message handling operations.
ClientHandler methods:
-
Message GetMessage()
Dequeue message from receive queue if available, else blocks.
-
Message ReceiveMessage()
Read message directly from socket if available, else blocks.
-
void PostMessage(const Message& m)
Enqueues message for connected client.
-
void SendMessage(const Message& m)
Write message directly to connected socket.
-
virtual void AppProc() = 0
Application message processing supplied by derived ClientHandler class.
-
virtual ClientHandler* Clone() = 0;
Create application defined ClientHandler instances.