Diag-Client-Lib
vd_conversation.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 <sstream>
12 #include <string>
13 #include <utility>
14 
17 
18 namespace diag {
19 namespace client {
20 namespace conversation {
21 
22 std::string ConvertToHexString(std::uint8_t char_start, std::uint8_t char_count,
23  std::vector<std::uint8_t> &input_buffer) {
24  std::string hex_string{};
25  std::uint8_t total_char_count{static_cast<uint8_t>(char_start + char_count)};
26 
27  for (std::uint8_t char_start_count = char_start; char_start_count < total_char_count;
28  char_start_count++) {
29  std::stringstream vehicle_info_data_eid{};
30  int payload_byte{input_buffer[char_start_count]};
31  if ((payload_byte <= 15)) {
32  // "0" appended in case of value upto 15/0xF
33  vehicle_info_data_eid << "0";
34  }
35  vehicle_info_data_eid << std::hex << payload_byte << ":";
36  hex_string.append(vehicle_info_data_eid.str());
37  }
38  hex_string.pop_back(); // remove last ":" appended before
39  return hex_string;
40 }
41 
42 std::string ConvertToAsciiString(std::uint8_t char_start, std::uint8_t char_count,
43  std::vector<std::uint8_t> &input_buffer) {
44  std::string ascii_string{};
45  std::uint8_t total_char_count{static_cast<uint8_t>(char_start + char_count)};
46 
47  for (std::uint8_t char_start_count = char_start; char_start_count < total_char_count;
48  char_start_count++) {
49  std::stringstream vehicle_info_data_vin{};
50  vehicle_info_data_vin << input_buffer[char_start_count];
51  ascii_string.append(vehicle_info_data_vin.str());
52  }
53  return ascii_string;
54 }
55 
56 void SerializeEIDGIDFromString(std::string &input_string, std::vector<uint8_t> &output_buffer,
57  std::uint8_t total_size, std::uint8_t substring_range) {
58 
59  for (auto char_count = 0U; char_count < total_size; char_count += substring_range) {
60  std::string input_string_new{
61  input_string.substr(char_count, static_cast<std::uint8_t>(substring_range))};
62  std::stringstream input_string_stream{input_string_new};
63  int get_byte;
64  input_string_stream >> std::hex >> get_byte;
65  output_buffer.emplace_back(static_cast<std::uint8_t>(get_byte));
66  }
67 }
68 
69 void SerializeVINFromString(std::string &input_string, std::vector<uint8_t> &output_buffer,
70  std::uint8_t total_size, std::uint8_t substring_range) {
71 
72  for (auto char_count = 0U; char_count < total_size; char_count += substring_range) {
73  std::string input_string_new{
74  input_string.substr(char_count, static_cast<std::uint8_t>(substring_range))};
75  std::stringstream input_string_stream{input_string_new};
76  int get_byte{input_string_stream.get()};
77  output_buffer.emplace_back(static_cast<std::uint8_t>(get_byte));
78  }
79 }
80 
81 // Vehicle Info Message implementation class
83  public:
85  std::map<std::uint16_t, vehicle_info::VehicleAddrInfoResponse> &vehicle_info_collection)
87  for (std::pair<std::uint16_t, vehicle_info::VehicleAddrInfoResponse> vehicle_info:
88  vehicle_info_collection) {
89  Push(vehicle_info.second);
90  }
91  }
92 
93  ~VehicleInfoMessageImpl() override = default;
94 
96 
97  private:
98  // Function to push the vehicle address info received
99  void Push(vehicle_info::VehicleAddrInfoResponse &vehicle_addr_info_response) {
100  vehicle_info_messages_.emplace_back(vehicle_addr_info_response);
101  }
102 
103  // store the vehicle info message list
105 };
106 
111  public:
120  VdConversation &vd_conversion)
121  : ::uds_transport::ConversionHandler{handler_id},
122  vd_conversation_{vd_conversion} {}
123 
127  VdConversationHandler(const VdConversationHandler &other) noexcept = delete;
128  VdConversationHandler &operator=(const VdConversationHandler &other) noexcept = delete;
129 
133  VdConversationHandler(VdConversationHandler &&other) noexcept = delete;
135 
139  ~VdConversationHandler() override = default;
140 
171  ::uds_transport::ChannelID channel_id, std::size_t size,
172  ::uds_transport::Priority priority, ::uds_transport::ProtocolKind protocol_kind,
173  core_type::Span<std::uint8_t const> payloadInfo) const noexcept override {
174  return (vd_conversation_.IndicateMessage(source_addr, target_addr, type, channel_id, size,
175  priority, protocol_kind, payloadInfo));
176  }
177 
184  void HandleMessage(::uds_transport::UdsMessagePtr message) const noexcept override {
185  vd_conversation_.HandleMessage(std::move(message));
186  }
187 
188  private:
193 };
194 
195 // Conversation class
196 VdConversation::VdConversation(std::string_view conversion_name,
197  VDConversationType &conversion_identifier)
198  : vd_conversion_handler_{std::make_unique<VdConversationHandler>(
199  conversion_identifier.handler_id, *this)},
200  conversation_name_{conversion_name},
201  broadcast_address_{conversion_identifier.udp_broadcast_address},
202  connection_ptr_{},
203  vehicle_info_collection_{},
204  vehicle_info_container_mutex_{} {}
205 
207 
208 void VdConversation::Startup() noexcept {
209  // initialize the connection
210  static_cast<void>(connection_ptr_->Initialize());
211  // start the connection
212  connection_ptr_->Start();
213  // Change the state to Active
215  logger::DiagClientLogger::GetDiagClientLogger().GetLogger().LogInfo(
216  FILE_NAME, __LINE__, __func__, [&](std::stringstream &msg) {
217  msg << "'" << conversation_name_ << "'"
218  << "-> "
219  << "Startup completed";
220  });
221 }
222 
223 void VdConversation::Shutdown() noexcept {
225  // shutdown connection
226  connection_ptr_->Stop();
227  // Change the state to InActive
229  logger::DiagClientLogger::GetDiagClientLogger().GetLogger().LogInfo(
230  FILE_NAME, __LINE__, __func__, [&](std::stringstream &msg) {
231  msg << "'" << conversation_name_ << "'"
232  << "-> "
233  << "Shutdown completed";
234  });
235  }
236 }
237 
239  std::unique_ptr<uds_transport::Connection> connection) noexcept {
240  connection_ptr_ = std::move(connection);
241 }
242 
246  vehicle_info::VehicleInfoListRequestType vehicle_info_request) noexcept {
247  using VehicleIdentificationResponseResult =
250 
251  VehicleIdentificationResponseResult result{VehicleIdentificationResponseResult::FromError(
253 
254  // Deserialize first , Todo: Add optional when deserialize fails
255  std::pair<PreselectionMode, PreselectionValue> vehicle_info_request_deserialized_value{
256  DeserializeVehicleInfoRequest(vehicle_info_request)};
257 
258  if (VerifyVehicleInfoRequest(
259  vehicle_info_request_deserialized_value.first,
260  static_cast<uint8_t>(vehicle_info_request_deserialized_value.second.size()))) {
261  if (connection_ptr_->Transmit(std::make_unique<diag::client::vd_message::VdMessage>(
262  vehicle_info_request_deserialized_value.first,
263  vehicle_info_request_deserialized_value.second, broadcast_address_)) !=
265  // Check if any response received
266  if (vehicle_info_collection_.empty()) {
267  // no response received
269  logger::DiagClientLogger::GetDiagClientLogger().GetLogger().LogWarn(
270  FILE_NAME, __LINE__, __func__, [&](std::stringstream &msg) {
271  msg << "'" << conversation_name_ << "'"
272  << "-> "
273  << "No vehicle identification response received, timed out "
274  "without response";
275  });
276  } else {
277  result.EmplaceValue(std::make_unique<VehicleInfoMessageImpl>(vehicle_info_collection_));
278  // all the responses are copied, now clear the map
279  vehicle_info_collection_.clear();
280  }
281  }
282  } else {
284  }
285  return result;
286 }
287 
289  return nullptr;
290 }
291 
295  uds_transport::UdsMessage::Address /* target_addr */,
299  core_type::Span<std::uint8_t const> payload_info) noexcept {
300  using IndicationResult = std::pair<::uds_transport::UdsTransportProtocolMgr::IndicationResult,
302  IndicationResult ret_val{
304  if (!payload_info.empty()) {
306  ret_val.second = std::make_unique<diag::client::vd_message::VdMessage>();
307  ret_val.second->GetPayload().resize(size);
308  }
309  return ret_val;
310 }
311 
313  if (message != nullptr) {
314  std::lock_guard<std::mutex> const lock{vehicle_info_container_mutex_};
315  std::pair<std::uint16_t, VehicleAddrInfoResponseStruct> vehicle_info_request{
316  DeserializeVehicleInfoResponse(std::move(message))};
317 
318  vehicle_info_collection_.emplace(vehicle_info_request.first, vehicle_info_request.second);
319  }
320 }
321 
323  std::uint8_t preselection_value_length) {
324  bool is_veh_info_valid{false};
325  // 0U : No preselection
326  if (preselection_mode == 0U && (preselection_value_length == 0U)) {
327  is_veh_info_valid = true;
328  }
329  // 1U : DoIP Entities with given VIN
330  else if (preselection_mode == 1U && (preselection_value_length == 17U)) {
331  is_veh_info_valid = true;
332  }
333  // 2U : DoIP Entities with given EID
334  else if (preselection_mode == 2U && (preselection_value_length == 6U)) {
335  is_veh_info_valid = true;
336  } else {
337  // do nothing
338  }
339 
340  return is_veh_info_valid;
341 }
342 
343 std::pair<VdConversation::LogicalAddress, VdConversation::VehicleAddrInfoResponseStruct>
345  constexpr std::uint8_t start_index_vin{0U};
346  constexpr std::uint8_t total_vin_length{17U};
347  constexpr std::uint8_t start_index_eid{19U};
348  constexpr std::uint8_t start_index_gid{25U};
349  constexpr std::uint8_t total_eid_gid_length{6U};
350 
351  std::string const vehicle_info_data_vin{
352  ConvertToAsciiString(start_index_vin, total_vin_length, message->GetPayload())};
353  std::string const vehicle_info_data_eid{
354  ConvertToHexString(start_index_eid, total_eid_gid_length, message->GetPayload())};
355  std::string const vehicle_info_data_gid{
356  ConvertToHexString(start_index_gid, total_eid_gid_length, message->GetPayload())};
357 
358  LogicalAddress const logical_address{(static_cast<std::uint16_t>(
359  ((message->GetPayload()[17U] & 0xFF) << 8) | (message->GetPayload()[18U] & 0xFF)))};
360 
361  // Create the structure out of the extracted string
362  VehicleAddrInfoResponseStruct const vehicle_addr_info{
363  std::string{message->GetHostIpAddress()}, // remote ip address
364  logical_address, // logical address
365  vehicle_info_data_vin, // vin
366  vehicle_info_data_eid, // eid
367  vehicle_info_data_gid}; // gid
368 
369  return std::pair<std::uint16_t, VdConversation::VehicleAddrInfoResponseStruct>{logical_address,
370  vehicle_addr_info};
371 }
372 
374  return *vd_conversion_handler_;
375 }
376 
377 std::pair<VdConversation::PreselectionMode, VdConversation::PreselectionValue>
379  vehicle_info::VehicleInfoListRequestType &vehicle_info_request) {
380 
381  std::pair<VdConversation::PreselectionMode, VdConversation::PreselectionValue> ret_val{};
382  ret_val.first = vehicle_info_request.preselection_mode;
383 
384  if (ret_val.first == 1U) {
385  // 1U : DoIP Entities with given VIN
386  SerializeVINFromString(vehicle_info_request.preselection_value, ret_val.second,
387  static_cast<uint8_t>(vehicle_info_request.preselection_value.length()),
388  1U);
389  } else if (ret_val.first == 2U) {
390  // 2U : DoIP Entities with given EID
391  vehicle_info_request.preselection_value.erase(
392  remove(vehicle_info_request.preselection_value.begin(),
393  vehicle_info_request.preselection_value.end(), ':'),
394  vehicle_info_request.preselection_value.end());
396  vehicle_info_request.preselection_value, ret_val.second,
397  static_cast<uint8_t>(vehicle_info_request.preselection_value.length()), 2U);
398  } else {
399  // log failure
400  }
401  return ret_val;
402 }
403 
404 } // namespace conversation
405 } // namespace client
406 } // namespace diag
Class type to contains a value (of type ValueType), or an error (of type ErrorType)
Definition: result.h:29
A view over a contiguous sequence of objects.
Definition: span.h:191
VehicleInfoResponseError
Definitions of Vehicle Identification response error code.
auto GetActivityStatus() const noexcept -> ActivityStatusType
Get the current activity status of this conversation.
Definition: conversation.h:196
ActivityStatusType activity_status_
Store the conversation activity status.
Definition: conversation.h:202
Class to manage reception from transport protocol handler to vd connection handler.
~VdConversationHandler() override=default
Destructs an instance of DmConversationHandler.
VdConversationHandler(VdConversationHandler &&other) noexcept=delete
Deleted move assignment and move constructor.
void HandleMessage(::uds_transport::UdsMessagePtr message) const noexcept override
Function to Hands over a valid received Uds message.
VdConversationHandler & operator=(const VdConversationHandler &other) noexcept=delete
VdConversation & vd_conversation_
Store the reference of vd conversation.
VdConversationHandler(::uds_transport::conversion_manager::ConversionHandlerID handler_id, VdConversation &vd_conversion)
Constructs an instance of VdConversationHandler.
std::pair<::uds_transport::UdsTransportProtocolMgr::IndicationResult, ::uds_transport::UdsMessagePtr > IndicateMessage(::uds_transport::UdsMessage::Address source_addr, ::uds_transport::UdsMessage::Address target_addr, ::uds_transport::UdsMessage::TargetAddressType type, ::uds_transport::ChannelID channel_id, std::size_t size, ::uds_transport::Priority priority, ::uds_transport::ProtocolKind protocol_kind, core_type::Span< std::uint8_t const > payloadInfo) const noexcept override
Function to indicate a start of reception of message.
VdConversationHandler(const VdConversationHandler &other) noexcept=delete
Deleted copy assignment and copy constructor.
VdConversationHandler & operator=(VdConversationHandler &&other) noexcept=delete
Class to search for available diagnostic server over a network.
::uds_transport::ConversionHandler & GetConversationHandler() noexcept override
Function to get the conversation handler from conversation object.
std::pair<::uds_transport::UdsTransportProtocolMgr::IndicationResult, ::uds_transport::UdsMessagePtr > IndicateMessage(::uds_transport::UdsMessage::Address source_addr, ::uds_transport::UdsMessage::Address target_addr, ::uds_transport::UdsMessage::TargetAddressType type, ::uds_transport::ChannelID channel_id, std::size_t size, ::uds_transport::Priority priority, ::uds_transport::ProtocolKind protocol_kind, core_type::Span< std::uint8_t const > payloadInfo) noexcept override
Function to indicate a start of reception of message.
static std::pair< PreselectionMode, PreselectionValue > DeserializeVehicleInfoRequest(vehicle_info::VehicleInfoListRequestType &vehicle_info_request)
Function to deserialize the Vehicle Information request from user.
static std::pair< LogicalAddress, VehicleAddrInfoResponseStruct > DeserializeVehicleInfoResponse(::uds_transport::UdsMessagePtr message)
Function to deserialize the received Vehicle Identification Response/ Announcement.
~VdConversation() override
Destructs an instance of DmConversation.
VdConversation(std::string_view conversion_name, VDConversationType &conversion_identifier)
Constructs an instance of VdConversation.
void Shutdown() noexcept override
Function to shutdown the DmConversation.
std::unique_ptr<::uds_transport::Connection > connection_ptr_
Store the underlying transport protocol connection object.
void RegisterConnection(std::unique_ptr<::uds_transport::Connection > connection) noexcept override
Function to register the conversation to underlying transport protocol handler.
std::uint16_t LogicalAddress
Type alias of logical address.
std::unique_ptr<::uds_transport::ConversionHandler > vd_conversion_handler_
Store the vd conversation handler.
void Startup() noexcept override
Function to start the DmConversation.
bool VerifyVehicleInfoRequest(PreselectionMode preselection_mode, std::uint8_t preselection_value_length)
Function to verify the received vehicle info request.
vehicle_info::VehicleInfoMessageResponseUniquePtr GetDiagnosticServerList()
Function to get the list of available diagnostic server.
void HandleMessage(::uds_transport::UdsMessagePtr message) noexcept override
Function to Hands over a valid received Uds message.
core_type::Result< diag::client::vehicle_info::VehicleInfoMessageResponseUniquePtr, DiagClient::VehicleInfoResponseError > SendVehicleIdentificationRequest(vehicle_info::VehicleInfoListRequestType vehicle_info_request) noexcept override
Function to send vehicle identification request and get the Diagnostic Server list.
std::string conversation_name_
Store the conversation name.
std::uint8_t PreselectionMode
Type alias of pre-selection mode.
void Push(vehicle_info::VehicleAddrInfoResponse &vehicle_addr_info_response)
VehicleInfoListResponseType & GetVehicleList() override
Function to get the list of vehicle available in the network.
VehicleInfoMessageImpl(std::map< std::uint16_t, vehicle_info::VehicleAddrInfoResponse > &vehicle_info_collection)
static auto GetDiagClientLogger() noexcept -> DiagClientLogger &
Get the diag client logger instance.
Definition: logger.h:32
Class provide storage of list of all available vehicle entity.
std::vector< VehicleAddrInfoResponse > VehicleInfoListResponseType
Alias to collection of Vehicle info response.
Class to manage reception from transport protocol handler to connection handler.
ConversionHandler(conversion_manager::ConversionHandlerID handler_id)
Constructs an instance of ConversionHandler.
#define FILE_NAME
Definition: file_path.h:14
void SerializeVINFromString(std::string &input_string, std::vector< uint8_t > &output_buffer, std::uint8_t total_size, std::uint8_t substring_range)
void SerializeEIDGIDFromString(std::string &input_string, std::vector< uint8_t > &output_buffer, std::uint8_t total_size, std::uint8_t substring_range)
std::string ConvertToAsciiString(std::uint8_t char_start, std::uint8_t char_count, std::vector< std::uint8_t > &input_buffer)
std::string ConvertToHexString(std::uint8_t char_start, std::uint8_t char_count, std::vector< std::uint8_t > &input_buffer)
std::unique_ptr< VehicleInfoMessage > VehicleInfoMessageResponseUniquePtr
The unique_ptr for Vehicle Identification Response Message.
std::string_view ProtocolKind
std::unique_ptr< UdsMessage > UdsMessagePtr
Definition: uds_message.h:71
uint32_t ChannelID
std::uint8_t Priority
Structure containing VD conversation type.
std::uint8_t preselection_mode
Mode to be used during sending of Vehicle Identification request. 0U : No preselection 1U : DoIP Enti...
std::string preselection_value
Value to be used based on preselection mode. VIN when preselection_mode = 1U EID when preselection_mo...
Structure containing available Vehicle Address Information.