Diag-Client-Lib
tcp_socket.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 <utility>
12 
14 
15 namespace boost_support {
16 namespace socket {
17 namespace tcp {
18 
19 TcpSocket::TcpSocket(std::string_view local_ip_address, std::uint16_t local_port_num,
20  IoContext &io_context) noexcept
21  : tcp_socket_{io_context.GetContext()},
22  local_endpoint_{boost::asio::ip::make_address(local_ip_address), local_port_num} {}
23 
25  : tcp_socket_{std::move(socket)},
26  local_endpoint_{tcp_socket_.local_endpoint()} {}
27 
28 TcpSocket::~TcpSocket() noexcept = default;
29 
30 core_type::Result<void, TcpSocket::SocketError> TcpSocket::Open() noexcept {
31  core_type::Result<void, SocketError> result{SocketError::kGenericError};
32  TcpErrorCodeType ec{};
33 
34  // Open the socket
35  tcp_socket_.open(local_endpoint_.protocol(), ec);
36  if (ec.value() == boost::system::errc::success) {
37  // reuse address
38  tcp_socket_.set_option(boost::asio::socket_base::reuse_address{true});
39  // Set socket to non blocking
40  tcp_socket_.non_blocking(false);
41  // Bind to local ip address and random port
42  tcp_socket_.bind(local_endpoint_, ec);
43 
44  if (ec.value() == boost::system::errc::success) {
45  local_endpoint_ = tcp_socket_.local_endpoint();
46  // Socket binding success
48  FILE_NAME, __LINE__, __func__, [this](std::stringstream &msg) {
49  Tcp::endpoint const endpoint_{tcp_socket_.local_endpoint()};
50  msg << "Tcp Socket opened and bound to "
51  << "<" << endpoint_.address().to_string() << "," << endpoint_.port() << ">";
52  });
53  result.EmplaceValue();
54  } else {
55  // Socket binding failed
57  FILE_NAME, __LINE__, __func__, [ec](std::stringstream &msg) {
58  msg << "Tcp Socket binding failed with message: " << ec.message();
59  });
60  result.EmplaceError(SocketError::kBindingFailed);
61  }
62  } else {
64  FILE_NAME, __LINE__, __func__, [ec](std::stringstream &msg) {
65  msg << "Tcp Socket opening failed with error: " << ec.message();
66  });
67  result.EmplaceError(SocketError::kOpenFailed);
68  }
69  return result;
70 }
71 
73  std::string_view host_ip_address, std::uint16_t host_port_num) noexcept {
74  core_type::Result<void, SocketError> result{SocketError::kGenericError};
75  TcpErrorCodeType ec{};
76 
77  // Connect to provided Ip address
78  tcp_socket_.connect(
79  Tcp::endpoint(TcpIpAddress::from_string(std::string{host_ip_address}), host_port_num), ec);
80  if (ec.value() == boost::system::errc::success) {
82  FILE_NAME, __LINE__, __func__, [this](std::stringstream &msg) {
83  Tcp::endpoint const endpoint_{tcp_socket_.remote_endpoint()};
84  msg << "Tcp Socket connected to host "
85  << "<" << endpoint_.address().to_string() << "," << endpoint_.port() << ">";
86  });
87  result.EmplaceValue();
88  } else {
90  FILE_NAME, __LINE__, __func__, [ec](std::stringstream &msg) {
91  msg << "Tcp Socket connect to host failed with error: " << ec.message();
92  });
93  }
94  return result;
95 }
96 
99  TcpErrorCodeType ec{};
100 
101  // Graceful shutdown
102  tcp_socket_.shutdown(Socket::shutdown_both, ec);
103  if (ec.value() == boost::system::errc::success) {
104  // Socket shutdown success
105  result.EmplaceValue();
106  } else {
108  FILE_NAME, __LINE__, __func__, [ec](std::stringstream &msg) {
109  msg << "Tcp Socket disconnection from host failed with error: " << ec.message();
110  });
111  }
112  return result;
113 }
114 
116  TcpMessageConstPtr tcp_message) noexcept {
117  core_type::Result<void, SocketError> result{SocketError::kGenericError};
118  TcpErrorCodeType ec{};
119 
120  boost::asio::write(
121  tcp_socket_,
122  boost::asio::buffer(tcp_message->GetPayload().data(), tcp_message->GetPayload().size()), ec);
123  // Check for error
124  if (ec.value() == boost::system::errc::success) {
126  FILE_NAME, __LINE__, __func__, [this](std::stringstream &msg) {
127  Tcp::endpoint const endpoint_{tcp_socket_.remote_endpoint()};
128  msg << "Tcp message sent to "
129  << "<" << endpoint_.address().to_string() << "," << endpoint_.port() << ">";
130  });
131  result.EmplaceValue();
132  } else {
134  FILE_NAME, __LINE__, __func__, [ec](std::stringstream &msg) {
135  msg << "Tcp message sending failed with error: " << ec.message();
136  });
137  }
138  return result;
139 }
140 
143  // destroy the socket
144  tcp_socket_.shutdown(boost::asio::socket_base::shutdown_receive);
145  tcp_socket_.close();
146  result.EmplaceValue();
147  return result;
148 }
149 
152  TcpErrorCodeType ec{};
153  // create and reserve the buffer
154  TcpMessage::BufferType rx_buffer{};
155  rx_buffer.resize(message::tcp::kDoipheadrSize);
156  // start blocking read to read Header first
157  boost::asio::read(tcp_socket_, boost::asio::buffer(&rx_buffer[0u], message::tcp::kDoipheadrSize),
158  ec);
159  // Check for error
160  if (ec.value() == boost::system::errc::success) {
161  // read the next bytes to read
162  std::uint32_t const read_next_bytes{[&rx_buffer]() noexcept -> std::uint32_t {
163  return static_cast<std::uint32_t>(
164  (static_cast<std::uint32_t>(rx_buffer[4u] << 24u) & 0xFF000000) |
165  (static_cast<std::uint32_t>(rx_buffer[5u] << 16u) & 0x00FF0000) |
166  (static_cast<std::uint32_t>(rx_buffer[6u] << 8u) & 0x0000FF00) |
167  (static_cast<std::uint32_t>(rx_buffer[7u] & 0x000000FF)));
168  }()};
169 
170  if (read_next_bytes != 0u) {
171  // reserve the buffer
172  rx_buffer.resize(message::tcp::kDoipheadrSize + std::size_t(read_next_bytes));
173  boost::asio::read(
174  tcp_socket_,
175  boost::asio::buffer(&rx_buffer[message::tcp::kDoipheadrSize], read_next_bytes), ec);
176 
177  Tcp::endpoint const remote_endpoint{tcp_socket_.remote_endpoint()};
178  TcpMessagePtr tcp_rx_message{std::make_unique<TcpMessage>(
179  remote_endpoint.address().to_string(), remote_endpoint.port(), std::move(rx_buffer))};
181  FILE_NAME, __LINE__, __func__, [&remote_endpoint](std::stringstream &msg) {
182  msg << "Tcp Message received from "
183  << "<" << remote_endpoint.address().to_string() << "," << remote_endpoint.port()
184  << ">";
185  });
186  result.EmplaceValue(std::move(tcp_rx_message));
187  } else {
189  FILE_NAME, __LINE__, __func__,
190  [](std::stringstream &msg) { msg << "Tcp Message read ignored as header size is zero"; });
191  }
192  } else if (ec.value() == boost::asio::error::eof) {
194  FILE_NAME, __LINE__, __func__,
195  [ec](std::stringstream &msg) { msg << "Remote Disconnected with: " << ec.message(); });
196  } else {
198  FILE_NAME, __LINE__, __func__, [ec](std::stringstream &msg) {
199  msg << "Remote Disconnected with undefined error: " << ec.message();
200  });
201  }
202  return result;
203 }
204 } // namespace tcp
205 } // namespace socket
206 } // namespace boost_support
static auto GetLibBoostLogger() noexcept -> LibBoostLogger &
Definition: logger.h:20
std::vector< std::uint8_t > BufferType
Type alias for underlying buffer.
Definition: tcp_message.h:49
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
TcpSocket(std::string_view local_ip_address, std::uint16_t local_port_num, IoContext &io_context) noexcept
Constructs an instance of TcpSocket.
Definition: tcp_socket.cpp:19
core_type::Result< void, SocketError > Connect(std::string_view host_ip_address, std::uint16_t host_port_num) noexcept
Function to connect to remote ip address and port number.
Definition: tcp_socket.cpp:72
boost_support::message::tcp::TcpMessagePtr TcpMessagePtr
Type alias for Tcp message pointer.
Definition: tcp_socket.h:44
core_type::Result< void, SocketError > Close() noexcept
Function to destroy the socket.
Definition: tcp_socket.cpp:141
core_type::Result< TcpMessagePtr, SocketError > Read() noexcept
Function to read message from socket.
Definition: tcp_socket.cpp:150
core_type::Result< void, SocketError > Disconnect() noexcept
Function to Disconnect from host.
Definition: tcp_socket.cpp:97
boost_support::message::tcp::TcpMessageConstPtr TcpMessageConstPtr
Type alias for Tcp message const pointer.
Definition: tcp_socket.h:49
Tcp::socket Socket
Type alias for tcp socket.
Definition: tcp_socket.h:59
core_type::Result< void, SocketError > Transmit(TcpMessageConstPtr tcp_message) noexcept
Function to trigger transmission.
Definition: tcp_socket.cpp:115
~TcpSocket() noexcept
Destruct an instance of TcpSocket.
Socket tcp_socket_
Store the underlying tcp socket.
Definition: tcp_socket.h:155
boost::system::error_code TcpErrorCodeType
Type alias for tcp error codes.
Definition: tcp_socket.h:150
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
constexpr std::uint8_t kDoipheadrSize
Doip HeaderSize.
Definition: tcp_message.h:156
core_type::Result< T, E > Result
Class type to contains a value (of type ValueType), or an error (of type ErrorType)