Diag-Client-Lib
tls_server_.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 */
9 
10 #include <utility>
11 
13 
14 namespace boost_support {
15 namespace socket {
16 namespace tcp {
17 namespace {
18 using TcpErrorCodeType = boost::system::error_code;
19 } // namespace
20 
21 TlsServerSocket::TlsServerSocket(std::string_view local_ip_address, std::uint16_t local_port_num)
22  : local_ip_address_{local_ip_address},
23  local_port_num_{local_port_num},
24  io_context_{},
25  io_ssl_context_{boost::asio::ssl::context::tlsv13_server},
26  tcp_acceptor_{io_context_, Tcp::endpoint(Tcp::v4(), local_port_num_), true} {
28  FILE_NAME, __LINE__, __func__, [&local_ip_address, &local_port_num](std::stringstream &msg) {
29  msg << "Tcp Socket Acceptor created at "
30  << "<" << local_ip_address << "," << local_port_num << ">";
31  });
32  // Load certificate and private key from provided locations
33  io_ssl_context_.use_certificate_chain_file("../../../openssl/DiagClientLib.crt");
34  io_ssl_context_.use_private_key_file("../../../openssl/DiagClientLib.key",
35  boost::asio::ssl::context::pem);
36  SSL_CTX_set_ciphersuites(io_ssl_context_.native_handle(),
37  "TLS_AES_256_GCM_SHA384:TLS_CHACHA20_POLY1305_SHA256");
38 }
39 
40 std::optional<TcpServerConnection> TlsServerSocket::GetTcpServerConnection(
41  TcpHandlerRead tcp_handler_read) {
42  std::optional<TcpServerConnection> tcp_connection{std::nullopt};
43  TcpErrorCodeType ec{};
44  Tcp::endpoint endpoint{};
46 
47  // blocking accept
48  tcp_acceptor_.accept(tls_socket.lowest_layer(), endpoint, ec);
49  if (ec.value() == boost::system::errc::success) {
50  tcp_connection.emplace(std::move(tls_socket), std::move(tcp_handler_read));
52  FILE_NAME, __LINE__, __func__, [&endpoint](std::stringstream &msg) {
53  msg << "TLS Socket connection received from client "
54  << "<" << endpoint.address().to_string() << "," << endpoint.port() << ">";
55  });
56  } else {
58  FILE_NAME, __LINE__, __func__, [ec](std::stringstream &msg) {
59  msg << "TLS Socket Connect to client failed with error: " << ec.message();
60  });
61  }
62  return tcp_connection;
63 }
64 
66  : tls_socket_{std::move(tls_socket)},
67  tcp_handler_read_{std::move(tcp_handler_read)} {}
68 
69 TcpServerConnection::TlsStream::lowest_layer_type &TcpServerConnection::GetNativeTcpSocket() {
70  return tls_socket_.lowest_layer();
71 }
72 
74  message::tcp::TcpMessageConstPtr tcp_tx_message) {
75  TcpErrorCodeType ec{};
77 
78  boost::asio::write(tls_socket_,
79  boost::asio::buffer(tcp_tx_message->GetPayload().data(),
80  std::size_t(tcp_tx_message->GetPayload().size())),
81  ec);
82  // Check for error
83  if (ec.value() == boost::system::errc::success) {
84  Tcp::endpoint endpoint_{GetNativeTcpSocket().remote_endpoint()};
86  FILE_NAME, __LINE__, __func__, [endpoint_](std::stringstream &msg) {
87  msg << "Tcp message sent to "
88  << "<" << endpoint_.address().to_string() << "," << endpoint_.port() << ">";
89  });
90  result.EmplaceValue();
91  } else {
93  FILE_NAME, __LINE__, __func__, [ec](std::stringstream &msg) {
94  msg << "Tcp message sending failed with error: " << ec.message();
95  });
96  }
97  return result;
98 }
99 
101  TcpErrorCodeType ec{};
102  bool connection_closed{false};
103 
104  // Perform TLS handshake
105  tls_socket_.handshake(boost::asio::ssl::stream_base::server, ec);
106 
107  if (ec.value() == boost::system::errc::success) {
108  // Create and reserve the buffer
110  rx_buffer.resize(message::tcp::kDoipheadrSize);
111  // Start blocking read to read Header first
112  boost::asio::read(tls_socket_, boost::asio::buffer(&rx_buffer[0], message::tcp::kDoipheadrSize),
113  ec);
114  // Check for error
115  if (ec.value() == boost::system::errc::success) {
116  // Read the next bytes to read
117  std::uint32_t const read_next_bytes = [&rx_buffer]() noexcept -> std::uint32_t {
118  return static_cast<std::uint32_t>(
119  (static_cast<std::uint32_t>(rx_buffer[4u] << 24u) & 0xFF000000) |
120  (static_cast<std::uint32_t>(rx_buffer[5u] << 16u) & 0x00FF0000) |
121  (static_cast<std::uint32_t>(rx_buffer[6u] << 8u) & 0x0000FF00) |
122  (static_cast<std::uint32_t>(rx_buffer[7u] & 0x000000FF)));
123  }();
124  // reserve the buffer
125  rx_buffer.resize(message::tcp::kDoipheadrSize + std::size_t(read_next_bytes));
126  boost::asio::read(
127  tls_socket_,
128  boost::asio::buffer(&rx_buffer[message::tcp::kDoipheadrSize], read_next_bytes), ec);
129 
130  // all message received, transfer to upper layer
131  Tcp::endpoint endpoint_{GetNativeTcpSocket().remote_endpoint()};
132  message::tcp::TcpMessagePtr tcp_rx_message{std::make_unique<message::tcp::TcpMessage>(
133  endpoint_.address().to_string(), endpoint_.port(), std::move(rx_buffer))};
135  FILE_NAME, __LINE__, __func__, [endpoint_](std::stringstream &msg) {
136  msg << "Tcp Message received from "
137  << "<" << endpoint_.address().to_string() << "," << endpoint_.port() << ">";
138  });
139  // send data to upper layer
140  tcp_handler_read_(std::move(tcp_rx_message));
141  } else if (ec.value() == boost::asio::error::eof) {
143  FILE_NAME, __LINE__, __func__,
144  [ec](std::stringstream &msg) { msg << "Remote Disconnected with: " << ec.message(); });
145  connection_closed = true;
146  } else {
148  FILE_NAME, __LINE__, __func__, [ec](std::stringstream &msg) {
149  msg << "Remote Disconnected with undefined error: " << ec.message();
150  });
151  connection_closed = true;
152  }
153  } else {
155  FILE_NAME, __LINE__, __func__, [ec](std::stringstream &msg) {
156  msg << "Tls server handshake with host failed with error: " << ec.message();
157  });
158  connection_closed = true;
159  }
160  return connection_closed;
161 }
162 
165  TcpErrorCodeType ec{};
166 
167  // Graceful shutdown
168  if (GetNativeTcpSocket().is_open()) {
169  // Shutdown TLS connection
170  tls_socket_.shutdown(ec);
171  // Shutdown of TCP connection
172  GetNativeTcpSocket().shutdown(TcpSocket::shutdown_both, ec);
173  if (ec.value() == boost::system::errc::success) {
174  // Socket shutdown success
175  result.EmplaceValue();
176  } else {
178  FILE_NAME, __LINE__, __func__, [ec](std::stringstream &msg) {
179  msg << "Tcp Socket Disconnection failed with error: " << ec.message();
180  });
181  }
182  GetNativeTcpSocket().close();
183  }
184  return result;
185 }
186 
187 } // namespace tcp
188 } // namespace socket
189 } // 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
boost::asio::ssl::stream< TcpSocket > TlsStream
Type alias for tls stream wrapping tcp socket.
Definition: tls_server_.h:53
core_type::Result< void, TcpErrorCode > Shutdown()
Function to shutdown the socket.
TcpServerConnection(TlsStream tls_socket, TcpHandlerRead tcp_handler_read)
Constructs an instance of TcpServerConnection.
Definition: tls_server_.cpp:65
std::function< void(message::tcp::TcpMessagePtr)> TcpHandlerRead
Tcp function template used for reception.
Definition: tls_server_.h:38
TlsStream::lowest_layer_type & GetNativeTcpSocket()
Function to get the native tcp socket under tls socket.
Definition: tls_server_.cpp:69
TcpHandlerRead tcp_handler_read_
Store the handler.
Definition: tls_server_.h:110
core_type::Result< void, TcpErrorCode > Transmit(message::tcp::TcpMessageConstPtr tcp_message)
Function to trigger transmission.
Definition: tls_server_.cpp:73
bool TryReceivingMessage()
Function to initiate reception of tcp message.
boost::asio::ssl::context io_ssl_context_
boost io ssl context
Definition: tls_server_.h:179
TcpAcceptor tcp_acceptor_
Store tcp acceptor.
Definition: tls_server_.h:184
boost::asio::ip::tcp Tcp
Type alias for tcp protocol.
Definition: tls_server_.h:159
std::optional< TcpServerConnection > GetTcpServerConnection(TcpHandlerRead tcp_handler_read)
Get the tcp connection to communicate.
Definition: tls_server_.cpp:40
boost::asio::io_context io_context_
boost io context
Definition: tls_server_.h:174
std::function< void(message::tcp::TcpMessagePtr)> TcpHandlerRead
Tcp function template used for reception.
Definition: tls_server_.h:126
TlsServerSocket(std::string_view local_ip_address, std::uint16_t local_port_num)
Constructs an instance of TlsServerSocket.
Definition: tls_server_.cpp:21
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::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
constexpr std::uint8_t kDoipheadrSize
Doip HeaderSize.
Definition: tcp_message.h:156