-
Notifications
You must be signed in to change notification settings - Fork 333
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
added tests to the passthrough controller
- Loading branch information
Showing
3 changed files
with
434 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
357 changes: 357 additions & 0 deletions
357
passthrough_controller/test/test_passthrough_controller.cpp
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,357 @@ | ||
// Copyright (c) 2023, PAL Robotics | ||
// | ||
// 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. | ||
/// \author Sai Kishor Kothakota | ||
|
||
#include <functional> | ||
#include <memory> | ||
#include <string> | ||
#include <utility> | ||
#include <vector> | ||
|
||
#include "gmock/gmock.h" | ||
|
||
#include "hardware_interface/loaned_command_interface.hpp" | ||
#include "lifecycle_msgs/msg/state.hpp" | ||
#include "rclcpp/node.hpp" | ||
#include "rclcpp/qos.hpp" | ||
#include "rclcpp/subscription.hpp" | ||
#include "rclcpp/utilities.hpp" | ||
#include "rclcpp/wait_set.hpp" | ||
#include "rclcpp_lifecycle/node_interfaces/lifecycle_node_interface.hpp" | ||
#include "test_passthrough_controller.hpp" | ||
|
||
using hardware_interface::LoanedCommandInterface; | ||
|
||
namespace | ||
{ | ||
rclcpp::WaitResultKind wait_for(rclcpp::SubscriptionBase::SharedPtr subscription) | ||
{ | ||
rclcpp::WaitSet wait_set; | ||
wait_set.add_subscription(subscription); | ||
const auto timeout = std::chrono::seconds(10); | ||
return wait_set.wait(timeout).kind(); | ||
} | ||
} // namespace | ||
|
||
void PassthroughControllerTest::SetUpTestCase() { rclcpp::init(0, nullptr); } | ||
|
||
void PassthroughControllerTest::TearDownTestCase() { rclcpp::shutdown(); } | ||
|
||
void PassthroughControllerTest::SetUp() | ||
{ | ||
// initialize controller | ||
controller_ = std::make_unique<FriendPassthroughController>(); | ||
} | ||
|
||
void PassthroughControllerTest::TearDown() { controller_.reset(nullptr); } | ||
|
||
void PassthroughControllerTest::SetUpController() | ||
{ | ||
const auto result = controller_->init("passthrough_controller"); | ||
ASSERT_EQ(result, controller_interface::return_type::OK); | ||
|
||
std::vector<LoanedCommandInterface> command_ifs; | ||
command_ifs.emplace_back(joint_1_pos_cmd_); | ||
command_ifs.emplace_back(joint_2_pos_cmd_); | ||
command_ifs.emplace_back(joint_3_pos_cmd_); | ||
controller_->assign_interfaces(std::move(command_ifs), {}); | ||
} | ||
|
||
TEST_F(PassthroughControllerTest, InterfaceParameterNotSet) | ||
{ | ||
SetUpController(); | ||
// configure failed, 'interfaces' parameter not set | ||
ASSERT_EQ( | ||
controller_->on_configure(rclcpp_lifecycle::State()), | ||
controller_interface::CallbackReturn::ERROR); | ||
} | ||
|
||
TEST_F(PassthroughControllerTest, InterfaceParameterEmpty) | ||
{ | ||
SetUpController(); | ||
controller_->get_node()->set_parameter({"interfaces", std::vector<std::string>()}); | ||
|
||
// configure failed, 'interfaces' is empty | ||
ASSERT_EQ( | ||
controller_->on_configure(rclcpp_lifecycle::State()), | ||
controller_interface::CallbackReturn::ERROR); | ||
} | ||
|
||
TEST_F(PassthroughControllerTest, ConfigureParamsSuccess) | ||
{ | ||
SetUpController(); | ||
|
||
controller_->get_node()->set_parameter( | ||
{"interfaces", std::vector<std::string>{"joint1/interface", "joint2/interface"}}); | ||
|
||
// configure successful | ||
ASSERT_EQ( | ||
controller_->on_configure(rclcpp_lifecycle::State()), | ||
controller_interface::CallbackReturn::SUCCESS); | ||
} | ||
|
||
TEST_F(PassthroughControllerTest, ActivateWithWrongInterfaceNameFails) | ||
{ | ||
SetUpController(); | ||
|
||
controller_->get_node()->set_parameter( | ||
{"interfaces", std::vector<std::string>{"joint1/input", "joint2/input"}}); | ||
|
||
// activate failed, 'input' is not a registered interface for `joint1` | ||
ASSERT_EQ( | ||
controller_->on_configure(rclcpp_lifecycle::State()), | ||
controller_interface::CallbackReturn::SUCCESS); | ||
ASSERT_EQ( | ||
controller_->on_activate(rclcpp_lifecycle::State()), | ||
controller_interface::CallbackReturn::ERROR); | ||
} | ||
|
||
TEST_F(PassthroughControllerTest, ActivateSuccess) | ||
{ | ||
SetUpController(); | ||
|
||
controller_->get_node()->set_parameter( | ||
{"interfaces", std::vector<std::string>{"joint1/interface", "joint2/interface"}}); | ||
|
||
// activate successful | ||
ASSERT_EQ( | ||
controller_->on_configure(rclcpp_lifecycle::State()), | ||
controller_interface::CallbackReturn::SUCCESS); | ||
ASSERT_EQ( | ||
controller_->on_activate(rclcpp_lifecycle::State()), | ||
controller_interface::CallbackReturn::SUCCESS); | ||
} | ||
|
||
TEST_F(PassthroughControllerTest, CommandSuccessTest) | ||
{ | ||
SetUpController(); | ||
|
||
// configure controller | ||
controller_->get_node()->set_parameter( | ||
{"interfaces", | ||
std::vector<std::string>{"joint1/interface", "joint2/interface", "joint3/interface"}}); | ||
|
||
ASSERT_EQ( | ||
controller_->on_configure(rclcpp_lifecycle::State()), | ||
controller_interface::CallbackReturn::SUCCESS); | ||
|
||
// update successful though no command has been send yet | ||
ASSERT_EQ( | ||
controller_->update(rclcpp::Time(0), rclcpp::Duration::from_seconds(0.01)), | ||
controller_interface::return_type::OK); | ||
|
||
// check joint commands are still the default ones | ||
ASSERT_EQ(joint_1_pos_cmd_.get_value(), 1.1); | ||
ASSERT_EQ(joint_2_pos_cmd_.get_value(), 2.1); | ||
ASSERT_EQ(joint_3_pos_cmd_.get_value(), 3.1); | ||
|
||
// send command | ||
auto command_ptr = std::make_shared<passthrough_controller::DataType>(); | ||
command_ptr->data = {10.0, 20.0, 30.0}; | ||
controller_->rt_buffer_ptr_.writeFromNonRT(command_ptr); | ||
|
||
// update successful, command received | ||
ASSERT_EQ( | ||
controller_->update(rclcpp::Time(0.1), rclcpp::Duration::from_seconds(0.01)), | ||
controller_interface::return_type::OK); | ||
|
||
// check joint commands have been modified | ||
ASSERT_EQ(joint_1_pos_cmd_.get_value(), 10.0); | ||
ASSERT_EQ(joint_2_pos_cmd_.get_value(), 20.0); | ||
ASSERT_EQ(joint_3_pos_cmd_.get_value(), 30.0); | ||
} | ||
|
||
TEST_F(PassthroughControllerTest, WrongCommandCheckTest) | ||
{ | ||
SetUpController(); | ||
|
||
// configure controller | ||
controller_->get_node()->set_parameter( | ||
{"interfaces", std::vector<std::string>{"joint1/interface", "joint2/interface"}}); | ||
|
||
ASSERT_EQ( | ||
controller_->on_configure(rclcpp_lifecycle::State()), | ||
controller_interface::CallbackReturn::SUCCESS); | ||
|
||
// send command with wrong number of joints | ||
auto command_ptr = std::make_shared<passthrough_controller::DataType>(); | ||
command_ptr->data = {10.0}; | ||
controller_->rt_buffer_ptr_.writeFromNonRT(command_ptr); | ||
|
||
// update failed, command size does not match number of joints | ||
ASSERT_EQ( | ||
controller_->update(rclcpp::Time(0), rclcpp::Duration::from_seconds(0.01)), | ||
controller_interface::return_type::ERROR); | ||
|
||
// check joint commands are still the default ones | ||
ASSERT_EQ(joint_1_pos_cmd_.get_value(), 1.1); | ||
ASSERT_EQ(joint_2_pos_cmd_.get_value(), 2.1); | ||
} | ||
|
||
TEST_F(PassthroughControllerTest, NoCommandCheckTest) | ||
{ | ||
SetUpController(); | ||
|
||
// configure controller | ||
controller_->get_node()->set_parameter( | ||
{"interfaces", std::vector<std::string>{"joint1/interface", "joint2/interface"}}); | ||
|
||
ASSERT_EQ( | ||
controller_->on_configure(rclcpp_lifecycle::State()), | ||
controller_interface::CallbackReturn::SUCCESS); | ||
|
||
// update successful, no command received yet | ||
ASSERT_EQ( | ||
controller_->update(rclcpp::Time(0), rclcpp::Duration::from_seconds(0.01)), | ||
controller_interface::return_type::OK); | ||
|
||
// check joint commands are still the default ones | ||
ASSERT_EQ(joint_1_pos_cmd_.get_value(), 1.1); | ||
ASSERT_EQ(joint_2_pos_cmd_.get_value(), 2.1); | ||
} | ||
|
||
TEST_F(PassthroughControllerTest, CommandCallbackTest) | ||
{ | ||
SetUpController(); | ||
|
||
// configure controller | ||
controller_->get_node()->set_parameter( | ||
{"interfaces", | ||
std::vector<std::string>{"joint1/interface", "joint2/interface", "joint3/interface"}}); | ||
|
||
// default values | ||
ASSERT_EQ(joint_1_pos_cmd_.get_value(), 1.1); | ||
ASSERT_EQ(joint_2_pos_cmd_.get_value(), 2.1); | ||
ASSERT_EQ(joint_3_pos_cmd_.get_value(), 3.1); | ||
|
||
auto node_state = controller_->get_node()->configure(); | ||
ASSERT_EQ(node_state.id(), lifecycle_msgs::msg::State::PRIMARY_STATE_INACTIVE); | ||
|
||
node_state = controller_->get_node()->activate(); | ||
ASSERT_EQ(node_state.id(), lifecycle_msgs::msg::State::PRIMARY_STATE_ACTIVE); | ||
|
||
// send a new command | ||
rclcpp::Node test_node("test_node"); | ||
auto command_pub = test_node.create_publisher<std_msgs::msg::Float64MultiArray>( | ||
std::string(controller_->get_node()->get_name()) + "/commands", rclcpp::SystemDefaultsQoS()); | ||
std_msgs::msg::Float64MultiArray command_msg; | ||
command_msg.data = {10.0, 20.0, 30.0}; | ||
command_pub->publish(command_msg); | ||
|
||
// wait for command message to be passed | ||
ASSERT_EQ(wait_for(controller_->joints_cmd_sub_), rclcpp::WaitResultKind::Ready); | ||
|
||
// process callbacks | ||
rclcpp::spin_some(controller_->get_node()->get_node_base_interface()); | ||
|
||
// update successful | ||
ASSERT_EQ( | ||
controller_->update(rclcpp::Time(0), rclcpp::Duration::from_seconds(0.01)), | ||
controller_interface::return_type::OK); | ||
|
||
// check command in handle was set | ||
ASSERT_EQ(joint_1_pos_cmd_.get_value(), 10.0); | ||
ASSERT_EQ(joint_2_pos_cmd_.get_value(), 20.0); | ||
ASSERT_EQ(joint_3_pos_cmd_.get_value(), 30.0); | ||
} | ||
|
||
TEST_F(PassthroughControllerTest, ActivateDeactivateCommandsResetSuccess) | ||
{ | ||
SetUpController(); | ||
|
||
// configure controller | ||
controller_->get_node()->set_parameter( | ||
{"interfaces", | ||
std::vector<std::string>{"joint1/interface", "joint2/interface", "joint3/interface"}}); | ||
|
||
// default values | ||
ASSERT_EQ(joint_1_pos_cmd_.get_value(), 1.1); | ||
ASSERT_EQ(joint_2_pos_cmd_.get_value(), 2.1); | ||
ASSERT_EQ(joint_3_pos_cmd_.get_value(), 3.1); | ||
|
||
auto node_state = controller_->configure(); | ||
ASSERT_EQ(node_state.id(), lifecycle_msgs::msg::State::PRIMARY_STATE_INACTIVE); | ||
|
||
node_state = controller_->get_node()->activate(); | ||
ASSERT_EQ(node_state.id(), lifecycle_msgs::msg::State::PRIMARY_STATE_ACTIVE); | ||
|
||
auto command_msg = std::make_shared<std_msgs::msg::Float64MultiArray>(); | ||
command_msg->data = {10.0, 20.0, 30.0}; | ||
|
||
controller_->rt_buffer_ptr_.writeFromNonRT(command_msg); | ||
|
||
// update successful | ||
ASSERT_EQ( | ||
controller_->update(rclcpp::Time(0), rclcpp::Duration::from_seconds(0.01)), | ||
controller_interface::return_type::OK); | ||
|
||
// check command in handle was set | ||
ASSERT_EQ(joint_1_pos_cmd_.get_value(), 10); | ||
ASSERT_EQ(joint_2_pos_cmd_.get_value(), 20); | ||
ASSERT_EQ(joint_3_pos_cmd_.get_value(), 30); | ||
|
||
node_state = controller_->get_node()->deactivate(); | ||
ASSERT_EQ(node_state.id(), lifecycle_msgs::msg::State::PRIMARY_STATE_INACTIVE); | ||
|
||
// command ptr should be reset (nullptr) after deactivation - same check as in `update` | ||
ASSERT_FALSE( | ||
controller_->rt_buffer_ptr_.readFromNonRT() && *(controller_->rt_buffer_ptr_.readFromNonRT())); | ||
ASSERT_FALSE( | ||
controller_->rt_buffer_ptr_.readFromRT() && *(controller_->rt_buffer_ptr_.readFromRT())); | ||
|
||
// Controller is inactive but let's put some data into buffer (simulate callback when inactive) | ||
command_msg = std::make_shared<std_msgs::msg::Float64MultiArray>(); | ||
command_msg->data = {5.5, 6.6, 7.7}; | ||
|
||
controller_->rt_buffer_ptr_.writeFromNonRT(command_msg); | ||
|
||
// command ptr should be available and message should be there - same check as in `update` | ||
ASSERT_TRUE( | ||
controller_->rt_buffer_ptr_.readFromNonRT() && *(controller_->rt_buffer_ptr_.readFromNonRT())); | ||
ASSERT_TRUE( | ||
controller_->rt_buffer_ptr_.readFromRT() && *(controller_->rt_buffer_ptr_.readFromRT())); | ||
|
||
// Now activate again | ||
node_state = controller_->get_node()->activate(); | ||
ASSERT_EQ(node_state.id(), lifecycle_msgs::msg::State::PRIMARY_STATE_ACTIVE); | ||
|
||
// command ptr should be reset (nullptr) after activation - same check as in `update` | ||
ASSERT_FALSE( | ||
controller_->rt_buffer_ptr_.readFromNonRT() && *(controller_->rt_buffer_ptr_.readFromNonRT())); | ||
ASSERT_FALSE( | ||
controller_->rt_buffer_ptr_.readFromRT() && *(controller_->rt_buffer_ptr_.readFromRT())); | ||
|
||
// update successful | ||
ASSERT_EQ( | ||
controller_->update(rclcpp::Time(0), rclcpp::Duration::from_seconds(0.01)), | ||
controller_interface::return_type::OK); | ||
|
||
// values should not change | ||
ASSERT_EQ(joint_1_pos_cmd_.get_value(), 10); | ||
ASSERT_EQ(joint_2_pos_cmd_.get_value(), 20); | ||
ASSERT_EQ(joint_3_pos_cmd_.get_value(), 30); | ||
|
||
// set commands again | ||
controller_->rt_buffer_ptr_.writeFromNonRT(command_msg); | ||
|
||
// update successful | ||
ASSERT_EQ( | ||
controller_->update(rclcpp::Time(0), rclcpp::Duration::from_seconds(0.01)), | ||
controller_interface::return_type::OK); | ||
|
||
// check command in handle was set | ||
ASSERT_EQ(joint_1_pos_cmd_.get_value(), 5.5); | ||
ASSERT_EQ(joint_2_pos_cmd_.get_value(), 6.6); | ||
ASSERT_EQ(joint_3_pos_cmd_.get_value(), 7.7); | ||
} |
Oops, something went wrong.