Diag-Client-Lib
udp_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 <boost/asio/ip/address.hpp>
12 
15 
16 namespace boost_support {
17 namespace socket {
18 namespace udp {
19 
20 UdpSocket::UdpSocket(std::string_view local_ip_address, std::uint16_t local_port_num,
21  boost::asio::io_context &io_context) noexcept
22  : udp_socket_{io_context},
23  local_endpoint_{boost::asio::ip::make_address(local_ip_address), local_port_num} {
24  rx_buffer_.resize(message::udp::kMaxUdpResSize);
25 }
26 
27 UdpSocket::~UdpSocket() noexcept = default;
28 
29 void UdpSocket::SetReadHandler(UdpSocket::UdpHandlerRead read_handler) {
30  udp_handler_read_ = std::move(read_handler);
31 }
32 
35  UdpErrorCodeType ec{};
36  udp_socket_.open(local_endpoint_.protocol(), ec);
37  if (ec.value() == boost::system::errc::success) {
38  // set broadcast option
39  udp_socket_.set_option(boost::asio::socket_base::broadcast{true});
40  // reuse address
41  udp_socket_.set_option(boost::asio::socket_base::reuse_address{true});
42  udp_socket_.bind(local_endpoint_, ec);
43  }
44  if (ec.value() == boost::system::errc::success) {
45  local_endpoint_ = udp_socket_.local_endpoint();
47  FILE_NAME, __LINE__, __func__, [this](std::stringstream &msg) {
48  msg << "Udp Socket opened and bound to "
49  << "<" << local_endpoint_.address() << "," << local_endpoint_.port() << ">";
50  });
51  // start async receive
53  result.EmplaceValue();
54  } else {
55  // Socket binding failed
56  result.EmplaceError(SocketError::kBindingFailed);
58  FILE_NAME, __LINE__, __func__, [ec](std::stringstream &msg) {
59  msg << "Udp Socket Bind failed with message: " << ec.message();
60  });
61  }
62  return result;
63 }
64 
66  UdpSocket::UdpMessageConstPtr udp_message) noexcept {
67  core_type::Result<void, SocketError> result{SocketError::kGenericError};
68  UdpErrorCodeType ec{};
69 
70  // Transmit to remote endpoints
71  std::size_t const send_size{udp_socket_.send_to(
72  boost::asio::buffer(udp_message->GetPayload().data(), udp_message->GetPayload().size()),
73  Udp::endpoint{boost::asio::ip::make_address(udp_message->GetHostIpAddress()),
74  udp_message->GetHostPortNumber()},
75  {}, ec)};
76  // Check for error
77  if (ec.value() == boost::system::errc::success && send_size == udp_message->GetPayload().size()) {
78  // successful
80  FILE_NAME, __LINE__, __func__, [this, &udp_message](std::stringstream &msg) {
81  msg << "Udp message sent : "
82  << "<" << local_endpoint_.address() << "," << local_endpoint_.port() << ">"
83  << " -> "
84  << "<" << udp_message->GetHostIpAddress() << "," << udp_message->GetHostPortNumber()
85  << ">";
86  });
87  result.EmplaceValue();
88  // start async receive
89  StartReceivingMessage();
90  } else {
92  FILE_NAME, __LINE__, __func__, [&ec, &udp_message](std::stringstream &msg) {
93  msg << "Udp message sending to "
94  << "<" << udp_message->GetHostIpAddress() << "> "
95  << "failed with error: " << ec.message();
96  });
97  }
98  return result;
99 }
100 
103  // destroy the socket
104  udp_socket_.close();
105  result.EmplaceValue();
106  return result;
107 }
108 
112  // Ignore self reception
113  if (local_endpoint_.address().to_string() != remote_endpoint_.address().to_string()) {
114  UdpMessage::BufferType received_data{};
115  received_data.reserve(total_bytes_received);
116 
117  // Received message must be less than max udp message
118  assert(total_bytes_received <= message::udp::kMaxUdpResSize);
119  // copy the received bytes into local buffer
120  received_data.insert(received_data.begin(), rx_buffer_.begin(),
121  rx_buffer_.begin() + static_cast<std::uint8_t>(total_bytes_received));
122 
123  UdpMessagePtr udp_rx_message{std::make_unique<UdpMessage>(
124  remote_endpoint_.address().to_string(), remote_endpoint_.port(), std::move(received_data))};
125 
127  FILE_NAME, __LINE__, __func__, [this](std::stringstream &msg) {
128  msg << "Udp Message received from: "
129  << "<" << remote_endpoint_.address() << "," << remote_endpoint_.port() << ">";
130  });
131  result.EmplaceValue(std::move(udp_rx_message));
132  } else {
133  common::logger::LibBoostLogger::GetLibBoostLogger().GetLogger().LogVerbose(
134  FILE_NAME, __LINE__, __func__, [this](std::stringstream &msg) {
135  msg << "Udp Message received from "
136  << "<" << remote_endpoint_.address() << "," << remote_endpoint_.port() << ">"
137  << " ignored as received by self ip"
138  << " <" << local_endpoint_.address() << "," << local_endpoint_.port() << ">";
139  });
140  }
141  // start async receive
143  return result;
144 }
145 
147  // start async receive
148  udp_socket_.async_receive_from(
149  boost::asio::buffer(rx_buffer_), remote_endpoint_,
150  [this](const UdpErrorCodeType &error, std::size_t bytes_received) {
151  if (error.value() == boost::system::errc::success) {
152  static_cast<void>(Read(bytes_received).AndThen([this](UdpMessagePtr udp_message) {
153  // send data to upper layer
154  if (udp_handler_read_) { udp_handler_read_(std::move(udp_message)); }
155  return core_type::Result<void>::FromValue();
156  }));
157  } else {
158  if (error.value() != boost::asio::error::operation_aborted) {
160  FILE_NAME, __LINE__, __func__, [error](std::stringstream &msg) {
161  msg << "Remote Disconnected with undefined error: " << error.message();
162  });
163  }
164  }
165  });
166 }
167 
168 } // namespace udp
169 } // namespace socket
170 } // namespace boost_support
static auto GetLibBoostLogger() noexcept -> LibBoostLogger &
Definition: logger.h:20
std::vector< uint8_t > BufferType
Type alias for underlying buffer.
Definition: udp_message.h:37
Class used to create a udp socket for handling transmission and reception of udp message from driver.
Definition: udp_socket.h:24
boost_support::message::udp::UdpMessageConstPtr UdpMessageConstPtr
Type alias for Udp message const pointer.
Definition: udp_socket.h:44
void StartReceivingMessage()
Function to start reception of Udp dataframe.
Definition: udp_socket.cpp:146
core_type::Result< void, SocketError > Close() noexcept
Function to destroy the socket.
Definition: udp_socket.cpp:101
std::vector< std::uint8_t > rx_buffer_
Reception buffer needed for async reception of udp data.
Definition: udp_socket.h:143
~UdpSocket() noexcept
Destruct an instance of TcpSocket.
Udp::endpoint local_endpoint_
Store the local endpoints.
Definition: udp_socket.h:133
std::function< void(UdpMessagePtr)> UdpHandlerRead
Udp function template used for reception.
Definition: udp_socket.h:59
core_type::Result< UdpMessagePtr > Read(std::size_t bytes_received)
Function to handle the reception of tcp message.
Definition: udp_socket.cpp:109
boost::system::error_code UdpErrorCodeType
Type alias for udp error codes.
Definition: udp_socket.h:123
core_type::Result< void, SocketError > Transmit(UdpMessageConstPtr udp_message) noexcept
Function to trigger transmission.
Definition: udp_socket.cpp:65
Socket udp_socket_
Store the underlying udp socket.
Definition: udp_socket.h:128
UdpSocket(std::string_view local_ip_address, std::uint16_t local_port_num, boost::asio::io_context &io_context) noexcept
Constructs an instance of TcpSocket.
Definition: udp_socket.cpp:20
core_type::Result< void, SocketError > Open() noexcept
Function to open and bind the socket to provided ip address & port.
Definition: udp_socket.cpp:33
boost_support::message::udp::UdpMessagePtr UdpMessagePtr
Type alias for Udp message pointer.
Definition: udp_socket.h:39
Udp::endpoint remote_endpoint_
Store the remote endpoints.
Definition: udp_socket.h:138
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
auto MakeErrorCode(BoostSupportErrorDomain::Errc code, BoostSupportErrorDomain::SupportDataType data) noexcept -> core_type::ErrorCode
Create a new ErrorCode within DoipErrorDomain.
constexpr std::uint8_t kMaxUdpResSize
Maximum response size.
Definition: udp_message.h:27