Diag-Client-Lib
tcp_connection.h
Go to the documentation of this file.
1 /* Diagnostic Client library
2  * Copyright (C) 2024 Avijit Dey
3  *
4  * This Source Code Form is subject to the terms of the Mozilla Public
5  * License, v. 2.0. If a copy of the MPL was not distributed with this
6  * file, You can obtain one at http://mozilla.org/MPL/2.0/.
7  */
8 #ifndef DIAG_CLIENT_LIB_LIB_BOOST_SUPPORT_SRC_BOOST_SUPPORT_CONNECTION_TCP_TCP_CONNECTION_H_
9 #define DIAG_CLIENT_LIB_LIB_BOOST_SUPPORT_SRC_BOOST_SUPPORT_CONNECTION_TCP_TCP_CONNECTION_H_
10 
11 #include <atomic>
12 #include <condition_variable>
13 #include <functional>
14 #include <memory>
15 #include <mutex>
16 #include <string>
17 #include <string_view>
18 #include <utility>
19 
21 #include "core/include/result.h"
22 #include "utility/thread.h"
23 
24 namespace boost_support {
25 namespace connection {
26 namespace tcp {
27 
31 enum class ConnectionType : std::uint8_t { kClient, kServer };
32 
40 template<ConnectionType Connection, typename Socket>
42 
48 template<typename Socket>
49 class TcpConnection<ConnectionType::kClient, Socket> final {
50  public:
54  using TcpMessage = typename Socket::TcpMessage;
55 
60 
65 
69  using HandlerRead = std::function<void(TcpMessagePtr)>;
70 
71  public:
77  explicit TcpConnection(std::string_view connection_name, Socket socket) noexcept
78  : socket_{std::move(socket)},
79  handler_read_{},
80  exit_request_{false},
81  running_{false},
82  cond_var_{},
83  connection_name_{connection_name},
84  thread_{},
85  mutex_{} {}
86 
90  TcpConnection(const TcpConnection &other) noexcept = delete;
91  TcpConnection &operator=(const TcpConnection &other) &noexcept = delete;
92 
96  TcpConnection(TcpConnection &&other) noexcept = default;
97  TcpConnection &operator=(TcpConnection &&other) &noexcept = default;
98 
102  ~TcpConnection() noexcept = default;
103 
110  void SetReadHandler(HandlerRead read_handler) { handler_read_ = std::move(read_handler); }
111 
115  void Initialize() noexcept {
116  // Open socket
117  socket_.Open();
118  // Start thread to receive messages
119  thread_ = utility::thread::Thread{
120  connection_name_, [this]() {
121  std::unique_lock<std::mutex> lck(mutex_);
122  while (!exit_request_) {
123  if (!running_) {
124  cond_var_.wait(lck, [this]() { return exit_request_ || running_; });
125  }
126  if (!exit_request_.load() && running_) {
127  lck.unlock();
128  running_ = ReadMessage();
129  lck.lock();
130  }
131  }
132  }};
133  }
134 
138  void DeInitialize() noexcept {
139  socket_.Close();
140  {
141  std::unique_lock<std::mutex> lck(mutex_);
142  exit_request_ = true;
143  running_ = false;
144  }
145  cond_var_.notify_all();
146  thread_.Join();
147  }
148 
157  auto ConnectToHost(std::string_view host_ip_address, std::uint16_t host_port_num) noexcept
158  -> bool {
159  return socket_.Connect(host_ip_address, host_port_num)
160  .AndThen([this]() noexcept {
161  { // start reading
162  std::lock_guard<std::mutex> lock{mutex_};
163  running_ = true;
164  cond_var_.notify_all();
165  }
166  })
167  .HasValue();
168  }
169 
174  void DisconnectFromHost() noexcept {
175  { // stop reading
176  std::lock_guard<std::mutex> lock{mutex_};
177  running_ = false;
178  }
179  socket_.Disconnect();
180  }
181 
189  return socket_.Transmit(std::move(message))
190  .AndThen([]() { return core_type::Result<void>::FromValue(); })
191  .MapError([](typename Socket::SocketError const &) {
193  });
194  }
195 
196  private:
200  Socket socket_;
201 
206 
210  std::atomic_bool exit_request_;
211 
215  std::atomic_bool running_;
216 
220  std::condition_variable cond_var_;
221 
225  std::string connection_name_;
226 
231 
235  std::mutex mutex_;
236 
237  private:
242  auto ReadMessage() -> bool {
243  // Try reading from socket
244  return socket_.Read()
245  .AndThen([this](TcpMessagePtr tcp_message) {
246  if (handler_read_) { handler_read_(std::move(tcp_message)); }
248  })
249  .HasValue();
250  }
251 };
252 
258 template<typename Socket>
259 class TcpConnection<ConnectionType::kServer, Socket> final {
260  public:
264  using TcpMessage = typename Socket::TcpMessage;
265 
270 
275 
279  using HandlerRead = std::function<void(TcpMessagePtr)>;
280 
281  public:
287  explicit TcpConnection(std::string_view connection_name, Socket socket) noexcept
288  : socket_{std::move(socket)},
289  handler_read_{},
290  exit_request_{false},
291  running_{false},
292  cond_var_{},
293  connection_name_{connection_name},
294  thread_{},
295  mutex_{} {}
296 
300  TcpConnection(const TcpConnection &other) noexcept = delete;
301  TcpConnection &operator=(const TcpConnection &other) &noexcept = delete;
302 
306  TcpConnection(TcpConnection &&other) noexcept
307  : socket_{std::move(other.socket_)},
308  handler_read_{std::move(other.handler_read_)},
309  exit_request_{other.exit_request_.load()},
310  running_{other.running_.load()},
311  thread_{std::move(other.thread_)} {}
312 
316  TcpConnection &operator=(TcpConnection &&other) &noexcept {
317  socket_ = std::move(other.socket_);
318  handler_read_ = std::move(other.handler_read_);
319  exit_request_.store(other.exit_request_.load());
320  running_.store(other.running_.load());
321  thread_ = std::move(other.thread_);
322  return *this;
323  }
324 
328  ~TcpConnection() noexcept = default;
329 
336  void SetReadHandler(HandlerRead read_handler) { handler_read_ = std::move(read_handler); }
337 
341  void Initialize() noexcept {
342  // Start thread to receive messages
343  thread_ = utility::thread::Thread(connection_name_, [this]() {
344  std::unique_lock<std::mutex> lck(mutex_);
345  while (!exit_request_) {
346  if (!running_) {
347  cond_var_.wait(lck, [this]() { return exit_request_ || running_; });
348  }
349  if (!exit_request_ && running_) {
350  lck.unlock();
351  running_ = ReadMessage();
352  lck.lock();
353  }
354  }
355  });
356  { // start reading
357  std::lock_guard<std::mutex> lock{mutex_};
358  running_ = true;
359  cond_var_.notify_all();
360  }
361  }
362 
366  void DeInitialize() noexcept {
367  socket_.Close();
368  {
369  std::unique_lock<std::mutex> lck(mutex_);
370  exit_request_ = true;
371  running_ = false;
372  }
373  cond_var_.notify_all();
374  thread_.Join();
375  }
376 
384  return socket_.Transmit(std::move(message))
385  .AndThen([]() { return core_type::Result<void>::FromValue(); })
386  .MapError([](typename Socket::SocketError const &) {
388  });
389  }
390 
391  private:
395  Socket socket_;
396 
401 
405  std::atomic_bool exit_request_;
406 
410  std::atomic_bool running_;
411 
415  std::condition_variable cond_var_;
416 
420  std::string connection_name_;
421 
426 
430  std::mutex mutex_;
431 
432  private:
437  auto ReadMessage() noexcept -> bool {
438  // Try reading from socket
439  return socket_.Read()
440  .AndThen([this](TcpMessagePtr tcp_message) {
441  if (handler_read_) { handler_read_(std::move(tcp_message)); }
443  })
444  .HasValue();
445  }
446 };
447 
448 } // namespace tcp
449 } // namespace connection
450 } // namespace boost_support
451 #endif // DIAG_CLIENT_LIB_LIB_BOOST_SUPPORT_SRC_BOOST_SUPPORT_CONNECTION_TCP_TCP_CONNECTION_H_
typename Socket::TcpMessage TcpMessage
Type alias for Tcp message.
typename Socket::TcpMessagePtr TcpMessagePtr
Type alias for Tcp message pointer.
TcpConnection(std::string_view connection_name, Socket socket) noexcept
Constructs an instance of TcpConnection.
Socket socket_
Store socket used for reading and writing tcp message.
typename Socket::TcpMessageConstPtr TcpMessageConstPtr
Type alias for Tcp message const pointer.
~TcpConnection() noexcept=default
Destruct an instance of TcpConnection.
std::condition_variable cond_var_
Conditional variable to block the thread.
TcpConnection & operator=(TcpConnection &&other) &noexcept=default
core_type::Result< void > Transmit(TcpMessageConstPtr message) noexcept
Function to trigger transmission.
TcpConnection & operator=(const TcpConnection &other) &noexcept=delete
auto ConnectToHost(std::string_view host_ip_address, std::uint16_t host_port_num) noexcept -> bool
Function to connect to remote ip address and port number.
std::function< void(TcpMessagePtr)> HandlerRead
Tcp function template used for reception.
TcpConnection(const TcpConnection &other) noexcept=delete
Deleted copy assignment and copy constructor.
TcpConnection(TcpConnection &&other) noexcept=default
Move assignment and move constructor.
auto ReadMessage() -> bool
Function to send the received message to stored handler.
auto ReadMessage() noexcept -> bool
Function to send the received message to stored handler.
TcpConnection(const TcpConnection &other) noexcept=delete
Deleted copy assignment and copy constructor.
core_type::Result< void > Transmit(TcpMessageConstPtr message) noexcept
Function to trigger transmission.
Socket socket_
Store socket used for reading and writing tcp message.
std::function< void(TcpMessagePtr)> HandlerRead
Tcp function template used for reception.
std::condition_variable cond_var_
Conditional variable to block the thread.
~TcpConnection() noexcept=default
Destruct an instance of TcpConnection.
typename Socket::TcpMessage TcpMessage
Type alias for Tcp message.
TcpConnection & operator=(TcpConnection &&other) &noexcept
Move constructor.
typename Socket::TcpMessageConstPtr TcpMessageConstPtr
Type alias for Tcp message const pointer.
typename Socket::TcpMessagePtr TcpMessagePtr
Type alias for Tcp message pointer.
TcpConnection(std::string_view connection_name, Socket socket) noexcept
Constructs an instance of TcpConnection.
TcpConnection & operator=(const TcpConnection &other) &noexcept=delete
Primary connection template to handle transmission and reception of tcp message from socket.
Class type to contains a value (of type ValueType), or an error (of type ErrorType)
Definition: result.h:29
static Result FromValue(T &t) noexcept
Build a new Result from the specified value (given as lvalue)
Definition: result.h:48
ConnectionType
Type of connection Client/ Server.
auto MakeErrorCode(BoostSupportErrorDomain::Errc code, BoostSupportErrorDomain::SupportDataType data) noexcept -> core_type::ErrorCode
Create a new ErrorCode within DoipErrorDomain.
std::unique_ptr< TcpMessage > TcpMessagePtr
The unique pointer to TcpMessage.
Definition: tcp_message.h:151
std::unique_ptr< TcpMessage const > TcpMessageConstPtr
The unique pointer to const TcpMessage.
Definition: tcp_message.h:146