moveit2
The MoveIt Motion Planning Framework for ROS 2.
cartesian_path_service_capability.cpp
Go to the documentation of this file.
1 /*********************************************************************
2  * Software License Agreement (BSD License)
3  *
4  * Copyright (c) 2012, Willow Garage, Inc.
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: Ioan Sucan */
36 
42 #include <tf2_eigen/tf2_eigen.hpp>
47 #include <moveit_msgs/msg/display_trajectory.hpp>
49 #include <moveit/utils/logger.hpp>
50 
51 using moveit::getLogger;
52 
53 namespace
54 {
55 
57 const std::string DISPLAY_PATH_TOPIC = std::string("display_planned_path");
58 
59 bool isStateValid(const planning_scene::PlanningScene* planning_scene,
61  const moveit::core::JointModelGroup* group, const double* ik_solution)
62 {
63  state->setJointGroupPositions(group, ik_solution);
64  state->update();
65  return (!planning_scene || !planning_scene->isStateColliding(*state, group->getName())) &&
66  (!constraint_set || constraint_set->decide(*state).satisfied);
67 }
68 
69 rclcpp::Logger getLogger()
70 {
71  return moveit::getLogger("moveit.ros.move_group.cartesian_path_service_capability");
72 }
73 } // namespace
74 
75 namespace move_group
76 {
78  : MoveGroupCapability("CartesianPathService"), display_computed_paths_(true)
79 {
80 }
81 
83 {
84  display_path_ =
85  context_->moveit_cpp_->getNode()->create_publisher<moveit_msgs::msg::DisplayTrajectory>(DISPLAY_PATH_TOPIC, 10);
86 
87  cartesian_path_service_ = context_->moveit_cpp_->getNode()->create_service<moveit_msgs::srv::GetCartesianPath>(
88 
89  CARTESIAN_PATH_SERVICE_NAME,
90  [this](const std::shared_ptr<rmw_request_id_t>& req_id,
91  const std::shared_ptr<moveit_msgs::srv::GetCartesianPath::Request>& req,
92  const std::shared_ptr<moveit_msgs::srv::GetCartesianPath::Response>& res) -> bool {
93  return computeService(req_id, req, res);
94  });
95 }
96 
97 bool MoveGroupCartesianPathService::computeService(
98  const std::shared_ptr<rmw_request_id_t>& /* unused */,
99  const std::shared_ptr<moveit_msgs::srv::GetCartesianPath::Request>& req,
100  const std::shared_ptr<moveit_msgs::srv::GetCartesianPath::Response>& res)
101 {
102  RCLCPP_INFO(getLogger(), "Received request to compute Cartesian path");
103  context_->planning_scene_monitor_->updateFrameTransforms();
104 
105  moveit::core::RobotState start_state =
106  planning_scene_monitor::LockedPlanningSceneRO(context_->planning_scene_monitor_)->getCurrentState();
107  moveit::core::robotStateMsgToRobotState(req->start_state, start_state);
108  if (const moveit::core::JointModelGroup* jmg = start_state.getJointModelGroup(req->group_name))
109  {
110  std::string link_name = req->link_name;
111  if (link_name.empty() && !jmg->getLinkModelNames().empty())
112  link_name = jmg->getLinkModelNames().back();
113 
114  bool ok = true;
115  EigenSTL::vector_Isometry3d waypoints(req->waypoints.size());
116  const std::string& default_frame = context_->planning_scene_monitor_->getRobotModel()->getModelFrame();
117  bool no_transform = req->header.frame_id.empty() ||
118  moveit::core::Transforms::sameFrame(req->header.frame_id, default_frame) ||
119  moveit::core::Transforms::sameFrame(req->header.frame_id, link_name);
120 
121  for (std::size_t i = 0; i < req->waypoints.size(); ++i)
122  {
123  if (no_transform)
124  {
125  tf2::fromMsg(req->waypoints[i], waypoints[i]);
126  }
127  else
128  {
129  geometry_msgs::msg::PoseStamped p;
130  p.header = req->header;
131  p.pose = req->waypoints[i];
132  if (performTransform(p, default_frame))
133  {
134  tf2::fromMsg(p.pose, waypoints[i]);
135  }
136  else
137  {
138  RCLCPP_ERROR(getLogger(), "Error encountered transforming waypoints to frame '%s'", default_frame.c_str());
139  ok = false;
140  break;
141  }
142  }
143  }
144 
145  if (ok)
146  {
147  if (req->max_step < std::numeric_limits<double>::epsilon())
148  {
149  RCLCPP_ERROR(getLogger(), "Maximum step to take between consecutive configurations along Cartesian path"
150  "was not specified (this value needs to be > 0)");
151  res->error_code.val = moveit_msgs::msg::MoveItErrorCodes::FAILURE;
152  }
153  else
154  {
155  if (!waypoints.empty())
156  {
158  std::unique_ptr<planning_scene_monitor::LockedPlanningSceneRO> ls;
159  std::unique_ptr<kinematic_constraints::KinematicConstraintSet> kset;
160  if (req->avoid_collisions || !moveit::core::isEmpty(req->path_constraints))
161  {
162  ls = std::make_unique<planning_scene_monitor::LockedPlanningSceneRO>(context_->planning_scene_monitor_);
163  kset = std::make_unique<kinematic_constraints::KinematicConstraintSet>((*ls)->getRobotModel());
164  kset->add(req->path_constraints, (*ls)->getTransforms());
165  constraint_fn =
166  [scene = req->avoid_collisions ? static_cast<const planning_scene::PlanningSceneConstPtr&>(*ls).get() :
167  nullptr,
168  kset = kset->empty() ? nullptr : kset.get()](moveit::core::RobotState* robot_state,
169  const moveit::core::JointModelGroup* joint_group,
170  const double* joint_group_variable_values) {
171  return isStateValid(scene, kset, robot_state, joint_group, joint_group_variable_values);
172  };
173  }
174  bool global_frame = !moveit::core::Transforms::sameFrame(link_name, req->header.frame_id);
175  RCLCPP_INFO(getLogger(),
176  "Attempting to follow %u waypoints for link '%s' using a step of %lf m "
177  "and jump threshold %lf (in %s reference frame)",
178  static_cast<unsigned int>(waypoints.size()), link_name.c_str(), req->max_step,
179  req->jump_threshold, global_frame ? "global" : "link");
181  if (req->jump_threshold > 0.0)
182  {
183  jump_threshold = moveit::core::JumpThreshold::relative(req->jump_threshold);
184  }
185  std::vector<moveit::core::RobotStatePtr> traj;
187  &start_state, jmg, traj, start_state.getLinkModel(link_name), waypoints, global_frame,
188  moveit::core::MaxEEFStep(req->max_step), moveit::core::CartesianPrecision{}, constraint_fn);
189  moveit::core::robotStateToRobotStateMsg(start_state, res->start_state);
190 
191  robot_trajectory::RobotTrajectory rt(context_->planning_scene_monitor_->getRobotModel(), req->group_name);
192  for (const moveit::core::RobotStatePtr& traj_state : traj)
193  rt.addSuffixWayPoint(traj_state, 0.0);
194 
195  // time trajectory
196  // \todo optionally compute timing to move the eef with constant speed
198  time_param.computeTimeStamps(rt, req->max_velocity_scaling_factor, req->max_acceleration_scaling_factor);
199 
200  rt.getRobotTrajectoryMsg(res->solution);
201  RCLCPP_INFO(getLogger(), "Computed Cartesian path with %u points (followed %lf%% of requested trajectory)",
202  static_cast<unsigned int>(traj.size()), res->fraction * 100.0);
203  if (display_computed_paths_ && rt.getWayPointCount() > 0)
204  {
205  moveit_msgs::msg::DisplayTrajectory disp;
206  disp.model_id = context_->planning_scene_monitor_->getRobotModel()->getName();
207  disp.trajectory.resize(1, res->solution);
208  moveit::core::robotStateToRobotStateMsg(rt.getFirstWayPoint(), disp.trajectory_start);
209  display_path_->publish(disp);
210  }
211  }
212  res->error_code.val = moveit_msgs::msg::MoveItErrorCodes::SUCCESS;
213  }
214  }
215  else
216  res->error_code.val = moveit_msgs::msg::MoveItErrorCodes::FRAME_TRANSFORM_FAILURE;
217  }
218  else
219  res->error_code.val = moveit_msgs::msg::MoveItErrorCodes::INVALID_GROUP_NAME;
220 
221  return true;
222 }
223 
224 } // namespace move_group
225 
226 #include <pluginlib/class_list_macros.hpp>
227 
PLUGINLIB_EXPORT_CLASS(cached_ik_kinematics_plugin::CachedIKKinematicsPlugin< kdl_kinematics_plugin::KDLKinematicsPlugin >, kinematics::KinematicsBase)
A class that contains many different constraints, and can check RobotState *versus the full set....
ConstraintEvaluationResult decide(const moveit::core::RobotState &state, bool verbose=false) const
Determines whether all constraints are satisfied by state, returning a single evaluation result.
bool performTransform(geometry_msgs::msg::PoseStamped &pose_msg, const std::string &target_frame) const
static Distance computeCartesianPath(const RobotState *start_state, const JointModelGroup *group, std::vector< std::shared_ptr< RobotState >> &traj, const LinkModel *link, const Eigen::Vector3d &translation, bool global_reference_frame, const MaxEEFStep &max_step, const CartesianPrecision &precision, const GroupStateValidityCallbackFn &validCallback=GroupStateValidityCallbackFn(), const kinematics::KinematicsQueryOptions &options=kinematics::KinematicsQueryOptions(), const kinematics::KinematicsBase::IKCostFn &cost_function=kinematics::KinematicsBase::IKCostFn())
Compute the sequence of joint values that correspond to a straight Cartesian path for a particular li...
const std::string & getName() const
Get the name of the joint group.
Representation of a robot's state. This includes position, velocity, acceleration and effort.
Definition: robot_state.hpp:90
const LinkModel * getLinkModel(const std::string &link) const
Get the model of a particular link.
void setJointGroupPositions(const std::string &joint_group_name, const double *gstate)
Given positions for the variables that make up a group, in the order found in the group (including va...
const JointModelGroup * getJointModelGroup(const std::string &group) const
Get the model of a particular joint group.
void update(bool force=false)
Update all transforms.
static bool sameFrame(const std::string &frame1, const std::string &frame2)
Check if two frames end up being the same once the missing / are added as prefix (if they are missing...
Definition: transforms.cpp:70
This class maintains the representation of the environment as seen by a planning instance....
This is a convenience class for obtaining access to an instance of a locked PlanningScene.
Maintain a sequence of waypoints and the time durations between these waypoints.
bool computeTimeStamps(robot_trajectory::RobotTrajectory &trajectory, const double max_velocity_scaling_factor=1.0, const double max_acceleration_scaling_factor=1.0) const override
Compute a trajectory with waypoints spaced equally in time (according to resample_dt_)....
void robotStateToRobotStateMsg(const RobotState &state, moveit_msgs::msg::RobotState &robot_state, bool copy_attached_bodies=true)
Convert a MoveIt robot state to a robot state message.
std::function< bool(RobotState *robot_state, const JointModelGroup *joint_group, const double *joint_group_variable_values)> GroupStateValidityCallbackFn
Signature for functions that can verify that if the group joint_group in robot_state is set to joint_...
Definition: robot_state.hpp:70
bool isEmpty(const moveit_msgs::msg::PlanningScene &msg)
Check if a message includes any information about a planning scene, or whether it is empty.
bool robotStateMsgToRobotState(const Transforms &tf, const moveit_msgs::msg::RobotState &robot_state, RobotState &state, bool copy_attached_bodies=true)
Convert a robot state msg (with accompanying extra transforms) to a MoveIt robot state.
rclcpp::Logger getLogger(const std::string &name)
Creates a namespaced logger.
Definition: logger.cpp:79
This namespace includes the central class for representing planning contexts.
bool satisfied
Whether or not the constraint or constraints were satisfied.
Struct with options for defining joint-space jump thresholds.
static JumpThreshold relative(double relative_factor)
Detect joint-space jumps relative to the average joint-space displacement along the path.
static JumpThreshold disabled()
Do not define any jump threshold, i.e., disable joint-space jump detection.
Struct for containing max_step for computeCartesianPath.