* reorder events executor constructor arguments to have the first args match the other executors Signed-off-by: Alberto Soragna <alberto.soragna@gmail.com> * deprecate rclcpp::spin_some and rclcpp::spin_all Signed-off-by: Alberto Soragna <alberto.soragna@gmail.com> * use rcpputils to ignore deprecation and remove deprecated APIs from comments Signed-off-by: Alberto Soragna <alberto.soragna@gmail.com> * fix rclcpp_lifecycle build failures. Signed-off-by: Tomoya Fujita <Tomoya.Fujita@sony.com> * make linters happy Signed-off-by: Alejandro Hernandez Cordero <ahcorde@gmail.com> * fix test_service_introspection Signed-off-by: Alberto Soragna <alberto.soragna@gmail.com> * fix lifecycle unit-test Signed-off-by: Alberto Soragna <alberto.soragna@gmail.com> --------- Signed-off-by: Alberto Soragna <alberto.soragna@gmail.com> Signed-off-by: Tomoya Fujita <Tomoya.Fujita@sony.com> Signed-off-by: Alejandro Hernandez Cordero <ahcorde@gmail.com> Co-authored-by: Tomoya Fujita <Tomoya.Fujita@sony.com> Co-authored-by: Alejandro Hernandez Cordero <ahcorde@gmail.com>
486 lines
17 KiB
C++
486 lines
17 KiB
C++
// Copyright 2020 Open Source Robotics Foundation, Inc.
|
|
//
|
|
// 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.
|
|
|
|
/**
|
|
* Service client test was adopted from:
|
|
* https://github.com/ros2/demos/blob/master/lifecycle/src/lifecycle_service_client.cpp
|
|
*/
|
|
|
|
#include <gtest/gtest.h>
|
|
#include <algorithm>
|
|
#include <array>
|
|
#include <chrono>
|
|
#include <memory>
|
|
#include <mutex>
|
|
#include <string>
|
|
#include <thread>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
#include "lifecycle_msgs/msg/state.hpp"
|
|
#include "lifecycle_msgs/msg/transition.hpp"
|
|
#include "lifecycle_msgs/msg/transition_description.hpp"
|
|
#include "lifecycle_msgs/srv/change_state.hpp"
|
|
#include "lifecycle_msgs/srv/get_available_states.hpp"
|
|
#include "lifecycle_msgs/srv/get_available_transitions.hpp"
|
|
#include "lifecycle_msgs/srv/get_state.hpp"
|
|
|
|
#include "rcl_lifecycle/rcl_lifecycle.h"
|
|
|
|
#include "rclcpp/node_interfaces/node_graph.hpp"
|
|
#include "rclcpp/rclcpp.hpp"
|
|
#include "rclcpp_lifecycle/lifecycle_node.hpp"
|
|
|
|
#include "rcpputils/scope_exit.hpp"
|
|
|
|
#include "./mocking_utils/patch.hpp"
|
|
|
|
using namespace std::chrono_literals;
|
|
|
|
constexpr char const * lifecycle_node_name = "lc_talker";
|
|
|
|
constexpr char const * node_get_state_topic = "/lc_talker/get_state";
|
|
constexpr char const * node_change_state_topic = "/lc_talker/change_state";
|
|
constexpr char const * node_get_available_states_topic = "/lc_talker/get_available_states";
|
|
constexpr char const * node_get_available_transitions_topic =
|
|
"/lc_talker/get_available_transitions";
|
|
constexpr char const * node_get_transition_graph_topic =
|
|
"/lc_talker/get_transition_graph";
|
|
const lifecycle_msgs::msg::State unknown_state = lifecycle_msgs::msg::State();
|
|
|
|
class EmptyLifecycleNode : public rclcpp_lifecycle::LifecycleNode
|
|
{
|
|
public:
|
|
EmptyLifecycleNode()
|
|
: rclcpp_lifecycle::LifecycleNode(lifecycle_node_name)
|
|
{}
|
|
};
|
|
|
|
class LifecycleServiceClient : public rclcpp::Node
|
|
{
|
|
public:
|
|
explicit LifecycleServiceClient(std::string node_name)
|
|
: Node(node_name)
|
|
{
|
|
client_get_available_states_ = this->create_client<lifecycle_msgs::srv::GetAvailableStates>(
|
|
node_get_available_states_topic);
|
|
client_get_available_transitions_ =
|
|
this->create_client<lifecycle_msgs::srv::GetAvailableTransitions>(
|
|
node_get_available_transitions_topic);
|
|
client_get_transition_graph_ =
|
|
this->create_client<lifecycle_msgs::srv::GetAvailableTransitions>(
|
|
node_get_transition_graph_topic);
|
|
client_get_state_ = this->create_client<lifecycle_msgs::srv::GetState>(
|
|
node_get_state_topic);
|
|
client_change_state_ = this->create_client<lifecycle_msgs::srv::ChangeState>(
|
|
node_change_state_topic);
|
|
}
|
|
|
|
lifecycle_msgs::msg::State
|
|
get_state(std::chrono::seconds time_out = 1s)
|
|
{
|
|
auto request = std::make_shared<lifecycle_msgs::srv::GetState::Request>();
|
|
|
|
if (!client_get_state_->wait_for_service(time_out)) {
|
|
return unknown_state;
|
|
}
|
|
|
|
auto future_result = client_get_state_->async_send_request(request);
|
|
auto future_status = future_result.wait_for(time_out);
|
|
|
|
if (future_status != std::future_status::ready) {
|
|
return unknown_state;
|
|
}
|
|
|
|
auto result = future_result.get();
|
|
if (result) {
|
|
return result->current_state;
|
|
} else {
|
|
return unknown_state;
|
|
}
|
|
}
|
|
|
|
bool
|
|
change_state(std::uint8_t transition, std::chrono::seconds time_out = 1s)
|
|
{
|
|
auto request = std::make_shared<lifecycle_msgs::srv::ChangeState::Request>();
|
|
request->transition.id = transition;
|
|
|
|
if (!client_change_state_->wait_for_service(time_out)) {
|
|
return false;
|
|
}
|
|
|
|
auto future_result = client_change_state_->async_send_request(request);
|
|
auto future_status = future_result.wait_for(time_out);
|
|
|
|
if (future_status != std::future_status::ready) {
|
|
return false;
|
|
}
|
|
|
|
return future_result.get()->success;
|
|
}
|
|
|
|
std::vector<lifecycle_msgs::msg::State>
|
|
get_available_states(std::chrono::seconds time_out = 1s)
|
|
{
|
|
auto request = std::make_shared<lifecycle_msgs::srv::GetAvailableStates::Request>();
|
|
|
|
if (!client_get_available_states_->wait_for_service(time_out)) {
|
|
return std::vector<lifecycle_msgs::msg::State>();
|
|
}
|
|
|
|
auto future_result = client_get_available_states_->async_send_request(request);
|
|
auto future_status = future_result.wait_for(time_out);
|
|
|
|
if (future_status != std::future_status::ready) {
|
|
return std::vector<lifecycle_msgs::msg::State>();
|
|
}
|
|
auto result = future_result.get();
|
|
if (result) {
|
|
return result->available_states;
|
|
}
|
|
|
|
return std::vector<lifecycle_msgs::msg::State>();
|
|
}
|
|
|
|
std::vector<lifecycle_msgs::msg::TransitionDescription>
|
|
get_available_transitions(std::chrono::seconds time_out = 1s)
|
|
{
|
|
auto request = std::make_shared<lifecycle_msgs::srv::GetAvailableTransitions::Request>();
|
|
|
|
if (!client_get_available_transitions_->wait_for_service(time_out)) {
|
|
return std::vector<lifecycle_msgs::msg::TransitionDescription>();
|
|
}
|
|
|
|
auto future_result = client_get_available_transitions_->async_send_request(request);
|
|
auto future_status = future_result.wait_for(time_out);
|
|
|
|
if (future_status != std::future_status::ready) {
|
|
return std::vector<lifecycle_msgs::msg::TransitionDescription>();
|
|
}
|
|
|
|
auto result = future_result.get();
|
|
if (result) {
|
|
return result->available_transitions;
|
|
}
|
|
|
|
return std::vector<lifecycle_msgs::msg::TransitionDescription>();
|
|
}
|
|
|
|
std::vector<lifecycle_msgs::msg::TransitionDescription>
|
|
get_transition_graph(std::chrono::seconds time_out = 1s)
|
|
{
|
|
auto request = std::make_shared<lifecycle_msgs::srv::GetAvailableTransitions::Request>();
|
|
|
|
if (!client_get_transition_graph_->wait_for_service(time_out)) {
|
|
return std::vector<lifecycle_msgs::msg::TransitionDescription>();
|
|
}
|
|
|
|
auto future_result = client_get_transition_graph_->async_send_request(request);
|
|
auto future_status = future_result.wait_for(time_out);
|
|
|
|
if (future_status != std::future_status::ready) {
|
|
return std::vector<lifecycle_msgs::msg::TransitionDescription>();
|
|
}
|
|
|
|
auto result = future_result.get();
|
|
if (result) {
|
|
return result->available_transitions;
|
|
}
|
|
|
|
return std::vector<lifecycle_msgs::msg::TransitionDescription>();
|
|
}
|
|
|
|
private:
|
|
std::shared_ptr<rclcpp::Client<lifecycle_msgs::srv::GetAvailableStates>>
|
|
client_get_available_states_;
|
|
std::shared_ptr<rclcpp::Client<lifecycle_msgs::srv::GetAvailableTransitions>>
|
|
client_get_available_transitions_;
|
|
std::shared_ptr<rclcpp::Client<lifecycle_msgs::srv::GetAvailableTransitions>>
|
|
client_get_transition_graph_;
|
|
std::shared_ptr<rclcpp::Client<lifecycle_msgs::srv::GetState>> client_get_state_;
|
|
std::shared_ptr<rclcpp::Client<lifecycle_msgs::srv::ChangeState>> client_change_state_;
|
|
};
|
|
|
|
|
|
class TestLifecycleServiceClient : public ::testing::Test
|
|
{
|
|
protected:
|
|
EmptyLifecycleNode * lifecycle_node() {return lifecycle_node_.get();}
|
|
LifecycleServiceClient * lifecycle_client() {return lifecycle_client_.get();}
|
|
|
|
private:
|
|
void SetUp() override
|
|
{
|
|
rclcpp::init(0, nullptr);
|
|
lifecycle_node_ = std::make_shared<EmptyLifecycleNode>();
|
|
lifecycle_client_ = std::make_shared<LifecycleServiceClient>("client");
|
|
|
|
executor_ = std::make_shared<rclcpp::executors::SingleThreadedExecutor>();
|
|
spinner_ = std::thread([this]() {
|
|
executor_->add_node(lifecycle_node_->get_node_base_interface());
|
|
executor_->add_node(lifecycle_client_);
|
|
executor_->spin();
|
|
});
|
|
}
|
|
|
|
void TearDown() override
|
|
{
|
|
while(!executor_->is_spinning()) {
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(1));
|
|
}
|
|
executor_->cancel();
|
|
spinner_.join();
|
|
rclcpp::shutdown();
|
|
}
|
|
|
|
std::shared_ptr<EmptyLifecycleNode> lifecycle_node_;
|
|
std::shared_ptr<LifecycleServiceClient> lifecycle_client_;
|
|
std::shared_ptr<rclcpp::executors::SingleThreadedExecutor> executor_;
|
|
std::thread spinner_;
|
|
};
|
|
|
|
|
|
TEST_F(TestLifecycleServiceClient, construct_destruct) {
|
|
EXPECT_NE(nullptr, lifecycle_client());
|
|
EXPECT_NE(nullptr, lifecycle_node());
|
|
}
|
|
|
|
TEST_F(TestLifecycleServiceClient, available_states) {
|
|
auto states = lifecycle_client()->get_available_states();
|
|
EXPECT_EQ(states.size(), 11u);
|
|
EXPECT_EQ(states[0].id, lifecycle_msgs::msg::State::PRIMARY_STATE_UNKNOWN);
|
|
EXPECT_EQ(states[1].id, lifecycle_msgs::msg::State::PRIMARY_STATE_UNCONFIGURED);
|
|
EXPECT_EQ(states[2].id, lifecycle_msgs::msg::State::PRIMARY_STATE_INACTIVE);
|
|
EXPECT_EQ(states[3].id, lifecycle_msgs::msg::State::PRIMARY_STATE_ACTIVE);
|
|
EXPECT_EQ(states[4].id, lifecycle_msgs::msg::State::PRIMARY_STATE_FINALIZED);
|
|
EXPECT_EQ(states[5].id, lifecycle_msgs::msg::State::TRANSITION_STATE_CONFIGURING);
|
|
EXPECT_EQ(states[6].id, lifecycle_msgs::msg::State::TRANSITION_STATE_CLEANINGUP);
|
|
EXPECT_EQ(states[7].id, lifecycle_msgs::msg::State::TRANSITION_STATE_SHUTTINGDOWN);
|
|
EXPECT_EQ(states[8].id, lifecycle_msgs::msg::State::TRANSITION_STATE_ACTIVATING);
|
|
EXPECT_EQ(states[9].id, lifecycle_msgs::msg::State::TRANSITION_STATE_DEACTIVATING);
|
|
EXPECT_EQ(states[10].id, lifecycle_msgs::msg::State::TRANSITION_STATE_ERRORPROCESSING);
|
|
}
|
|
|
|
TEST_F(TestLifecycleServiceClient, transition_graph) {
|
|
auto transitions = lifecycle_client()->get_transition_graph();
|
|
EXPECT_EQ(transitions.size(), 25u);
|
|
}
|
|
|
|
TEST_F(TestLifecycleServiceClient, available_transitions) {
|
|
auto transitions = lifecycle_client()->get_available_transitions();
|
|
EXPECT_EQ(transitions.size(), 2u);
|
|
EXPECT_EQ(transitions[0].transition.id, lifecycle_msgs::msg::Transition::TRANSITION_CONFIGURE);
|
|
EXPECT_EQ(
|
|
transitions[1].transition.id,
|
|
lifecycle_msgs::msg::Transition::TRANSITION_UNCONFIGURED_SHUTDOWN);
|
|
}
|
|
|
|
TEST_F(TestLifecycleServiceClient, lifecycle_transitions) {
|
|
EXPECT_EQ(
|
|
lifecycle_client()->get_state().id, lifecycle_msgs::msg::State::PRIMARY_STATE_UNCONFIGURED);
|
|
|
|
auto transitions = lifecycle_client()->get_available_transitions();
|
|
EXPECT_EQ(transitions.size(), 2u);
|
|
EXPECT_EQ(transitions[0].transition.id, lifecycle_msgs::msg::Transition::TRANSITION_CONFIGURE);
|
|
EXPECT_EQ(
|
|
transitions[1].transition.id,
|
|
lifecycle_msgs::msg::Transition::TRANSITION_UNCONFIGURED_SHUTDOWN);
|
|
|
|
EXPECT_TRUE(
|
|
lifecycle_client()->change_state(
|
|
lifecycle_msgs::msg::Transition::TRANSITION_CONFIGURE));
|
|
EXPECT_EQ(
|
|
lifecycle_client()->get_state().id,
|
|
lifecycle_msgs::msg::State::PRIMARY_STATE_INACTIVE);
|
|
transitions = lifecycle_client()->get_available_transitions();
|
|
EXPECT_EQ(transitions.size(), 3u);
|
|
EXPECT_EQ(transitions[0].transition.id, lifecycle_msgs::msg::Transition::TRANSITION_CLEANUP);
|
|
EXPECT_EQ(transitions[1].transition.id, lifecycle_msgs::msg::Transition::TRANSITION_ACTIVATE);
|
|
EXPECT_EQ(
|
|
transitions[2].transition.id,
|
|
lifecycle_msgs::msg::Transition::TRANSITION_INACTIVE_SHUTDOWN);
|
|
|
|
EXPECT_TRUE(
|
|
lifecycle_client()->change_state(
|
|
lifecycle_msgs::msg::Transition::TRANSITION_ACTIVATE));
|
|
EXPECT_EQ(lifecycle_client()->get_state().id, lifecycle_msgs::msg::State::PRIMARY_STATE_ACTIVE);
|
|
transitions = lifecycle_client()->get_available_transitions();
|
|
EXPECT_EQ(transitions.size(), 2u);
|
|
EXPECT_EQ(transitions[0].transition.id, lifecycle_msgs::msg::Transition::TRANSITION_DEACTIVATE);
|
|
EXPECT_EQ(
|
|
transitions[1].transition.id,
|
|
lifecycle_msgs::msg::Transition::TRANSITION_ACTIVE_SHUTDOWN);
|
|
|
|
EXPECT_TRUE(
|
|
lifecycle_client()->change_state(
|
|
lifecycle_msgs::msg::Transition::
|
|
TRANSITION_DEACTIVATE));
|
|
EXPECT_EQ(
|
|
lifecycle_client()->get_state().id,
|
|
lifecycle_msgs::msg::State::PRIMARY_STATE_INACTIVE);
|
|
transitions = lifecycle_client()->get_available_transitions();
|
|
EXPECT_EQ(transitions.size(), 3u);
|
|
EXPECT_EQ(transitions[0].transition.id, lifecycle_msgs::msg::Transition::TRANSITION_CLEANUP);
|
|
EXPECT_EQ(transitions[1].transition.id, lifecycle_msgs::msg::Transition::TRANSITION_ACTIVATE);
|
|
EXPECT_EQ(
|
|
transitions[2].transition.id,
|
|
lifecycle_msgs::msg::Transition::TRANSITION_INACTIVE_SHUTDOWN);
|
|
|
|
EXPECT_TRUE(
|
|
lifecycle_client()->change_state(
|
|
lifecycle_msgs::msg::Transition::TRANSITION_CLEANUP));
|
|
EXPECT_EQ(
|
|
lifecycle_client()->get_state().id, lifecycle_msgs::msg::State::PRIMARY_STATE_UNCONFIGURED);
|
|
transitions = lifecycle_client()->get_available_transitions();
|
|
EXPECT_EQ(transitions.size(), 2u);
|
|
EXPECT_EQ(transitions[0].transition.id, lifecycle_msgs::msg::Transition::TRANSITION_CONFIGURE);
|
|
EXPECT_EQ(
|
|
transitions[1].transition.id,
|
|
lifecycle_msgs::msg::Transition::TRANSITION_UNCONFIGURED_SHUTDOWN);
|
|
|
|
EXPECT_TRUE(
|
|
lifecycle_client()->change_state(
|
|
lifecycle_msgs::msg::Transition::
|
|
TRANSITION_UNCONFIGURED_SHUTDOWN));
|
|
EXPECT_EQ(
|
|
lifecycle_client()->get_state().id,
|
|
lifecycle_msgs::msg::State::PRIMARY_STATE_FINALIZED);
|
|
transitions = lifecycle_client()->get_available_transitions();
|
|
EXPECT_EQ(transitions.size(), 0u);
|
|
}
|
|
|
|
TEST_F(TestLifecycleServiceClient, get_service_names_and_types_by_node)
|
|
{
|
|
EXPECT_THROW(
|
|
lifecycle_node()->get_service_names_and_types_by_node("not_a_node", "not_absolute_namespace"),
|
|
std::runtime_error);
|
|
auto service_names_and_types =
|
|
lifecycle_node()->get_service_names_and_types_by_node(lifecycle_node_name, "/");
|
|
auto start = std::chrono::steady_clock::now();
|
|
while (0 == service_names_and_types.size() ||
|
|
(std::chrono::steady_clock::now() - start) < std::chrono::seconds(1))
|
|
{
|
|
service_names_and_types =
|
|
lifecycle_node()->get_service_names_and_types_by_node(lifecycle_node_name, "/");
|
|
}
|
|
const std::array services = {
|
|
std::make_pair(node_get_state_topic, "lifecycle_msgs/srv/GetState"),
|
|
std::make_pair(node_change_state_topic, "lifecycle_msgs/srv/ChangeState"),
|
|
std::make_pair(node_get_available_states_topic, "lifecycle_msgs/srv/GetAvailableStates"),
|
|
std::make_pair(
|
|
node_get_available_transitions_topic, "lifecycle_msgs/srv/GetAvailableTransitions"),
|
|
std::make_pair(node_get_transition_graph_topic, "lifecycle_msgs/srv/GetAvailableTransitions"),
|
|
};
|
|
for (const auto & [service_name, service_type] : services) {
|
|
ASSERT_TRUE(service_names_and_types.find(service_name) != service_names_and_types.end())
|
|
<< service_name;
|
|
const auto service_types = service_names_and_types.at(service_name);
|
|
EXPECT_TRUE(
|
|
std::find(service_types.cbegin(), service_types.cend(), service_type) != service_types.cend())
|
|
<< service_name;
|
|
}
|
|
}
|
|
|
|
TEST_F(TestLifecycleServiceClient, declare_parameter_with_no_initial_values)
|
|
{
|
|
auto node1 = std::make_shared<LifecycleServiceClient>("client1");
|
|
|
|
auto on_set_parameters =
|
|
[](const std::vector<rclcpp::Parameter> &) {
|
|
rcl_interfaces::msg::SetParametersResult result;
|
|
result.successful = true;
|
|
return result;
|
|
};
|
|
|
|
auto handler = node1->add_on_set_parameters_callback(on_set_parameters);
|
|
RCPPUTILS_SCOPE_EXIT(
|
|
{node1->remove_on_set_parameters_callback(handler.get());}); // always reset
|
|
}
|
|
|
|
TEST_F(TestLifecycleServiceClient, wait_for_graph_change)
|
|
{
|
|
auto node = std::make_shared<LifecycleServiceClient>("client_wait_for_graph_change");
|
|
auto node_graph = node->get_node_graph_interface();
|
|
ASSERT_NE(nullptr, node_graph);
|
|
|
|
EXPECT_NO_THROW(node_graph->notify_graph_change());
|
|
EXPECT_THROW(
|
|
node_graph->wait_for_graph_change(nullptr, std::chrono::milliseconds(1)),
|
|
rclcpp::exceptions::InvalidEventError);
|
|
|
|
auto event = std::make_shared<rclcpp::Event>();
|
|
EXPECT_THROW(
|
|
node_graph->wait_for_graph_change(event, std::chrono::milliseconds(0)),
|
|
rclcpp::exceptions::EventNotRegisteredError);
|
|
}
|
|
|
|
class TestLifecycleServiceClientRCLErrors : public ::testing::Test
|
|
{
|
|
protected:
|
|
void SetUp() override
|
|
{
|
|
rclcpp::init(0, nullptr);
|
|
}
|
|
|
|
void TearDown() override
|
|
{
|
|
rclcpp::shutdown();
|
|
}
|
|
};
|
|
|
|
TEST_F(TestLifecycleServiceClientRCLErrors, call_services_rcl_errors) {
|
|
auto lifecycle_node = std::make_shared<EmptyLifecycleNode>();
|
|
auto lifecycle_client = std::make_shared<LifecycleServiceClient>("client_with_errors");
|
|
|
|
auto mock = mocking_utils::patch_and_return(
|
|
"lib:rclcpp_lifecycle", rcl_lifecycle_state_machine_is_initialized, RCL_RET_ERROR);
|
|
|
|
rclcpp::executors::SingleThreadedExecutor lifecycle_node_executor;
|
|
lifecycle_node_executor.add_node(lifecycle_node->get_node_base_interface());
|
|
rclcpp::executors::SingleThreadedExecutor client_executor;
|
|
client_executor.add_node(lifecycle_client);
|
|
|
|
// on_change_state
|
|
lifecycle_client->change_state(
|
|
lifecycle_msgs::msg::Transition::TRANSITION_CONFIGURE);
|
|
client_executor.spin_some();
|
|
EXPECT_THROW(
|
|
lifecycle_node_executor.spin_some(), std::runtime_error);
|
|
|
|
// on_get_state
|
|
lifecycle_client->get_state();
|
|
client_executor.spin_some();
|
|
EXPECT_THROW(
|
|
lifecycle_node_executor.spin_some(), std::runtime_error);
|
|
|
|
// on_get_avilable_states
|
|
lifecycle_client->get_available_states();
|
|
client_executor.spin_some();
|
|
EXPECT_THROW(
|
|
lifecycle_node_executor.spin_some(), std::runtime_error);
|
|
|
|
// on_get_available_transitions
|
|
lifecycle_client->get_available_transitions();
|
|
client_executor.spin_some();
|
|
EXPECT_THROW(
|
|
lifecycle_node_executor.spin_some(), std::runtime_error);
|
|
|
|
// on_get_transition_graph
|
|
lifecycle_client->get_transition_graph();
|
|
client_executor.spin_some();
|
|
EXPECT_THROW(
|
|
lifecycle_node_executor.spin_some(), std::runtime_error);
|
|
}
|