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;

  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)
  : m_topic(client_name), m_node(node), m_pAgent(agent), isRunning(true)
  {
    m_client = m_node.get()->create_client<T>(client_name);
    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_