Diag-Client-Lib
tcp_client.cpp
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 
10 
11 #include <variant>
12 
20 
21 namespace boost_support {
22 namespace client {
23 namespace tcp {
24 namespace {
28 std::string AppendIpAddressAndPort(std::string_view client_name, std::string_view ip_address,
29  std::uint16_t port_num) {
30  std::string connection_name{client_name};
31  connection_name.append("_");
32  connection_name.append(ip_address);
33  connection_name.append("_");
34  connection_name.append(std::to_string(port_num));
35  return connection_name;
36 }
37 } // namespace
38 
43  private:
49 
53  enum class State : std::uint8_t {
54  kConnected = 0U,
55  kDisconnected = 1U
56  };
57 
62 
63  public:
71  TcpClientImpl(std::string_view client_name, std::string_view local_ip_address,
72  std::uint16_t local_port_num) noexcept
73  : io_context_{},
75  client_name_{AppendIpAddressAndPort(client_name, local_ip_address, local_port_num)},
76  tcp_connection_{client_name,
77  socket::tcp::TcpSocket{local_ip_address, local_port_num, io_context_}} {}
78 
82  TcpClientImpl(const TcpClientImpl &other) noexcept = delete;
83  TcpClientImpl &operator=(const TcpClientImpl &other) noexcept = delete;
84 
88  TcpClientImpl(TcpClientImpl &&other) noexcept = delete;
89  TcpClientImpl &operator=(TcpClientImpl &&other) noexcept = delete;
90 
94  ~TcpClientImpl() noexcept = default;
95 
99  void Initialize() noexcept { tcp_connection_.Initialize(); }
100 
104  void DeInitialize() noexcept { tcp_connection_.DeInitialize(); }
105 
112  void SetReadHandler(HandlerRead read_handler) noexcept {
113  tcp_connection_.SetReadHandler(std::move(read_handler));
114  }
115 
124  core_type::Result<void> ConnectToHost(std::string_view host_ip_address,
125  std::uint16_t host_port_num) {
128  if (connection_state_.load(std::memory_order_seq_cst) != State::kConnected) {
129  if (tcp_connection_.ConnectToHost(host_ip_address, host_port_num)) {
130  connection_state_.store(State::kConnected, std::memory_order_seq_cst);
131  result.EmplaceValue();
132  } // else, connect failed
133  } else {
134  // already connected
135  result.EmplaceValue();
136  common::logger::LibBoostLogger::GetLibBoostLogger().GetLogger().LogVerbose(
137  FILE_NAME, __LINE__, __func__,
138  [](std::stringstream &msg) { msg << "Tcp client is already connected"; });
139  }
140  return result;
141  }
142 
150  if (connection_state_.load(std::memory_order_seq_cst) == State::kConnected) {
151  tcp_connection_.DisconnectFromHost();
152  connection_state_.store(State::kDisconnected, std::memory_order_seq_cst);
153  result.EmplaceValue();
154  } else {
155  // Not connected
157  FILE_NAME, __LINE__, __func__,
158  [](std::stringstream &msg) { msg << "Tcp client is in disconnected state"; });
159  }
160  return result;
161  }
162 
167  auto IsConnectedToHost() const noexcept -> bool {
168  return (connection_state_.load(std::memory_order_seq_cst) == State::kConnected);
169  }
170 
180  if (connection_state_.load(std::memory_order_seq_cst) == State::kConnected) {
181  if (tcp_connection_.Transmit(std::move(tcp_message))) { result.EmplaceValue(); }
182  } else {
183  // not connected
185  FILE_NAME, __LINE__, __func__, [](std::stringstream &msg) {
186  msg << "Tcp client is Offline, please connect to server first";
187  });
188  }
189  return result;
190  }
191 
192  private:
197 
201  std::atomic<State> connection_state_;
202 
206  std::string client_name_;
207 
212 };
213 
214 TcpClient::TcpClient(std::string_view client_name, std::string_view local_ip_address,
215  std::uint16_t local_port_num) noexcept
216  : tcp_client_impl_{
217  std::make_unique<TcpClientImpl>(client_name, local_ip_address, local_port_num)} {}
218 
219 TcpClient::TcpClient(TcpClient &&other) noexcept = default;
220 
221 TcpClient &TcpClient::operator=(TcpClient &&other) noexcept = default;
222 
223 TcpClient::~TcpClient() noexcept = default;
224 
225 void TcpClient::Initialize() noexcept { tcp_client_impl_->Initialize(); }
226 
227 void TcpClient::DeInitialize() noexcept { tcp_client_impl_->DeInitialize(); }
228 
230  tcp_client_impl_->SetReadHandler(std::move(read_handler));
231 }
232 
233 core_type::Result<void> TcpClient::ConnectToHost(std::string_view host_ip_address,
234  std::uint16_t host_port_num) {
235  return tcp_client_impl_->ConnectToHost(host_ip_address, host_port_num);
236 }
237 
239  return tcp_client_impl_->DisconnectFromHost();
240 }
241 
242 auto TcpClient::IsConnectedToHost() const noexcept -> bool {
243  return tcp_client_impl_->IsConnectedToHost();
244 }
245 
247  return tcp_client_impl_->Transmit(std::move(tcp_message));
248 }
249 
250 } // namespace tcp
251 } // namespace client
252 } // namespace boost_support
Class to provide implementation of tcp client.
Definition: tcp_client.cpp:42
TcpClientImpl & operator=(TcpClientImpl &&other) noexcept=delete
void SetReadHandler(HandlerRead read_handler) noexcept
Function to set the read handler that is invoked when message is received.
Definition: tcp_client.cpp:112
~TcpClientImpl() noexcept=default
Destruct an instance of TcpClientImpl.
TcpClientImpl(const TcpClientImpl &other) noexcept=delete
Deleted copy assignment and copy constructor.
State
Definitions of different connection state.
Definition: tcp_client.cpp:53
TcpConnection tcp_connection_
Store the tcp connection.
Definition: tcp_client.cpp:211
TcpClientImpl & operator=(const TcpClientImpl &other) noexcept=delete
core_type::Result< void > ConnectToHost(std::string_view host_ip_address, std::uint16_t host_port_num)
Function to connect to remote ip address and port number.
Definition: tcp_client.cpp:124
void DeInitialize() noexcept
De-initialize the client.
Definition: tcp_client.cpp:104
void Initialize() noexcept
Initialize the client.
Definition: tcp_client.cpp:99
core_type::Result< void > DisconnectFromHost()
Function to disconnect from remote host if already connected.
Definition: tcp_client.cpp:147
TcpClientImpl(std::string_view client_name, std::string_view local_ip_address, std::uint16_t local_port_num) noexcept
Constructs an instance of TcpClientImpl.
Definition: tcp_client.cpp:71
auto IsConnectedToHost() const noexcept -> bool
Function to get the connection status.
Definition: tcp_client.cpp:167
std::atomic< State > connection_state_
Store the state of tcp connection.
Definition: tcp_client.cpp:201
TcpClientImpl(TcpClientImpl &&other) noexcept=delete
Deleted move assignment and move constructor.
core_type::Result< void > Transmit(MessageConstPtr tcp_message)
Function to transmit the provided tcp message.
Definition: tcp_client.cpp:177
Client that manages unsecured/ secured tcp connection.
Definition: tcp_client.h:24
TcpClient & operator=(const TcpClient &other) noexcept=delete
TcpClient(std::string_view client_name, std::string_view local_ip_address, std::uint16_t local_port_num) noexcept
Constructs an instance of TcpClient.
Definition: tcp_client.cpp:214
core_type::Result< void > DisconnectFromHost()
Function to disconnect from remote host if already connected.
Definition: tcp_client.cpp:238
boost_support::message::tcp::TcpMessageConstPtr MessageConstPtr
Type alias for Tcp message const pointer.
Definition: tcp_client.h:39
void Initialize() noexcept
Initialize the client.
Definition: tcp_client.cpp:225
void SetReadHandler(HandlerRead read_handler) noexcept
Function to set the read handler that is invoked when message is received.
Definition: tcp_client.cpp:229
core_type::Result< void > Transmit(MessageConstPtr tcp_message)
Function to transmit the provided tcp message.
Definition: tcp_client.cpp:246
void DeInitialize() noexcept
De-initialize the client.
Definition: tcp_client.cpp:227
~TcpClient() noexcept
Destruct an instance of TcpClient.
auto IsConnectedToHost() const noexcept -> bool
Function to get the connection status.
Definition: tcp_client.cpp:242
core_type::Result< void > ConnectToHost(std::string_view host_ip_address, std::uint16_t host_port_num)
Function to connect to remote ip address and port number.
Definition: tcp_client.cpp:233
std::function< void(MessagePtr)> HandlerRead
Tcp function template used for reception.
Definition: tcp_client.h:44
std::unique_ptr< TcpClientImpl > tcp_client_impl_
Unique pointer to tcp client implementation.
Definition: tcp_client.h:129
static auto GetLibBoostLogger() noexcept -> LibBoostLogger &
Definition: logger.h:20
Wrapper class to hold boost io context required for io object( sockets)
Definition: io_context.h:22
Class used to create a tcp socket for handling transmission and reception of tcp message from driver.
Definition: tcp_socket.h:24
Class type to contains a value (of type ValueType), or an error (of type ErrorType)
Definition: result.h:29
#define FILE_NAME
Definition: file_path.h:14
std::string AppendIpAddressAndPort(std::string_view client_name, std::string_view ip_address, std::uint16_t port_num)
Function to append the ip address and port number to the connection name.
Definition: tcp_client.cpp:28
auto MakeErrorCode(BoostSupportErrorDomain::Errc code, BoostSupportErrorDomain::SupportDataType data) noexcept -> core_type::ErrorCode
Create a new ErrorCode within DoipErrorDomain.