moveit2
The MoveIt Motion Planning Framework for ROS 2.
Loading...
Searching...
No Matches
unittest_planning_context.cpp
Go to the documentation of this file.
1/*********************************************************************
2 * Software License Agreement (BSD License)
3 *
4 * Copyright (c) 2018 Pilz GmbH & Co. KG
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 Pilz GmbH & Co. KG 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#include <boost/core/demangle.hpp>
36#include <gtest/gtest.h>
37
39
44
49
50#include "test_utils.hpp"
51
52// parameters from parameter server
53const std::string PARAM_PLANNING_GROUP_NAME("planning_group");
54const std::string PARAM_TARGET_LINK_NAME("target_link");
55
62template <typename T, int N>
64{
65public:
66 typedef T Type_;
67 static const int VALUE = N;
68};
69template <typename T, int N>
71
78
82
86template <typename T>
87class PlanningContextTest : public ::testing::Test
88{
89protected:
90 void SetUp() override
91 {
92 rclcpp::NodeOptions node_options;
93 node_options.automatically_declare_parameters_from_overrides(true);
94 node_ = rclcpp::Node::make_shared("unittest_planning_context", node_options);
95
96 // load robot model
97 rm_loader_ = std::make_unique<robot_model_loader::RobotModelLoader>(node_);
98 robot_model_ = rm_loader_->getModel();
99 ASSERT_FALSE(robot_model_ == nullptr) << "There is no robot model!";
100
101 // get parameters
102 ASSERT_TRUE(node_->has_parameter(PARAM_PLANNING_GROUP_NAME)) << "Could not find parameter 'planning_group'";
103 node_->get_parameter<std::string>(PARAM_PLANNING_GROUP_NAME, planning_group_);
104
105 ASSERT_TRUE(node_->has_parameter(PARAM_TARGET_LINK_NAME)) << "Could not find parameter 'target_link'";
106 node_->get_parameter<std::string>(PARAM_TARGET_LINK_NAME, target_link_);
107
109 testutils::createFakeLimits(robot_model_->getVariableNames());
110
111 cartesian_limits::Params cartesian_limit;
112 cartesian_limit.max_trans_vel = 1.0 * M_PI;
113 cartesian_limit.max_trans_acc = 1.0 * M_PI;
114 cartesian_limit.max_trans_dec = 1.0 * M_PI;
115 cartesian_limit.max_rot_vel = 1.0 * M_PI;
116
119 limits.setCartesianLimits(cartesian_limit);
120
121 planning_context_ = std::unique_ptr<typename T::Type_>(
122 new typename T::Type_("TestPlanningContext", planning_group_, robot_model_, limits));
123
124 // Define and set the current scene
125 auto scene = std::make_shared<planning_scene::PlanningScene>(robot_model_);
127 current_state.setToDefaultValues();
128 current_state.setJointGroupPositions(planning_group_, std::vector<double>{ 0, 1.57, 1.57, 0, 0.2, 0 });
129 scene->setCurrentState(current_state);
130 planning_context_->setPlanningScene(scene); // TODO Check what happens if this is missing
131 }
132
133 void TearDown() override
134 {
135 robot_model_.reset();
136 }
137
141 planning_interface::MotionPlanRequest getValidRequest(const std::string& context_name) const
142 {
144
145 req.planner_id =
146 std::string(context_name).erase(0, std::string("pilz_industrial_motion_planner::PlanningContext").length());
147 req.group_name = this->planning_group_;
148 req.max_velocity_scaling_factor = 0.01;
149 req.max_acceleration_scaling_factor = 0.01;
150
151 // start state
153 rstate.setToDefaultValues();
154 // state state in joint space, used as initial positions, since IK does not
155 // work at zero positions
157 std::vector<double>{ 4.430233957464225e-12, 0.007881892504574495, -1.8157263253868452,
158 1.1801525390026025e-11, 1.8236082178909834,
159 8.591793942969161e-12 });
160 Eigen::Isometry3d start_pose(Eigen::Isometry3d::Identity());
161 start_pose.translation() = Eigen::Vector3d(0.3, 0, 0.65);
162 rstate.setFromIK(this->robot_model_->getJointModelGroup(this->planning_group_), start_pose);
163 moveit::core::robotStateToRobotStateMsg(rstate, req.start_state, false);
164
165 // goal constraint
166 Eigen::Isometry3d goal_pose(Eigen::Isometry3d::Identity());
167 goal_pose.translation() = Eigen::Vector3d(0, 0.3, 0.65);
168 Eigen::Matrix3d goal_rotation;
169 goal_rotation = Eigen::AngleAxisd(0 * M_PI, Eigen::Vector3d::UnitZ());
170 goal_pose.linear() = goal_rotation;
171 rstate.setFromIK(this->robot_model_->getJointModelGroup(this->planning_group_), goal_pose);
172 req.goal_constraints.push_back(kinematic_constraints::constructGoalConstraints(
173 rstate, this->robot_model_->getJointModelGroup(this->planning_group_)));
174
175 // path constraint
176 req.path_constraints.name = "center";
177 moveit_msgs::msg::PositionConstraint center_point;
178 center_point.link_name = this->target_link_;
179 geometry_msgs::msg::Pose center_position;
180 center_position.position.x = 0.0;
181 center_position.position.y = 0.0;
182 center_position.position.z = 0.65;
183 center_point.constraint_region.primitive_poses.push_back(center_position);
184 req.path_constraints.position_constraints.push_back(center_point);
185
186 return req;
187 }
188
189protected:
190 // ros stuff
191 rclcpp::Node::SharedPtr node_;
192 moveit::core::RobotModelConstPtr robot_model_;
193 std::unique_ptr<robot_model_loader::RobotModelLoader> rm_loader_;
194
195 std::unique_ptr<planning_interface::PlanningContext> planning_context_;
196
198};
199
200// Define the types we need to test
202
208{
210 this->planning_context_->solve(res);
211
212 EXPECT_EQ(moveit_msgs::msg::MoveItErrorCodes::INVALID_MOTION_PLAN, res.error_code.val)
213 << testutils::demangle(typeid(TypeParam).name());
214}
215
220{
222 planning_interface::MotionPlanRequest req = this->getValidRequest(testutils::demangle(typeid(TypeParam).name()));
223
224 this->planning_context_->setMotionPlanRequest(req);
225
226 // TODO Formulate valid request
227 this->planning_context_->solve(res);
228
229 EXPECT_EQ(moveit_msgs::msg::MoveItErrorCodes::SUCCESS, res.error_code.val)
230 << testutils::demangle(typeid(TypeParam).name());
231
233 this->planning_context_->solve(res_detailed);
234
235 EXPECT_EQ(moveit_msgs::msg::MoveItErrorCodes::SUCCESS, res.error_code.val)
236 << testutils::demangle(typeid(TypeParam).name());
237}
238
242TYPED_TEST(PlanningContextTest, SolveValidRequestDetailedResponse)
243{
245 planning_interface::MotionPlanRequest req = this->getValidRequest(testutils::demangle(typeid(TypeParam).name()));
246
247 this->planning_context_->setMotionPlanRequest(req);
248 this->planning_context_->solve(res);
249
250 EXPECT_EQ(moveit_msgs::msg::MoveItErrorCodes::SUCCESS, res.error_code.val)
251 << testutils::demangle(typeid(TypeParam).name());
252}
253
258{
260 planning_interface::MotionPlanRequest req = this->getValidRequest(testutils::demangle(typeid(TypeParam).name()));
261
262 this->planning_context_->setMotionPlanRequest(req);
263
264 bool result_termination = this->planning_context_->terminate();
265 EXPECT_TRUE(result_termination) << testutils::demangle(typeid(TypeParam).name());
266
267 this->planning_context_->solve(res);
268
269 EXPECT_EQ(moveit_msgs::msg::MoveItErrorCodes::PLANNING_FAILED, res.error_code.val)
270 << testutils::demangle(typeid(TypeParam).name());
271}
272
277{
278 EXPECT_NO_THROW(this->planning_context_->clear()) << testutils::demangle(typeid(TypeParam).name());
279}
280
281int main(int argc, char** argv)
282{
283 rclcpp::init(argc, argv);
284 testing::InitGoogleTest(&argc, argv);
285 return RUN_ALL_TESTS();
286}
planning_interface::MotionPlanRequest getValidRequest(const std::string &context_name) const
Generate a valid fully defined request.
moveit::core::RobotModelConstPtr robot_model_
std::unique_ptr< planning_interface::PlanningContext > planning_context_
std::unique_ptr< robot_model_loader::RobotModelLoader > rm_loader_
rclcpp::Node::SharedPtr node_
Representation of a robot's state. This includes position, velocity, acceleration and effort.
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 setToDefaultValues()
Set all joints to their default positions. The default position is 0, or if that is not within bounds...
bool setFromIK(const JointModelGroup *group, const geometry_msgs::msg::Pose &pose, double timeout=0.0, const GroupStateValidityCallbackFn &constraint=GroupStateValidityCallbackFn(), const kinematics::KinematicsQueryOptions &options=kinematics::KinematicsQueryOptions(), const kinematics::KinematicsBase::IKCostFn &cost_function=kinematics::KinematicsBase::IKCostFn())
If the group this state corresponds to is a chain and a solver is available, then the joint values ca...
Container for JointLimits, essentially a map with convenience functions. Adds the ability to as for l...
This class combines CartesianLimit and JointLimits into on single class.
void setCartesianLimits(cartesian_limits::Params &cartesian_limit)
Set cartesian limits.
void setJointLimits(JointLimitsContainer &joint_limits)
Set joint limits.
moveit_msgs::msg::Constraints constructGoalConstraints(const moveit::core::RobotState &state, const moveit::core::JointModelGroup *jmg, double tolerance_below, double tolerance_above)
Generates a constraint message intended to be used as a goal constraint for a joint group....
Definition utils.cpp:152
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.
moveit_msgs::msg::MotionPlanRequest MotionPlanRequest
std::string demangle(const char *name)
pilz_industrial_motion_planner::JointLimitsContainer createFakeLimits(const std::vector< std::string > &joint_names)
Create limits for tests to avoid the need to get the limits from the node parameter.
moveit_msgs::msg::MoveItErrorCodes error_code
moveit::core::MoveItErrorCode error_code
#define TYPED_TEST_SUITE(...)
const std::string PARAM_PLANNING_GROUP_NAME("planning_group")
ValueTypeContainer< pilz_industrial_motion_planner::PlanningContextCIRC, 0 > CIRC_NO_GRIPPER
ValueTypeContainer< pilz_industrial_motion_planner::PlanningContextPTP, 1 > PTP_WITH_GRIPPER
int main(int argc, char **argv)
TYPED_TEST(PlanningContextTest, NoRequest)
No request is set. Check the output of solve. Using robot model without gripper.
const std::string PARAM_TARGET_LINK_NAME("target_link")
::testing::Types< PTP_NO_GRIPPER, PTP_WITH_GRIPPER, LIN_NO_GRIPPER, LIN_WITH_GRIPPER, CIRC_NO_GRIPPER, CIRC_WITH_GRIPPER > PlanningContextTestTypes
ValueTypeContainer< pilz_industrial_motion_planner::PlanningContextLIN, 0 > LIN_NO_GRIPPER
ValueTypeContainer< pilz_industrial_motion_planner::PlanningContextPTP, 0 > PTP_NO_GRIPPER
ValueTypeContainer< pilz_industrial_motion_planner::PlanningContextCIRC, 1 > CIRC_WITH_GRIPPER
ValueTypeContainer< pilz_industrial_motion_planner::PlanningContextLIN, 1 > LIN_WITH_GRIPPER