ADD: Changes to run in ROS2 Humble on Ubuntu 22.04 (#40)

* FIX: fix building for ROS2 Humble on Ubuntu 22.04

* FIX: do not need to add .py in Humble

refer to https://discourse.articulatedrobotics.xyz/t/ros2-control-spawner-py-not-found/124/2

* FIX: fix std::future_error

when running `ros2 run rl_sar rl_sim`, get:

```bash
terminate called after throwing an instance of 'std::future_error'
  what():  std::future_error: No associated state
```

* fix: multi ros version support

* FIX: typo

* FIX: let CMake find Python version that corresponds to ROS distro

* ADD: Update CONTRIBUTORS.md

---------

Co-authored-by: fan-ziqi <fanziqi614@gmail.com>
This commit is contained in:
-T.K.- 2024-11-14 01:30:10 -08:00 committed by GitHub
parent 4124f78e1c
commit 914c3af023
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
11 changed files with 72 additions and 40 deletions

View File

@ -16,6 +16,7 @@ Guidelines for modifications:
## Contributors ## Contributors
* Jijie Huang * Jijie Huang
* Yufeng Chi (-T.K.-)
## Acknowledgements ## Acknowledgements

View File

@ -18,7 +18,7 @@ set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${GAZEBO_CXX_FLAGS}")
find_package(gazebo REQUIRED) find_package(gazebo REQUIRED)
find_package(ament_cmake REQUIRED) find_package(ament_cmake REQUIRED)
find_package(joint_state_controller REQUIRED) find_package(joint_state_broadcaster REQUIRED)
find_package(robot_state_publisher REQUIRED) find_package(robot_state_publisher REQUIRED)
find_package(rclcpp REQUIRED) find_package(rclcpp REQUIRED)
find_package(gazebo_ros REQUIRED) find_package(gazebo_ros REQUIRED)
@ -29,8 +29,19 @@ find_package(rclpy REQUIRED)
find_package(gazebo_msgs REQUIRED) find_package(gazebo_msgs REQUIRED)
find_package(std_srvs REQUIRED) find_package(std_srvs REQUIRED)
# Make sure to find Python 3.8 # Make sure to find correct Python version based on ROS distro
find_package(Python3 3.8 EXACT COMPONENTS Interpreter Development REQUIRED) if(DEFINED ENV{ROS_DISTRO})
set(ROS_DISTRO_ENV $ENV{ROS_DISTRO})
if(ROS_DISTRO_ENV STREQUAL "foxy")
find_package(Python3 3.8 EXACT COMPONENTS Interpreter Development REQUIRED)
elseif(ROS_DISTRO_ENV STREQUAL "humble")
find_package(Python3 3.10 EXACT COMPONENTS Interpreter Development REQUIRED)
else()
# Not very sure what would the default case be. A fuzzy match?
find_package(Python3 COMPONENTS Interpreter Development REQUIRED)
endif()
endif()
link_directories(/usr/local/lib) link_directories(/usr/local/lib)
include_directories(${YAML_CPP_INCLUDE_DIR}) include_directories(${YAML_CPP_INCLUDE_DIR})
@ -101,7 +112,7 @@ target_link_libraries(rl_real_go2
) )
ament_target_dependencies(rl_sim ament_target_dependencies(rl_sim
joint_state_controller joint_state_broadcaster
robot_state_publisher robot_state_publisher
rclcpp rclcpp
gazebo_ros gazebo_ros

View File

@ -52,14 +52,14 @@ def generate_launch_description():
joint_state_broadcaster_node = Node( joint_state_broadcaster_node = Node(
package="controller_manager", package="controller_manager",
executable="spawner.py", executable='spawner.py' if os.environ.get('ROS_DISTRO', '') == 'foxy' else 'spawner',
arguments=["joint_state_broadcaster"], arguments=["joint_state_broadcaster"],
output="screen", output="screen",
) )
robot_joint_controller_node = Node( robot_joint_controller_node = Node(
package="controller_manager", package="controller_manager",
executable="spawner.py", executable='spawner.py' if os.environ.get('ROS_DISTRO', '') == 'foxy' else 'spawner',
arguments=["robot_joint_controller"], arguments=["robot_joint_controller"],
output="screen", output="screen",
) )

View File

@ -19,7 +19,7 @@ from ament_index_python.packages import get_package_share_directory
from rl_sdk import * from rl_sdk import *
from observation_buffer import * from observation_buffer import *
CSV_LOGGER = True CSV_LOGGER = False
class RL_Sim(RL, Node): class RL_Sim(RL, Node):
def __init__(self): def __init__(self):

View File

@ -1,7 +1,7 @@
#include "rl_sim.hpp" #include "rl_sim.hpp"
// #define PLOT // #define PLOT
#define CSV_LOGGER // #define CSV_LOGGER
RL_Sim::RL_Sim() RL_Sim::RL_Sim()
: rclcpp::Node("rl_sim_node") : rclcpp::Node("rl_sim_node")
@ -12,30 +12,38 @@ RL_Sim::RL_Sim()
param_client = this->create_client<rcl_interfaces::srv::GetParameters>("/param_node/get_parameters"); param_client = this->create_client<rcl_interfaces::srv::GetParameters>("/param_node/get_parameters");
while (!param_client->wait_for_service(std::chrono::seconds(1))) while (!param_client->wait_for_service(std::chrono::seconds(1)))
{ {
if (!rclcpp::ok()) {
std::cout << LOGGER::ERROR << "Interrupted while waiting for param_node service. Exiting." << std::endl;
return;
}
std::cout << LOGGER::WARNING << "Waiting for param_node service to be available..." << std::endl; std::cout << LOGGER::WARNING << "Waiting for param_node service to be available..." << std::endl;
} }
auto request = std::make_shared<rcl_interfaces::srv::GetParameters::Request>(); auto request = std::make_shared<rcl_interfaces::srv::GetParameters::Request>();
request->names.push_back("robot_name"); request->names = {"robot_name", "gazebo_model_name"};
request->names.push_back("gazebo_model_name");
// Use a timeout for the future
auto future = param_client->async_send_request(request); auto future = param_client->async_send_request(request);
rclcpp::spin_until_future_complete(this->get_node_base_interface(), future); auto status = rclcpp::spin_until_future_complete(this->get_node_base_interface(), future, std::chrono::seconds(5));
if (future.get()->values.size() < 2)
if (status == rclcpp::FutureReturnCode::SUCCESS)
{ {
std::cout << LOGGER::WARNING << "Failed to get all parameters from param_node" << std::endl; auto result = future.get();
if (result->values.size() < 2)
{
std::cout << LOGGER::ERROR << "Failed to get all parameters from param_node" << std::endl;
} }
else else
{ {
this->robot_name = future.get()->values[0].string_value; this->robot_name = result->values[0].string_value;
this->gazebo_model_name = future.get()->values[1].string_value; this->gazebo_model_name = result->values[1].string_value;
std::cout << LOGGER::INFO << "Get param robot_name: " << this->robot_name << std::endl; std::cout << LOGGER::INFO << "Get param robot_name: " << this->robot_name << std::endl;
std::cout << LOGGER::INFO << "Get param gazebo_model_name: " << this->gazebo_model_name << std::endl; std::cout << LOGGER::INFO << "Get param gazebo_model_name: " << this->gazebo_model_name << std::endl;
} }
}
// // get params from this node else
// this->declare_parameter<std::string>("robot_name", ""); {
// this->get_parameter("robot_name", this->robot_name); std::cout << LOGGER::ERROR << "Failed to call param_node service" << std::endl;
// this->declare_parameter<std::string>("gazebo_model_name", ""); }
// this->get_parameter("gazebo_model_name", this->gazebo_model_name);
// read params from yaml // read params from yaml
this->ReadYaml(this->robot_name); this->ReadYaml(this->robot_name);

View File

@ -37,7 +37,11 @@ typedef struct
namespace robot_joint_controller namespace robot_joint_controller
{ {
using CallbackReturn = rclcpp_lifecycle::node_interfaces::LifecycleNodeInterface::CallbackReturn; #if defined(ROS_DISTRO_FOXY)
using CallbackReturn = rclcpp_lifecycle::node_interfaces::LifecycleNodeInterface::CallbackReturn;
#elif defined(ROS_DISTRO_HUMBLE)
using CallbackReturn = controller_interface::CallbackReturn;
#endif
class RobotJointController : public controller_interface::ControllerInterface class RobotJointController : public controller_interface::ControllerInterface
{ {
@ -51,13 +55,13 @@ public:
#elif defined(ROS_DISTRO_HUMBLE) #elif defined(ROS_DISTRO_HUMBLE)
ROBOT_JOINT_CONTROLLER_PUBLIC ROBOT_JOINT_CONTROLLER_PUBLIC
controller_interface::return_type update(const rclcpp::Time &time, const rclcpp::Duration &period) override; controller_interface::return_type update(const rclcpp::Time &time, const rclcpp::Duration &period) override;
ROBOT_JOINT_CONTROLLER_PUBLIC
CallbackReturn on_init() override;
#endif #endif
ROBOT_JOINT_CONTROLLER_PUBLIC ROBOT_JOINT_CONTROLLER_PUBLIC
controller_interface::InterfaceConfiguration command_interface_configuration() const override; controller_interface::InterfaceConfiguration command_interface_configuration() const override;
ROBOT_JOINT_CONTROLLER_PUBLIC ROBOT_JOINT_CONTROLLER_PUBLIC
controller_interface::InterfaceConfiguration state_interface_configuration() const override; controller_interface::InterfaceConfiguration state_interface_configuration() const override;
// ROBOT_JOINT_CONTROLLER_PUBLIC
// CallbackReturn on_init() override;
ROBOT_JOINT_CONTROLLER_PUBLIC ROBOT_JOINT_CONTROLLER_PUBLIC
CallbackReturn on_configure(const rclcpp_lifecycle::State &previous_state) override; CallbackReturn on_configure(const rclcpp_lifecycle::State &previous_state) override;
ROBOT_JOINT_CONTROLLER_PUBLIC ROBOT_JOINT_CONTROLLER_PUBLIC

View File

@ -37,7 +37,11 @@ typedef struct
namespace robot_joint_controller namespace robot_joint_controller
{ {
using CallbackReturn = rclcpp_lifecycle::node_interfaces::LifecycleNodeInterface::CallbackReturn; #if defined(ROS_DISTRO_FOXY)
using CallbackReturn = rclcpp_lifecycle::node_interfaces::LifecycleNodeInterface::CallbackReturn;
#elif defined(ROS_DISTRO_HUMBLE)
using CallbackReturn = controller_interface::CallbackReturn;
#endif
class RobotJointControllerGroup : public controller_interface::ControllerInterface class RobotJointControllerGroup : public controller_interface::ControllerInterface
{ {
@ -51,13 +55,13 @@ public:
#elif defined(ROS_DISTRO_HUMBLE) #elif defined(ROS_DISTRO_HUMBLE)
ROBOT_JOINT_CONTROLLER_PUBLIC ROBOT_JOINT_CONTROLLER_PUBLIC
controller_interface::return_type update(const rclcpp::Time &time, const rclcpp::Duration &period) override; controller_interface::return_type update(const rclcpp::Time &time, const rclcpp::Duration &period) override;
ROBOT_JOINT_CONTROLLER_PUBLIC
CallbackReturn on_init() override;
#endif #endif
ROBOT_JOINT_CONTROLLER_PUBLIC ROBOT_JOINT_CONTROLLER_PUBLIC
controller_interface::InterfaceConfiguration command_interface_configuration() const override; controller_interface::InterfaceConfiguration command_interface_configuration() const override;
ROBOT_JOINT_CONTROLLER_PUBLIC ROBOT_JOINT_CONTROLLER_PUBLIC
controller_interface::InterfaceConfiguration state_interface_configuration() const override; controller_interface::InterfaceConfiguration state_interface_configuration() const override;
// ROBOT_JOINT_CONTROLLER_PUBLIC
// CallbackReturn on_init() override;
ROBOT_JOINT_CONTROLLER_PUBLIC ROBOT_JOINT_CONTROLLER_PUBLIC
CallbackReturn on_configure(const rclcpp_lifecycle::State &previous_state) override; CallbackReturn on_configure(const rclcpp_lifecycle::State &previous_state) override;
ROBOT_JOINT_CONTROLLER_PUBLIC ROBOT_JOINT_CONTROLLER_PUBLIC

View File

@ -18,10 +18,12 @@ RobotJointController::RobotJointController()
memset(&servo_command_, 0, sizeof(ServoCommand)); memset(&servo_command_, 0, sizeof(ServoCommand));
} }
// CallbackReturn RobotJointController::on_init() #if defined(ROS_DISTRO_HUMBLE)
// { CallbackReturn RobotJointController::on_init()
// return CallbackReturn::SUCCESS; {
// } return CallbackReturn::SUCCESS;
}
#endif
CallbackReturn RobotJointController::on_configure(const rclcpp_lifecycle::State &previous_state) CallbackReturn RobotJointController::on_configure(const rclcpp_lifecycle::State &previous_state)
{ {

View File

@ -16,10 +16,12 @@ RobotJointControllerGroup::RobotJointControllerGroup()
memset(&servo_command_, 0, sizeof(ServoCommand)); memset(&servo_command_, 0, sizeof(ServoCommand));
} }
// CallbackReturn RobotJointControllerGroup::on_init() #if defined(ROS_DISTRO_HUMBLE)
// { CallbackReturn RobotJointControllerGroup::on_init()
// return CallbackReturn::SUCCESS; {
// } return CallbackReturn::SUCCESS;
}
#endif
CallbackReturn RobotJointControllerGroup::on_configure(const rclcpp_lifecycle::State &previous_state) CallbackReturn RobotJointControllerGroup::on_configure(const rclcpp_lifecycle::State &previous_state)
{ {

View File

@ -38,14 +38,14 @@ def generate_launch_description():
joint_state_broadcaster_node = Node( joint_state_broadcaster_node = Node(
package="controller_manager", package="controller_manager",
executable="spawner.py", executable='spawner.py' if os.environ.get('ROS_DISTRO', '') == 'foxy' else 'spawner',
arguments=["joint_state_broadcaster"], arguments=["joint_state_broadcaster"],
output='screen', output='screen',
) )
robot_joint_controller_node = Node( robot_joint_controller_node = Node(
package="controller_manager", package="controller_manager",
executable="spawner.py", executable='spawner.py' if os.environ.get('ROS_DISTRO', '') == 'foxy' else 'spawner',
arguments=["robot_joint_controller"], arguments=["robot_joint_controller"],
output='screen', output='screen',
) )

View File

@ -46,7 +46,7 @@ def generate_launch_description():
controller_nodes = [Node( controller_nodes = [Node(
package="controller_manager", package="controller_manager",
executable="spawner.py", executable='spawner.py' if os.environ.get('ROS_DISTRO', '') == 'foxy' else 'spawner',
arguments=[name], arguments=[name],
output='screen', output='screen',
) for name in controller_names] ) for name in controller_names]