Program Listing for File Client.hpp

Return to documentation for file (include/soar_ros/Client.hpp)

// Copyright 2024 Moritz Schmidt
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//     http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

#ifndef SOAR_ROS__CLIENT_HPP_
#define SOAR_ROS__CLIENT_HPP_

namespace soar_ros
{

#include <rclcpp/rclcpp.hpp>
#include <string>

#include "Interface.hpp"
#include "SafeQueue.hpp"
#include "sml_Client.h"

template <typename T, typename pRequestType = typename T::Request::SharedPtr,
          typename pResponseType = typename T::Response::SharedPtr>
class Client : public virtual Output<pRequestType>, public virtual Input<pResponseType>, public Interface
{
protected:
  typename rclcpp::Client<T>::SharedPtr m_client;
  std::string m_topic;
  rclcpp::Node::SharedPtr m_node;
  sml::Agent* m_pAgent;
  std::thread m_send_client_requests;
  std::atomic<bool> isRunning;
  rclcpp::CallbackGroup::SharedPtr m_callback_group;
  rclcpp::QoS m_qos;

  void run()
  {
    while (isRunning.load())
    {
      auto timeout_duration = std::chrono::seconds(10);
      auto start_time = std::chrono::steady_clock::now();
      while (!m_client->wait_for_service(std::chrono::seconds(1)))
      {
        if (!rclcpp::ok())
        {
          RCLCPP_ERROR(this->m_node->get_logger(), "Interrupted while waiting for the service. Exiting.");
          isRunning.store(false);
          break;
        }

        auto current_time = std::chrono::steady_clock::now();
        if (current_time - start_time >= timeout_duration)
        {
          RCLCPP_ERROR(this->m_node->get_logger(), "Timeout while waiting for service %s", m_topic.c_str());
          // rclcpp::shutdown();
          // break;
        }

        RCLCPP_WARN(this->m_node->get_logger(), "%s service not available, waiting again...", m_topic.c_str());
      }

      auto msg = this->m_s2rQueue.pop();

      auto result = m_client->async_send_request(msg);

      auto response = result.get();
      this->m_r2sQueue.push(response);
    }
  }

public:
  Client(sml::Agent* agent, rclcpp::Node::SharedPtr node, const std::string& client_name,
         rclcpp::QoS qos = rclcpp::QoS(rclcpp::QoSInitialization::from_rmw(rmw_qos_profile_services_default)),
         rclcpp::CallbackGroup::SharedPtr callback_group = nullptr)
    : m_topic(client_name), m_node(node), m_pAgent(agent), isRunning(true), m_callback_group(callback_group), m_qos(qos)
  {
    if (!m_callback_group)
    {
      m_callback_group = m_node->get_node_base_interface()->get_default_callback_group();
    }
    m_client = m_node.get()->create_client<T>(client_name, qos);
    m_send_client_requests = std::thread(&Client::run, this);
  }

  ~Client()
  {
    isRunning.store(false);
    if (m_send_client_requests.joinable())
    {
      m_send_client_requests.join();
    }
  }

  virtual void parse(pResponseType msg) = 0;
  pRequestType parse(sml::Identifier* id) override = 0;

  std::string getTopic() override
  {
    return m_topic;
  }

  sml::Agent* getAgent() override
  {
    return m_pAgent;
  }
};
}  // namespace soar_ros

#endif  // SOAR_ROS__CLIENT_HPP_