moveit2
The MoveIt Motion Planning Framework for ROS 2.
execute_trajectory_action_capability.cpp
Go to the documentation of this file.
1 /*********************************************************************
2  * Software License Agreement (BSD License)
3  *
4  * Copyright (c) 2016, Kentaro Wada.
5  * All rights reserved.
6  *
7  * Redistribution and use in source and binary forms, with or without
8  * modification, are permitted provided that the following conditions
9  * are met:
10  *
11  * * Redistributions of source code must retain the above copyright
12  * notice, this list of conditions and the following disclaimer.
13  * * Redistributions in binary form must reproduce the above
14  * copyright notice, this list of conditions and the following
15  * disclaimer in the documentation and/or other materials provided
16  * with the distribution.
17  * * Neither the name of Willow Garage nor the names of its
18  * contributors may be used to endorse or promote products derived
19  * from this software without specific prior written permission.
20  *
21  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
22  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
23  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
24  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
25  * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
26  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
27  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
28  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
29  * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
30  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
31  * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
32  * POSSIBILITY OF SUCH DAMAGE.
33  *********************************************************************/
34 
35 /* Author: Kentaro Wada */
36 
38 
44 
45 namespace move_group
46 {
47 static const rclcpp::Logger LOGGER =
48  rclcpp::get_logger("moveit_move_group_default_capabilities.execute_trajectory_action_capability");
49 
51 {
52 }
53 
55 {
56  callback_executor_.cancel();
57 
58  if (callback_thread_.joinable())
59  callback_thread_.join();
60 }
61 
63 {
64  using std::placeholders::_1;
65  using std::placeholders::_2;
66 
67  auto node = context_->moveit_cpp_->getNode();
68  callback_group_ = node->create_callback_group(rclcpp::CallbackGroupType::MutuallyExclusive,
69  false /* don't spin with node executor */);
70  callback_executor_.add_callback_group(callback_group_, node->get_node_base_interface());
71  callback_thread_ = std::thread([this]() { callback_executor_.spin(); });
72  // start the move action server
73  execute_action_server_ = rclcpp_action::create_server<ExecTrajectory>(
74  node->get_node_base_interface(), node->get_node_clock_interface(), node->get_node_logging_interface(),
75  node->get_node_waitables_interface(), EXECUTE_ACTION_NAME,
76  [](const rclcpp_action::GoalUUID& /*unused*/, const std::shared_ptr<const ExecTrajectory::Goal>& /*unused*/) {
77  RCLCPP_INFO(LOGGER, "Received goal request");
78  return rclcpp_action::GoalResponse::ACCEPT_AND_EXECUTE;
79  },
80  [](const std::shared_ptr<ExecTrajectoryGoal>& /* unused */) {
81  RCLCPP_INFO(LOGGER, "Received request to cancel goal");
82  return rclcpp_action::CancelResponse::ACCEPT;
83  },
84  [this](const auto& goal) { executePathCallback(goal); }, rcl_action_server_get_default_options(),
85  callback_group_);
86 }
87 
88 void MoveGroupExecuteTrajectoryAction::executePathCallback(const std::shared_ptr<ExecTrajectoryGoal>& goal)
89 {
90  auto action_res = std::make_shared<ExecTrajectory::Result>();
91  if (!context_->trajectory_execution_manager_)
92  {
93  const std::string response = "Cannot execute trajectory since ~allow_trajectory_execution was set to false";
94  action_res->error_code.val = moveit_msgs::msg::MoveItErrorCodes::CONTROL_FAILED;
95  goal->abort(action_res);
96  return;
97  }
98 
99  executePath(goal, action_res);
100 
101  const std::string response = getActionResultString(action_res->error_code, false, false);
102  auto fb = std::make_shared<ExecTrajectory::Feedback>();
103  fb->state = response;
104  if (action_res->error_code.val == moveit_msgs::msg::MoveItErrorCodes::SUCCESS)
105  {
106  goal->publish_feedback(fb);
107  goal->succeed(action_res);
108  }
109  else
110  {
111  goal->publish_feedback(fb);
112  goal->abort(action_res);
113  }
114 
115  setExecuteTrajectoryState(IDLE, goal);
116 }
117 
118 void MoveGroupExecuteTrajectoryAction::executePath(const std::shared_ptr<ExecTrajectoryGoal>& goal,
119  std::shared_ptr<ExecTrajectory::Result>& action_res)
120 {
121  RCLCPP_INFO(LOGGER, "Execution request received");
122 
123  if (context_->trajectory_execution_manager_->push(goal->get_goal()->trajectory))
124  {
125  setExecuteTrajectoryState(MONITOR, goal);
126  context_->trajectory_execution_manager_->execute();
127  moveit_controller_manager::ExecutionStatus status = context_->trajectory_execution_manager_->waitForExecution();
129  {
130  action_res->error_code.val = moveit_msgs::msg::MoveItErrorCodes::SUCCESS;
131  }
133  {
134  action_res->error_code.val = moveit_msgs::msg::MoveItErrorCodes::PREEMPTED;
135  }
137  {
138  action_res->error_code.val = moveit_msgs::msg::MoveItErrorCodes::TIMED_OUT;
139  }
140  else
141  {
142  action_res->error_code.val = moveit_msgs::msg::MoveItErrorCodes::CONTROL_FAILED;
143  }
144  RCLCPP_INFO_STREAM(LOGGER, "Execution completed: " << status.asString());
145  }
146  else
147  {
148  action_res->error_code.val = moveit_msgs::msg::MoveItErrorCodes::CONTROL_FAILED;
149  }
150 }
151 
152 void MoveGroupExecuteTrajectoryAction::preemptExecuteTrajectoryCallback()
153 {
154  context_->trajectory_execution_manager_->stopExecution(true);
155 }
156 
157 void MoveGroupExecuteTrajectoryAction::setExecuteTrajectoryState(MoveGroupState state,
158  const std::shared_ptr<ExecTrajectoryGoal>& goal)
159 {
160  auto execute_feedback = std::make_shared<ExecTrajectory::Feedback>();
161  execute_feedback->state = stateToStr(state);
162  goal->publish_feedback(execute_feedback);
163 }
164 
165 } // namespace move_group
166 
167 #include <pluginlib/class_list_macros.hpp>
168 
PLUGINLIB_EXPORT_CLASS(cached_ik_kinematics_plugin::CachedIKKinematicsPlugin< kdl_kinematics_plugin::KDLKinematicsPlugin >, kinematics::KinematicsBase)
std::string getActionResultString(const moveit_msgs::msg::MoveItErrorCodes &error_code, bool planned_trajectory_empty, bool plan_only)
std::string stateToStr(MoveGroupState state) const
const rclcpp::Logger LOGGER
Definition: async_test.h:31
std::string asString() const
Convert the execution status to a string.