21 int idx = -1, success = 0;
22 unsigned char *utf8 = NULL;
27 idx = X509_NAME_get_index_by_NID(name, NID_commonName, -1);
28 if (!(idx > -1))
break;
30 X509_NAME_ENTRY *entry = X509_NAME_get_entry(name, idx);
33 ASN1_STRING *data = X509_NAME_ENTRY_get_data(entry);
36 int length = ASN1_STRING_to_UTF8(&utf8, data);
37 if (!utf8 || !(length > 0))
break;
39 fprintf(stdout,
" %s: %s\n", label, utf8);
44 if (utf8) OPENSSL_free(utf8);
46 if (!success) fprintf(stdout,
" %s: <not available>\n", label);
53 std::string_view ca_certification_path)
54 : local_ip_address_{local_ip_address},
55 local_port_num_{local_port_num},
57 io_ssl_context_{boost::asio::ssl::context::tlsv13_client},
58 tls_socket_{io_context_, io_ssl_context_},
64 tcp_handler_read_{std::move(tcp_handler_read)} {
66 tls_socket_.set_verify_mode(boost::asio::ssl::verify_peer);
69 [](
bool pre_verified, boost::asio::ssl::verify_context &ctx) noexcept ->
bool {
70 X509 *cert = X509_STORE_CTX_get_current_cert(ctx.native_handle());
71 int depth = X509_STORE_CTX_get_error_depth(ctx.native_handle());
73 fprintf(stdout,
"verify_callback (depth=%d)(preverify=%d)\n", depth, pre_verified);
75 X509_NAME *iname = cert ? X509_get_issuer_name(cert) :
nullptr;
76 X509_NAME *sname = cert ? X509_get_subject_name(cert) :
nullptr;
88 "TLS_AES_128_GCM_SHA256:TLS_AES_256_GCM_SHA384:TLS_"
89 "CHACHA20_POLY1305_SHA256");
92 thread_ = std::thread([
this]() {
93 std::unique_lock<std::mutex> lck(
mutex_);
111 std::unique_lock<std::mutex> lck(
mutex_);
125 if (ec.value() == boost::system::errc::success) {
134 if (ec.value() == boost::system::errc::success) {
137 FILE_NAME, __LINE__, __func__, [
this](std::stringstream &msg) {
139 msg <<
"Tcp Socket opened and bound to "
140 <<
"<" << endpoint_.address().to_string() <<
"," << endpoint_.port() <<
">";
142 result.EmplaceValue();
146 FILE_NAME, __LINE__, __func__, [ec](std::stringstream &msg) {
147 msg <<
"Tcp Socket binding failed with message: " << ec.message();
153 FILE_NAME, __LINE__, __func__, [ec](std::stringstream &msg) {
154 msg <<
"Tcp Socket opening failed with error: " << ec.message();
162 std::string_view host_ip_address, std::uint16_t host_port_num) {
168 Tcp::endpoint(TcpIpAddress::from_string(std::string{host_ip_address}), host_port_num), ec);
169 if (ec.value() == boost::system::errc::success) {
171 FILE_NAME, __LINE__, __func__, [
this](std::stringstream &msg) {
173 msg <<
"Tcp Socket connected to host "
174 <<
"<" << endpoint_.address().to_string() <<
"," << endpoint_.port() <<
">";
177 tls_socket_.handshake(boost::asio::ssl::stream_base::client, ec);
178 if (ec.value() == boost::system::errc::success) {
180 std::lock_guard<std::mutex> lock{
mutex_};
184 printf(
"Connected with %s encryption\n", SSL_get_cipher(
tls_socket_.native_handle()));
185 result.EmplaceValue();
188 FILE_NAME, __LINE__, __func__, [ec](std::stringstream &msg) {
189 msg <<
"Tls client handshake with host failed with error: " << ec.message();
195 FILE_NAME, __LINE__, __func__, [ec](std::stringstream &msg) {
196 msg <<
"Tcp Socket connect to host failed with error: " << ec.message();
211 if (ec.value() == boost::system::errc::success) {
213 std::lock_guard<std::mutex> lock{
mutex_};
218 result.EmplaceValue();
221 FILE_NAME, __LINE__, __func__, [ec](std::stringstream &msg) {
222 msg <<
"Tcp Socket disconnection from host failed with error: " << ec.message();
235 boost::asio::buffer(tcp_message->GetPayload().data(), tcp_message->GetPayload().size()), ec);
237 if (ec.value() == boost::system::errc::success) {
239 FILE_NAME, __LINE__, __func__, [
this](std::stringstream &msg) {
241 msg <<
"Tcp message sent to "
242 <<
"<" << endpoint_.address().to_string() <<
"," << endpoint_.port() <<
">";
244 result.EmplaceValue();
247 FILE_NAME, __LINE__, __func__, [ec](std::stringstream &msg) {
248 msg <<
"Tcp message sending failed with error: " << ec.message();
258 result.EmplaceValue();
271 if (ec.value() == boost::system::errc::success) {
273 std::uint32_t
const read_next_bytes = [&rx_buffer]() noexcept -> std::uint32_t {
274 return static_cast<std::uint32_t
>(
275 (
static_cast<std::uint32_t
>(rx_buffer[4u] << 24u) & 0xFF000000) |
276 (
static_cast<std::uint32_t
>(rx_buffer[5u] << 16u) & 0x00FF0000) |
277 (
static_cast<std::uint32_t
>(rx_buffer[6u] << 8u) & 0x0000FF00) |
278 (
static_cast<std::uint32_t
>(rx_buffer[7u] & 0x000000FF)));
281 if (read_next_bytes != 0u) {
291 endpoint_.address().to_string(), endpoint_.port(), std::move(rx_buffer))};
293 FILE_NAME, __LINE__, __func__, [endpoint_](std::stringstream &msg) {
294 msg <<
"Tcp Message received from "
295 <<
"<" << endpoint_.address().to_string() <<
"," << endpoint_.port() <<
">";
302 [](std::stringstream &msg) { msg <<
"Tcp Message read ignored as header size is zero"; });
304 }
else if (ec.value() == boost::asio::error::eof) {
308 [ec](std::stringstream &msg) { msg <<
"Remote Disconnected with: " << ec.message(); });
312 FILE_NAME, __LINE__, __func__, [ec](std::stringstream &msg) {
313 msg <<
"Remote Disconnected with undefined error: " << ec.message();
static auto GetLibBoostLogger() noexcept -> LibBoostLogger &
std::vector< std::uint8_t > BufferType
Type alias for underlying buffer.
TlsStream::lowest_layer_type & GetNativeTcpSocket()
Function to get the native tcp socket under tls socket.
core_type::Result< void, TlsErrorCode > DisconnectFromHost()
Function to Disconnect from host.
std::uint16_t local_port_num_
Store local port number.
std::atomic_bool exit_request_
Flag to terminate the thread.
TlsClientSocket(std::string_view local_ip_address, std::uint16_t local_port_num, TcpHandlerRead tcp_handler_read, std::string_view ca_certification_path)
Constructs an instance of TlsClientSocket.
std::thread thread_
The thread itself.
core_type::Result< void, TlsErrorCode > ConnectToHost(std::string_view host_ip_address, std::uint16_t host_port_num)
Function to connect to remote ip address and port number.
core_type::Result< void, TlsErrorCode > Open()
Function to Open the socket.
std::function< void(message::tcp::TcpMessagePtr)> TcpHandlerRead
Tcp function template used for reception.
core_type::Result< void, TlsErrorCode > Transmit(message::tcp::TcpMessageConstPtr tcp_message)
Function to trigger transmission.
core_type::Result< void, TlsErrorCode > Destroy()
Function to destroy the socket.
boost::asio::ssl::context io_ssl_context_
boost io ssl context
boost::system::error_code TcpErrorCodeType
Type alias for tcp error codes.
~TlsClientSocket()
Destruct an instance of TcpClientSocket.
std::mutex mutex_
mutex to lock critical section
void HandleMessage()
Function to handle the reception of tcp message.
std::string local_ip_address_
Store local ip address.
std::atomic_bool running_
Flag to start the thread.
TlsStream tls_socket_
Store ssl socket.
std::condition_variable cond_var_
Conditional variable to block the thread.
TcpHandlerRead tcp_handler_read_
Store the handler.
Class type to contains a value (of type ValueType), or an error (of type ErrorType)
std::unique_ptr< TcpMessage > TcpMessagePtr
The unique pointer to TcpMessage.
std::unique_ptr< TcpMessage const > TcpMessageConstPtr
The unique pointer to const TcpMessage.
constexpr std::uint8_t kDoipheadrSize
Doip HeaderSize.
void print_cn_name(const char *label, X509_NAME *const name)