diff --git a/controller_manager/include/controller_manager/controller_manager.hpp b/controller_manager/include/controller_manager/controller_manager.hpp index 845fa2dee0..b2cf2197ea 100644 --- a/controller_manager/include/controller_manager/controller_manager.hpp +++ b/controller_manager/include/controller_manager/controller_manager.hpp @@ -193,7 +193,24 @@ class ControllerManager : public rclcpp::Node // the executor (see issue #260). // rclcpp::CallbackGroup::SharedPtr deterministic_callback_group_; - // Per controller update rate support + /// Interface for external components to check if Resource Manager is initialized. + /** + * Checks if components in Resource Manager are loaded and initialized. + * \returns true if they are initialized, false otherwise. + */ + CONTROLLER_MANAGER_PUBLIC + bool is_resource_manager_initialized() const + { + return resource_manager_ && resource_manager_->are_components_initialized(); + } + + /// Update rate of the main control loop in the controller manager. + /** + * Update rate of the main control loop in the controller manager. + * The method is used for per-controller update rate support. + * + * \returns update rate of the controller manager. + */ CONTROLLER_MANAGER_PUBLIC unsigned int get_update_rate() const; diff --git a/controller_manager/src/controller_manager.cpp b/controller_manager/src/controller_manager.cpp index 3b84fc07e4..f55f23190e 100644 --- a/controller_manager/src/controller_manager.cpp +++ b/controller_manager/src/controller_manager.cpp @@ -187,8 +187,8 @@ ControllerManager::ControllerManager( std::shared_ptr executor, const std::string & manager_node_name, const std::string & node_namespace, const rclcpp::NodeOptions & options) : rclcpp::Node(manager_node_name, node_namespace, options), - resource_manager_(std::make_unique( - update_rate_, this->get_node_clock_interface())), + resource_manager_( + std::make_unique(this->get_node_clock_interface())), diagnostics_updater_(this), executor_(executor), loader_(std::make_shared>( @@ -216,7 +216,13 @@ ControllerManager::ControllerManager( "[Deprecated] Passing the robot description parameter directly to the control_manager node " "is deprecated. Use the 'robot_description' topic from 'robot_state_publisher' instead."); init_resource_manager(robot_description_); - init_services(); + if (is_resource_manager_initialized()) + { + RCLCPP_WARN( + get_logger(), + "You have to restart the framework when using robot description from parameter!"); + init_services(); + } } diagnostics_updater_.setHardwareID("ros2_control"); @@ -243,7 +249,7 @@ ControllerManager::ControllerManager( RCLCPP_WARN(get_logger(), "'update_rate' parameter not set, using default value."); } - if (resource_manager_->is_urdf_already_loaded()) + if (is_resource_manager_initialized()) { init_services(); } @@ -271,36 +277,32 @@ void ControllerManager::robot_description_callback(const std_msgs::msg::String & RCLCPP_INFO(get_logger(), "Received robot description from topic."); RCLCPP_DEBUG( get_logger(), "'Content of robot description file: %s", robot_description.data.c_str()); - // TODO(mamueluth): errors should probably be caught since we don't want controller_manager node - // to die if a non valid urdf is passed. However, should maybe be fine tuned. - try + robot_description_ = robot_description.data; + if (is_resource_manager_initialized()) { - robot_description_ = robot_description.data; - if (resource_manager_->is_urdf_already_loaded()) - { - RCLCPP_WARN( - get_logger(), - "ResourceManager has already loaded an urdf file. Ignoring attempt to reload a robot " - "description file."); - return; - } - init_resource_manager(robot_description_); - init_services(); + RCLCPP_WARN( + get_logger(), + "ResourceManager has already loaded an urdf file. Ignoring attempt to reload a robot " + "description file."); + return; } - catch (std::runtime_error & e) + init_resource_manager(robot_description_); + if (is_resource_manager_initialized()) { - RCLCPP_ERROR_STREAM( - get_logger(), - "The published robot description file (urdf) seems not to be genuine. The following error " - "was caught:" - << e.what()); + init_services(); } } void ControllerManager::init_resource_manager(const std::string & robot_description) { - // TODO(destogl): manage this when there is an error - CM should not die because URDF is wrong... - resource_manager_->load_urdf(robot_description); + if (!resource_manager_->load_and_initialize_components(robot_description, update_rate_)) + { + RCLCPP_WARN( + get_logger(), + "Could not load and initialize hardware. Please check previous output for more details. " + "After you have corrected your URDF, try to publish robot description again."); + return; + } // Get all components and if they are not defined in parameters activate them automatically auto components_to_activate = resource_manager_->get_components_status(); diff --git a/controller_manager/test/test_controller_manager_urdf_passing.cpp b/controller_manager/test/test_controller_manager_urdf_passing.cpp index 5c2ebd14f6..5db4cfc683 100644 --- a/controller_manager/test/test_controller_manager_urdf_passing.cpp +++ b/controller_manager/test/test_controller_manager_urdf_passing.cpp @@ -29,11 +29,13 @@ class TestableControllerManager : public controller_manager::ControllerManager { friend TestControllerManagerWithTestableCM; - FRIEND_TEST(TestControllerManagerWithTestableCM, initial_no_load_urdf_called); - FRIEND_TEST(TestControllerManagerWithTestableCM, load_urdf_called_after_callback); - FRIEND_TEST(TestControllerManagerWithTestableCM, load_urdf_called_after_invalid_urdf_passed); - FRIEND_TEST(TestControllerManagerWithTestableCM, load_urdf_called_after_callback); - FRIEND_TEST(TestControllerManagerWithTestableCM, load_urdf_called_after_callback); + FRIEND_TEST( + TestControllerManagerWithTestableCM, initial_no_load_and_initialize_components_called); + FRIEND_TEST( + TestControllerManagerWithTestableCM, load_and_initialize_components_called_after_callback); + FRIEND_TEST( + TestControllerManagerWithTestableCM, + expect_to_failure_when_invalid_urdf_is_given_and_be_able_to_submit_new_robot_description); public: TestableControllerManager( @@ -59,29 +61,36 @@ class TestControllerManagerWithTestableCM } }; -TEST_P(TestControllerManagerWithTestableCM, initial_no_load_urdf_called) +TEST_P(TestControllerManagerWithTestableCM, initial_no_load_and_initialize_components_called) { - ASSERT_FALSE(cm_->resource_manager_->is_urdf_already_loaded()); + ASSERT_FALSE(cm_->resource_manager_->are_components_initialized()); } -TEST_P(TestControllerManagerWithTestableCM, load_urdf_called_after_callback) +TEST_P(TestControllerManagerWithTestableCM, load_and_initialize_components_called_after_callback) { - ASSERT_FALSE(cm_->resource_manager_->is_urdf_already_loaded()); + ASSERT_FALSE(cm_->resource_manager_->are_components_initialized()); // mimic callback auto msg = std_msgs::msg::String(); msg.data = ros2_control_test_assets::minimal_robot_urdf; cm_->robot_description_callback(msg); - ASSERT_TRUE(cm_->resource_manager_->is_urdf_already_loaded()); + ASSERT_TRUE(cm_->resource_manager_->are_components_initialized()); } -TEST_P(TestControllerManagerWithTestableCM, load_urdf_called_after_invalid_urdf_passed) +TEST_P( + TestControllerManagerWithTestableCM, + expect_to_failure_when_invalid_urdf_is_given_and_be_able_to_submit_new_robot_description) { - ASSERT_FALSE(cm_->resource_manager_->is_urdf_already_loaded()); - // mimic callback + ASSERT_FALSE(cm_->resource_manager_->are_components_initialized()); + // wrong urdf auto msg = std_msgs::msg::String(); - msg.data = ros2_control_test_assets::minimal_robot_missing_command_keys_urdf; + msg.data = ros2_control_test_assets::minimal_uninitializable_robot_urdf; + cm_->robot_description_callback(msg); + ASSERT_FALSE(cm_->resource_manager_->are_components_initialized()); + // correct urdf + msg = std_msgs::msg::String(); + msg.data = ros2_control_test_assets::minimal_robot_urdf; cm_->robot_description_callback(msg); - ASSERT_TRUE(cm_->resource_manager_->is_urdf_already_loaded()); + ASSERT_TRUE(cm_->resource_manager_->are_components_initialized()); } INSTANTIATE_TEST_SUITE_P( diff --git a/hardware_interface/include/hardware_interface/resource_manager.hpp b/hardware_interface/include/hardware_interface/resource_manager.hpp index 31def4598b..37baf96388 100644 --- a/hardware_interface/include/hardware_interface/resource_manager.hpp +++ b/hardware_interface/include/hardware_interface/resource_manager.hpp @@ -49,8 +49,7 @@ class HARDWARE_INTERFACE_PUBLIC ResourceManager { public: /// Default constructor for the Resource Manager. - ResourceManager( - unsigned int update_rate = 100, + explicit ResourceManager( rclcpp::node_interfaces::NodeClockInterface::SharedPtr clock_interface = nullptr); /// Constructor for the Resource Manager. @@ -59,18 +58,16 @@ class HARDWARE_INTERFACE_PUBLIC ResourceManager * hardware components listed within as well as populate their respective * state and command interfaces. * - * If the interfaces ought to be validated, the constructor throws an exception - * in case the URDF lists interfaces which are not available. - * * \param[in] urdf string containing the URDF. - * \param[in] validate_interfaces boolean argument indicating whether the exported - * interfaces ought to be validated. Defaults to true. * \param[in] activate_all boolean argument indicating if all resources should be immediately * activated. Currently used only in tests. + * \param[in] update_rate Update rate of the controller manager to calculate calling frequency + * of async components. + * \param[in] clock_interface reference to the clock interface of the CM node for getting time + * used for triggering async components. */ explicit ResourceManager( - const std::string & urdf, bool validate_interfaces = true, bool activate_all = false, - unsigned int update_rate = 100, + const std::string & urdf, bool activate_all = false, const unsigned int update_rate = 100, rclcpp::node_interfaces::NodeClockInterface::SharedPtr clock_interface = nullptr); ResourceManager(const ResourceManager &) = delete; @@ -79,29 +76,26 @@ class HARDWARE_INTERFACE_PUBLIC ResourceManager /// Load resources from on a given URDF. /** - * The resource manager can be post initialized with a given URDF. + * The resource manager can be post-initialized with a given URDF. * This is mainly used in conjunction with the default constructor * in which the URDF might not be present at first initialization. * * \param[in] urdf string containing the URDF. - * \param[in] validate_interfaces boolean argument indicating whether the exported - * interfaces ought to be validated. Defaults to true. - * \param[in] load_and_initialize_components boolean argument indicating whether to load and - * initialize the components present in the parsed URDF. Defaults to true. + * \param[in] update_rate update rate of the main control loop, i.e., of the controller manager. + * \returns false if URDF validation has failed. */ - void load_urdf( - const std::string & urdf, bool validate_interfaces = true, - bool load_and_initialize_components = true); + virtual bool load_and_initialize_components( + const std::string & urdf, const unsigned int update_rate = 100); /** - * @brief if the resource manager load_urdf(...) function has been called this returns true. - * We want to permit to load the urdf later on but we currently don't want to permit multiple - * calls to load_urdf (reloading/loading different urdf). + * @brief if the resource manager load_and_initialize_components(...) function has been called + * this returns true. We want to permit to loading the urdf later on, but we currently don't want + * to permit multiple calls to load_and_initialize_components (reloading/loading different urdf). * - * @return true if resource manager's load_urdf() has been already called. - * @return false if resource manager's load_urdf() has not been yet called. + * @return true if the resource manager has successfully loaded and initialized the components + * @return false if the resource manager doesn't have any components loaded and initialized. */ - bool is_urdf_already_loaded() const; + bool are_components_initialized() const; /// Claim a state interface given its key. /** @@ -413,23 +407,24 @@ class HARDWARE_INTERFACE_PUBLIC ResourceManager */ bool state_interface_exists(const std::string & key) const; +protected: + bool components_are_loaded_and_initialized_ = false; + + mutable std::recursive_mutex resource_interfaces_lock_; + mutable std::recursive_mutex claimed_command_interfaces_lock_; + mutable std::recursive_mutex resources_lock_; + private: - void validate_storage(const std::vector & hardware_info) const; + bool validate_storage(const std::vector & hardware_info) const; void release_command_interface(const std::string & key); std::unordered_map claimed_command_interface_map_; - mutable std::recursive_mutex resource_interfaces_lock_; - mutable std::recursive_mutex claimed_command_interfaces_lock_; - mutable std::recursive_mutex resources_lock_; - std::unique_ptr resource_storage_; // Structure to store read and write status so it is not initialized in the real-time loop HardwareReadWriteStatus read_write_status; - - bool is_urdf_loaded__ = false; }; } // namespace hardware_interface diff --git a/hardware_interface/src/resource_manager.cpp b/hardware_interface/src/resource_manager.cpp index d77f915eee..7be0cd0cb6 100644 --- a/hardware_interface/src/resource_manager.cpp +++ b/hardware_interface/src/resource_manager.cpp @@ -60,7 +60,7 @@ auto trigger_and_print_hardware_state_transition = } else { - RCUTILS_LOG_INFO_NAMED( + RCUTILS_LOG_ERROR_NAMED( "resource_manager", "Failed to '%s' hardware '%s'", transition_name.c_str(), hardware_name.c_str()); } @@ -98,13 +98,11 @@ class ResourceStorage public: // TODO(VX792): Change this when HW ifs get their own update rate, // because the ResourceStorage really shouldn't know about the cm's parameters - ResourceStorage( - unsigned int update_rate = 100, + explicit ResourceStorage( rclcpp::node_interfaces::NodeClockInterface::SharedPtr clock_interface = nullptr) : actuator_loader_(pkg_name, actuator_interface_name), sensor_loader_(pkg_name, sensor_interface_name), system_loader_(pkg_name, system_interface_name), - cm_update_rate_(update_rate), clock_interface_(clock_interface) { } @@ -699,26 +697,14 @@ class ResourceStorage } } - void check_for_duplicates(const HardwareInfo & hardware_info) - { - // Check for identical names - if (hardware_info_map_.find(hardware_info.name) != hardware_info_map_.end()) - { - throw std::runtime_error( - "Hardware name " + hardware_info.name + - " is duplicated. Please provide a unique 'name' in the URDF."); - } - } - // TODO(destogl): Propagate "false" up, if happens in initialize_hardware - void load_and_initialize_actuator(const HardwareInfo & hardware_info) + bool load_and_initialize_actuator(const HardwareInfo & hardware_info) { auto load_and_init_actuators = [&](auto & container) { - check_for_duplicates(hardware_info); if (!load_hardware(hardware_info, actuator_loader_, container)) { - return; + return false; } if (initialize_hardware(hardware_info, container.back())) { @@ -727,31 +713,32 @@ class ResourceStorage } else { - RCUTILS_LOG_WARN_NAMED( + RCUTILS_LOG_ERROR_NAMED( "resource_manager", "Actuator hardware component '%s' from plugin '%s' failed to initialize.", hardware_info.name.c_str(), hardware_info.hardware_plugin_name.c_str()); + return false; } + return true; }; if (hardware_info.is_async) { - load_and_init_actuators(async_actuators_); + return load_and_init_actuators(async_actuators_); } else { - load_and_init_actuators(actuators_); + return load_and_init_actuators(actuators_); } } - void load_and_initialize_sensor(const HardwareInfo & hardware_info) + bool load_and_initialize_sensor(const HardwareInfo & hardware_info) { auto load_and_init_sensors = [&](auto & container) { - check_for_duplicates(hardware_info); if (!load_hardware(hardware_info, sensor_loader_, container)) { - return; + return false; } if (initialize_hardware(hardware_info, container.back())) { @@ -759,31 +746,32 @@ class ResourceStorage } else { - RCUTILS_LOG_WARN_NAMED( + RCUTILS_LOG_ERROR_NAMED( "resource_manager", "Sensor hardware component '%s' from plugin '%s' failed to initialize.", hardware_info.name.c_str(), hardware_info.hardware_plugin_name.c_str()); + return false; } + return true; }; if (hardware_info.is_async) { - load_and_init_sensors(async_sensors_); + return load_and_init_sensors(async_sensors_); } else { - load_and_init_sensors(sensors_); + return load_and_init_sensors(sensors_); } } - void load_and_initialize_system(const HardwareInfo & hardware_info) + bool load_and_initialize_system(const HardwareInfo & hardware_info) { auto load_and_init_systems = [&](auto & container) { - check_for_duplicates(hardware_info); if (!load_hardware(hardware_info, system_loader_, container)) { - return; + return false; } if (initialize_hardware(hardware_info, container.back())) { @@ -792,20 +780,22 @@ class ResourceStorage } else { - RCUTILS_LOG_WARN_NAMED( + RCUTILS_LOG_ERROR_NAMED( "resource_manager", "System hardware component '%s' from plugin '%s' failed to initialize.", hardware_info.name.c_str(), hardware_info.hardware_plugin_name.c_str()); + return false; } + return true; }; if (hardware_info.is_async) { - load_and_init_systems(async_systems_); + return load_and_init_systems(async_systems_); } else { - load_and_init_systems(systems_); + return load_and_init_systems(systems_); } } @@ -898,6 +888,26 @@ class ResourceStorage } } + void clear() + { + actuators_.clear(); + sensors_.clear(); + systems_.clear(); + + async_actuators_.clear(); + async_sensors_.clear(); + async_systems_.clear(); + + hardware_info_map_.clear(); + state_interface_map_.clear(); + command_interface_map_.clear(); + + available_state_interfaces_.clear(); + available_command_interfaces_.clear(); + + claimed_command_interface_map_.clear(); + } + /** * Returns the return type of the hardware component group state, if the return type is other * than OK, then updates the return type of the group to the respective one @@ -958,24 +968,24 @@ class ResourceStorage // Update rate of the controller manager, and the clock interface of its node // Used by async components. - unsigned int cm_update_rate_; + unsigned int cm_update_rate_ = 100; rclcpp::node_interfaces::NodeClockInterface::SharedPtr clock_interface_; }; ResourceManager::ResourceManager( - unsigned int update_rate, rclcpp::node_interfaces::NodeClockInterface::SharedPtr clock_interface) -: resource_storage_(std::make_unique(update_rate, clock_interface)) + rclcpp::node_interfaces::NodeClockInterface::SharedPtr clock_interface) +: resource_storage_(std::make_unique(clock_interface)) { } ResourceManager::~ResourceManager() = default; ResourceManager::ResourceManager( - const std::string & urdf, bool validate_interfaces, bool activate_all, unsigned int update_rate, + const std::string & urdf, bool activate_all, const unsigned int update_rate, rclcpp::node_interfaces::NodeClockInterface::SharedPtr clock_interface) -: resource_storage_(std::make_unique(update_rate, clock_interface)) +: resource_storage_(std::make_unique(clock_interface)) { - load_urdf(urdf, validate_interfaces); + load_and_initialize_components(urdf, update_rate); if (activate_all) { @@ -989,50 +999,86 @@ ResourceManager::ResourceManager( } // CM API: Called in "callback/slow"-thread -void ResourceManager::load_urdf( - const std::string & urdf, bool validate_interfaces, bool load_and_initialize_components) +bool ResourceManager::load_and_initialize_components( + const std::string & urdf, const unsigned int update_rate) { - is_urdf_loaded__ = true; + components_are_loaded_and_initialized_ = true; + + resource_storage_->cm_update_rate_ = update_rate; + + const auto hardware_info = hardware_interface::parse_control_resources_from_urdf(urdf); + const std::string system_type = "system"; const std::string sensor_type = "sensor"; const std::string actuator_type = "actuator"; - const auto hardware_info = hardware_interface::parse_control_resources_from_urdf(urdf); - if (load_and_initialize_components) + for (const auto & individual_hardware_info : hardware_info) { - for (const auto & individual_hardware_info : hardware_info) + // Check for identical names + if ( + resource_storage_->hardware_info_map_.find(individual_hardware_info.name) != + resource_storage_->hardware_info_map_.end()) + { + RCUTILS_LOG_ERROR_NAMED( + "resource_manager", + "Hardware name %s is duplicated. Please provide a unique 'name' " + "in the URDF.", + individual_hardware_info.name.c_str()); + components_are_loaded_and_initialized_ = false; + break; + } + + if (individual_hardware_info.type == actuator_type) { - if (individual_hardware_info.type == actuator_type) + std::scoped_lock guard(resource_interfaces_lock_, claimed_command_interfaces_lock_); + if (!resource_storage_->load_and_initialize_actuator(individual_hardware_info)) { - std::scoped_lock guard(resource_interfaces_lock_, claimed_command_interfaces_lock_); - resource_storage_->load_and_initialize_actuator(individual_hardware_info); + components_are_loaded_and_initialized_ = false; + break; } - if (individual_hardware_info.type == sensor_type) + } + if (individual_hardware_info.type == sensor_type) + { + std::lock_guard guard(resource_interfaces_lock_); + if (!resource_storage_->load_and_initialize_sensor(individual_hardware_info)) { - std::lock_guard guard(resource_interfaces_lock_); - resource_storage_->load_and_initialize_sensor(individual_hardware_info); + components_are_loaded_and_initialized_ = false; + break; } - if (individual_hardware_info.type == system_type) + } + if (individual_hardware_info.type == system_type) + { + std::scoped_lock guard(resource_interfaces_lock_, claimed_command_interfaces_lock_); + if (!resource_storage_->load_and_initialize_system(individual_hardware_info)) { - std::scoped_lock guard(resource_interfaces_lock_, claimed_command_interfaces_lock_); - resource_storage_->load_and_initialize_system(individual_hardware_info); + components_are_loaded_and_initialized_ = false; + break; } } } - // throw on missing state and command interfaces, not specified keys are being ignored - if (validate_interfaces) + if (components_are_loaded_and_initialized_ && validate_storage(hardware_info)) + { + std::lock_guard guard(resources_lock_); + read_write_status.failed_hardware_names.reserve( + resource_storage_->actuators_.size() + resource_storage_->sensors_.size() + + resource_storage_->systems_.size()); + } + else { - validate_storage(hardware_info); + std::scoped_lock guard(resource_interfaces_lock_, claimed_command_interfaces_lock_); + resource_storage_->clear(); + // cleanup everything + components_are_loaded_and_initialized_ = false; } - std::lock_guard guard(resources_lock_); - read_write_status.failed_hardware_names.reserve( - resource_storage_->actuators_.size() + resource_storage_->sensors_.size() + - resource_storage_->systems_.size()); + return components_are_loaded_and_initialized_; } -bool ResourceManager::is_urdf_already_loaded() const { return is_urdf_loaded__; } +bool ResourceManager::are_components_initialized() const +{ + return components_are_loaded_and_initialized_; +} // CM API: Called in "update"-thread LoanedStateInterface ResourceManager::claim_state_interface(const std::string & key) @@ -1743,7 +1789,7 @@ bool ResourceManager::state_interface_exists(const std::string & key) const // BEGIN: private methods -void ResourceManager::validate_storage( +bool ResourceManager::validate_storage( const std::vector & hardware_info) const { std::vector missing_state_keys = {}; @@ -1799,20 +1845,28 @@ void ResourceManager::validate_storage( if (!missing_state_keys.empty() || !missing_command_keys.empty()) { - std::string err_msg = "Wrong state or command interface configuration.\n"; - err_msg += "missing state interfaces:\n"; + std::string message = "Wrong state or command interface configuration.\n"; + message += "missing state interfaces:\n"; for (const auto & missing_key : missing_state_keys) { - err_msg += "' " + missing_key + " '" + "\t"; + message += " " + missing_key + "\t"; } - err_msg += "\nmissing command interfaces:\n"; + message += "\nmissing command interfaces:\n"; for (const auto & missing_key : missing_command_keys) { - err_msg += "' " + missing_key + " '" + "\t"; + message += " " + missing_key + "\t"; } - throw std::runtime_error(err_msg); + RCUTILS_LOG_ERROR_NAMED( + "resource_manager", + "Discrepancy between robot description file (urdf) and actually exported HW interfaces.\n" + "Details: %s", + message.c_str()); + + return false; } + + return true; } // END: private methods diff --git a/hardware_interface/test/mock_components/test_generic_system.cpp b/hardware_interface/test/mock_components/test_generic_system.cpp index c7777b3e21..08a3640bd8 100644 --- a/hardware_interface/test/mock_components/test_generic_system.cpp +++ b/hardware_interface/test/mock_components/test_generic_system.cpp @@ -696,9 +696,8 @@ class TestableResourceManager : public hardware_interface::ResourceManager TestableResourceManager() : hardware_interface::ResourceManager() {} - TestableResourceManager( - const std::string & urdf, bool validate_interfaces = true, bool activate_all = false) - : hardware_interface::ResourceManager(urdf, validate_interfaces, activate_all) + explicit TestableResourceManager(const std::string & urdf, bool activate_all = false) + : hardware_interface::ResourceManager(urdf, activate_all) { } }; diff --git a/hardware_interface_testing/test/test_components/test_actuator.cpp b/hardware_interface_testing/test/test_components/test_actuator.cpp index a01ccd879f..1277fecfb6 100644 --- a/hardware_interface_testing/test/test_components/test_actuator.cpp +++ b/hardware_interface_testing/test/test_components/test_actuator.cpp @@ -32,6 +32,11 @@ class TestActuator : public ActuatorInterface return CallbackReturn::ERROR; } + if (info_.joints[0].state_interfaces[1].name == "does_not_exist") + { + return CallbackReturn::ERROR; + } + /* * a hardware can optional prove for incorrect info here. * diff --git a/hardware_interface_testing/test/test_components/test_sensor.cpp b/hardware_interface_testing/test/test_components/test_sensor.cpp index 2ea36ac5c1..3fc2ef2445 100644 --- a/hardware_interface_testing/test/test_components/test_sensor.cpp +++ b/hardware_interface_testing/test/test_components/test_sensor.cpp @@ -30,7 +30,7 @@ class TestSensor : public SensorInterface return CallbackReturn::ERROR; } // can only give feedback state for velocity - if (info_.sensors[0].state_interfaces.size() != 1) + if (info_.sensors[0].state_interfaces.size() == 2) { return CallbackReturn::ERROR; } diff --git a/hardware_interface_testing/test/test_components/test_system.cpp b/hardware_interface_testing/test/test_components/test_system.cpp index 20e606ee6d..502f5b4c21 100644 --- a/hardware_interface_testing/test/test_components/test_system.cpp +++ b/hardware_interface_testing/test/test_components/test_system.cpp @@ -27,6 +27,22 @@ using hardware_interface::SystemInterface; class TestSystem : public SystemInterface { + CallbackReturn on_init(const hardware_interface::HardwareInfo & info) override + { + if (SystemInterface::on_init(info) != CallbackReturn::SUCCESS) + { + return CallbackReturn::ERROR; + } + + // Simulating initialization error + if (info_.joints[0].state_interfaces[1].name == "does_not_exist") + { + return CallbackReturn::ERROR; + } + + return CallbackReturn::SUCCESS; + } + std::vector export_state_interfaces() override { std::vector state_interfaces; diff --git a/hardware_interface_testing/test/test_resource_manager.cpp b/hardware_interface_testing/test/test_resource_manager.cpp index c8b077229e..5a2ea0210c 100644 --- a/hardware_interface_testing/test/test_resource_manager.cpp +++ b/hardware_interface_testing/test/test_resource_manager.cpp @@ -99,26 +99,20 @@ TEST_F(ResourceManagerTest, initialization_with_urdf) TEST_F(ResourceManagerTest, post_initialization_with_urdf) { TestableResourceManager rm; - ASSERT_NO_THROW(rm.load_urdf(ros2_control_test_assets::minimal_robot_urdf)); + ASSERT_NO_THROW(rm.load_and_initialize_components(ros2_control_test_assets::minimal_robot_urdf)); } -TEST_F(ResourceManagerTest, test_uninitializable_hardware_validation) +void test_load_and_initialized_components_failure(const std::string & urdf) { - // If the the hardware can not be initialized and load_urdf tried to validate - // the interfaces a runtime exception is thrown TestableResourceManager rm; - ASSERT_THROW( - rm.load_urdf(ros2_control_test_assets::minimal_uninitializable_robot_urdf, true), - std::runtime_error); -} + ASSERT_FALSE(rm.load_and_initialize_components(urdf)); -TEST_F(ResourceManagerTest, test_uninitializable_hardware_no_validation) -{ - // If the the hardware can not be initialized and load_urdf didn't try to - // validate the interfaces, the interface should not show up - TestableResourceManager rm; - EXPECT_NO_THROW( - rm.load_urdf(ros2_control_test_assets::minimal_uninitializable_robot_urdf, false)); + ASSERT_FALSE(rm.are_components_initialized()); + + // resource manager should also not have any components + EXPECT_EQ(rm.actuator_components_size(), 0); + EXPECT_EQ(rm.sensor_components_size(), 0); + EXPECT_EQ(rm.system_components_size(), 0); // test actuator EXPECT_FALSE(rm.state_interface_exists("joint1/position")); @@ -142,7 +136,16 @@ TEST_F(ResourceManagerTest, test_uninitializable_hardware_no_validation) EXPECT_FALSE(rm.command_interface_exists("joint3/max_acceleration")); } -TEST_F(ResourceManagerTest, initialization_with_urdf_manual_validation) +TEST_F(ResourceManagerTest, test_unitilizable_hardware) +{ + SCOPED_TRACE("test_unitilizable_hardware_no_validation"); + // If the the hardware can not be initialized and load_and_initialize_components didn't try to + // validate the interfaces, the interface should not show up + test_load_and_initialized_components_failure( + ros2_control_test_assets::minimal_uninitializable_robot_urdf); +} + +TEST_F(ResourceManagerTest, initialization_with_urdf_and_manual_validation) { // we validate the results manually TestableResourceManager rm(ros2_control_test_assets::minimal_robot_urdf, false); @@ -166,20 +169,17 @@ TEST_F(ResourceManagerTest, initialization_with_urdf_manual_validation) EXPECT_TRUE(rm.command_interface_exists("joint3/velocity")); } -TEST_F(ResourceManagerTest, initialization_with_wrong_urdf) +TEST_F(ResourceManagerTest, expect_validation_failure_if_not_all_interfaces_are_exported) { + SCOPED_TRACE("missing state keys"); // missing state keys - { - EXPECT_THROW( - TestableResourceManager rm(ros2_control_test_assets::minimal_robot_missing_state_keys_urdf), - std::exception); - } + test_load_and_initialized_components_failure( + ros2_control_test_assets::minimal_robot_missing_state_keys_urdf); + + SCOPED_TRACE("missing command keys"); // missing command keys - { - EXPECT_THROW( - TestableResourceManager rm(ros2_control_test_assets::minimal_robot_missing_command_keys_urdf), - std::exception); - } + test_load_and_initialized_components_failure( + ros2_control_test_assets::minimal_robot_missing_command_keys_urdf); } TEST_F(ResourceManagerTest, initialization_with_urdf_unclaimed) @@ -201,32 +201,75 @@ TEST_F(ResourceManagerTest, initialization_with_urdf_unclaimed) } } -TEST_F(ResourceManagerTest, no_load_urdf_function_called) +TEST_F(ResourceManagerTest, no_load_and_initialize_components_function_called) { TestableResourceManager rm; - ASSERT_FALSE(rm.is_urdf_already_loaded()); + ASSERT_FALSE(rm.are_components_initialized()); } -TEST_F(ResourceManagerTest, load_urdf_called_if_urdf_is_invalid) +TEST_F( + ResourceManagerTest, expect_load_and_initialize_to_fail_when_a_hw_component_plugin_does_not_exist) { - TestableResourceManager rm; - EXPECT_THROW( - rm.load_urdf(ros2_control_test_assets::minimal_robot_missing_state_keys_urdf), std::exception); - ASSERT_TRUE(rm.is_urdf_already_loaded()); + SCOPED_TRACE("Actuator plugin does not exist"); + // Actuator + test_load_and_initialized_components_failure( + ros2_control_test_assets::minimal_robot_not_existing_actuator_plugin); + + SCOPED_TRACE("Sensor plugin does not exist"); + // Sensor + test_load_and_initialized_components_failure( + ros2_control_test_assets::minimal_robot_not_existing_sensors_plugin); + + SCOPED_TRACE("System plugin does not exist"); + // System + test_load_and_initialized_components_failure( + ros2_control_test_assets::minimal_robot_not_existing_system_plugin); } -TEST_F(ResourceManagerTest, load_urdf_called_if_urdf_is_valid) +TEST_F(ResourceManagerTest, expect_load_and_initialize_to_fail_when_there_are_dupplicate_of_hw_comp) +{ + SCOPED_TRACE("Duplicated components"); + test_load_and_initialized_components_failure( + ros2_control_test_assets::minimal_robot_duplicated_component); +} + +TEST_F( + ResourceManagerTest, expect_load_and_initialize_to_fail_when_a_hw_component_initialization_fails) +{ + SCOPED_TRACE("Actuator initialization fails"); + // Actuator + test_load_and_initialized_components_failure( + ros2_control_test_assets::minimal_robot_actuator_initialization_error); + + SCOPED_TRACE("Sensor initialization fails"); + // Sensor + test_load_and_initialized_components_failure( + ros2_control_test_assets::minimal_robot_sensor_initialization_error); + + SCOPED_TRACE("System initialization fails"); + // System + test_load_and_initialized_components_failure( + ros2_control_test_assets::minimal_robot_system_initialization_error); +} + +TEST_F(ResourceManagerTest, load_and_initialize_components_called_if_urdf_is_valid) { TestableResourceManager rm(ros2_control_test_assets::minimal_robot_urdf); - ASSERT_TRUE(rm.is_urdf_already_loaded()); + ASSERT_TRUE(rm.are_components_initialized()); +} + +TEST_F(ResourceManagerTest, load_and_initialize_components_called_if_async_urdf_is_valid) +{ + TestableResourceManager rm(ros2_control_test_assets::minimal_async_robot_urdf); + ASSERT_TRUE(rm.are_components_initialized()); } -TEST_F(ResourceManagerTest, can_load_urdf_later) +TEST_F(ResourceManagerTest, can_load_and_initialize_components_later) { TestableResourceManager rm; - ASSERT_FALSE(rm.is_urdf_already_loaded()); - rm.load_urdf(ros2_control_test_assets::minimal_robot_urdf); - ASSERT_TRUE(rm.is_urdf_already_loaded()); + ASSERT_FALSE(rm.are_components_initialized()); + rm.load_and_initialize_components(ros2_control_test_assets::minimal_robot_urdf); + ASSERT_TRUE(rm.are_components_initialized()); } TEST_F(ResourceManagerTest, resource_claiming) @@ -345,7 +388,7 @@ class ExternalComponent : public hardware_interface::ActuatorInterface TEST_F(ResourceManagerTest, post_initialization_add_components) { // we validate the results manually - TestableResourceManager rm(ros2_control_test_assets::minimal_robot_urdf, false); + TestableResourceManager rm(ros2_control_test_assets::minimal_robot_urdf); // Activate components to get all interfaces available activate_components(rm); diff --git a/hardware_interface_testing/test/test_resource_manager.hpp b/hardware_interface_testing/test/test_resource_manager.hpp index 554dfe2c86..4eb7b8a799 100644 --- a/hardware_interface_testing/test/test_resource_manager.hpp +++ b/hardware_interface_testing/test/test_resource_manager.hpp @@ -44,7 +44,7 @@ class TestableResourceManager : public hardware_interface::ResourceManager public: friend ResourceManagerTest; - FRIEND_TEST(ResourceManagerTest, initialization_with_urdf_manual_validation); + FRIEND_TEST(ResourceManagerTest, initialization_with_urdf_and_manual_validation); FRIEND_TEST(ResourceManagerTest, post_initialization_add_components); FRIEND_TEST(ResourceManagerTest, managing_controllers_reference_interfaces); FRIEND_TEST(ResourceManagerTest, resource_availability_and_claiming_in_lifecycle); @@ -52,9 +52,8 @@ class TestableResourceManager : public hardware_interface::ResourceManager TestableResourceManager() : hardware_interface::ResourceManager() {} - TestableResourceManager( - const std::string & urdf, bool validate_interfaces = true, bool activate_all = false) - : hardware_interface::ResourceManager(urdf, validate_interfaces, activate_all) + explicit TestableResourceManager(const std::string & urdf, bool activate_all = false) + : hardware_interface::ResourceManager(urdf, activate_all) { } }; diff --git a/ros2_control_test_assets/include/ros2_control_test_assets/descriptions.hpp b/ros2_control_test_assets/include/ros2_control_test_assets/descriptions.hpp index 31e630059b..aeb7dcec5b 100644 --- a/ros2_control_test_assets/include/ros2_control_test_assets/descriptions.hpp +++ b/ros2_control_test_assets/include/ros2_control_test_assets/descriptions.hpp @@ -660,6 +660,55 @@ const auto hardware_resources = )"; +const auto async_hardware_resources = + R"( + + + test_actuator + + + + + + + + + + + test_sensor + 2 + 2 + + + + + + + + test_system + 2 + 2 + + + + + + + + + + + + + + + + + + + +)"; + const auto uninitializable_hardware_resources = R"( @@ -705,7 +754,56 @@ const auto uninitializable_hardware_resources = )"; -const auto hardware_resources_missing_state_keys = +const auto hardware_resources_not_existing_actuator_plugin = + R"( + + + test_actuator23 + + + + + + + + + + + test_sensor + 2 + 2 + + + + + + + + test_system + 2 + 2 + + + + + + + + + + + + + + + + + + + +)"; + +const auto hardware_resources_not_existing_sensor_plugin = R"( @@ -715,6 +813,55 @@ const auto hardware_resources_missing_state_keys = + + + + + + test_sensor23 + 2 + 2 + + + + + + + + test_system + 2 + 2 + + + + + + + + + + + + + + + + + + + +)"; +const auto hardware_resources_not_existing_system_plugin = + R"( + + + test_actuator + + + + + + @@ -725,7 +872,103 @@ const auto hardware_resources_missing_state_keys = + + + + + test_system23 + 2 + 2 + + + + + + + + + + + + + + + + + + + +)"; + +const auto hardware_resources_duplicated_component = + R"( + + + test_actuator + + + + + + + + + + + test_sensor + 2 + 2 + + + + + + + + test_system + 2 + 2 + + + + + + + + + + + + + + + + + + + +)"; + +const auto hardware_resources_actuator_initializaion_error = + R"( + + + test_actuator + + + + + + + + + test_sensor + 2 + 2 + + + @@ -737,11 +980,140 @@ const auto hardware_resources_missing_state_keys = + + + + + + + + +)"; + +const auto hardware_resources_sensor_initializaion_error = + R"( + + + test_actuator + + + + + + + + + + test_sensor + 2 + 2 + + + + + + + + test_system + 2 + 2 + + + + + + + + +)"; + +const auto hardware_resources_system_initializaion_error = + R"( + + + test_actuator + + + + + + + + + + test_sensor + 2 + 2 + + + + + + + + test_system + 2 + 2 + + + + + + + + + + + + +)"; + +const auto hardware_resources_missing_state_keys = + R"( + + + test_actuator + + + + + + + + + + + test_sensor + 2 + 2 + + + + + + + + + + test_system + 2 + 2 + + + + + + + + + + + @@ -778,13 +1150,15 @@ const auto hardware_resources_missing_command_keys = - + + + )"; @@ -1367,9 +1741,35 @@ const auto gripper_hardware_resources_mimic_false_command_if = const auto minimal_robot_urdf = std::string(urdf_head) + std::string(hardware_resources) + std::string(urdf_tail); +const auto minimal_async_robot_urdf = + std::string(urdf_head) + std::string(async_hardware_resources) + std::string(urdf_tail); const auto minimal_uninitializable_robot_urdf = std::string(urdf_head) + std::string(uninitializable_hardware_resources) + std::string(urdf_tail); +const auto minimal_robot_not_existing_actuator_plugin = + std::string(urdf_head) + std::string(hardware_resources_not_existing_actuator_plugin) + + std::string(urdf_tail); +const auto minimal_robot_not_existing_sensors_plugin = + std::string(urdf_head) + std::string(hardware_resources_not_existing_sensor_plugin) + + std::string(urdf_tail); +const auto minimal_robot_not_existing_system_plugin = + std::string(urdf_head) + std::string(hardware_resources_not_existing_system_plugin) + + std::string(urdf_tail); + +const auto minimal_robot_duplicated_component = + std::string(urdf_head) + std::string(hardware_resources_duplicated_component) + + std::string(urdf_tail); + +const auto minimal_robot_actuator_initialization_error = + std::string(urdf_head) + std::string(hardware_resources_not_existing_actuator_plugin) + + std::string(urdf_tail); +const auto minimal_robot_sensor_initialization_error = + std::string(urdf_head) + std::string(hardware_resources_sensor_initializaion_error) + + std::string(urdf_tail); +const auto minimal_robot_system_initialization_error = + std::string(urdf_head) + std::string(hardware_resources_system_initializaion_error) + + std::string(urdf_tail); + const auto minimal_robot_missing_state_keys_urdf = std::string(urdf_head) + std::string(hardware_resources_missing_state_keys) + std::string(urdf_tail);