From f4371b473c0f32d6c625b7db73b2b27c6ae637d9 Mon Sep 17 00:00:00 2001 From: Beatriz Leon Date: Thu, 23 Jul 2015 16:14:58 +0100 Subject: [PATCH 01/18] moving but only with one goal --- .../init_joint_trajectory.h | 233 ++++++++++++------ .../joint_trajectory_controller.h | 6 +- .../joint_trajectory_controller_impl.h | 167 +++++++------ .../joint_trajectory_segment.h | 32 +++ 4 files changed, 288 insertions(+), 150 deletions(-) diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h index 75cf1889f..f56c9d422 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h @@ -67,7 +67,7 @@ inline std::vector permutation(const T& t1, const T& t2) typedef unsigned int SizeType; // Arguments must have the same size - if (t1.size() != t2.size()) {return std::vector();} + //if (t1.size() != t2.size()) {return std::vector();} std::vector permutation_vector(t1.size()); // Return value for (typename T::const_iterator t1_it = t1.begin(); t1_it != t1.end(); ++t1_it) @@ -95,7 +95,9 @@ struct InitJointTrajectoryOptions { typedef realtime_tools::RealtimeServerGoalHandle RealtimeGoalHandle; typedef boost::shared_ptr RealtimeGoalHandlePtr; - typedef typename Trajectory::value_type::Scalar Scalar; + typedef typename Trajectory::value_type TrajectoryPerJoint; + typedef typename TrajectoryPerJoint::value_type Segment; + typedef typename Segment::Scalar Scalar; InitJointTrajectoryOptions() : current_trajectory(0), @@ -184,14 +186,15 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg const InitJointTrajectoryOptions& options = InitJointTrajectoryOptions()) { - typedef typename Trajectory::value_type Segment; + typedef typename Trajectory::value_type TrajectoryPerJoint; + typedef typename TrajectoryPerJoint::value_type Segment; typedef typename Segment::Scalar Scalar; const unsigned int n_joints = msg.joint_names.size(); const ros::Time msg_start_time = internal::startTime(msg, time); // Message start time - ROS_DEBUG_STREAM("Figuring out new trajectory starting at time " + ROS_INFO_STREAM("Figuring out new trajectory starting at time " << std::fixed << std::setprecision(3) << msg_start_time.toSec()); // Empty trajectory @@ -243,7 +246,32 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg // If unspecified, a trivial map (no permutation) is computed const std::vector joint_names = has_joint_names ? *(options.joint_names) : msg.joint_names; - std::vector permutation_vector = internal::permutation(joint_names, msg.joint_names); + ROS_INFO_STREAM("Joint names size" << joint_names.size()); + + std::stringstream log_str; + log_str.str(""); + log_str << "Joint names size:" ; + for (unsigned int i=0; i < joint_names.size();i++) log_str << joint_names[i] << "\t"; + log_str << "\n"; + ROS_INFO_STREAM(log_str.str()); + + log_str.str(""); + log_str << "msg.joint_names size:" ; + for (unsigned int i=0; i < msg.joint_names.size();i++) log_str << msg.joint_names[i] << "\t"; + log_str << "\n"; + ROS_INFO_STREAM(log_str.str()); + + log_str.str(""); + log_str << "msg.position size:" ; + for (unsigned int i=0; i < msg.points.size();i++) + { + for (unsigned int j=0; j < msg.points[i].positions.size();j++) + log_str << msg.points[i].positions[j] << "\t"; + log_str << "\n"; + } + ROS_INFO_STREAM(log_str.str()); + + std::vector permutation_vector = internal::permutation(msg.joint_names,joint_names); if (permutation_vector.empty()) { @@ -251,6 +279,13 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg return Trajectory(); } + //std::stringstream log_str; + log_str.str(""); + log_str << "permutation_vector:" ; + for (unsigned int i=0; i < permutation_vector.size();i++) log_str << permutation_vector[i] << "\t"; + log_str << "\n"; + ROS_INFO_STREAM(log_str.str()); + // Tolerances to be used in all new segments SegmentTolerances tolerances = has_default_tolerances ? *(options.default_tolerances) : SegmentTolerances(n_joints); @@ -268,6 +303,7 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg if (it == msg.points.end()) { it = msg.points.begin(); // Entire trajectory is after current time + ROS_WARN_STREAM("Entire trajectory is after current time"); } else { @@ -292,96 +328,145 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg } } + + // Initialize result trajectory: combination of: // - Useful segments of currently followed trajectory // - Useful segments of new trajectory (contained in ROS message) - Trajectory result_traj; // Currently empty + Trajectory result_traj = *(options.current_trajectory); + ROS_WARN_STREAM("********** result_traj size" << result_traj.size()); - // Initialize offsets due to wrapping joints to zero - std::vector position_offset(n_joints, 0.0); + std::vector permutation_vector_per_joint(1,0); //refactor this and remove it as it is not needed - // Bridge current trajectory to new one - if (has_current_trajectory) + //Iterate through the joints that are in the message, in the order of the permutation vector + //for (unsigned int joint_id=0; joint_id < joint_names.size();joint_id++) + for (unsigned int msg_joint_it=0; msg_joint_it < permutation_vector.size();msg_joint_it++) { - const Trajectory& curr_traj = *(options.current_trajectory); - - // Get the last time and state that will be executed from the current trajectory - const typename Segment::Time last_curr_time = std::max(o_msg_start_time.toSec(), o_time.toSec()); // Important! - typename Segment::State last_curr_state; - sample(curr_traj, last_curr_time, last_curr_state); + TrajectoryPerJoint result_traj_per_joint; // Currently empty + unsigned int joint_id = permutation_vector[msg_joint_it]; - // Get the first time and state that will be executed from the new trajectory - const typename Segment::Time first_new_time = o_msg_start_time.toSec() + (it->time_from_start).toSec(); - typename Segment::State first_new_state(*it, permutation_vector); // Here offsets are not yet applied + ROS_WARN_STREAM("********** Processing trajectory for joint " << joint_names[joint_id]); + // Initialize offsets due to wrapping joints to zero + std::vector position_offset(1, 0.0); + //Scalar position_offset = 0.0; - // Compute offsets due to wrapping joints - if (has_angle_wraparound) + // Bridge current trajectory to new one + if (has_current_trajectory) { - position_offset = wraparoundOffset(last_curr_state.position, - first_new_state.position, - *(options.angle_wraparound)); - if (position_offset.empty()) + const TrajectoryPerJoint& curr_joint_traj = (*options.current_trajectory)[joint_id]; + + // Get the last time and state that will be executed from the current trajectory + const typename Segment::Time last_curr_time = std::max(o_msg_start_time.toSec(), o_time.toSec()); // Important! + typename Segment::State last_curr_state; + sample(curr_joint_traj, last_curr_time, last_curr_state); + + log_str.str(""); + log_str << "last_curr_state.position:" ; + for (unsigned int i=0; i < last_curr_state.position.size();i++) log_str << last_curr_state.position[i] << "\t"; + log_str << "\n"; + ROS_INFO_STREAM(log_str.str()); + + // Get the first time and state that will be executed from the new trajectory + trajectory_msgs::JointTrajectoryPoint point_per_joint; + point_per_joint.positions.resize(1, it->positions[msg_joint_it]); + point_per_joint.velocities.resize(1, it->velocities[msg_joint_it]); + point_per_joint.accelerations.resize(1, it->accelerations[msg_joint_it]); + point_per_joint.time_from_start = it->time_from_start; + + + const typename Segment::Time first_new_time = o_msg_start_time.toSec() + (it->time_from_start).toSec(); + typename Segment::State first_new_state(point_per_joint, permutation_vector_per_joint); // Here offsets are not yet applied + + log_str.str(""); + log_str << "first_new_state.position:" ; + for (unsigned int i=0; i < first_new_state.position.size();i++) log_str << first_new_state.position[i] << "\t"; + log_str << "\n"; + ROS_INFO_STREAM(log_str.str()); + + // Compute offsets due to wrapping joints + if (has_angle_wraparound) { - ROS_ERROR("Cannot create trajectory from message. " - "Vector specifying whether joints wrap around has an invalid size."); - return Trajectory(); + position_offset[0] = wraparoundJointOffset(last_curr_state.position[0], + first_new_state.position[0], + (*options.angle_wraparound)[joint_id]); } - } - // Apply offset to first state that will be executed from the new trajectory - first_new_state = typename Segment::State(*it, permutation_vector, position_offset); // Now offsets are applied + // Apply offset to first state that will be executed from the new trajectory + first_new_state = typename Segment::State(point_per_joint, permutation_vector_per_joint, position_offset); // Now offsets are applied - // Add useful segments of current trajectory to result - { - typedef typename Trajectory::const_iterator TrajIter; - TrajIter first = findSegment(curr_traj, o_time.toSec()); // Currently active segment - TrajIter last = findSegment(curr_traj, last_curr_time); // Segment active when new trajectory starts - if (first == curr_traj.end() || last == curr_traj.end()) + // Add useful segments of current trajectory to result { - ROS_ERROR("Unexpected error: Could not find segments in current trajectory. Please contact the package maintainer."); - return Trajectory(); + typedef typename TrajectoryPerJoint::const_iterator TrajIter; + TrajIter first = findSegment(curr_joint_traj, o_time.toSec()); // Currently active segment + TrajIter last = findSegment(curr_joint_traj, last_curr_time); // Segment active when new trajectory starts + if (first == curr_joint_traj.end() || last == curr_joint_traj.end()) + { + ROS_ERROR("Unexpected error: Could not find segments in current trajectory. Please contact the package maintainer."); + return Trajectory(); + } + result_traj_per_joint.insert(result_traj_per_joint.begin(), first, ++last); // Range [first,last) will still be executed } - result_traj.insert(result_traj.begin(), first, ++last); // Range [first,last) will still be executed + + // Add segment bridging current and new trajectories to result + Segment bridge_seg(last_curr_time, last_curr_state, + first_new_time, first_new_state); + bridge_seg.setGoalHandle(options.rt_goal_handle); + if (has_rt_goal_handle) {bridge_seg.setTolerances(tolerances);} + result_traj_per_joint.push_back(bridge_seg); + ROS_INFO_STREAM("Finish adding old trajectory segment"); } - // Add segment bridging current and new trajectories to result - Segment bridge_seg(last_curr_time, last_curr_state, - first_new_time, first_new_state); - bridge_seg.setGoalHandle(options.rt_goal_handle); - if (has_rt_goal_handle) {bridge_seg.setTolerances(tolerances);} - result_traj.push_back(bridge_seg); - } + // Constants used in log statement at the end + const unsigned int num_old_segments = result_traj_per_joint.size() -1; + const unsigned int num_new_segments = std::distance(it, msg.points.end()) -1; - // Constants used in log statement at the end - const unsigned int num_old_segments = result_traj.size() -1; - const unsigned int num_new_segments = std::distance(it, msg.points.end()) -1; + // Add useful segments of new trajectory to result + // - Construct all trajectory segments occurring after current time + // - As long as there remain two trajectory points we can construct the next trajectory segment + while (std::distance(it, msg.points.end()) >= 2) + { + std::vector::const_iterator next_it = it; ++next_it; + + ROS_INFO_STREAM("Inside while"); + trajectory_msgs::JointTrajectoryPoint it_point_per_joint, next_it_point_per_joint; + it_point_per_joint.positions.resize(1, it->positions[msg_joint_it]); + it_point_per_joint.velocities.resize(1, it->velocities[msg_joint_it]); + it_point_per_joint.accelerations.resize(1, it->accelerations[msg_joint_it]); + it_point_per_joint.time_from_start = it->time_from_start; + + next_it_point_per_joint.positions.resize(1, next_it->positions[msg_joint_it]); + next_it_point_per_joint.velocities.resize(1, next_it->velocities[msg_joint_it]); + next_it_point_per_joint.accelerations.resize(1, next_it->accelerations[msg_joint_it]); + next_it_point_per_joint.time_from_start = next_it->time_from_start; + + //if (!permutation.empty() && joint_dim != permutation.size()) + // unsigned int joint_dim = point.positions.size(); + + Segment segment(o_msg_start_time, it_point_per_joint, next_it_point_per_joint, permutation_vector_per_joint, position_offset); + segment.setGoalHandle(options.rt_goal_handle); + if (has_rt_goal_handle) {segment.setTolerances(tolerances);} + result_traj_per_joint.push_back(segment); + ++it; + } - // Add useful segments of new trajectory to result - // - Construct all trajectory segments occurring after current time - // - As long as there remain two trajectory points we can construct the next trajectory segment - while (std::distance(it, msg.points.end()) >= 2) - { - std::vector::const_iterator next_it = it; ++next_it; - Segment segment(o_msg_start_time, *it, *next_it, permutation_vector, position_offset); - segment.setGoalHandle(options.rt_goal_handle); - if (has_rt_goal_handle) {segment.setTolerances(tolerances);} - result_traj.push_back(segment); - ++it; - } + // Useful debug info + std::stringstream log_str; + log_str << "Trajectory of joint " << joint_names[joint_id] << "has " << result_traj_per_joint.size() << " segments"; + if (has_current_trajectory) + { + log_str << ":"; + log_str << "\n- " << num_old_segments << " segment(s) will still be executed from previous trajectory."; + log_str << "\n- 1 segment added for transitioning between the current trajectory and first point of the input message."; + if (num_new_segments > 0) {log_str << "\n- " << num_new_segments << " new segments (" << (num_new_segments + 1) << + " points) taken from the input trajectory.";} + } + else {log_str << ".";} + ROS_WARN_STREAM(log_str.str()); + + ROS_WARN_STREAM("********** result_traj_per_joint" << result_traj[joint_id].size()); + result_traj[joint_id] = result_traj_per_joint; - // Useful debug info - std::stringstream log_str; - log_str << "Trajectory has " << result_traj.size() << " segments"; - if (has_current_trajectory) - { - log_str << ":"; - log_str << "\n- " << num_old_segments << " segment(s) will still be executed from previous trajectory."; - log_str << "\n- 1 segment added for transitioning between the current trajectory and first point of the input message."; - if (num_new_segments > 0) {log_str << "\n- " << num_new_segments << " new segments (" << (num_new_segments + 1) << - " points) taken from the input trajectory.";} } - else {log_str << ".";} - ROS_DEBUG_STREAM(log_str.str()); return result_traj; } diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h index c3b9d63c9..615efcbf3 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h @@ -166,8 +166,10 @@ class JointTrajectoryController : public controller_interface::Controller StatePublisherPtr; typedef JointTrajectorySegment Segment; - typedef std::vector Trajectory; + typedef std::vector TrajectoryPerJoint; + typedef std::vector Trajectory; typedef boost::shared_ptr TrajectoryPtr; + typedef boost::shared_ptr TrajectoryPerJointPtr; typedef realtime_tools::RealtimeBox TrajectoryBox; typedef typename Segment::Scalar Scalar; @@ -193,12 +195,14 @@ class JointTrajectoryController : public controller_interface::Controller time_data_; diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h index 3b155ad91..20e52c932 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h @@ -245,6 +245,7 @@ bool JointTrajectoryController::init(HardwareInt ros::NodeHandle& root_nh, ros::NodeHandle& controller_nh) { + ROS_ERROR_NAMED(name_,"*********************init"); using namespace internal; // Cache controller node handle @@ -347,11 +348,24 @@ bool JointTrajectoryController::init(HardwareInt current_state_ = typename Segment::State(n_joints); desired_state_ = typename Segment::State(n_joints); state_error_ = typename Segment::State(n_joints); - hold_start_state_ = typename Segment::State(n_joints); - hold_end_state_ = typename Segment::State(n_joints); - Segment hold_segment(0.0, current_state_, 0.0, current_state_); - hold_trajectory_ptr_->resize(1, hold_segment); + desired_joint_state_ = typename Segment::State(1); + hold_start_state_ = typename Segment::State(1); + hold_end_state_ = typename Segment::State(1); + + // Initialize trajectory with all joints + typename Segment::State current_joint_state_ = typename Segment::State(1); + for (unsigned int i = 0; i < n_joints; ++i) + { + current_joint_state_.position[0]= current_state_.position[i]; + current_joint_state_.velocity[0]= current_state_.velocity[i]; + Segment hold_segment(0.0, current_joint_state_, 0.0, current_joint_state_); + + TrajectoryPerJoint joint_segment; + joint_segment.resize(1, hold_segment); + hold_trajectory_ptr_->push_back(joint_segment); + } + ROS_ERROR_STREAM_NAMED(name_,"hold_trajectory_ptr_->size():" << hold_trajectory_ptr_->size()); { state_publisher_->lock(); @@ -365,7 +379,7 @@ bool JointTrajectoryController::init(HardwareInt state_publisher_->msg_.error.velocities.resize(n_joints); state_publisher_->unlock(); } - + ROS_ERROR_NAMED(name_,"********************* finish init"); return true; } @@ -393,16 +407,6 @@ update(const ros::Time& time, const ros::Duration& period) // fetch the currently followed trajectory, it has been updated by the non-rt thread with something that starts in the // next control cycle, leaving the current cycle without a valid trajectory. - // Update desired state: sample trajectory at current time - typename Trajectory::const_iterator segment_it = sample(curr_traj, time_data.uptime.toSec(), desired_state_); - if (curr_traj.end() == segment_it) - { - // Non-realtime safe, but should never happen under normal operation - ROS_ERROR_NAMED(name_, - "Unexpected error: No trajectory defined at current time. Please contact the package maintainer."); - return; - } - // Update current state and state error for (unsigned int i = 0; i < joints_.size(); ++i) { @@ -410,33 +414,45 @@ update(const ros::Time& time, const ros::Duration& period) current_state_.velocity[i] = joints_[i].getVelocity(); // There's no acceleration data available in a joint handle - state_error_.position[i] = desired_state_.position[i] - current_state_.position[i]; - state_error_.velocity[i] = desired_state_.velocity[i] - current_state_.velocity[i]; - state_error_.acceleration[i] = 0.0; - } - - // Check tolerances if segment corresponds to currently active action goal - const RealtimeGoalHandlePtr rt_segment_goal = segment_it->getGoalHandle(); - if (rt_segment_goal && rt_segment_goal == rt_active_goal_) - { - // Check tolerances - if (time_data.uptime.toSec() < segment_it->endTime()) + typename TrajectoryPerJoint::const_iterator segment_it = sample(curr_traj[i], time_data.uptime.toSec(), desired_joint_state_); + if (curr_traj[i].end() == segment_it) { - // Currently executing a segment: check path tolerances - checkPathTolerances(state_error_, - *segment_it); + // Non-realtime safe, but should never happen under normal operation + ROS_ERROR_NAMED(name_, + "Unexpected error: No trajectory defined at current time. Please contact the package maintainer."); + return; } - else if (segment_it == --curr_traj.end()) - { - if (verbose_) - ROS_DEBUG_STREAM_THROTTLE_NAMED(1,name_,"Finished executing last segement, checking goal tolerances"); + desired_state_.position[i] = desired_joint_state_.position[0]; + desired_state_.velocity[i] = desired_joint_state_.velocity[0]; + desired_state_.acceleration[i] = 0.0; - // Finished executing the LAST segment: check goal tolerances - checkGoalTolerances(state_error_, - *segment_it); - } + state_error_.position[i] = desired_joint_state_.position[0] - current_state_.position[i]; + state_error_.velocity[i] = desired_joint_state_.velocity[0] - current_state_.velocity[i]; + state_error_.acceleration[i] = 0.0; } - +// +// // Check tolerances if segment corresponds to currently active action goal +// const RealtimeGoalHandlePtr rt_segment_goal = segment_it->getGoalHandle(); +// if (rt_segment_goal && rt_segment_goal == rt_active_goal_) +// { +// // Check tolerances +// if (time_data.uptime.toSec() < segment_it->endTime()) +// { +// // Currently executing a segment: check path tolerances +// checkPathTolerances(state_error_, +// *segment_it); +// } +// else if (segment_it == --curr_traj.end()) +// { +// if (verbose_) +// ROS_DEBUG_STREAM_THROTTLE_NAMED(1,name_,"Finished executing last segement, checking goal tolerances"); +// +// // Finished executing the LAST segment: check goal tolerances +// checkGoalTolerances(state_error_, +// *segment_it); +// } +// } +// // Hardware interface adapter: Generate and send commands hw_iface_adapter_.updateCommand(time_data.uptime, time_data.period, desired_state_, state_error_); @@ -526,7 +542,7 @@ template void JointTrajectoryController:: goalCB(GoalHandle gh) { - ROS_DEBUG_STREAM_NAMED(name_,"Recieved new action goal"); + ROS_DEBUG_STREAM_NAMED(name_,"Received new action goal"); // Precondition: Running controller if (!this->isRunning()) @@ -540,7 +556,7 @@ goalCB(GoalHandle gh) // Goal should specify all controller joints (they can be ordered differently). Reject if this is not the case using internal::permutation; - std::vector permutation_vector = permutation(joint_names_, gh.getGoal()->trajectory.joint_names); + std::vector permutation_vector = permutation(gh.getGoal()->trajectory.joint_names, joint_names_); if (permutation_vector.empty()) { @@ -619,26 +635,26 @@ queryStateService(control_msgs::QueryTrajectoryState::Request& req, const ros::Duration time_offset = req.time - time_data->time; const ros::Time sample_time = time_data->uptime + time_offset; - // Sample trajectory at requested time - TrajectoryPtr curr_traj_ptr; - curr_trajectory_box_.get(curr_traj_ptr); - Trajectory& curr_traj = *curr_traj_ptr; - - typename Segment::State state; - typename Trajectory::const_iterator segment_it = sample(curr_traj, sample_time.toSec(), state); - if (curr_traj.end() == segment_it) - { - ROS_ERROR_STREAM_NAMED(name_, "Requested sample time preceeds trajectory start time."); - return false; - } - - // Populate response - resp.name = joint_names_; - resp.position = state.position; - resp.velocity = state.velocity; - resp.acceleration = state.acceleration; - - return true; +// // Sample trajectory at requested time +// TrajectoryPtr curr_traj_ptr; +// curr_trajectory_box_.get(curr_traj_ptr); +// Trajectory& curr_traj = *curr_traj_ptr; + return false; +// typename Segment::State state; +// typename Trajectory::const_iterator segment_it = sample(curr_traj, sample_time.toSec(), state); +// if (curr_traj.end() == segment_it) +// { +// ROS_ERROR_STREAM_NAMED(name_, "Requested sample time preceeds trajectory start time."); +// return false; +// } +// +// // Populate response +// resp.name = joint_names_; +// resp.position = state.position; +// resp.velocity = state.velocity; +// resp.acceleration = state.acceleration; +// +// return true; } template @@ -676,7 +692,7 @@ setHoldPosition(const ros::Time& time) // - Create segment that goes from current state to above zero velocity state, in the desired time // NOTE: The symmetry assumption from the second point above might not hold for all possible segment types - assert(1 == hold_trajectory_ptr_->size()); + assert(joint_names_.size() == hold_trajectory_ptr_->size()); const typename Segment::Time start_time = time.toSec(); const typename Segment::Time end_time = time.toSec() + stop_trajectory_duration_; @@ -686,24 +702,25 @@ setHoldPosition(const ros::Time& time) const unsigned int n_joints = joints_.size(); for (unsigned int i = 0; i < n_joints; ++i) { - hold_start_state_.position[i] = joints_[i].getPosition(); - hold_start_state_.velocity[i] = joints_[i].getVelocity(); - hold_start_state_.acceleration[i] = 0.0; + hold_start_state_.position[0] = joints_[i].getPosition(); + hold_start_state_.velocity[0] = joints_[i].getVelocity(); + hold_start_state_.acceleration[0] = 0.0; - hold_end_state_.position[i] = joints_[i].getPosition(); - hold_end_state_.velocity[i] = -joints_[i].getVelocity(); - hold_end_state_.acceleration[i] = 0.0; - } - hold_trajectory_ptr_->front().init(start_time, hold_start_state_, - end_time_2x, hold_end_state_); + hold_end_state_.position[0] = joints_[i].getPosition(); + hold_end_state_.velocity[0] = -joints_[i].getVelocity(); + hold_end_state_.acceleration[0] = 0.0; - // Sample segment at its midpoint, that should have zero velocity - hold_trajectory_ptr_->front().sample(end_time, hold_end_state_); + //hold_traj[i].joint_trajectory.front() + (*hold_trajectory_ptr_)[i].front().init(start_time, hold_start_state_, + end_time_2x, hold_end_state_); - // Now create segment that goes from current state to one with zero end velocity - hold_trajectory_ptr_->front().init(start_time, hold_start_state_, - end_time, hold_end_state_); + // Sample segment at its midpoint, that should have zero velocity + (*hold_trajectory_ptr_)[i].front().sample(end_time, hold_end_state_); + // Now create segment that goes from current state to one with zero end velocity + (*hold_trajectory_ptr_)[i].front().init(start_time, hold_start_state_, + end_time, hold_end_state_); + } curr_trajectory_box_.set(hold_trajectory_ptr_); } diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_segment.h b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_segment.h index 3c024c7cf..07c2d8160 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_segment.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_segment.h @@ -275,6 +275,38 @@ std::vector wraparoundOffset(const std::vector& prev_position, return pos_offset; } +/** + * \param prev_position Previous position from which to compute the wraparound offset. + * \param next_position Next position from which to compute the wraparound offset. + * \param angle_wraparound Vector of booleans where true values correspond to joints that wrap around + * (ie. are continuous). Offsets will be computed only for these joints, otherwise they are set to zero. + * \return Wraparound offsets that should be applied to \p next_position such that no multi-turns are performed when + * transitioning from \p prev_position. + * \tparam Scalar Scalar type. + */ +template +Scalar wraparoundJointOffset(const Scalar& prev_position, + const Scalar& next_position, + const bool& angle_wraparound) +{ + // Return value + Scalar pos_offset = 0.0; + + if (angle_wraparound) + { + Scalar dist = angles::shortest_angular_distance(prev_position, next_position); + + // Deal with singularity at M_PI shortest distance + if (std::abs(dist) - M_PI < 1e-9) + { + dist = next_position > prev_position ? std::abs(dist) : -std::abs(dist); + } + pos_offset = (prev_position + dist) - next_position; + } + + return pos_offset; +} + } // namespace #endif // header guard From 38f6a1f922dac1ad7ae468f29d0ffececc32f86b Mon Sep 17 00:00:00 2001 From: Beatriz Leon Date: Fri, 24 Jul 2015 14:28:44 +0100 Subject: [PATCH 02/18] working with one goal, still needs to set the goal handlers correctly --- .../init_joint_trajectory.h | 10 +- .../joint_trajectory_controller_impl.h | 106 ++++++++++-------- .../joint_trajectory_controller/tolerances.h | 3 + 3 files changed, 68 insertions(+), 51 deletions(-) diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h index f56c9d422..71390e1e7 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h @@ -194,7 +194,7 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg const ros::Time msg_start_time = internal::startTime(msg, time); // Message start time - ROS_INFO_STREAM("Figuring out new trajectory starting at time " + ROS_DEBUG_STREAM("Figuring out new trajectory starting at time " << std::fixed << std::setprecision(3) << msg_start_time.toSec()); // Empty trajectory @@ -328,8 +328,6 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg } } - - // Initialize result trajectory: combination of: // - Useful segments of currently followed trajectory // - Useful segments of new trajectory (contained in ROS message) @@ -346,9 +344,10 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg unsigned int joint_id = permutation_vector[msg_joint_it]; ROS_WARN_STREAM("********** Processing trajectory for joint " << joint_names[joint_id]); + // Initialize offsets due to wrapping joints to zero std::vector position_offset(1, 0.0); - //Scalar position_offset = 0.0; + // Bridge current trajectory to new one if (has_current_trajectory) @@ -439,9 +438,6 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg next_it_point_per_joint.accelerations.resize(1, next_it->accelerations[msg_joint_it]); next_it_point_per_joint.time_from_start = next_it->time_from_start; - //if (!permutation.empty() && joint_dim != permutation.size()) - // unsigned int joint_dim = point.positions.size(); - Segment segment(o_msg_start_time, it_point_per_joint, next_it_point_per_joint, permutation_vector_per_joint, position_offset); segment.setGoalHandle(options.rt_goal_handle); if (has_rt_goal_handle) {segment.setTolerances(tolerances);} diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h index 20e52c932..2bf45a2f6 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h @@ -379,7 +379,6 @@ bool JointTrajectoryController::init(HardwareInt state_publisher_->msg_.error.velocities.resize(n_joints); state_publisher_->unlock(); } - ROS_ERROR_NAMED(name_,"********************* finish init"); return true; } @@ -429,30 +428,35 @@ update(const ros::Time& time, const ros::Duration& period) state_error_.position[i] = desired_joint_state_.position[0] - current_state_.position[i]; state_error_.velocity[i] = desired_joint_state_.velocity[0] - current_state_.velocity[i]; state_error_.acceleration[i] = 0.0; + + //ROS_WARN_STREAM_NAMED(name_, "Check tolerances for joint: "<getGoalHandle(); + if (rt_segment_goal && rt_segment_goal == rt_active_goal_) + { + // Check tolerances + if (time_data.uptime.toSec() < segment_it->endTime()) + { + ROS_WARN_NAMED(name_, "Currently executing a segment: check path tolerances"); + // Currently executing a segment: check path tolerances + checkPathTolerances(state_error_, + *segment_it); + } + else if (segment_it == --curr_traj[i].end()) + { + ROS_WARN_NAMED(name_, "Finished executing the LAST segment: check goal tolerances"); + if (verbose_) + ROS_DEBUG_STREAM_THROTTLE_NAMED(1,name_,"Finished executing last segment, checking goal tolerances"); + + // Finished executing the LAST segment: check goal tolerances + checkGoalTolerances(state_error_, + *segment_it); + } + } } -// -// // Check tolerances if segment corresponds to currently active action goal -// const RealtimeGoalHandlePtr rt_segment_goal = segment_it->getGoalHandle(); -// if (rt_segment_goal && rt_segment_goal == rt_active_goal_) -// { -// // Check tolerances -// if (time_data.uptime.toSec() < segment_it->endTime()) -// { -// // Currently executing a segment: check path tolerances -// checkPathTolerances(state_error_, -// *segment_it); -// } -// else if (segment_it == --curr_traj.end()) -// { -// if (verbose_) -// ROS_DEBUG_STREAM_THROTTLE_NAMED(1,name_,"Finished executing last segement, checking goal tolerances"); -// -// // Finished executing the LAST segment: check goal tolerances -// checkGoalTolerances(state_error_, -// *segment_it); -// } -// } -// + + + // Hardware interface adapter: Generate and send commands hw_iface_adapter_.updateCommand(time_data.uptime, time_data.period, desired_state_, state_error_); @@ -575,6 +579,11 @@ goalCB(GoalHandle gh) if (update_ok) { // Accept new goal + + // If new goal has less joints than current active goal + // create a new one and do not preempt the other + // store them in an array of active goals + preemptActiveGoal(); gh.setAccepted(); rt_active_goal_ = rt_goal; @@ -635,26 +644,35 @@ queryStateService(control_msgs::QueryTrajectoryState::Request& req, const ros::Duration time_offset = req.time - time_data->time; const ros::Time sample_time = time_data->uptime + time_offset; -// // Sample trajectory at requested time -// TrajectoryPtr curr_traj_ptr; -// curr_trajectory_box_.get(curr_traj_ptr); -// Trajectory& curr_traj = *curr_traj_ptr; - return false; -// typename Segment::State state; -// typename Trajectory::const_iterator segment_it = sample(curr_traj, sample_time.toSec(), state); -// if (curr_traj.end() == segment_it) -// { -// ROS_ERROR_STREAM_NAMED(name_, "Requested sample time preceeds trajectory start time."); -// return false; -// } -// -// // Populate response -// resp.name = joint_names_; -// resp.position = state.position; -// resp.velocity = state.velocity; -// resp.acceleration = state.acceleration; -// -// return true; + // Sample trajectory at requested time + TrajectoryPtr curr_traj_ptr; + curr_trajectory_box_.get(curr_traj_ptr); + Trajectory& curr_traj = *curr_traj_ptr; + + typename Segment::State response_point = typename Segment::State(joint_names_.size()); + + for (unsigned int i = 0; i < joints_.size(); ++i) + { + typename Segment::State state; + typename TrajectoryPerJoint::const_iterator segment_it = sample(curr_traj[i], sample_time.toSec(), state); + if (curr_traj[i].end() == segment_it) + { + ROS_ERROR_STREAM_NAMED(name_, "Requested sample time precedes trajectory start time."); + return false; + } + + response_point.position[i] = state.position[0]; + response_point.velocity[i] = state.velocity[0]; + response_point.acceleration[i] = state.acceleration[0]; + } + + // Populate response + resp.name = joint_names_; + resp.position = response_point.position; + resp.velocity = response_point.velocity; + resp.acceleration = response_point.acceleration; + + return true; } template diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h b/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h index efc06580a..69172f9f4 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h @@ -244,6 +244,9 @@ SegmentTolerances getSegmentTolerances(const ros::NodeHandle& nh, nh.param(joint_names[i] + "/trajectory", tolerances.state_tolerance[i].position, 0.0); nh.param(joint_names[i] + "/goal", tolerances.goal_state_tolerance[i].position, 0.0); tolerances.goal_state_tolerance[i].velocity = stopped_velocity_tolerance; + + //ROS_ERROR_STREAM("**************** "< Date: Wed, 29 Jul 2015 15:30:00 +0100 Subject: [PATCH 03/18] Trial with vector of goals but it turns out that the action server can only hold one goal at a time --- .../init_joint_trajectory.h | 117 +++--- .../joint_trajectory_controller.h | 12 +- .../joint_trajectory_controller_impl.h | 345 +++++++++++++----- .../joint_trajectory_segment.h | 10 +- .../joint_trajectory_controller/tolerances.h | 61 ++++ 5 files changed, 387 insertions(+), 158 deletions(-) diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h index 71390e1e7..52ef6997d 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h @@ -246,30 +246,32 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg // If unspecified, a trivial map (no permutation) is computed const std::vector joint_names = has_joint_names ? *(options.joint_names) : msg.joint_names; - ROS_INFO_STREAM("Joint names size" << joint_names.size()); - std::stringstream log_str; - log_str.str(""); - log_str << "Joint names size:" ; - for (unsigned int i=0; i < joint_names.size();i++) log_str << joint_names[i] << "\t"; - log_str << "\n"; - ROS_INFO_STREAM(log_str.str()); - - log_str.str(""); - log_str << "msg.joint_names size:" ; - for (unsigned int i=0; i < msg.joint_names.size();i++) log_str << msg.joint_names[i] << "\t"; - log_str << "\n"; - ROS_INFO_STREAM(log_str.str()); - - log_str.str(""); - log_str << "msg.position size:" ; - for (unsigned int i=0; i < msg.points.size();i++) - { - for (unsigned int j=0; j < msg.points[i].positions.size();j++) - log_str << msg.points[i].positions[j] << "\t"; - log_str << "\n"; - } - ROS_INFO_STREAM(log_str.str()); + + +// ROS_INFO_STREAM("Joint names size" << joint_names.size()); +// +// log_str.str(""); +// log_str << "Joint names size:" ; +// for (unsigned int i=0; i < joint_names.size();i++) log_str << joint_names[i] << "\t"; +// log_str << "\n"; +// ROS_INFO_STREAM(log_str.str()); +// +// log_str.str(""); +// log_str << "msg.joint_names size:" ; +// for (unsigned int i=0; i < msg.joint_names.size();i++) log_str << msg.joint_names[i] << "\t"; +// log_str << "\n"; +// ROS_INFO_STREAM(log_str.str()); +// +// log_str.str(""); +// log_str << "msg.position size:" ; +// for (unsigned int i=0; i < msg.points.size();i++) +// { +// for (unsigned int j=0; j < msg.points[i].positions.size();j++) +// log_str << msg.points[i].positions[j] << "\t"; +// log_str << "\n"; +// } +// ROS_INFO_STREAM(log_str.str()); std::vector permutation_vector = internal::permutation(msg.joint_names,joint_names); @@ -279,12 +281,12 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg return Trajectory(); } - //std::stringstream log_str; - log_str.str(""); - log_str << "permutation_vector:" ; - for (unsigned int i=0; i < permutation_vector.size();i++) log_str << permutation_vector[i] << "\t"; - log_str << "\n"; - ROS_INFO_STREAM(log_str.str()); +// //std::stringstream log_str; +// log_str.str(""); +// log_str << "permutation_vector:" ; +// for (unsigned int i=0; i < permutation_vector.size();i++) log_str << permutation_vector[i] << "\t"; +// log_str << "\n"; +// ROS_INFO_STREAM(log_str.str()); // Tolerances to be used in all new segments SegmentTolerances tolerances = has_default_tolerances ? @@ -299,18 +301,18 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg // This point is used later on in this function, but is computed here, in advance because if the trajectory message // contains a trajectory in the past, we can quickly return without spending additional computational resources std::vector::const_iterator - it = findPoint(msg, time); // Points to last point occurring before current time (NOTE: Using time, not o_time) - if (it == msg.points.end()) + msg_it = findPoint(msg, time); // Points to last point occurring before current time (NOTE: Using time, not o_time) + if (msg_it == msg.points.end()) { - it = msg.points.begin(); // Entire trajectory is after current time + msg_it = msg.points.begin(); // Entire trajectory is after current time ROS_WARN_STREAM("Entire trajectory is after current time"); } else { - ++it; // Points to first point after current time OR sequence end - if (it == msg.points.end()) + ++msg_it; // Points to first point after current time OR sequence end + if (msg_it == msg.points.end()) { - ros::Duration last_point_dur = time - (msg_start_time + (--it)->time_from_start); + ros::Duration last_point_dur = time - (msg_start_time + (--msg_it)->time_from_start); ROS_WARN_STREAM("Dropping all " << msg.points.size() << " trajectory point(s), as they occur before the current time.\n" << "Last point is " << std::fixed << std::setprecision(3) << last_point_dur.toSec() << @@ -319,8 +321,8 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg } else { - ros::Duration next_point_dur = msg_start_time + it->time_from_start - time; - ROS_WARN_STREAM("Dropping first " << std::distance(msg.points.begin(), it) << + ros::Duration next_point_dur = msg_start_time + msg_it->time_from_start - time; + ROS_WARN_STREAM("Dropping first " << std::distance(msg.points.begin(), msg_it) << " trajectory point(s) out of " << msg.points.size() << ", as they occur before the current time.\n" << "First valid point will be reached in " << std::fixed << std::setprecision(3) << @@ -332,7 +334,6 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg // - Useful segments of currently followed trajectory // - Useful segments of new trajectory (contained in ROS message) Trajectory result_traj = *(options.current_trajectory); - ROS_WARN_STREAM("********** result_traj size" << result_traj.size()); std::vector permutation_vector_per_joint(1,0); //refactor this and remove it as it is not needed @@ -340,14 +341,18 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg //for (unsigned int joint_id=0; joint_id < joint_names.size();joint_id++) for (unsigned int msg_joint_it=0; msg_joint_it < permutation_vector.size();msg_joint_it++) { + std::vector::const_iterator it = msg_it; TrajectoryPerJoint result_traj_per_joint; // Currently empty unsigned int joint_id = permutation_vector[msg_joint_it]; - ROS_WARN_STREAM("********** Processing trajectory for joint " << joint_names[joint_id]); - // Initialize offsets due to wrapping joints to zero std::vector position_offset(1, 0.0); + //Initialize segment tolerance per joint + SegmentTolerancesPerJoint tolerances_per_joint; + tolerances_per_joint.state_tolerance = tolerances.state_tolerance[joint_id]; + tolerances_per_joint.goal_state_tolerance = tolerances.goal_state_tolerance[joint_id]; + tolerances_per_joint.goal_time_tolerance = tolerances.goal_time_tolerance; // Bridge current trajectory to new one if (has_current_trajectory) @@ -359,11 +364,11 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg typename Segment::State last_curr_state; sample(curr_joint_traj, last_curr_time, last_curr_state); - log_str.str(""); - log_str << "last_curr_state.position:" ; - for (unsigned int i=0; i < last_curr_state.position.size();i++) log_str << last_curr_state.position[i] << "\t"; - log_str << "\n"; - ROS_INFO_STREAM(log_str.str()); +// log_str.str(""); +// log_str << "last_curr_state.position:" ; +// for (unsigned int i=0; i < last_curr_state.position.size();i++) log_str << last_curr_state.position[i] << "\t"; +// log_str << "\n"; +// ROS_INFO_STREAM(log_str.str()); // Get the first time and state that will be executed from the new trajectory trajectory_msgs::JointTrajectoryPoint point_per_joint; @@ -376,11 +381,11 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg const typename Segment::Time first_new_time = o_msg_start_time.toSec() + (it->time_from_start).toSec(); typename Segment::State first_new_state(point_per_joint, permutation_vector_per_joint); // Here offsets are not yet applied - log_str.str(""); - log_str << "first_new_state.position:" ; - for (unsigned int i=0; i < first_new_state.position.size();i++) log_str << first_new_state.position[i] << "\t"; - log_str << "\n"; - ROS_INFO_STREAM(log_str.str()); +// log_str.str(""); +// log_str << "first_new_state.position:" ; +// for (unsigned int i=0; i < first_new_state.position.size();i++) log_str << first_new_state.position[i] << "\t"; +// log_str << "\n"; +// ROS_INFO_STREAM(log_str.str()); // Compute offsets due to wrapping joints if (has_angle_wraparound) @@ -410,9 +415,9 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg Segment bridge_seg(last_curr_time, last_curr_state, first_new_time, first_new_state); bridge_seg.setGoalHandle(options.rt_goal_handle); - if (has_rt_goal_handle) {bridge_seg.setTolerances(tolerances);} + if (has_rt_goal_handle) {bridge_seg.setTolerances(tolerances_per_joint);} result_traj_per_joint.push_back(bridge_seg); - ROS_INFO_STREAM("Finish adding old trajectory segment"); + //ROS_INFO_STREAM("Finish adding old trajectory segment"); } // Constants used in log statement at the end @@ -426,7 +431,7 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg { std::vector::const_iterator next_it = it; ++next_it; - ROS_INFO_STREAM("Inside while"); + //ROS_INFO_STREAM("Inside while"); trajectory_msgs::JointTrajectoryPoint it_point_per_joint, next_it_point_per_joint; it_point_per_joint.positions.resize(1, it->positions[msg_joint_it]); it_point_per_joint.velocities.resize(1, it->velocities[msg_joint_it]); @@ -440,7 +445,7 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg Segment segment(o_msg_start_time, it_point_per_joint, next_it_point_per_joint, permutation_vector_per_joint, position_offset); segment.setGoalHandle(options.rt_goal_handle); - if (has_rt_goal_handle) {segment.setTolerances(tolerances);} + if (has_rt_goal_handle) {segment.setTolerances(tolerances_per_joint);} result_traj_per_joint.push_back(segment); ++it; } @@ -457,9 +462,9 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg " points) taken from the input trajectory.";} } else {log_str << ".";} - ROS_WARN_STREAM(log_str.str()); + //ROS_WARN_STREAM(log_str.str()); - ROS_WARN_STREAM("********** result_traj_per_joint" << result_traj[joint_id].size()); + //ROS_WARN_STREAM("********** result_traj_per_joint" << result_traj[joint_id].size()); result_traj[joint_id] = result_traj_per_joint; } diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h index 615efcbf3..de0badae6 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h @@ -156,6 +156,8 @@ class JointTrajectoryController : public controller_interface::Controller ActionServer; typedef boost::shared_ptr ActionServerPtr; typedef ActionServer::GoalHandle GoalHandle; @@ -184,7 +186,8 @@ class JointTrajectoryController : public controller_interface::Controller default_tolerances_; ///< Default trajectory segment tolerances. HwIfaceAdapter hw_iface_adapter_; ///< Adapts desired trajectory state to HW interface. - RealtimeGoalHandlePtr rt_active_goal_; ///< Currently active action goal, if any. + //RealtimeGoalHandlePtr rt_active_goal_; ///< Currently active action goal, if any. + std::vector rt_active_goals_; ///< Currently active action goal, if any. /** * Thread-safe container with a smart pointer to trajectory currently being followed. @@ -202,7 +205,6 @@ class JointTrajectoryController : public controller_interface::Controller time_data_; @@ -225,7 +227,7 @@ class JointTrajectoryController : public controller_interface::Controller inline void JointTrajectoryController:: stopping(const ros::Time& time) { - preemptActiveGoal(); + preemptActiveGoals(); } template @@ -161,78 +161,99 @@ inline void JointTrajectoryController:: trajectoryCommandCB(const JointTrajectoryConstPtr& msg) { const bool update_ok = updateTrajectoryCommand(msg, RealtimeGoalHandlePtr()); - if (update_ok) {preemptActiveGoal();} + if (update_ok) {preemptActiveGoals();} } template inline void JointTrajectoryController:: -preemptActiveGoal() +preemptActiveGoals() { - RealtimeGoalHandlePtr current_active_goal(rt_active_goal_); - - // Cancels the currently active goal - if (current_active_goal) - { - // Marks the current goal as canceled - rt_active_goal_.reset(); - current_active_goal->gh_.setCanceled(); - } -} - -template -inline void JointTrajectoryController:: -checkPathTolerances(const typename Segment::State& state_error, - const Segment& segment) -{ - const RealtimeGoalHandlePtr rt_segment_goal = segment.getGoalHandle(); - const SegmentTolerances& tolerances = segment.getTolerances(); - if (!checkStateTolerance(state_error, tolerances.state_tolerance)) + // Check that cancel request refers to currently active goals (if any) + for (unsigned int i = 0; i < rt_active_goals_.size(); ++i) { - rt_segment_goal->preallocated_result_->error_code = - control_msgs::FollowJointTrajectoryResult::PATH_TOLERANCE_VIOLATED; - rt_segment_goal->setAborted(rt_segment_goal->preallocated_result_); - rt_active_goal_.reset(); - } -} - -template -inline void JointTrajectoryController:: -checkGoalTolerances(const typename Segment::State& state_error, - const Segment& segment) -{ - // Controller uptime - const ros::Time uptime = time_data_.readFromRT()->uptime; + RealtimeGoalHandlePtr current_active_goal(rt_active_goals_[i]); - // Checks that we have ended inside the goal tolerances - const RealtimeGoalHandlePtr rt_segment_goal = segment.getGoalHandle(); - const SegmentTolerances& tolerances = segment.getTolerances(); - const bool inside_goal_tolerances = checkStateTolerance(state_error, tolerances.goal_state_tolerance); - - if (inside_goal_tolerances) - { - rt_segment_goal->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::SUCCESSFUL; - rt_segment_goal->setSucceeded(rt_segment_goal->preallocated_result_); - rt_active_goal_.reset(); - } - else if (uptime.toSec() < segment.endTime() + tolerances.goal_time_tolerance) - { - // Still have some time left to meet the goal state tolerances - } - else - { - if (verbose_) + // Cancels the currently active goal + if (current_active_goal) { - ROS_ERROR_STREAM_NAMED(name_,"Goal tolerances failed"); - // Check the tolerances one more time to output the errors that occures - checkStateTolerance(state_error, tolerances.goal_state_tolerance, true); + // Marks the current goal as canceled + rt_active_goals_[i].reset(); + rt_active_goals_.erase(rt_active_goals_.begin()+i); + current_active_goal->gh_.setCanceled(); } - - rt_segment_goal->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::GOAL_TOLERANCE_VIOLATED; - rt_segment_goal->setAborted(rt_segment_goal->preallocated_result_); - rt_active_goal_.reset(); } } +//template +//inline void JointTrajectoryController:: +//preemptActiveGoal() +//{ +// +// RealtimeGoalHandlePtr current_active_goal(rt_active_goal_); +// +// // Cancels the currently active goal +// if (current_active_goal) +// { +// // Marks the current goal as canceled +// rt_active_goal_.reset(); +// current_active_goal->gh_.setCanceled(); +// } +//} + +//template +//inline void JointTrajectoryController:: +//checkPathTolerances(const typename Segment::State& state_error, +// const Segment& segment) +//{ +// const RealtimeGoalHandlePtr rt_segment_goal = segment.getGoalHandle(); +// const SegmentTolerancesPerJoint& tolerances = segment.getTolerances(); +// if (!checkStateTolerancePerJoint(state_error, tolerances.state_tolerance)) +// { +// rt_segment_goal->preallocated_result_->error_code = +// control_msgs::FollowJointTrajectoryResult::PATH_TOLERANCE_VIOLATED; +// rt_segment_goal->setAborted(rt_segment_goal->preallocated_result_); +// rt_active_goal_.reset(); +// } +//} + +//template +//inline void JointTrajectoryController:: +//checkGoalTolerances(const typename Segment::State& state_error, +// const Segment& segment) +//{ +// // Controller uptime +// const ros::Time uptime = time_data_.readFromRT()->uptime; +// +// // Checks that we have ended inside the goal tolerances +// const RealtimeGoalHandlePtr rt_segment_goal = segment.getGoalHandle(); +// const SegmentTolerancesPerJoint& tolerances = segment.getTolerances(); +// const bool inside_goal_tolerances = checkStateTolerancePerJoint(state_error, tolerances.goal_state_tolerance); +// +// if (inside_goal_tolerances) +// { +// rt_segment_goal->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::SUCCESSFUL; +// rt_segment_goal->setSucceeded(rt_segment_goal->preallocated_result_); +// rt_active_goal_.reset(); +// } +// else if (uptime.toSec() < segment.endTime() + tolerances.goal_time_tolerance) +// { +// // Still have some time left to meet the goal state tolerances +// } +// else +// { +// if (verbose_) +// { +// ROS_ERROR_STREAM_NAMED(name_,"Goal tolerances failed"); +// // Check the tolerances one more time to output the errors that occures +// checkStateTolerance(state_error, tolerances.goal_state_tolerance, true); +// } +// +// rt_segment_goal->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::GOAL_TOLERANCE_VIOLATED; +// rt_segment_goal->setAborted(rt_segment_goal->preallocated_result_); +// rt_active_goal_.reset(); +// } +//} + template JointTrajectoryController:: JointTrajectoryController() @@ -349,7 +370,6 @@ bool JointTrajectoryController::init(HardwareInt desired_state_ = typename Segment::State(n_joints); state_error_ = typename Segment::State(n_joints); - desired_joint_state_ = typename Segment::State(1); hold_start_state_ = typename Segment::State(1); hold_end_state_ = typename Segment::State(1); @@ -409,6 +429,9 @@ update(const ros::Time& time, const ros::Duration& period) // Update current state and state error for (unsigned int i = 0; i < joints_.size(); ++i) { + typename Segment::State desired_joint_state_ = typename Segment::State(1); + typename Segment::State state_joint_error_= typename Segment::State(1); + current_state_.position[i] = joints_[i].getPosition(); current_state_.velocity[i] = joints_[i].getVelocity(); // There's no acceleration data available in a joint handle @@ -425,37 +448,167 @@ update(const ros::Time& time, const ros::Duration& period) desired_state_.velocity[i] = desired_joint_state_.velocity[0]; desired_state_.acceleration[i] = 0.0; + state_joint_error_.position[0] = desired_joint_state_.position[0] - current_state_.position[i]; + state_joint_error_.velocity[0] = desired_joint_state_.velocity[0] - current_state_.velocity[i]; + state_joint_error_.acceleration[0] = 0.0; + state_error_.position[i] = desired_joint_state_.position[0] - current_state_.position[i]; state_error_.velocity[i] = desired_joint_state_.velocity[0] - current_state_.velocity[i]; state_error_.acceleration[i] = 0.0; - //ROS_WARN_STREAM_NAMED(name_, "Check tolerances for joint: "<getGoalHandle(); - if (rt_segment_goal && rt_segment_goal == rt_active_goal_) + if (i == 22) + if (rt_segment_goal) + { + int new_status = int(rt_segment_goal->gh_.getGoalStatus().status); + if (previous_status != new_status) + ROS_WARN_STREAM_NAMED(name_,"Status " << int(new_status) << " for goal: "<< rt_segment_goal->gh_.getGoalStatus().goal_id.id); + previous_status = new_status; + } + } + + //if (rt_active_goals_.size()>0) + // ROS_ERROR_STREAM_NAMED(name_,"****************** Number rt_active_goals_:" << rt_active_goals_.size()); + + // Check tolerances if segment corresponds to currently active action goal + for (unsigned int active_goal_id = 0; active_goal_id < rt_active_goals_.size(); ++active_goal_id) + { + std::vector permutation_vector = internal::permutation(rt_active_goals_[active_goal_id]->gh_.getGoal()->trajectory.joint_names, joint_names_); + std::vector successful_joint_traj; + successful_joint_traj.resize(permutation_vector.size(), false); + + //Check names in active goal + for (unsigned int j = 0; j < permutation_vector.size(); ++j) { - // Check tolerances - if (time_data.uptime.toSec() < segment_it->endTime()) + unsigned int joint_id = permutation_vector[j]; + //ROS_ERROR_STREAM_NAMED(name_,"Joint_name:" << joint_names_[joint_id]); + + typename Segment::State desired_joint_state_ = typename Segment::State(1); + typename Segment::State state_joint_error_= typename Segment::State(1); + + state_joint_error_.position[0] = state_error_.position[joint_id]; + state_joint_error_.velocity[0] = state_error_.velocity[joint_id]; + state_joint_error_.acceleration[0] = state_error_.acceleration[joint_id]; + + //sample trajectory and see if it is active + typename TrajectoryPerJoint::const_iterator segment_it = sample(curr_traj[joint_id], time_data.uptime.toSec(), desired_joint_state_); + const RealtimeGoalHandlePtr rt_segment_goal = segment_it->getGoalHandle(); + + if (rt_segment_goal && rt_segment_goal == rt_active_goals_[active_goal_id]) { - ROS_WARN_NAMED(name_, "Currently executing a segment: check path tolerances"); - // Currently executing a segment: check path tolerances - checkPathTolerances(state_error_, - *segment_it); + // Check tolerances + if (time_data.uptime.toSec() < segment_it->endTime()) + { + //ROS_WARN_NAMED(name_, "Currently executing a segment: check path tolerances"); + // Currently executing a segment: check path tolerances + const SegmentTolerancesPerJoint& joint_tolerances = segment_it->getTolerances(); + if (!checkStateTolerancePerJoint(state_joint_error_, joint_tolerances.state_tolerance)) + { + ROS_ERROR_STREAM_NAMED(name_, "PATH_TOLERANCE_VIOLATED! joint_id:" << joint_id); + rt_segment_goal->preallocated_result_->error_code = + control_msgs::FollowJointTrajectoryResult::PATH_TOLERANCE_VIOLATED; + rt_segment_goal->setAborted(rt_segment_goal->preallocated_result_); + + rt_active_goals_[active_goal_id].reset(); + //ROS_ERROR_STREAM_NAMED(name_,"Trajectory failed #active_goals: " << rt_active_goals_.size()); + rt_active_goals_.erase(rt_active_goals_.begin()+active_goal_id); + //ROS_ERROR_STREAM_NAMED(name_,"Trajectory failed after #active_goals: " << rt_active_goals_.size()); + break; + } + } + else if (segment_it == --curr_traj[joint_id].end()) + { + //ROS_WARN_NAMED(name_, "Finished executing the LAST segment: check goal tolerances"); + if (verbose_) + ROS_DEBUG_STREAM_THROTTLE_NAMED(1,name_,"Finished executing last segment, checking goal tolerances"); + + // Controller uptime + const ros::Time uptime = time_data_.readFromRT()->uptime; + + // Checks that we have ended inside the goal tolerances + const SegmentTolerancesPerJoint& tolerances = segment_it->getTolerances(); + const bool inside_goal_tolerances = checkStateTolerancePerJoint(state_joint_error_, tolerances.goal_state_tolerance); + + if (inside_goal_tolerances) + { + //ROS_WARN_STREAM_NAMED(name_, "TrajectoryResult::SUCCESSFUL joint_id: "<< joint_id); + successful_joint_traj[joint_id] = true; + + } + else if (uptime.toSec() < segment_it->endTime() + tolerances.goal_time_tolerance) + { + //ROS_WARN_STREAM_NAMED(name_, "Still have some time left joint_id: "<< joint_id); + // Still have some time left to meet the goal state tolerances + } + else + { + if (verbose_) + { + ROS_ERROR_STREAM_NAMED(name_,"Goal tolerances failed"); + // Check the tolerances one more time to output the errors that occurs + checkStateTolerancePerJoint(state_joint_error_, tolerances.goal_state_tolerance, true); + } + + //ROS_ERROR_STREAM_NAMED(name_,"Aborting active_goal: " << rt_active_goals_[active_goal_id]->gh_.getGoalID().id); + ROS_ERROR_STREAM_NAMED(name_,"Aborting active_goal: " << rt_segment_goal->gh_.getGoalID().id); + + + //ROS_ERROR_STREAM_NAMED(name_, "GOAL_TOLERANCE_VIOLATED goal_id:"<< active_goal_id << " joint_id:" << joint_id); + //ROS_ERROR_STREAM_NAMED(name_,"Status before aborting: " << rt_segment_goal->gh_.getGoalStatus() ); + //rt_active_goals_[active_goal_id]->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::GOAL_TOLERANCE_VIOLATED; + //rt_active_goals_[active_goal_id]->setAborted(rt_segment_goal->preallocated_result_); + + RealtimeGoalHandlePtr current_active_goal(rt_active_goals_[active_goal_id]); + current_active_goal->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::GOAL_TOLERANCE_VIOLATED; + current_active_goal->setAborted(current_active_goal->preallocated_result_); + + //ROS_ERROR_STREAM_NAMED(name_,"Status after aborting: " << rt_segment_goal->gh_.getGoalStatus() ); + //rt_active_goals_[active_goal_id].reset(); + //ROS_ERROR_STREAM_NAMED(name_,"Trajectory failed #active_goals: " << rt_active_goals_.size()); + + //rt_active_goals_.erase(rt_active_goals_.begin()+active_goal_id); + //ROS_ERROR_STREAM_NAMED(name_,"Trajectory failed after #active_goals: " << rt_active_goals_.size()); + + break; + } + } } - else if (segment_it == --curr_traj[i].end()) + } + + if (std::find(successful_joint_traj.begin(), successful_joint_traj.end(), false) == successful_joint_traj.end()) // All true + { + ROS_WARN_NAMED(name_, "Whole TrajectoryResult::SUCCESSFUL !!!"); + + for (unsigned int j = 0; j < permutation_vector.size(); ++j) { - ROS_WARN_NAMED(name_, "Finished executing the LAST segment: check goal tolerances"); - if (verbose_) - ROS_DEBUG_STREAM_THROTTLE_NAMED(1,name_,"Finished executing last segment, checking goal tolerances"); + unsigned int joint_id = permutation_vector[j]; + + typename Segment::State desired_joint_state_ = typename Segment::State(1); + typename TrajectoryPerJoint::const_iterator segment_it = sample(curr_traj[joint_id], time_data.uptime.toSec(), desired_joint_state_); + const RealtimeGoalHandlePtr rt_segment_goal = segment_it->getGoalHandle(); - // Finished executing the LAST segment: check goal tolerances - checkGoalTolerances(state_error_, - *segment_it); + rt_segment_goal->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::SUCCESSFUL; + rt_segment_goal->setSucceeded(rt_segment_goal->preallocated_result_); } + rt_active_goals_[active_goal_id].reset(); + rt_active_goals_.erase(rt_active_goals_.begin()+active_goal_id); + break; } - } + } + // Check which goals have finished and reset pointer + for (unsigned int active_goal_id = 0; active_goal_id < rt_active_goals_.size(); ++active_goal_id) + { + int status = int(rt_active_goals_[active_goal_id]->gh_.getGoalStatus().status); + if (status == 3 or status == 4) + { + ROS_WARN_STREAM_NAMED(name_,"**********Status " << int(status) << " for goal: "<< rt_active_goals_[active_goal_id]->gh_.getGoalStatus().goal_id.id); + rt_active_goals_[active_goal_id].reset(); + rt_active_goals_.erase(rt_active_goals_.begin()+active_goal_id); + break; // Find the way to erase pointer!!!!!!!!!!!!! + } + } // Hardware interface adapter: Generate and send commands hw_iface_adapter_.updateCommand(time_data.uptime, time_data.period, @@ -584,9 +737,11 @@ goalCB(GoalHandle gh) // create a new one and do not preempt the other // store them in an array of active goals - preemptActiveGoal(); + //preempt if same joints + //preemptActiveGoal(); gh.setAccepted(); - rt_active_goal_ = rt_goal; + rt_active_goals_.push_back(rt_goal); + ROS_ERROR_STREAM_NAMED(name_,"Trajectory accepted #active_goals: " << rt_active_goals_.size()); // Setup goal status checking timer goal_handle_timer_ = controller_nh_.createTimer(action_monitor_period_, @@ -607,23 +762,29 @@ template void JointTrajectoryController:: cancelCB(GoalHandle gh) { - RealtimeGoalHandlePtr current_active_goal(rt_active_goal_); + //RealtimeGoalHandlePtr current_active_goal(rt_active_goal_); + std::vector current_active_goals(rt_active_goals_); // Check that cancel request refers to currently active goal (if any) - if (current_active_goal && current_active_goal->gh_ == gh) + for (unsigned int i = 0; i < current_active_goals.size(); ++i) { - // Reset current goal - rt_active_goal_.reset(); + if (current_active_goals[i] && current_active_goals[i]->gh_ == gh) + { + // Reset current goal + rt_active_goals_[i].reset(); + rt_active_goals_.erase(rt_active_goals_.begin()+i); - // Controller uptime - const ros::Time uptime = time_data_.readFromRT()->uptime; + // Controller uptime + const ros::Time uptime = time_data_.readFromRT()->uptime; - // Enter hold current position mode - setHoldPosition(uptime); - ROS_DEBUG_NAMED(name_, "Canceling active action goal because cancel callback recieved from actionlib."); + // Enter hold current position mode + setHoldPosition(uptime); + ROS_DEBUG_NAMED(name_, "Canceling active action goal because cancel callback received from actionlib."); - // Mark the current goal as canceled - current_active_goal->gh_.setCanceled(); + // Mark the current goal as canceled + current_active_goals[i]->gh_.setCanceled(); + return; + } } } diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_segment.h b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_segment.h index 07c2d8160..528f41d7f 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_segment.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_segment.h @@ -171,7 +171,7 @@ class JointTrajectorySegment : public Segment const Time& end_time, const State& end_state) : rt_goal_handle_(), - tolerances_(start_state.position.size()) + tolerances_() { Segment::init(start_time, start_state, end_time, end_state); } @@ -194,7 +194,7 @@ class JointTrajectorySegment : public Segment const std::vector& permutation = std::vector(), const std::vector& position_offset = std::vector()) : rt_goal_handle_(), - tolerances_(start_point.positions.size()) + tolerances_() { if (start_point.positions.size() != end_point.positions.size()) { @@ -227,14 +227,14 @@ class JointTrajectorySegment : public Segment void setGoalHandle(RealtimeGoalHandlePtr rt_goal_handle) {rt_goal_handle_ = rt_goal_handle;} /** \return Tolerances this segment is associated to. */ - const SegmentTolerances& getTolerances() const {return tolerances_;} + const SegmentTolerancesPerJoint& getTolerances() const {return tolerances_;} /** \brief Set the tolerances this segment is associated to. */ - void setTolerances(const SegmentTolerances& tolerances) {tolerances_ = tolerances;} + void setTolerances(const SegmentTolerancesPerJoint& tolerances) {tolerances_ = tolerances;} private: RealtimeGoalHandlePtr rt_goal_handle_; - SegmentTolerances tolerances_; + SegmentTolerancesPerJoint tolerances_; }; /** diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h b/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h index 69172f9f4..e10a198e2 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h @@ -88,6 +88,28 @@ struct SegmentTolerances Scalar goal_time_tolerance; }; +/** + * \brief Trajectory segment tolerances per Joint (to replace the previous function + */ +template +struct SegmentTolerancesPerJoint +{ + SegmentTolerancesPerJoint() + : state_tolerance(static_cast(0.0)), + goal_state_tolerance(static_cast(0.0)), + goal_time_tolerance(static_cast(0.0)) + {} + + /** State tolerances that appply during segment execution. */ + StateTolerances state_tolerance; + + /** State tolerances that apply for the goal state only.*/ + StateTolerances goal_state_tolerance; + + /** Extra time after the segment end time allowed to reach the goal state tolerances. */ + Scalar goal_time_tolerance; +}; + /** * \param state_error State error to check. * \param state_tolerance State tolerances to check \p state_error against. @@ -136,6 +158,45 @@ inline bool checkStateTolerance(const State& return true; } +/** + * \param state_error State error to check. + * \param state_tolerance State tolerances to check \p state_error against. + * \param show_errors If the joints that violate their tolerances should be output to console. NOT REALTIME if true + * \return True if \p state_error fulfills \p state_tolerance. + */ +template +inline bool checkStateTolerancePerJoint(const State& state_error, + const StateTolerances& state_tolerance, + bool show_errors = false) +{ + + using std::abs; + + const bool is_valid = !(state_tolerance.position > 0.0 && abs(state_error.position[0]) > state_tolerance.position) && + !(state_tolerance.velocity > 0.0 && abs(state_error.velocity[0]) > state_tolerance.velocity) && + !(state_tolerance.acceleration > 0.0 && abs(state_error.acceleration[0]) > state_tolerance.acceleration); + + if (!is_valid) + { + if( show_errors ) + { + ROS_ERROR_STREAM_NAMED("tolerances","Path state tolerances failed on joint"); + + if (state_tolerance.position > 0.0 && abs(state_error.position[0]) > state_tolerance.position) + ROS_ERROR_STREAM_NAMED("tolerances","Position Error: " << state_error.position[0] << + " Position Tolerance: " << state_tolerance.position); + if (state_tolerance.velocity > 0.0 && abs(state_error.velocity[0]) > state_tolerance.velocity) + ROS_ERROR_STREAM_NAMED("tolerances","Velocity Error: " << state_error.velocity[0] << + " Velocity Tolerance: " << state_tolerance.velocity); + if (state_tolerance.acceleration > 0.0 && abs(state_error.acceleration[0]) > state_tolerance.acceleration) + ROS_ERROR_STREAM_NAMED("tolerances","Acceleration Error: " << state_error.acceleration[0] << + " Acceleration Tolerance: " << state_tolerance.acceleration); + } + return false; + } + return true; +} + /** * \brief Update data in \p tols from data in \p msg_tol. * From dc156d5bd3bbc097aebc73f3bb32153ebf04a4f5 Mon Sep 17 00:00:00 2001 From: Beatriz Leon Date: Thu, 30 Jul 2015 11:50:39 +0100 Subject: [PATCH 04/18] Partial trajectory working. Needs cleaning and check tolerances --- .../init_joint_trajectory.h | 11 + .../joint_trajectory_controller.h | 6 +- .../joint_trajectory_controller_impl.h | 262 ++++++------------ .../joint_trajectory_controller/tolerances.h | 2 +- 4 files changed, 97 insertions(+), 184 deletions(-) diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h index 52ef6997d..b4bc1f217 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h @@ -330,11 +330,22 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg } } + + // Initialize result trajectory: combination of: // - Useful segments of currently followed trajectory // - Useful segments of new trajectory (contained in ROS message) Trajectory result_traj = *(options.current_trajectory); + //Iterate to all segments to set the new goal handler + for (unsigned int joint_id=0; joint_id < joint_names.size();joint_id++) + { + for (unsigned int segment_id=0; segment_id < result_traj[joint_id].size(); segment_id++) + { + (result_traj[joint_id])[segment_id].setGoalHandle(options.rt_goal_handle); + } + } + std::vector permutation_vector_per_joint(1,0); //refactor this and remove it as it is not needed //Iterate through the joints that are in the message, in the order of the permutation vector diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h index de0badae6..f5b3845e7 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h @@ -186,8 +186,8 @@ class JointTrajectoryController : public controller_interface::Controller default_tolerances_; ///< Default trajectory segment tolerances. HwIfaceAdapter hw_iface_adapter_; ///< Adapts desired trajectory state to HW interface. - //RealtimeGoalHandlePtr rt_active_goal_; ///< Currently active action goal, if any. - std::vector rt_active_goals_; ///< Currently active action goal, if any. + RealtimeGoalHandlePtr rt_active_goal_; ///< Currently active action goal, if any. + //std::vector rt_active_goals_; ///< Currently active action goal, if any. /** * Thread-safe container with a smart pointer to trajectory currently being followed. @@ -227,7 +227,7 @@ class JointTrajectoryController : public controller_interface::Controller inline void JointTrajectoryController:: stopping(const ros::Time& time) { - preemptActiveGoals(); + preemptActiveGoal(); } template @@ -161,45 +161,24 @@ inline void JointTrajectoryController:: trajectoryCommandCB(const JointTrajectoryConstPtr& msg) { const bool update_ok = updateTrajectoryCommand(msg, RealtimeGoalHandlePtr()); - if (update_ok) {preemptActiveGoals();} + if (update_ok) {preemptActiveGoal();} } template inline void JointTrajectoryController:: -preemptActiveGoals() +preemptActiveGoal() { - // Check that cancel request refers to currently active goals (if any) - for (unsigned int i = 0; i < rt_active_goals_.size(); ++i) - { - RealtimeGoalHandlePtr current_active_goal(rt_active_goals_[i]); + RealtimeGoalHandlePtr current_active_goal(rt_active_goal_); - // Cancels the currently active goal - if (current_active_goal) - { - // Marks the current goal as canceled - rt_active_goals_[i].reset(); - rt_active_goals_.erase(rt_active_goals_.begin()+i); - current_active_goal->gh_.setCanceled(); - } + // Cancels the currently active goal + if (current_active_goal) + { + // Marks the current goal as canceled + rt_active_goal_.reset(); + current_active_goal->gh_.setCanceled(); } } -//template -//inline void JointTrajectoryController:: -//preemptActiveGoal() -//{ -// -// RealtimeGoalHandlePtr current_active_goal(rt_active_goal_); -// -// // Cancels the currently active goal -// if (current_active_goal) -// { -// // Marks the current goal as canceled -// rt_active_goal_.reset(); -// current_active_goal->gh_.setCanceled(); -// } -//} - //template //inline void JointTrajectoryController:: //checkPathTolerances(const typename Segment::State& state_error, @@ -266,7 +245,6 @@ bool JointTrajectoryController::init(HardwareInt ros::NodeHandle& root_nh, ros::NodeHandle& controller_nh) { - ROS_ERROR_NAMED(name_,"*********************init"); using namespace internal; // Cache controller node handle @@ -385,7 +363,6 @@ bool JointTrajectoryController::init(HardwareInt joint_segment.resize(1, hold_segment); hold_trajectory_ptr_->push_back(joint_segment); } - ROS_ERROR_STREAM_NAMED(name_,"hold_trajectory_ptr_->size():" << hold_trajectory_ptr_->size()); { state_publisher_->lock(); @@ -399,6 +376,7 @@ bool JointTrajectoryController::init(HardwareInt state_publisher_->msg_.error.velocities.resize(n_joints); state_publisher_->unlock(); } + return true; } @@ -467,149 +445,86 @@ update(const ros::Time& time, const ros::Duration& period) } } - //if (rt_active_goals_.size()>0) - // ROS_ERROR_STREAM_NAMED(name_,"****************** Number rt_active_goals_:" << rt_active_goals_.size()); - - // Check tolerances if segment corresponds to currently active action goal - for (unsigned int active_goal_id = 0; active_goal_id < rt_active_goals_.size(); ++active_goal_id) + std::vector successful_joint_traj; + successful_joint_traj.resize(joints_.size(), false); + //Check tolerances + for (unsigned int joint_id = 0; joint_id < joints_.size(); ++joint_id) { - std::vector permutation_vector = internal::permutation(rt_active_goals_[active_goal_id]->gh_.getGoal()->trajectory.joint_names, joint_names_); - std::vector successful_joint_traj; - successful_joint_traj.resize(permutation_vector.size(), false); - - //Check names in active goal - for (unsigned int j = 0; j < permutation_vector.size(); ++j) - { - unsigned int joint_id = permutation_vector[j]; - //ROS_ERROR_STREAM_NAMED(name_,"Joint_name:" << joint_names_[joint_id]); - - typename Segment::State desired_joint_state_ = typename Segment::State(1); - typename Segment::State state_joint_error_= typename Segment::State(1); + typename Segment::State desired_joint_state_ = typename Segment::State(1); + typename Segment::State state_joint_error_= typename Segment::State(1); - state_joint_error_.position[0] = state_error_.position[joint_id]; - state_joint_error_.velocity[0] = state_error_.velocity[joint_id]; - state_joint_error_.acceleration[0] = state_error_.acceleration[joint_id]; + state_joint_error_.position[0] = state_error_.position[joint_id]; + state_joint_error_.velocity[0] = state_error_.velocity[joint_id]; + state_joint_error_.acceleration[0] = state_error_.acceleration[joint_id]; - //sample trajectory and see if it is active - typename TrajectoryPerJoint::const_iterator segment_it = sample(curr_traj[joint_id], time_data.uptime.toSec(), desired_joint_state_); - const RealtimeGoalHandlePtr rt_segment_goal = segment_it->getGoalHandle(); + //sample trajectory and see if it is active + typename TrajectoryPerJoint::const_iterator segment_it = sample(curr_traj[joint_id], time_data.uptime.toSec(), desired_joint_state_); + const RealtimeGoalHandlePtr rt_segment_goal = segment_it->getGoalHandle(); - if (rt_segment_goal && rt_segment_goal == rt_active_goals_[active_goal_id]) + if (rt_segment_goal && rt_segment_goal == rt_active_goal_) + { + // Check tolerances + if (time_data.uptime.toSec() < segment_it->endTime()) { - // Check tolerances - if (time_data.uptime.toSec() < segment_it->endTime()) + // Currently executing a segment: check path tolerances + const SegmentTolerancesPerJoint& joint_tolerances = segment_it->getTolerances(); + if (!checkStateTolerancePerJoint(state_joint_error_, joint_tolerances.state_tolerance)) { - //ROS_WARN_NAMED(name_, "Currently executing a segment: check path tolerances"); - // Currently executing a segment: check path tolerances - const SegmentTolerancesPerJoint& joint_tolerances = segment_it->getTolerances(); - if (!checkStateTolerancePerJoint(state_joint_error_, joint_tolerances.state_tolerance)) - { - ROS_ERROR_STREAM_NAMED(name_, "PATH_TOLERANCE_VIOLATED! joint_id:" << joint_id); - rt_segment_goal->preallocated_result_->error_code = - control_msgs::FollowJointTrajectoryResult::PATH_TOLERANCE_VIOLATED; - rt_segment_goal->setAborted(rt_segment_goal->preallocated_result_); - - rt_active_goals_[active_goal_id].reset(); - //ROS_ERROR_STREAM_NAMED(name_,"Trajectory failed #active_goals: " << rt_active_goals_.size()); - rt_active_goals_.erase(rt_active_goals_.begin()+active_goal_id); - //ROS_ERROR_STREAM_NAMED(name_,"Trajectory failed after #active_goals: " << rt_active_goals_.size()); - break; - } + ROS_ERROR_STREAM_NAMED(name_, "PATH_TOLERANCE_VIOLATED! joint_id:" << joint_id); + rt_segment_goal->preallocated_result_->error_code = + control_msgs::FollowJointTrajectoryResult::PATH_TOLERANCE_VIOLATED; + rt_segment_goal->setAborted(rt_segment_goal->preallocated_result_); + rt_active_goal_.reset(); } - else if (segment_it == --curr_traj[joint_id].end()) - { - //ROS_WARN_NAMED(name_, "Finished executing the LAST segment: check goal tolerances"); - if (verbose_) - ROS_DEBUG_STREAM_THROTTLE_NAMED(1,name_,"Finished executing last segment, checking goal tolerances"); + } + else if (segment_it == --curr_traj[joint_id].end()) + { + if (verbose_) + ROS_DEBUG_STREAM_THROTTLE_NAMED(1,name_,"Finished executing last segment, checking goal tolerances"); - // Controller uptime - const ros::Time uptime = time_data_.readFromRT()->uptime; + // Controller uptime + const ros::Time uptime = time_data_.readFromRT()->uptime; - // Checks that we have ended inside the goal tolerances - const SegmentTolerancesPerJoint& tolerances = segment_it->getTolerances(); - const bool inside_goal_tolerances = checkStateTolerancePerJoint(state_joint_error_, tolerances.goal_state_tolerance); + // Checks that we have ended inside the goal tolerances + const SegmentTolerancesPerJoint& tolerances = segment_it->getTolerances(); + const bool inside_goal_tolerances = checkStateTolerancePerJoint(state_joint_error_, tolerances.goal_state_tolerance); - if (inside_goal_tolerances) - { - //ROS_WARN_STREAM_NAMED(name_, "TrajectoryResult::SUCCESSFUL joint_id: "<< joint_id); - successful_joint_traj[joint_id] = true; + if (inside_goal_tolerances) + { + //ROS_WARN_STREAM_NAMED(name_, "TrajectoryResult::SUCCESSFUL joint_id: "<< joint_id); + successful_joint_traj[joint_id] = true; - } - else if (uptime.toSec() < segment_it->endTime() + tolerances.goal_time_tolerance) + } + else if (uptime.toSec() < segment_it->endTime() + tolerances.goal_time_tolerance) + { + // Still have some time left to meet the goal state tolerances + } + else + { + if (verbose_) { - //ROS_WARN_STREAM_NAMED(name_, "Still have some time left joint_id: "<< joint_id); - // Still have some time left to meet the goal state tolerances + ROS_ERROR_STREAM_NAMED(name_,"Goal tolerances failed"); + // Check the tolerances one more time to output the errors that occurs + checkStateTolerancePerJoint(state_joint_error_, tolerances.goal_state_tolerance, true); } - else - { - if (verbose_) - { - ROS_ERROR_STREAM_NAMED(name_,"Goal tolerances failed"); - // Check the tolerances one more time to output the errors that occurs - checkStateTolerancePerJoint(state_joint_error_, tolerances.goal_state_tolerance, true); - } - - //ROS_ERROR_STREAM_NAMED(name_,"Aborting active_goal: " << rt_active_goals_[active_goal_id]->gh_.getGoalID().id); - ROS_ERROR_STREAM_NAMED(name_,"Aborting active_goal: " << rt_segment_goal->gh_.getGoalID().id); - - //ROS_ERROR_STREAM_NAMED(name_, "GOAL_TOLERANCE_VIOLATED goal_id:"<< active_goal_id << " joint_id:" << joint_id); - //ROS_ERROR_STREAM_NAMED(name_,"Status before aborting: " << rt_segment_goal->gh_.getGoalStatus() ); - //rt_active_goals_[active_goal_id]->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::GOAL_TOLERANCE_VIOLATED; - //rt_active_goals_[active_goal_id]->setAborted(rt_segment_goal->preallocated_result_); - - RealtimeGoalHandlePtr current_active_goal(rt_active_goals_[active_goal_id]); - current_active_goal->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::GOAL_TOLERANCE_VIOLATED; - current_active_goal->setAborted(current_active_goal->preallocated_result_); - - //ROS_ERROR_STREAM_NAMED(name_,"Status after aborting: " << rt_segment_goal->gh_.getGoalStatus() ); - //rt_active_goals_[active_goal_id].reset(); - //ROS_ERROR_STREAM_NAMED(name_,"Trajectory failed #active_goals: " << rt_active_goals_.size()); - - //rt_active_goals_.erase(rt_active_goals_.begin()+active_goal_id); - //ROS_ERROR_STREAM_NAMED(name_,"Trajectory failed after #active_goals: " << rt_active_goals_.size()); - - break; - } + rt_segment_goal->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::GOAL_TOLERANCE_VIOLATED; + rt_segment_goal->setAborted(rt_segment_goal->preallocated_result_); + rt_active_goal_.reset(); } } } - - if (std::find(successful_joint_traj.begin(), successful_joint_traj.end(), false) == successful_joint_traj.end()) // All true - { - ROS_WARN_NAMED(name_, "Whole TrajectoryResult::SUCCESSFUL !!!"); - - for (unsigned int j = 0; j < permutation_vector.size(); ++j) - { - unsigned int joint_id = permutation_vector[j]; - - typename Segment::State desired_joint_state_ = typename Segment::State(1); - typename TrajectoryPerJoint::const_iterator segment_it = sample(curr_traj[joint_id], time_data.uptime.toSec(), desired_joint_state_); - const RealtimeGoalHandlePtr rt_segment_goal = segment_it->getGoalHandle(); - - rt_segment_goal->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::SUCCESSFUL; - rt_segment_goal->setSucceeded(rt_segment_goal->preallocated_result_); - } - rt_active_goals_[active_goal_id].reset(); - rt_active_goals_.erase(rt_active_goals_.begin()+active_goal_id); - break; - } - } - // Check which goals have finished and reset pointer - for (unsigned int active_goal_id = 0; active_goal_id < rt_active_goals_.size(); ++active_goal_id) + if (std::find(successful_joint_traj.begin(), successful_joint_traj.end(), false) == successful_joint_traj.end()) // All true { - int status = int(rt_active_goals_[active_goal_id]->gh_.getGoalStatus().status); - if (status == 3 or status == 4) - { - ROS_WARN_STREAM_NAMED(name_,"**********Status " << int(status) << " for goal: "<< rt_active_goals_[active_goal_id]->gh_.getGoalStatus().goal_id.id); - rt_active_goals_[active_goal_id].reset(); - rt_active_goals_.erase(rt_active_goals_.begin()+active_goal_id); - break; // Find the way to erase pointer!!!!!!!!!!!!! - } + rt_active_goal_->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::SUCCESSFUL; + rt_active_goal_->setSucceeded(rt_active_goal_->preallocated_result_); + rt_active_goal_.reset(); } + //} + // Hardware interface adapter: Generate and send commands hw_iface_adapter_.updateCommand(time_data.uptime, time_data.period, desired_state_, state_error_); @@ -732,16 +647,9 @@ goalCB(GoalHandle gh) if (update_ok) { // Accept new goal - - // If new goal has less joints than current active goal - // create a new one and do not preempt the other - // store them in an array of active goals - - //preempt if same joints - //preemptActiveGoal(); + preemptActiveGoal(); gh.setAccepted(); - rt_active_goals_.push_back(rt_goal); - ROS_ERROR_STREAM_NAMED(name_,"Trajectory accepted #active_goals: " << rt_active_goals_.size()); + rt_active_goal_ = rt_goal; // Setup goal status checking timer goal_handle_timer_ = controller_nh_.createTimer(action_monitor_period_, @@ -762,29 +670,23 @@ template void JointTrajectoryController:: cancelCB(GoalHandle gh) { - //RealtimeGoalHandlePtr current_active_goal(rt_active_goal_); - std::vector current_active_goals(rt_active_goals_); + RealtimeGoalHandlePtr current_active_goal(rt_active_goal_); // Check that cancel request refers to currently active goal (if any) - for (unsigned int i = 0; i < current_active_goals.size(); ++i) + if (current_active_goal && current_active_goal->gh_ == gh) { - if (current_active_goals[i] && current_active_goals[i]->gh_ == gh) - { - // Reset current goal - rt_active_goals_[i].reset(); - rt_active_goals_.erase(rt_active_goals_.begin()+i); + // Reset current goal + rt_active_goal_.reset(); - // Controller uptime - const ros::Time uptime = time_data_.readFromRT()->uptime; + // Controller uptime + const ros::Time uptime = time_data_.readFromRT()->uptime; - // Enter hold current position mode - setHoldPosition(uptime); - ROS_DEBUG_NAMED(name_, "Canceling active action goal because cancel callback received from actionlib."); + // Enter hold current position mode + setHoldPosition(uptime); + ROS_DEBUG_NAMED(name_, "Canceling active action goal because cancel callback recieved from actionlib."); - // Mark the current goal as canceled - current_active_goals[i]->gh_.setCanceled(); - return; - } + // Mark the current goal as canceled + current_active_goal->gh_.setCanceled(); } } diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h b/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h index e10a198e2..2bf707411 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h @@ -172,7 +172,7 @@ inline bool checkStateTolerancePerJoint(const State& using std::abs; - const bool is_valid = !(state_tolerance.position > 0.0 && abs(state_error.position[0]) > state_tolerance.position) && + const bool is_valid = !(state_tolerance.position > 0.0 && abs(state_error.position[0]) > state_tolerance.position); //&& !(state_tolerance.velocity > 0.0 && abs(state_error.velocity[0]) > state_tolerance.velocity) && !(state_tolerance.acceleration > 0.0 && abs(state_error.acceleration[0]) > state_tolerance.acceleration); From c54846f454d0bc485e86097d201f582aa2064b52 Mon Sep 17 00:00:00 2001 From: Beatriz Leon Date: Thu, 30 Jul 2015 13:06:28 +0100 Subject: [PATCH 05/18] Working and cleaned --- .../init_joint_trajectory.h | 58 +-------------- .../joint_trajectory_controller.h | 3 +- .../joint_trajectory_controller_impl.h | 71 ++++++++----------- .../joint_trajectory_segment.h | 44 +----------- .../joint_trajectory_controller/tolerances.h | 5 +- 5 files changed, 36 insertions(+), 145 deletions(-) diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h index b4bc1f217..b51ce83da 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h @@ -66,9 +66,6 @@ inline std::vector permutation(const T& t1, const T& t2) { typedef unsigned int SizeType; - // Arguments must have the same size - //if (t1.size() != t2.size()) {return std::vector();} - std::vector permutation_vector(t1.size()); // Return value for (typename T::const_iterator t1_it = t1.begin(); t1_it != t1.end(); ++t1_it) { @@ -246,33 +243,7 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg // If unspecified, a trivial map (no permutation) is computed const std::vector joint_names = has_joint_names ? *(options.joint_names) : msg.joint_names; - std::stringstream log_str; - - -// ROS_INFO_STREAM("Joint names size" << joint_names.size()); -// -// log_str.str(""); -// log_str << "Joint names size:" ; -// for (unsigned int i=0; i < joint_names.size();i++) log_str << joint_names[i] << "\t"; -// log_str << "\n"; -// ROS_INFO_STREAM(log_str.str()); -// -// log_str.str(""); -// log_str << "msg.joint_names size:" ; -// for (unsigned int i=0; i < msg.joint_names.size();i++) log_str << msg.joint_names[i] << "\t"; -// log_str << "\n"; -// ROS_INFO_STREAM(log_str.str()); -// -// log_str.str(""); -// log_str << "msg.position size:" ; -// for (unsigned int i=0; i < msg.points.size();i++) -// { -// for (unsigned int j=0; j < msg.points[i].positions.size();j++) -// log_str << msg.points[i].positions[j] << "\t"; -// log_str << "\n"; -// } -// ROS_INFO_STREAM(log_str.str()); - + //std::stringstream log_str; std::vector permutation_vector = internal::permutation(msg.joint_names,joint_names); if (permutation_vector.empty()) @@ -281,13 +252,6 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg return Trajectory(); } -// //std::stringstream log_str; -// log_str.str(""); -// log_str << "permutation_vector:" ; -// for (unsigned int i=0; i < permutation_vector.size();i++) log_str << permutation_vector[i] << "\t"; -// log_str << "\n"; -// ROS_INFO_STREAM(log_str.str()); - // Tolerances to be used in all new segments SegmentTolerances tolerances = has_default_tolerances ? *(options.default_tolerances) : SegmentTolerances(n_joints); @@ -305,7 +269,6 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg if (msg_it == msg.points.end()) { msg_it = msg.points.begin(); // Entire trajectory is after current time - ROS_WARN_STREAM("Entire trajectory is after current time"); } else { @@ -330,8 +293,6 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg } } - - // Initialize result trajectory: combination of: // - Useful segments of currently followed trajectory // - Useful segments of new trajectory (contained in ROS message) @@ -375,12 +336,6 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg typename Segment::State last_curr_state; sample(curr_joint_traj, last_curr_time, last_curr_state); -// log_str.str(""); -// log_str << "last_curr_state.position:" ; -// for (unsigned int i=0; i < last_curr_state.position.size();i++) log_str << last_curr_state.position[i] << "\t"; -// log_str << "\n"; -// ROS_INFO_STREAM(log_str.str()); - // Get the first time and state that will be executed from the new trajectory trajectory_msgs::JointTrajectoryPoint point_per_joint; point_per_joint.positions.resize(1, it->positions[msg_joint_it]); @@ -392,12 +347,6 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg const typename Segment::Time first_new_time = o_msg_start_time.toSec() + (it->time_from_start).toSec(); typename Segment::State first_new_state(point_per_joint, permutation_vector_per_joint); // Here offsets are not yet applied -// log_str.str(""); -// log_str << "first_new_state.position:" ; -// for (unsigned int i=0; i < first_new_state.position.size();i++) log_str << first_new_state.position[i] << "\t"; -// log_str << "\n"; -// ROS_INFO_STREAM(log_str.str()); - // Compute offsets due to wrapping joints if (has_angle_wraparound) { @@ -428,7 +377,6 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg bridge_seg.setGoalHandle(options.rt_goal_handle); if (has_rt_goal_handle) {bridge_seg.setTolerances(tolerances_per_joint);} result_traj_per_joint.push_back(bridge_seg); - //ROS_INFO_STREAM("Finish adding old trajectory segment"); } // Constants used in log statement at the end @@ -442,7 +390,6 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg { std::vector::const_iterator next_it = it; ++next_it; - //ROS_INFO_STREAM("Inside while"); trajectory_msgs::JointTrajectoryPoint it_point_per_joint, next_it_point_per_joint; it_point_per_joint.positions.resize(1, it->positions[msg_joint_it]); it_point_per_joint.velocities.resize(1, it->velocities[msg_joint_it]); @@ -473,9 +420,8 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg " points) taken from the input trajectory.";} } else {log_str << ".";} - //ROS_WARN_STREAM(log_str.str()); + ROS_DEBUG_STREAM(log_str.str()); - //ROS_WARN_STREAM("********** result_traj_per_joint" << result_traj[joint_id].size()); result_traj[joint_id] = result_traj_per_joint; } diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h index f5b3845e7..ac361cd4d 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h @@ -203,8 +203,7 @@ class JointTrajectoryController : public controller_interface::Controller time_data_; diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h index eee823dd7..fb4107739 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h @@ -185,8 +185,8 @@ preemptActiveGoal() // const Segment& segment) //{ // const RealtimeGoalHandlePtr rt_segment_goal = segment.getGoalHandle(); -// const SegmentTolerancesPerJoint& tolerances = segment.getTolerances(); -// if (!checkStateTolerancePerJoint(state_error, tolerances.state_tolerance)) +// const SegmentTolerances& tolerances = segment.getTolerances(); +// if (!checkStateTolerance(state_error, tolerances.state_tolerance)) // { // rt_segment_goal->preallocated_result_->error_code = // control_msgs::FollowJointTrajectoryResult::PATH_TOLERANCE_VIOLATED; @@ -194,7 +194,7 @@ preemptActiveGoal() // rt_active_goal_.reset(); // } //} - +// //template //inline void JointTrajectoryController:: //checkGoalTolerances(const typename Segment::State& state_error, @@ -205,8 +205,8 @@ preemptActiveGoal() // // // Checks that we have ended inside the goal tolerances // const RealtimeGoalHandlePtr rt_segment_goal = segment.getGoalHandle(); -// const SegmentTolerancesPerJoint& tolerances = segment.getTolerances(); -// const bool inside_goal_tolerances = checkStateTolerancePerJoint(state_error, tolerances.goal_state_tolerance); +// const SegmentTolerances& tolerances = segment.getTolerances(); +// const bool inside_goal_tolerances = checkStateTolerance(state_error, tolerances.goal_state_tolerance); // // if (inside_goal_tolerances) // { @@ -348,9 +348,6 @@ bool JointTrajectoryController::init(HardwareInt desired_state_ = typename Segment::State(n_joints); state_error_ = typename Segment::State(n_joints); - hold_start_state_ = typename Segment::State(1); - hold_end_state_ = typename Segment::State(1); - // Initialize trajectory with all joints typename Segment::State current_joint_state_ = typename Segment::State(1); for (unsigned int i = 0; i < n_joints; ++i) @@ -404,6 +401,9 @@ update(const ros::Time& time, const ros::Duration& period) // fetch the currently followed trajectory, it has been updated by the non-rt thread with something that starts in the // next control cycle, leaving the current cycle without a valid trajectory. + std::vector successful_joint_traj; + successful_joint_traj.resize(joints_.size(), false); + // Update current state and state error for (unsigned int i = 0; i < joints_.size(); ++i) { @@ -434,33 +434,18 @@ update(const ros::Time& time, const ros::Duration& period) state_error_.velocity[i] = desired_joint_state_.velocity[0] - current_state_.velocity[i]; state_error_.acceleration[i] = 0.0; +// const RealtimeGoalHandlePtr rt_segment_goal = segment_it->getGoalHandle(); +// if (i == 22) +// if (rt_segment_goal) +// { +// int new_status = int(rt_segment_goal->gh_.getGoalStatus().status); +// if (previous_status != new_status) +// ROS_WARN_STREAM_NAMED(name_,"Status " << int(new_status) << " for goal: "<< rt_segment_goal->gh_.getGoalStatus().goal_id.id); +// previous_status = new_status; +// } + + //Check tolerances const RealtimeGoalHandlePtr rt_segment_goal = segment_it->getGoalHandle(); - if (i == 22) - if (rt_segment_goal) - { - int new_status = int(rt_segment_goal->gh_.getGoalStatus().status); - if (previous_status != new_status) - ROS_WARN_STREAM_NAMED(name_,"Status " << int(new_status) << " for goal: "<< rt_segment_goal->gh_.getGoalStatus().goal_id.id); - previous_status = new_status; - } - } - - std::vector successful_joint_traj; - successful_joint_traj.resize(joints_.size(), false); - //Check tolerances - for (unsigned int joint_id = 0; joint_id < joints_.size(); ++joint_id) - { - typename Segment::State desired_joint_state_ = typename Segment::State(1); - typename Segment::State state_joint_error_= typename Segment::State(1); - - state_joint_error_.position[0] = state_error_.position[joint_id]; - state_joint_error_.velocity[0] = state_error_.velocity[joint_id]; - state_joint_error_.acceleration[0] = state_error_.acceleration[joint_id]; - - //sample trajectory and see if it is active - typename TrajectoryPerJoint::const_iterator segment_it = sample(curr_traj[joint_id], time_data.uptime.toSec(), desired_joint_state_); - const RealtimeGoalHandlePtr rt_segment_goal = segment_it->getGoalHandle(); - if (rt_segment_goal && rt_segment_goal == rt_active_goal_) { // Check tolerances @@ -470,14 +455,14 @@ update(const ros::Time& time, const ros::Duration& period) const SegmentTolerancesPerJoint& joint_tolerances = segment_it->getTolerances(); if (!checkStateTolerancePerJoint(state_joint_error_, joint_tolerances.state_tolerance)) { - ROS_ERROR_STREAM_NAMED(name_, "PATH_TOLERANCE_VIOLATED! joint_id:" << joint_id); + ROS_ERROR_STREAM_NAMED(name_, "PATH_TOLERANCE_VIOLATED! joint_id:" << i); rt_segment_goal->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::PATH_TOLERANCE_VIOLATED; rt_segment_goal->setAborted(rt_segment_goal->preallocated_result_); rt_active_goal_.reset(); } } - else if (segment_it == --curr_traj[joint_id].end()) + else if (segment_it == --curr_traj[i].end()) { if (verbose_) ROS_DEBUG_STREAM_THROTTLE_NAMED(1,name_,"Finished executing last segment, checking goal tolerances"); @@ -491,8 +476,8 @@ update(const ros::Time& time, const ros::Duration& period) if (inside_goal_tolerances) { - //ROS_WARN_STREAM_NAMED(name_, "TrajectoryResult::SUCCESSFUL joint_id: "<< joint_id); - successful_joint_traj[joint_id] = true; + //ROS_WARN_STREAM_NAMED(name_, "TrajectoryResult::SUCCESSFUL i: "<< i); + successful_joint_traj[i] = true; } else if (uptime.toSec() < segment_it->endTime() + tolerances.goal_time_tolerance) @@ -516,15 +501,14 @@ update(const ros::Time& time, const ros::Duration& period) } } - if (std::find(successful_joint_traj.begin(), successful_joint_traj.end(), false) == successful_joint_traj.end()) // All true + //If all segments finished successfully then set goal as succeeded + if (std::find(successful_joint_traj.begin(), successful_joint_traj.end(), false) == successful_joint_traj.end()) { rt_active_goal_->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::SUCCESSFUL; rt_active_goal_->setSucceeded(rt_active_goal_->preallocated_result_); rt_active_goal_.reset(); } - //} - // Hardware interface adapter: Generate and send commands hw_iface_adapter_.updateCommand(time_data.uptime, time_data.period, desired_state_, state_error_); @@ -775,6 +759,10 @@ setHoldPosition(const ros::Time& time) assert(joint_names_.size() == hold_trajectory_ptr_->size()); + ROS_INFO_STREAM_NAMED(name_, "Setting hold position"); + typename Segment::State hold_start_state_ = typename Segment::State(1); + typename Segment::State hold_end_state_ = typename Segment::State(1); + const typename Segment::Time start_time = time.toSec(); const typename Segment::Time end_time = time.toSec() + stop_trajectory_duration_; const typename Segment::Time end_time_2x = time.toSec() + 2.0 * stop_trajectory_duration_; @@ -791,7 +779,6 @@ setHoldPosition(const ros::Time& time) hold_end_state_.velocity[0] = -joints_[i].getVelocity(); hold_end_state_.acceleration[0] = 0.0; - //hold_traj[i].joint_trajectory.front() (*hold_trajectory_ptr_)[i].front().init(start_time, hold_start_state_, end_time_2x, hold_end_state_); diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_segment.h b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_segment.h index 528f41d7f..58ea6e0f8 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_segment.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_segment.h @@ -240,47 +240,9 @@ class JointTrajectorySegment : public Segment /** * \param prev_position Previous position from which to compute the wraparound offset. * \param next_position Next position from which to compute the wraparound offset. - * \param angle_wraparound Vector of booleans where true values correspond to joints that wrap around - * (ie. are continuous). Offsets will be computed only for these joints, otherwise they are set to zero. - * \return Wraparound offsets that should be applied to \p next_position such that no multi-turns are performed when - * transitioning from \p prev_position. - * \tparam Scalar Scalar type. - */ -template -std::vector wraparoundOffset(const std::vector& prev_position, - const std::vector& next_position, - const std::vector& angle_wraparound) -{ - // Preconditions - const unsigned int n_joints = angle_wraparound.size(); - if (n_joints != prev_position.size() || n_joints != next_position.size()) {return std::vector();} - - // Return value - std::vector pos_offset(n_joints, 0.0); - - for (unsigned int i = 0; i < angle_wraparound.size(); ++i) - { - if (angle_wraparound[i]) - { - Scalar dist = angles::shortest_angular_distance(prev_position[i], next_position[i]); - - // Deal with singularity at M_PI shortest distance - if (std::abs(dist) - M_PI < 1e-9) - { - dist = next_position[i] > prev_position[i] ? std::abs(dist) : -std::abs(dist); - } - pos_offset[i] = (prev_position[i] + dist) - next_position[i]; - } - } - return pos_offset; -} - -/** - * \param prev_position Previous position from which to compute the wraparound offset. - * \param next_position Next position from which to compute the wraparound offset. - * \param angle_wraparound Vector of booleans where true values correspond to joints that wrap around - * (ie. are continuous). Offsets will be computed only for these joints, otherwise they are set to zero. - * \return Wraparound offsets that should be applied to \p next_position such that no multi-turns are performed when + * \param angle_wraparound Boolean where true value corresponds to a joint that wrap around + * (ie. is continuous). Offset will be computed only for this joint, otherwise it is set to zero. + * \return Wraparound offset that should be applied to \p next_position such that no multi-turns are performed when * transitioning from \p prev_position. * \tparam Scalar Scalar type. */ diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h b/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h index 2bf707411..0ab5f49e2 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h @@ -161,7 +161,7 @@ inline bool checkStateTolerance(const State& /** * \param state_error State error to check. * \param state_tolerance State tolerances to check \p state_error against. - * \param show_errors If the joints that violate their tolerances should be output to console. NOT REALTIME if true + * \param show_errors If the joint that violate its tolerance should be output to console. NOT REALTIME if true * \return True if \p state_error fulfills \p state_tolerance. */ template @@ -305,9 +305,6 @@ SegmentTolerances getSegmentTolerances(const ros::NodeHandle& nh, nh.param(joint_names[i] + "/trajectory", tolerances.state_tolerance[i].position, 0.0); nh.param(joint_names[i] + "/goal", tolerances.goal_state_tolerance[i].position, 0.0); tolerances.goal_state_tolerance[i].velocity = stopped_velocity_tolerance; - - //ROS_ERROR_STREAM("**************** "< Date: Thu, 30 Jul 2015 13:14:54 +0100 Subject: [PATCH 06/18] Working and cleaned --- .../joint_trajectory_controller_impl.h | 66 +------------------ 1 file changed, 1 insertion(+), 65 deletions(-) diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h index fb4107739..5a457a049 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h @@ -179,60 +179,6 @@ preemptActiveGoal() } } -//template -//inline void JointTrajectoryController:: -//checkPathTolerances(const typename Segment::State& state_error, -// const Segment& segment) -//{ -// const RealtimeGoalHandlePtr rt_segment_goal = segment.getGoalHandle(); -// const SegmentTolerances& tolerances = segment.getTolerances(); -// if (!checkStateTolerance(state_error, tolerances.state_tolerance)) -// { -// rt_segment_goal->preallocated_result_->error_code = -// control_msgs::FollowJointTrajectoryResult::PATH_TOLERANCE_VIOLATED; -// rt_segment_goal->setAborted(rt_segment_goal->preallocated_result_); -// rt_active_goal_.reset(); -// } -//} -// -//template -//inline void JointTrajectoryController:: -//checkGoalTolerances(const typename Segment::State& state_error, -// const Segment& segment) -//{ -// // Controller uptime -// const ros::Time uptime = time_data_.readFromRT()->uptime; -// -// // Checks that we have ended inside the goal tolerances -// const RealtimeGoalHandlePtr rt_segment_goal = segment.getGoalHandle(); -// const SegmentTolerances& tolerances = segment.getTolerances(); -// const bool inside_goal_tolerances = checkStateTolerance(state_error, tolerances.goal_state_tolerance); -// -// if (inside_goal_tolerances) -// { -// rt_segment_goal->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::SUCCESSFUL; -// rt_segment_goal->setSucceeded(rt_segment_goal->preallocated_result_); -// rt_active_goal_.reset(); -// } -// else if (uptime.toSec() < segment.endTime() + tolerances.goal_time_tolerance) -// { -// // Still have some time left to meet the goal state tolerances -// } -// else -// { -// if (verbose_) -// { -// ROS_ERROR_STREAM_NAMED(name_,"Goal tolerances failed"); -// // Check the tolerances one more time to output the errors that occures -// checkStateTolerance(state_error, tolerances.goal_state_tolerance, true); -// } -// -// rt_segment_goal->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::GOAL_TOLERANCE_VIOLATED; -// rt_segment_goal->setAborted(rt_segment_goal->preallocated_result_); -// rt_active_goal_.reset(); -// } -//} - template JointTrajectoryController:: JointTrajectoryController() @@ -434,16 +380,6 @@ update(const ros::Time& time, const ros::Duration& period) state_error_.velocity[i] = desired_joint_state_.velocity[0] - current_state_.velocity[i]; state_error_.acceleration[i] = 0.0; -// const RealtimeGoalHandlePtr rt_segment_goal = segment_it->getGoalHandle(); -// if (i == 22) -// if (rt_segment_goal) -// { -// int new_status = int(rt_segment_goal->gh_.getGoalStatus().status); -// if (previous_status != new_status) -// ROS_WARN_STREAM_NAMED(name_,"Status " << int(new_status) << " for goal: "<< rt_segment_goal->gh_.getGoalStatus().goal_id.id); -// previous_status = new_status; -// } - //Check tolerances const RealtimeGoalHandlePtr rt_segment_goal = segment_it->getGoalHandle(); if (rt_segment_goal && rt_segment_goal == rt_active_goal_) @@ -455,7 +391,7 @@ update(const ros::Time& time, const ros::Duration& period) const SegmentTolerancesPerJoint& joint_tolerances = segment_it->getTolerances(); if (!checkStateTolerancePerJoint(state_joint_error_, joint_tolerances.state_tolerance)) { - ROS_ERROR_STREAM_NAMED(name_, "PATH_TOLERANCE_VIOLATED! joint_id:" << i); + //ROS_ERROR_STREAM_NAMED(name_, "PATH_TOLERANCE_VIOLATED! joint_id:" << i); rt_segment_goal->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::PATH_TOLERANCE_VIOLATED; rt_segment_goal->setAborted(rt_segment_goal->preallocated_result_); From 6b8aa382ca3f1d2abdf12f1b9b49b1125dfb7db3 Mon Sep 17 00:00:00 2001 From: Beatriz Leon Date: Mon, 3 Aug 2015 10:58:43 +0100 Subject: [PATCH 07/18] Changes made after modifing tests --- .../init_joint_trajectory.h | 67 ++++++++++++++++--- 1 file changed, 59 insertions(+), 8 deletions(-) diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h index b51ce83da..84f46718e 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h @@ -113,6 +113,12 @@ struct InitJointTrajectoryOptions ros::Time* other_time_base; }; +template +bool IsNotEmpty(typename Trajectory::value_type trajPerJoint) +{ + return trajPerJoint.size()>0; +}; + /** * \brief Initialize a joint trajectory from ROS message data. * @@ -243,7 +249,31 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg // If unspecified, a trivial map (no permutation) is computed const std::vector joint_names = has_joint_names ? *(options.joint_names) : msg.joint_names; - //std::stringstream log_str; + if (has_angle_wraparound) + { + // Preconditions + const unsigned int n_angle_wraparound = options.angle_wraparound->size(); + if (n_angle_wraparound != joint_names.size()) + { + ROS_ERROR("Cannot create trajectory from message. " + "Vector specifying whether joints wrap around has an invalid size."); + return Trajectory(); + } + } + +// std::stringstream log_str; +// log_str.str(""); +// log_str << "Joint names size:" ; +// for (unsigned int i=0; i < joint_names.size();i++) log_str << joint_names[i] << "\t"; +// log_str << "\n"; +// ROS_INFO_STREAM(log_str.str()); +// +// log_str.str(""); +// log_str << "msg.joint_names size:" ; +// for (unsigned int i=0; i < msg.joint_names.size();i++) log_str << msg.joint_names[i] << "\t"; +// log_str << "\n"; +// ROS_INFO_STREAM(log_str.str()); + std::vector permutation_vector = internal::permutation(msg.joint_names,joint_names); if (permutation_vector.empty()) @@ -251,7 +281,14 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg ROS_ERROR("Cannot create trajectory from message. It does not contain the expected joints."); return Trajectory(); } - + + //std::stringstream log_str; +// log_str.str(""); +// log_str << "permutation_vector:" ; +// for (unsigned int i=0; i < permutation_vector.size();i++) log_str << permutation_vector[i] << "\t"; +// log_str << "\n"; +// ROS_INFO_STREAM(log_str.str()); + // Tolerances to be used in all new segments SegmentTolerances tolerances = has_default_tolerances ? *(options.default_tolerances) : SegmentTolerances(n_joints); @@ -296,16 +333,23 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg // Initialize result trajectory: combination of: // - Useful segments of currently followed trajectory // - Useful segments of new trajectory (contained in ROS message) - Trajectory result_traj = *(options.current_trajectory); + Trajectory result_traj; - //Iterate to all segments to set the new goal handler - for (unsigned int joint_id=0; joint_id < joint_names.size();joint_id++) + if (has_current_trajectory) { - for (unsigned int segment_id=0; segment_id < result_traj[joint_id].size(); segment_id++) + result_traj = *(options.current_trajectory); + + //Iterate to all segments to set the new goal handler + for (unsigned int joint_id=0; joint_id < joint_names.size();joint_id++) { - (result_traj[joint_id])[segment_id].setGoalHandle(options.rt_goal_handle); + for (unsigned int segment_id=0; segment_id < result_traj[joint_id].size(); segment_id++) + { + (result_traj[joint_id])[segment_id].setGoalHandle(options.rt_goal_handle); + } } } + else + result_traj.resize(joint_names.size()); std::vector permutation_vector_per_joint(1,0); //refactor this and remove it as it is not needed @@ -422,8 +466,15 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg else {log_str << ".";} ROS_DEBUG_STREAM(log_str.str()); - result_traj[joint_id] = result_traj_per_joint; + if (result_traj_per_joint.size() > 0) + result_traj[joint_id] = result_traj_per_joint; + } + // If the trajectory for all joints is empty, empty the trajectory vector + typename Trajectory::const_iterator trajIter = std::find_if (result_traj.begin(), result_traj.end(), IsNotEmpty); + if (trajIter == result_traj.end()) + { + result_traj.clear(); } return result_traj; From 37d0326fa49711ae2be03b4c8432fd8cbc8c2bd6 Mon Sep 17 00:00:00 2001 From: Beatriz Leon Date: Mon, 3 Aug 2015 12:03:40 +0100 Subject: [PATCH 08/18] working on tests --- .../init_joint_trajectory.h | 6 +- .../test/init_joint_trajectory_test.cpp | 129 ++++++++++-------- .../test/joint_trajectory_controller.test | 9 +- .../test/joint_trajectory_controller_test.cpp | 24 ++-- .../test/joint_trajectory_segment_test.cpp | 66 +++++---- 5 files changed, 130 insertions(+), 104 deletions(-) diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h index 84f46718e..6075c2bdf 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h @@ -193,6 +193,8 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg typedef typename TrajectoryPerJoint::value_type Segment; typedef typename Segment::Scalar Scalar; + ROS_ERROR("-----------------------*****HERE!"); + const unsigned int n_joints = msg.joint_names.size(); const ros::Time msg_start_time = internal::startTime(msg, time); // Message start time @@ -261,6 +263,7 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg } } + // std::stringstream log_str; // log_str.str(""); // log_str << "Joint names size:" ; @@ -380,13 +383,14 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg typename Segment::State last_curr_state; sample(curr_joint_traj, last_curr_time, last_curr_state); + ROS_ERROR("*****HERE1!"); // Get the first time and state that will be executed from the new trajectory trajectory_msgs::JointTrajectoryPoint point_per_joint; point_per_joint.positions.resize(1, it->positions[msg_joint_it]); point_per_joint.velocities.resize(1, it->velocities[msg_joint_it]); point_per_joint.accelerations.resize(1, it->accelerations[msg_joint_it]); point_per_joint.time_from_start = it->time_from_start; - + ROS_ERROR("*****HERE2!"); const typename Segment::Time first_new_time = o_msg_start_time.toSec() + (it->time_from_start).toSec(); typename Segment::State first_new_state(point_per_joint, permutation_vector_per_joint); // Here offsets are not yet applied diff --git a/joint_trajectory_controller/test/init_joint_trajectory_test.cpp b/joint_trajectory_controller/test/init_joint_trajectory_test.cpp index b26985dcf..2d63c51ad 100644 --- a/joint_trajectory_controller/test/init_joint_trajectory_test.cpp +++ b/joint_trajectory_controller/test/init_joint_trajectory_test.cpp @@ -45,7 +45,8 @@ using std::string; const double EPS = 1e-9; typedef JointTrajectorySegment > Segment; -typedef vector Trajectory; +typedef vector TrajectoryPerJoint; +typedef vector Trajectory; TEST(PermutationTest, Permutation) { @@ -140,11 +141,15 @@ class InitTrajectoryTest : public ::testing::Test state[1].velocity.push_back(1.0); state[1].acceleration.push_back(1.0); - curr_traj.push_back(Segment(0.0, state[0], + TrajectoryPerJoint joint_segment; + + joint_segment.push_back(Segment(0.0, state[0], 1.0, state[1])); - curr_traj.push_back(Segment(3.0, state[1], + joint_segment.push_back(Segment(3.0, state[1], 10.0, state[0])); // This is just junk that should be discarded in the tests + + curr_traj.push_back(joint_segment); } protected: @@ -154,13 +159,15 @@ class InitTrajectoryTest : public ::testing::Test // Currently executed trajectory Trajectory curr_traj; + }; +//Now it is possible to have different sizes TEST_F(InitTrajectoryTest, InitException) { // Invalid input should throw an exception trajectory_msg.points.back().positions.push_back(0.0); // Inconsistent size in last element - EXPECT_THROW(initJointTrajectory(trajectory_msg, trajectory_msg.header.stamp), std::invalid_argument); + EXPECT_NO_THROW(initJointTrajectory(trajectory_msg, trajectory_msg.header.stamp)); } // Test logic of parsing a trajectory message. No current trajectory is specified @@ -182,14 +189,14 @@ TEST_F(InitTrajectoryTest, InitLogic) { const ros::Time time = msg_start_time; Trajectory trajectory = initJointTrajectory(trajectory_msg, time); - EXPECT_EQ(points.size() - 1, trajectory.size()); + EXPECT_EQ(points.size() - 1, trajectory[0].size()); } // First point: Return partial trajectory (last segment) { const ros::Time time = msg_start_time + msg_points.begin()->time_from_start; Trajectory trajectory = initJointTrajectory(trajectory_msg, time); - EXPECT_EQ(points.size() - 2, trajectory.size()); + EXPECT_EQ(points.size() - 2, trajectory[0].size()); } // Between the first and second points: Return partial trajectory (last segment) @@ -197,7 +204,7 @@ TEST_F(InitTrajectoryTest, InitLogic) const ros::Time time = msg_start_time + ros::Duration((msg_points.begin()->time_from_start + (++msg_points.begin())->time_from_start).toSec() / 2.0); Trajectory trajectory = initJointTrajectory(trajectory_msg, time); - EXPECT_EQ(msg_points.size() - 2, trajectory.size()); + EXPECT_EQ(msg_points.size() - 2, trajectory[0].size()); } // Second point: Return empty trajectory @@ -249,23 +256,23 @@ TEST_F(InitTrajectoryTest, InitLogicCombine) // Before first point, starting the current trajectory: Return 4 segments: Last of current + bridge + full message { - const ros::Time time(curr_traj[0].startTime()); + const ros::Time time(curr_traj[0][0].startTime()); Trajectory trajectory = initJointTrajectory(trajectory_msg, time, options); - EXPECT_EQ(points.size() + 1, trajectory.size()); + EXPECT_EQ(points.size() + 1, trajectory[0].size()); } // Before first point: Return 4 segments: Last of current + bridge + full message { const ros::Time time = msg_start_time; Trajectory trajectory = initJointTrajectory(trajectory_msg, time, options); - EXPECT_EQ(points.size() + 1, trajectory.size()); + EXPECT_EQ(points.size() + 1, trajectory[0].size()); } // First point: Return partial trajectory, 3 segments: Last of current + bridge + last of message { const ros::Time time = msg_start_time + msg_points.begin()->time_from_start; Trajectory trajectory = initJointTrajectory(trajectory_msg, time, options); - EXPECT_EQ(points.size(), trajectory.size()); + EXPECT_EQ(points.size(), trajectory[0].size()); } // Between the first and second points: Same as before @@ -273,14 +280,14 @@ TEST_F(InitTrajectoryTest, InitLogicCombine) const ros::Time time = msg_start_time + ros::Duration((msg_points.begin()->time_from_start + (++msg_points.begin())->time_from_start).toSec() / 2.0); Trajectory trajectory = initJointTrajectory(trajectory_msg, time, options); - EXPECT_EQ(msg_points.size(), trajectory.size()); + EXPECT_EQ(msg_points.size(), trajectory[0].size()); } // Second point: Return partial trajectory, 2 segments: Last of current + bridge (only one point of message made it) { const ros::Time time = msg_start_time + (++msg_points.begin())->time_from_start; Trajectory trajectory = initJointTrajectory(trajectory_msg, time, options); - EXPECT_EQ(msg_points.size() - 1, trajectory.size()); + EXPECT_EQ(msg_points.size() - 1, trajectory[0].size()); } // Between the second and third points: Same as before @@ -288,7 +295,7 @@ TEST_F(InitTrajectoryTest, InitLogicCombine) const ros::Time time = msg_start_time + ros::Duration(((++msg_points.begin())->time_from_start + (--msg_points.end())->time_from_start).toSec() / 2.0); Trajectory trajectory = initJointTrajectory(trajectory_msg, time, options); - EXPECT_EQ(msg_points.size() - 1, trajectory.size()); + EXPECT_EQ(msg_points.size() - 1, trajectory[0].size()); } // Last point: Return empty trajectory @@ -314,12 +321,12 @@ TEST_F(InitTrajectoryTest, InitValues) // Input time is before first point: Return full trajectory (2 segments) const ros::Time time = msg_start_time; Trajectory trajectory = initJointTrajectory(trajectory_msg, time); - ASSERT_EQ(points.size() - 1, trajectory.size()); + ASSERT_EQ(points.size() - 1, trajectory[0].size()); // Check all segment start/end times and states - for (unsigned int i = 0; i < trajectory.size(); ++i) + for (unsigned int i = 0; i < trajectory[0].size(); ++i) { - const Segment& segment = trajectory[i]; + const Segment& segment = trajectory[0][i]; const JointTrajectoryPoint& p_start = msg_points[i]; const JointTrajectoryPoint& p_end = msg_points[i + 1]; @@ -354,17 +361,17 @@ TEST_F(InitTrajectoryTest, InitValues) TEST_F(InitTrajectoryTest, InitValuesCombine) { // Before first point: Return 4 segments: Last of current + bridge + full message - const ros::Time time(curr_traj[0].startTime()); + const ros::Time time(curr_traj[0][0].startTime()); InitJointTrajectoryOptions options; options.current_trajectory = &curr_traj; Trajectory trajectory = initJointTrajectory(trajectory_msg, time, options); - ASSERT_EQ(points.size() + 1, trajectory.size()); + ASSERT_EQ(points.size() + 1, trajectory[0].size()); // Check current trajectory segment start/end times and states (only one) { - const Segment& ref_segment = curr_traj[0]; - const Segment& segment = trajectory[0]; + const Segment& ref_segment = curr_traj[0][0]; + const Segment& segment = trajectory[0][0]; // Check start/end times { @@ -395,8 +402,8 @@ TEST_F(InitTrajectoryTest, InitValuesCombine) // Check bridge trajectory segment start/end times and states (only one) { - const Segment& ref_segment = curr_traj[0]; - const Segment& segment = trajectory[1]; + const Segment& ref_segment = curr_traj[0][0]; + const Segment& segment = trajectory[0][1]; const vector& msg_points = trajectory_msg.points; @@ -433,12 +440,12 @@ TEST_F(InitTrajectoryTest, InitValuesCombine) } // Check all segment start/end times and states (2 segments) - for (unsigned int traj_it = 2, msg_it = 0; traj_it < trajectory.size(); ++traj_it, ++msg_it) + for (unsigned int traj_it = 2, msg_it = 0; traj_it < trajectory[0].size(); ++traj_it, ++msg_it) { const ros::Time msg_start_time = trajectory_msg.header.stamp; const vector& msg_points = trajectory_msg.points; - const Segment& segment = trajectory[traj_it]; + const Segment& segment = trajectory[0][traj_it]; const JointTrajectoryPoint& p_start = msg_points[msg_it]; const JointTrajectoryPoint& p_end = msg_points[msg_it + 1]; @@ -492,11 +499,13 @@ TEST_F(InitTrajectoryTest, JointPermutation) Trajectory trajectory = initJointTrajectory(trajectory_msg, trajectory_msg.header.stamp); // Check position values only - const Segment& segment = trajectory[0]; + const Segment& segment_foo_joint = trajectory[0][0]; + const Segment& segment_bar_joint = trajectory[1][0]; typename Segment::State state; - segment.sample(segment.startTime(), state); + segment_foo_joint.sample(segment_foo_joint.startTime(), state); EXPECT_EQ(trajectory_msg.points[0].positions[0], state.position[0]); - EXPECT_EQ(trajectory_msg.points[0].positions[1], state.position[1]); + segment_bar_joint.sample(segment_bar_joint.startTime(), state); + EXPECT_EQ(trajectory_msg.points[0].positions[1], state.position[0]); } // Reference joint names vector that reverses joint order @@ -510,13 +519,17 @@ TEST_F(InitTrajectoryTest, JointPermutation) Trajectory trajectory = initJointTrajectory(trajectory_msg, trajectory_msg.header.stamp, options); // Check position values only - const Segment& segment = trajectory[0]; - typename Segment::State state; - segment.sample(segment.startTime(), state); - EXPECT_NE(trajectory_msg.points[0].positions[0], state.position[0]); - EXPECT_NE(trajectory_msg.points[0].positions[1], state.position[1]); - EXPECT_EQ(trajectory_msg.points[0].positions[0], state.position[1]); - EXPECT_EQ(trajectory_msg.points[0].positions[1], state.position[0]); + const Segment& segment_bar_joint = trajectory[0][0]; + const Segment& segment_foo_joint = trajectory[1][0]; + + typename Segment::State state_foo_joint; + typename Segment::State state_bar_joint; + segment_foo_joint.sample(segment_foo_joint.startTime(), state_foo_joint); + segment_bar_joint.sample(segment_bar_joint.startTime(), state_bar_joint); + EXPECT_NE(trajectory_msg.points[0].positions[0], state_bar_joint.position[0]); + EXPECT_NE(trajectory_msg.points[0].positions[1], state_foo_joint.position[0]); + EXPECT_EQ(trajectory_msg.points[0].positions[0], state_foo_joint.position[0]); + EXPECT_EQ(trajectory_msg.points[0].positions[1], state_bar_joint.position[0]); } // Reference joint names size mismatch @@ -553,19 +566,19 @@ TEST_F(InitTrajectoryTest, WrappingSpec) } // Before first point: Return 4 segments: Last of current + bridge + full message - const ros::Time time(curr_traj[0].startTime()); + const ros::Time time(curr_traj[0][0].startTime()); std::vector angle_wraparound(1, true); InitJointTrajectoryOptions options; options.current_trajectory = &curr_traj; options.angle_wraparound = &angle_wraparound; Trajectory trajectory = initJointTrajectory(trajectory_msg, time, options); - ASSERT_EQ(points.size() + 1, trajectory.size()); + ASSERT_EQ(points.size() + 1, trajectory[0].size()); // Check current trajectory segment start/end times and states (only one) { - const Segment& ref_segment = curr_traj[0]; - const Segment& segment = trajectory[0]; + const Segment& ref_segment = curr_traj[0][0]; + const Segment& segment = trajectory[0][0]; // Check start/end times { @@ -596,8 +609,8 @@ TEST_F(InitTrajectoryTest, WrappingSpec) // Check bridge trajectory segment start/end times and states (only one) { - const Segment& ref_segment = curr_traj[0]; - const Segment& segment = trajectory[1]; + const Segment& ref_segment = curr_traj[0][0]; + const Segment& segment = trajectory[0][1]; const vector& msg_points = trajectory_msg.points; @@ -634,12 +647,12 @@ TEST_F(InitTrajectoryTest, WrappingSpec) } // Check all segment start/end times and states (2 segments) - for (unsigned int traj_it = 2, msg_it = 0; traj_it < trajectory.size(); ++traj_it, ++msg_it) + for (unsigned int traj_it = 2, msg_it = 0; traj_it < trajectory[0].size(); ++traj_it, ++msg_it) { const ros::Time msg_start_time = trajectory_msg.header.stamp; const vector& msg_points = trajectory_msg.points; - const Segment& segment = trajectory[traj_it]; + const Segment& segment = trajectory[0][traj_it]; const JointTrajectoryPoint& p_start = msg_points[msg_it]; const JointTrajectoryPoint& p_end = msg_points[msg_it + 1]; @@ -696,18 +709,18 @@ TEST_F(InitTrajectoryTest, IgnoreWrappingSpec) } // Before first point: Return 2 segments: Full message - const ros::Time time(curr_traj[0].startTime()); + const ros::Time time(curr_traj[0][0].startTime()); std::vector angle_wraparound(1, true); InitJointTrajectoryOptions options; options.angle_wraparound = &angle_wraparound; Trajectory trajectory = initJointTrajectory(trajectory_msg, time, options); - ASSERT_EQ(points.size() - 1, trajectory.size()); + ASSERT_EQ(points.size() - 1, trajectory[0].size()); // Check all segment start/end times and states (2 segments) - for (unsigned int i = 0; i < trajectory.size(); ++i) + for (unsigned int i = 0; i < trajectory[0].size(); ++i) { - const Segment& segment = trajectory[i]; + const Segment& segment = trajectory[0][i]; const JointTrajectoryPoint& p_start = msg_points[i]; const JointTrajectoryPoint& p_end = msg_points[i + 1]; @@ -751,21 +764,21 @@ TEST_F(InitTrajectoryTest, GoalHandleTest) RealtimeGoalHandlePtr rt_goal(new RealtimeGoalHandle(gh)); // Before first point: Return 4 segments: Last of current + bridge + full message - const ros::Time time(curr_traj[0].startTime()); + const ros::Time time(curr_traj[0][0].startTime()); InitJointTrajectoryOptions options; options.current_trajectory = &curr_traj; options.rt_goal_handle = rt_goal; Trajectory trajectory = initJointTrajectory(trajectory_msg, time, options); - ASSERT_EQ(points.size() + 1, trajectory.size()); + ASSERT_EQ(points.size() + 1, trajectory[0].size()); // Segment of current trajectory preserves existing null goal handle - ASSERT_FALSE(trajectory[0].getGoalHandle()); + ASSERT_FALSE(trajectory[0][0].getGoalHandle()); // All further segments should be associated to the new goal handle - for (unsigned int i = 1; i < trajectory.size(); ++i) // NOTE: Start index != 0 + for (unsigned int i = 1; i < trajectory[0].size(); ++i) // NOTE: Start index != 0 { - ASSERT_EQ(rt_goal, trajectory[i].getGoalHandle()); + ASSERT_EQ(rt_goal, trajectory[0][i].getGoalHandle()); } } @@ -783,12 +796,12 @@ TEST_F(InitTrajectoryTest, OtherTimeBase) // Before first point: Return 4 segments: Last of current + bridge + full message const ros::Time time = msg_start_time; Trajectory trajectory = initJointTrajectory(trajectory_msg, time, options); - ASSERT_EQ(points.size() + 1, trajectory.size()); + ASSERT_EQ(points.size() + 1, trajectory[0].size()); // Check current trajectory segment start/end times (only one). Time offset does NOT apply { - const Segment& ref_segment = curr_traj[0]; - const Segment& segment = trajectory[0]; + const Segment& ref_segment = curr_traj[0][0]; + const Segment& segment = trajectory[0][0]; // Check start/end times { @@ -799,7 +812,7 @@ TEST_F(InitTrajectoryTest, OtherTimeBase) // Check bridge trajectory segment start/end times and states (only one). Time offset applies to end state only { - const Segment& segment = trajectory[1]; + const Segment& segment = trajectory[0][1]; const vector& msg_points = trajectory_msg.points; @@ -817,12 +830,12 @@ TEST_F(InitTrajectoryTest, OtherTimeBase) } // Check all segment start/end times and states (2 segments). Time offset does apply - for (unsigned int traj_it = 2, msg_it = 0; traj_it < trajectory.size(); ++traj_it, ++msg_it) + for (unsigned int traj_it = 2, msg_it = 0; traj_it < trajectory[0].size(); ++traj_it, ++msg_it) { const ros::Time msg_start_time = trajectory_msg.header.stamp; const vector& msg_points = trajectory_msg.points; - const Segment& segment = trajectory[traj_it]; + const Segment& segment = trajectory[0][traj_it]; const JointTrajectoryPoint& p_start = msg_points[msg_it]; const JointTrajectoryPoint& p_end = msg_points[msg_it + 1]; diff --git a/joint_trajectory_controller/test/joint_trajectory_controller.test b/joint_trajectory_controller/test/joint_trajectory_controller.test index 499e34d90..8ffc4e9d6 100644 --- a/joint_trajectory_controller/test/joint_trajectory_controller.test +++ b/joint_trajectory_controller/test/joint_trajectory_controller.test @@ -28,8 +28,15 @@ args="/rrbot_controller/state/desired/velocities[0]:velocities[1],/rrbot_controller/state/actual/velocities[0]:velocities[1]" /> + + + + + diff --git a/joint_trajectory_controller/test/joint_trajectory_controller_test.cpp b/joint_trajectory_controller/test/joint_trajectory_controller_test.cpp index bb357c5ba..b411cdcd8 100644 --- a/joint_trajectory_controller/test/joint_trajectory_controller_test.cpp +++ b/joint_trajectory_controller/test/joint_trajectory_controller_test.cpp @@ -112,6 +112,7 @@ class JointTrajectoryControllerTest : public ::testing::Test const std::string action_server_name = nh.getNamespace() + "/follow_joint_trajectory"; action_client.reset(new ActionClient(action_server_name)); action_client2.reset(new ActionClient(action_server_name)); + ROS_INFO("**********JointTrajectoryControllerTest finished!!!"); } ~JointTrajectoryControllerTest() @@ -152,18 +153,21 @@ class JointTrajectoryControllerTest : public ::testing::Test void stateCB(const StateConstPtr& state) { + //ROS_INFO("**********stateCB!!!"); boost::mutex::scoped_lock lock(mutex); controller_state = state; } StateConstPtr getState() { + //ROS_INFO("**********stateCB!!!"); boost::mutex::scoped_lock lock(mutex); return controller_state; } bool initState(const ros::Duration& timeout = ros::Duration(5.0)) { + //ROS_INFO("**********initState!!!"); bool init_ok = false; ros::Time start_time = ros::Time::now(); while (!init_ok && (ros::Time::now() - start_time) < timeout) @@ -181,6 +185,7 @@ class JointTrajectoryControllerTest : public ::testing::Test const actionlib::SimpleClientGoalState& state, const ros::Duration& timeout) { + //ROS_INFO("**********waitForState!!!"); using ros::Time; using ros::Duration; @@ -248,7 +253,7 @@ TEST_F(JointTrajectoryControllerTest, invalidMessages) ASSERT_TRUE(initState()); ASSERT_TRUE(action_client->waitForServer(long_timeout)); - // Invalid size + // Invalid size => Now this is possible. Partial trajectory { trajectory_msgs::JointTrajectoryPoint point; point.positions.resize(1, 0.0); @@ -264,8 +269,7 @@ TEST_F(JointTrajectoryControllerTest, invalidMessages) bad_goal.trajectory.header.stamp = ros::Time(0); // Start immediately action_client->sendGoal(bad_goal); - ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::REJECTED, long_timeout)); - EXPECT_EQ(action_client->getResult()->error_code, control_msgs::FollowJointTrajectoryResult::INVALID_JOINTS); + ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::ACTIVE, long_timeout)); } // Incompatible joint names @@ -291,7 +295,7 @@ TEST_F(JointTrajectoryControllerTest, invalidMessages) } // Incompatible data sizes - { + /*{ ActionGoal bad_goal = traj_home_goal; bad_goal.trajectory.points[0].positions.pop_back(); @@ -299,10 +303,10 @@ TEST_F(JointTrajectoryControllerTest, invalidMessages) action_client->sendGoal(bad_goal); ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::REJECTED, long_timeout)); EXPECT_EQ(action_client->getResult()->error_code, control_msgs::FollowJointTrajectoryResult::INVALID_GOAL); - } + }*/ // Non-strictly increasing waypoint times - { + /* { ActionGoal bad_goal = traj_goal; bad_goal.trajectory.points[2].time_from_start = bad_goal.trajectory.points[1].time_from_start; @@ -319,12 +323,12 @@ TEST_F(JointTrajectoryControllerTest, invalidMessages) action_client->sendGoal(empty_goal); ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::REJECTED, long_timeout)); EXPECT_EQ(action_client->getResult()->error_code, control_msgs::FollowJointTrajectoryResult::INVALID_JOINTS); - } + }*/ } // Uninterrupted trajectory execution ////////////////////////////////////////////////////////////////////////////////// -TEST_F(JointTrajectoryControllerTest, topicSingleTraj) +/*TEST_F(JointTrajectoryControllerTest, topicSingleTraj) { ASSERT_TRUE(initState()); @@ -924,13 +928,13 @@ TEST_F(JointTrajectoryControllerTest, goalToleranceViolation) smoothing_pub.publish(smoothing); ros::Duration(0.5).sleep(); } -} +}*/ int main(int argc, char** argv) { testing::InitGoogleTest(&argc, argv); ros::init(argc, argv, "joint_trajectory_controller_test"); - + ROS_WARN("**********Main!!!"); ros::AsyncSpinner spinner(1); spinner.start(); int ret = RUN_ALL_TESTS(); diff --git a/joint_trajectory_controller/test/joint_trajectory_segment_test.cpp b/joint_trajectory_controller/test/joint_trajectory_segment_test.cpp index 546dec449..5ee9e25f0 100644 --- a/joint_trajectory_controller/test/joint_trajectory_segment_test.cpp +++ b/joint_trajectory_controller/test/joint_trajectory_segment_test.cpp @@ -60,10 +60,10 @@ TEST(WraparoundOffsetTest, WrappingPositions) // No wrapping joints { std::vector angle_wraparound(2, false); - std::vector wraparound_offset = wraparoundOffset(pos1, pos2, angle_wraparound); - EXPECT_NEAR(pos1.size(), wraparound_offset.size(), EPS); - EXPECT_NEAR(0.0, wraparound_offset[0], EPS); - EXPECT_NEAR(0.0, wraparound_offset[1], EPS); + double wraparound_offset1 = wraparoundJointOffset(pos1[0], pos2[0], angle_wraparound[0]); + double wraparound_offset2 = wraparoundJointOffset(pos1[1], pos2[1], angle_wraparound[1]); + EXPECT_NEAR(0.0, wraparound_offset1, EPS); + EXPECT_NEAR(0.0, wraparound_offset2, EPS); } // Single wrapping joint @@ -74,18 +74,18 @@ TEST(WraparoundOffsetTest, WrappingPositions) // From state1 to state2 { - std::vector wraparound_offset = wraparoundOffset(pos1, pos2, angle_wraparound); - EXPECT_NEAR(pos1.size(), wraparound_offset.size(), EPS); - EXPECT_NEAR(-two_pi, wraparound_offset[0], EPS); - EXPECT_NEAR(0.0, wraparound_offset[1], EPS); + double wraparound_offset1 = wraparoundJointOffset(pos1[0], pos2[0], angle_wraparound[0]); + double wraparound_offset2 = wraparoundJointOffset(pos1[1], pos2[1], angle_wraparound[1]); + EXPECT_NEAR(-two_pi, wraparound_offset1, EPS); + EXPECT_NEAR(0.0, wraparound_offset2, EPS); } // From state2 to state1 { - std::vector wraparound_offset = wraparoundOffset(pos2, pos1, angle_wraparound); - EXPECT_NEAR(pos1.size(), wraparound_offset.size(), EPS); - EXPECT_NEAR(two_pi, wraparound_offset[0], EPS); - EXPECT_NEAR(0.0, wraparound_offset[1], EPS); + double wraparound_offset1 = wraparoundJointOffset(pos2[0], pos1[0], angle_wraparound[0]); + double wraparound_offset2 = wraparoundJointOffset(pos2[1], pos1[1], angle_wraparound[1]); + EXPECT_NEAR(two_pi, wraparound_offset1, EPS); + EXPECT_NEAR(0.0, wraparound_offset2, EPS); } } @@ -95,18 +95,18 @@ TEST(WraparoundOffsetTest, WrappingPositions) // From state1 to state2 { - std::vector wraparound_offset = wraparoundOffset(pos1, pos2, angle_wraparound); - EXPECT_NEAR(pos1.size(), wraparound_offset.size(), EPS); - EXPECT_NEAR(-two_pi, wraparound_offset[0], EPS); - EXPECT_NEAR(-2.0 * two_pi, wraparound_offset[1], EPS); + double wraparound_offset1 = wraparoundJointOffset(pos1[0], pos2[0], angle_wraparound[0]); + double wraparound_offset2 = wraparoundJointOffset(pos1[1], pos2[1], angle_wraparound[1]); + EXPECT_NEAR(-two_pi, wraparound_offset1, EPS); + EXPECT_NEAR(-2.0 * two_pi, wraparound_offset2, EPS); } // From state2 to state1 { - std::vector wraparound_offset = wraparoundOffset(pos2, pos1, angle_wraparound); - EXPECT_NEAR(pos1.size(), wraparound_offset.size(), EPS); - EXPECT_NEAR(two_pi, wraparound_offset[0], EPS); - EXPECT_NEAR(2.0 * two_pi, wraparound_offset[1], EPS); + double wraparound_offset1 = wraparoundJointOffset(pos2[0], pos1[0], angle_wraparound[0]); + double wraparound_offset2 = wraparoundJointOffset(pos2[1], pos1[1], angle_wraparound[1]); + EXPECT_NEAR(two_pi, wraparound_offset1, EPS); + EXPECT_NEAR(2.0 * two_pi, wraparound_offset2, EPS); } } } @@ -122,16 +122,14 @@ TEST(WraparoundOffsetTest, WrappingPositionsPiSingularity) std::vector angle_wraparound(1, true); // From state1 to state2 { - std::vector wraparound_offset = wraparoundOffset(pos1, pos2, angle_wraparound); - EXPECT_EQ(pos1.size(), wraparound_offset.size()); - EXPECT_NEAR(0.0, wraparound_offset[0], EPS); + double wraparound_offset1 = wraparoundJointOffset(pos1[0], pos2[0], angle_wraparound[0]); + EXPECT_NEAR(0.0, wraparound_offset1, EPS); } // From state2 to state1 { - std::vector wraparound_offset = wraparoundOffset(pos2, pos1, angle_wraparound); - EXPECT_EQ(pos1.size(), wraparound_offset.size()); - EXPECT_NEAR(0.0, wraparound_offset[0], EPS); + double wraparound_offset1 = wraparoundJointOffset(pos2[0], pos1[0], angle_wraparound[0]); + EXPECT_NEAR(0.0, wraparound_offset1, EPS); } } @@ -154,18 +152,18 @@ TEST(WraparoundOffsetTest, NonWrappingPositions) // From state1 to state2 { - std::vector wraparound_offset = wraparoundOffset(pos1, pos2, angle_wraparound); - EXPECT_NEAR(pos1.size(), wraparound_offset.size(), EPS); - EXPECT_NEAR(0.0, wraparound_offset[0], EPS); - EXPECT_NEAR(0.0, wraparound_offset[1], EPS); + double wraparound_offset1 = wraparoundJointOffset(pos1[0], pos2[0], angle_wraparound[0]); + double wraparound_offset2 = wraparoundJointOffset(pos1[1], pos2[1], angle_wraparound[1]); + EXPECT_NEAR(0.0, wraparound_offset1, EPS); + EXPECT_NEAR(0.0, wraparound_offset2, EPS); } // From state2 to state1 { - std::vector wraparound_offset = wraparoundOffset(pos2, pos1, angle_wraparound); - EXPECT_NEAR(pos1.size(), wraparound_offset.size(), EPS); - EXPECT_NEAR(0.0, wraparound_offset[0], EPS); - EXPECT_NEAR(0.0, wraparound_offset[1], EPS); + double wraparound_offset1 = wraparoundJointOffset(pos2[0], pos1[0], angle_wraparound[0]); + double wraparound_offset2 = wraparoundJointOffset(pos2[1], pos1[1], angle_wraparound[1]); + EXPECT_NEAR(0.0, wraparound_offset1, EPS); + EXPECT_NEAR(0.0, wraparound_offset2, EPS); } } } From 093104bdb3dd7df77484d2ccb92c2282296d22b1 Mon Sep 17 00:00:00 2001 From: Beatriz Leon Date: Mon, 3 Aug 2015 15:52:11 +0100 Subject: [PATCH 09/18] Fixed test and all passed --- .../init_joint_trajectory.h | 31 +++---------------- .../joint_trajectory_controller/tolerances.h | 2 +- .../test/joint_trajectory_controller.test | 9 +----- .../test/joint_trajectory_controller_test.cpp | 19 +++++------- 4 files changed, 14 insertions(+), 47 deletions(-) diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h index 6075c2bdf..41cc1ce62 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h @@ -193,8 +193,6 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg typedef typename TrajectoryPerJoint::value_type Segment; typedef typename Segment::Scalar Scalar; - ROS_ERROR("-----------------------*****HERE!"); - const unsigned int n_joints = msg.joint_names.size(); const ros::Time msg_start_time = internal::startTime(msg, time); // Message start time @@ -263,20 +261,6 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg } } - -// std::stringstream log_str; -// log_str.str(""); -// log_str << "Joint names size:" ; -// for (unsigned int i=0; i < joint_names.size();i++) log_str << joint_names[i] << "\t"; -// log_str << "\n"; -// ROS_INFO_STREAM(log_str.str()); -// -// log_str.str(""); -// log_str << "msg.joint_names size:" ; -// for (unsigned int i=0; i < msg.joint_names.size();i++) log_str << msg.joint_names[i] << "\t"; -// log_str << "\n"; -// ROS_INFO_STREAM(log_str.str()); - std::vector permutation_vector = internal::permutation(msg.joint_names,joint_names); if (permutation_vector.empty()) @@ -284,14 +268,7 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg ROS_ERROR("Cannot create trajectory from message. It does not contain the expected joints."); return Trajectory(); } - - //std::stringstream log_str; -// log_str.str(""); -// log_str << "permutation_vector:" ; -// for (unsigned int i=0; i < permutation_vector.size();i++) log_str << permutation_vector[i] << "\t"; -// log_str << "\n"; -// ROS_INFO_STREAM(log_str.str()); - + // Tolerances to be used in all new segments SegmentTolerances tolerances = has_default_tolerances ? *(options.default_tolerances) : SegmentTolerances(n_joints); @@ -383,14 +360,16 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg typename Segment::State last_curr_state; sample(curr_joint_traj, last_curr_time, last_curr_state); - ROS_ERROR("*****HERE1!"); // Get the first time and state that will be executed from the new trajectory + if (!isValid(*it, it->positions.size())) + { + throw(std::invalid_argument("Size mismatch in trajectory point position, velocity or acceleration data.")); + } trajectory_msgs::JointTrajectoryPoint point_per_joint; point_per_joint.positions.resize(1, it->positions[msg_joint_it]); point_per_joint.velocities.resize(1, it->velocities[msg_joint_it]); point_per_joint.accelerations.resize(1, it->accelerations[msg_joint_it]); point_per_joint.time_from_start = it->time_from_start; - ROS_ERROR("*****HERE2!"); const typename Segment::Time first_new_time = o_msg_start_time.toSec() + (it->time_from_start).toSec(); typename Segment::State first_new_state(point_per_joint, permutation_vector_per_joint); // Here offsets are not yet applied diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h b/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h index 0ab5f49e2..013329e46 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h @@ -172,7 +172,7 @@ inline bool checkStateTolerancePerJoint(const State& using std::abs; - const bool is_valid = !(state_tolerance.position > 0.0 && abs(state_error.position[0]) > state_tolerance.position); //&& + const bool is_valid = !(state_tolerance.position > 0.0 && abs(state_error.position[0]) > state_tolerance.position) && !(state_tolerance.velocity > 0.0 && abs(state_error.velocity[0]) > state_tolerance.velocity) && !(state_tolerance.acceleration > 0.0 && abs(state_error.acceleration[0]) > state_tolerance.acceleration); diff --git a/joint_trajectory_controller/test/joint_trajectory_controller.test b/joint_trajectory_controller/test/joint_trajectory_controller.test index 8ffc4e9d6..499e34d90 100644 --- a/joint_trajectory_controller/test/joint_trajectory_controller.test +++ b/joint_trajectory_controller/test/joint_trajectory_controller.test @@ -28,15 +28,8 @@ args="/rrbot_controller/state/desired/velocities[0]:velocities[1],/rrbot_controller/state/actual/velocities[0]:velocities[1]" /> - - - - - + time-limit="80.0"/> diff --git a/joint_trajectory_controller/test/joint_trajectory_controller_test.cpp b/joint_trajectory_controller/test/joint_trajectory_controller_test.cpp index b411cdcd8..688741ac1 100644 --- a/joint_trajectory_controller/test/joint_trajectory_controller_test.cpp +++ b/joint_trajectory_controller/test/joint_trajectory_controller_test.cpp @@ -112,7 +112,6 @@ class JointTrajectoryControllerTest : public ::testing::Test const std::string action_server_name = nh.getNamespace() + "/follow_joint_trajectory"; action_client.reset(new ActionClient(action_server_name)); action_client2.reset(new ActionClient(action_server_name)); - ROS_INFO("**********JointTrajectoryControllerTest finished!!!"); } ~JointTrajectoryControllerTest() @@ -153,21 +152,18 @@ class JointTrajectoryControllerTest : public ::testing::Test void stateCB(const StateConstPtr& state) { - //ROS_INFO("**********stateCB!!!"); boost::mutex::scoped_lock lock(mutex); controller_state = state; } StateConstPtr getState() { - //ROS_INFO("**********stateCB!!!"); boost::mutex::scoped_lock lock(mutex); return controller_state; } bool initState(const ros::Duration& timeout = ros::Duration(5.0)) { - //ROS_INFO("**********initState!!!"); bool init_ok = false; ros::Time start_time = ros::Time::now(); while (!init_ok && (ros::Time::now() - start_time) < timeout) @@ -185,7 +181,6 @@ class JointTrajectoryControllerTest : public ::testing::Test const actionlib::SimpleClientGoalState& state, const ros::Duration& timeout) { - //ROS_INFO("**********waitForState!!!"); using ros::Time; using ros::Duration; @@ -295,7 +290,7 @@ TEST_F(JointTrajectoryControllerTest, invalidMessages) } // Incompatible data sizes - /*{ + { ActionGoal bad_goal = traj_home_goal; bad_goal.trajectory.points[0].positions.pop_back(); @@ -303,10 +298,10 @@ TEST_F(JointTrajectoryControllerTest, invalidMessages) action_client->sendGoal(bad_goal); ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::REJECTED, long_timeout)); EXPECT_EQ(action_client->getResult()->error_code, control_msgs::FollowJointTrajectoryResult::INVALID_GOAL); - }*/ + } // Non-strictly increasing waypoint times - /* { + { ActionGoal bad_goal = traj_goal; bad_goal.trajectory.points[2].time_from_start = bad_goal.trajectory.points[1].time_from_start; @@ -323,12 +318,12 @@ TEST_F(JointTrajectoryControllerTest, invalidMessages) action_client->sendGoal(empty_goal); ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::REJECTED, long_timeout)); EXPECT_EQ(action_client->getResult()->error_code, control_msgs::FollowJointTrajectoryResult::INVALID_JOINTS); - }*/ + } } // Uninterrupted trajectory execution ////////////////////////////////////////////////////////////////////////////////// -/*TEST_F(JointTrajectoryControllerTest, topicSingleTraj) +TEST_F(JointTrajectoryControllerTest, topicSingleTraj) { ASSERT_TRUE(initState()); @@ -928,13 +923,13 @@ TEST_F(JointTrajectoryControllerTest, goalToleranceViolation) smoothing_pub.publish(smoothing); ros::Duration(0.5).sleep(); } -}*/ +} int main(int argc, char** argv) { testing::InitGoogleTest(&argc, argv); ros::init(argc, argv, "joint_trajectory_controller_test"); - ROS_WARN("**********Main!!!"); + ros::AsyncSpinner spinner(1); spinner.start(); int ret = RUN_ALL_TESTS(); From 324f85122c8e7cceaabaca1c17200cc003d817a4 Mon Sep 17 00:00:00 2001 From: Beatriz Leon Date: Mon, 3 Aug 2015 16:50:30 +0100 Subject: [PATCH 10/18] Cleaning --- .../joint_trajectory_controller.h | 32 ------------------- .../joint_trajectory_segment.h | 2 +- .../joint_trajectory_controller/tolerances.h | 2 +- .../test/init_joint_trajectory_test.cpp | 1 - 4 files changed, 2 insertions(+), 35 deletions(-) diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h index ac361cd4d..03850d095 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h @@ -156,8 +156,6 @@ class JointTrajectoryController : public controller_interface::Controller ActionServer; typedef boost::shared_ptr ActionServerPtr; typedef ActionServer::GoalHandle GoalHandle; @@ -187,7 +185,6 @@ class JointTrajectoryController : public controller_interface::Controller rt_active_goals_; ///< Currently active action goal, if any. /** * Thread-safe container with a smart pointer to trajectory currently being followed. @@ -198,13 +195,11 @@ class JointTrajectoryController : public controller_interface::Controller time_data_; ros::Duration state_publisher_period_; @@ -246,33 +241,6 @@ class JointTrajectoryController : public controller_interface::Controller"{B, D, A, C}", and we are interested in * constructing a segment with joints ordered as "{A, B, C, D}", the permutation vector should * be set to "{2, 0, 3, 1}". diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h b/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h index 013329e46..15b1ffd97 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h @@ -89,7 +89,7 @@ struct SegmentTolerances }; /** - * \brief Trajectory segment tolerances per Joint (to replace the previous function + * \brief Trajectory segment tolerances per Joint */ template struct SegmentTolerancesPerJoint diff --git a/joint_trajectory_controller/test/init_joint_trajectory_test.cpp b/joint_trajectory_controller/test/init_joint_trajectory_test.cpp index 2d63c51ad..5ca9d5787 100644 --- a/joint_trajectory_controller/test/init_joint_trajectory_test.cpp +++ b/joint_trajectory_controller/test/init_joint_trajectory_test.cpp @@ -159,7 +159,6 @@ class InitTrajectoryTest : public ::testing::Test // Currently executed trajectory Trajectory curr_traj; - }; //Now it is possible to have different sizes From 36d6bc5bc45a65f2fab7641c08fd4d81efe02002 Mon Sep 17 00:00:00 2001 From: Beatriz Leon Date: Fri, 7 Aug 2015 15:05:21 +0100 Subject: [PATCH 11/18] Fixed the import of new trajectories with accelerations empty and a test --- .../init_joint_trajectory.h | 30 +++++++++++-------- .../test/init_joint_trajectory_test.cpp | 3 +- 2 files changed, 18 insertions(+), 15 deletions(-) diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h index 41cc1ce62..8604dbd58 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h @@ -338,6 +338,9 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg for (unsigned int msg_joint_it=0; msg_joint_it < permutation_vector.size();msg_joint_it++) { std::vector::const_iterator it = msg_it; + if (!isValid(*it, it->positions.size())) + throw(std::invalid_argument("Size mismatch in trajectory point position, velocity or acceleration data.")); + TrajectoryPerJoint result_traj_per_joint; // Currently empty unsigned int joint_id = permutation_vector[msg_joint_it]; @@ -361,14 +364,10 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg sample(curr_joint_traj, last_curr_time, last_curr_state); // Get the first time and state that will be executed from the new trajectory - if (!isValid(*it, it->positions.size())) - { - throw(std::invalid_argument("Size mismatch in trajectory point position, velocity or acceleration data.")); - } trajectory_msgs::JointTrajectoryPoint point_per_joint; - point_per_joint.positions.resize(1, it->positions[msg_joint_it]); - point_per_joint.velocities.resize(1, it->velocities[msg_joint_it]); - point_per_joint.accelerations.resize(1, it->accelerations[msg_joint_it]); + if (!it->positions.empty()) {point_per_joint.positions.resize(1, it->positions[msg_joint_it]);} + if (!it->velocities.empty()) {point_per_joint.velocities.resize(1, it->velocities[msg_joint_it]);} + if (!it->accelerations.empty()) {point_per_joint.accelerations.resize(1, it->accelerations[msg_joint_it]);} point_per_joint.time_from_start = it->time_from_start; const typename Segment::Time first_new_time = o_msg_start_time.toSec() + (it->time_from_start).toSec(); @@ -418,14 +417,19 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg std::vector::const_iterator next_it = it; ++next_it; trajectory_msgs::JointTrajectoryPoint it_point_per_joint, next_it_point_per_joint; - it_point_per_joint.positions.resize(1, it->positions[msg_joint_it]); - it_point_per_joint.velocities.resize(1, it->velocities[msg_joint_it]); - it_point_per_joint.accelerations.resize(1, it->accelerations[msg_joint_it]); + + if (!isValid(*it, it->positions.size())) + throw(std::invalid_argument("Size mismatch in trajectory point position, velocity or acceleration data.")); + if (!it->positions.empty()) {it_point_per_joint.positions.resize(1, it->positions[msg_joint_it]);} + if (!it->velocities.empty()) {it_point_per_joint.velocities.resize(1, it->velocities[msg_joint_it]);} + if (!it->accelerations.empty()) {it_point_per_joint.accelerations.resize(1, it->accelerations[msg_joint_it]);} it_point_per_joint.time_from_start = it->time_from_start; - next_it_point_per_joint.positions.resize(1, next_it->positions[msg_joint_it]); - next_it_point_per_joint.velocities.resize(1, next_it->velocities[msg_joint_it]); - next_it_point_per_joint.accelerations.resize(1, next_it->accelerations[msg_joint_it]); + if (!isValid(*next_it, next_it->positions.size())) + throw(std::invalid_argument("Size mismatch in trajectory point position, velocity or acceleration data.")); + if (!next_it->positions.empty()) {next_it_point_per_joint.positions.resize(1, next_it->positions[msg_joint_it]);} + if (!next_it->velocities.empty()) {next_it_point_per_joint.velocities.resize(1, next_it->velocities[msg_joint_it]);} + if (!next_it->accelerations.empty()) {next_it_point_per_joint.accelerations.resize(1, next_it->accelerations[msg_joint_it]);} next_it_point_per_joint.time_from_start = next_it->time_from_start; Segment segment(o_msg_start_time, it_point_per_joint, next_it_point_per_joint, permutation_vector_per_joint, position_offset); diff --git a/joint_trajectory_controller/test/init_joint_trajectory_test.cpp b/joint_trajectory_controller/test/init_joint_trajectory_test.cpp index 5ca9d5787..fe1ea3884 100644 --- a/joint_trajectory_controller/test/init_joint_trajectory_test.cpp +++ b/joint_trajectory_controller/test/init_joint_trajectory_test.cpp @@ -161,12 +161,11 @@ class InitTrajectoryTest : public ::testing::Test Trajectory curr_traj; }; -//Now it is possible to have different sizes TEST_F(InitTrajectoryTest, InitException) { // Invalid input should throw an exception trajectory_msg.points.back().positions.push_back(0.0); // Inconsistent size in last element - EXPECT_NO_THROW(initJointTrajectory(trajectory_msg, trajectory_msg.header.stamp)); + EXPECT_THROW(initJointTrajectory(trajectory_msg, trajectory_msg.header.stamp), std::invalid_argument); } // Test logic of parsing a trajectory message. No current trajectory is specified From 2d939d788ecd9b339ea7930e7cc8d3390e393bfc Mon Sep 17 00:00:00 2001 From: Beatriz Leon Date: Tue, 11 Aug 2015 11:54:21 +0100 Subject: [PATCH 12/18] Small fix to error message --- .../joint_trajectory_controller_impl.h | 4 +--- .../include/joint_trajectory_controller/tolerances.h | 2 +- 2 files changed, 2 insertions(+), 4 deletions(-) diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h index 5a457a049..8fe6cc326 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h @@ -391,7 +391,6 @@ update(const ros::Time& time, const ros::Duration& period) const SegmentTolerancesPerJoint& joint_tolerances = segment_it->getTolerances(); if (!checkStateTolerancePerJoint(state_joint_error_, joint_tolerances.state_tolerance)) { - //ROS_ERROR_STREAM_NAMED(name_, "PATH_TOLERANCE_VIOLATED! joint_id:" << i); rt_segment_goal->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::PATH_TOLERANCE_VIOLATED; rt_segment_goal->setAborted(rt_segment_goal->preallocated_result_); @@ -412,7 +411,6 @@ update(const ros::Time& time, const ros::Duration& period) if (inside_goal_tolerances) { - //ROS_WARN_STREAM_NAMED(name_, "TrajectoryResult::SUCCESSFUL i: "<< i); successful_joint_traj[i] = true; } @@ -424,7 +422,7 @@ update(const ros::Time& time, const ros::Duration& period) { if (verbose_) { - ROS_ERROR_STREAM_NAMED(name_,"Goal tolerances failed"); + ROS_ERROR_STREAM_NAMED(name_,"Goal tolerances failed for joint: "<< joint_names_[i]); // Check the tolerances one more time to output the errors that occurs checkStateTolerancePerJoint(state_joint_error_, tolerances.goal_state_tolerance, true); } diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h b/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h index 15b1ffd97..02e03e0f4 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/tolerances.h @@ -180,7 +180,7 @@ inline bool checkStateTolerancePerJoint(const State& { if( show_errors ) { - ROS_ERROR_STREAM_NAMED("tolerances","Path state tolerances failed on joint"); + ROS_ERROR_STREAM_NAMED("tolerances","Path state tolerances failed:"); if (state_tolerance.position > 0.0 && abs(state_error.position[0]) > state_tolerance.position) ROS_ERROR_STREAM_NAMED("tolerances","Position Error: " << state_error.position[0] << From af5c23772dc5da0364384763a952949a8104dfbe Mon Sep 17 00:00:00 2001 From: Beatriz Leon Date: Mon, 17 Aug 2015 14:26:31 +0100 Subject: [PATCH 13/18] Fixed comments --- .../init_joint_trajectory.h | 45 ++++++------ .../joint_trajectory_controller.h | 9 ++- .../joint_trajectory_controller_impl.h | 29 ++++---- .../joint_trajectory_segment.h | 13 +--- .../test/init_joint_trajectory_test.cpp | 73 +++++++++++-------- .../test/joint_trajectory_segment_test.cpp | 52 ++++++------- 6 files changed, 114 insertions(+), 107 deletions(-) diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h index 8604dbd58..3a42b10f9 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h @@ -57,16 +57,19 @@ namespace joint_trajectory_controller namespace internal { /** - * \return The permutation vector between two containers. - * If \p t1 is "{A, B, C, D}" and \p t2 is "{B, D, A, C}", the associated permutation vector is - * "{2, 0, 3, 1}". + * \return The map between \p t1 indices (implicitly encoded in return vector indices) to \t2 indices. + * If \p t1 is "{C, B}" and \p t2 is "{A, B, C, D}", the associated mapping vector is + * "{2, 1}". */ template -inline std::vector permutation(const T& t1, const T& t2) +inline std::vector mapping(const T& t1, const T& t2) { typedef unsigned int SizeType; + + // t1 must be a subset of t2 + if (t1.size() > t2.size()) {return std::vector();} - std::vector permutation_vector(t1.size()); // Return value + std::vector mapping_vector(t1.size()); // Return value for (typename T::const_iterator t1_it = t1.begin(); t1_it != t1.end(); ++t1_it) { typename T::const_iterator t2_it = std::find(t2.begin(), t2.end(), *t1_it); @@ -75,10 +78,10 @@ inline std::vector permutation(const T& t1, const T& t2) { const SizeType t1_dist = std::distance(t1.begin(), t1_it); const SizeType t2_dist = std::distance(t2.begin(), t2_it); - permutation_vector[t1_dist] = t2_dist; + mapping_vector[t1_dist] = t2_dist; } } - return permutation_vector; + return mapping_vector; } } // namespace @@ -114,9 +117,9 @@ struct InitJointTrajectoryOptions }; template -bool IsNotEmpty(typename Trajectory::value_type trajPerJoint) +bool isNotEmpty(typename Trajectory::value_type trajPerJoint) { - return trajPerJoint.size()>0; + return !trajPerJoint.empty(); }; /** @@ -245,8 +248,8 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg o_msg_start_time = msg_start_time; } - // Permutation vector mapping the expected joint order to the message joint order - // If unspecified, a trivial map (no permutation) is computed + // Mapping vector contains the map between the message joint order and the expected joint order + // If unspecified, a trivial map is computed const std::vector joint_names = has_joint_names ? *(options.joint_names) : msg.joint_names; if (has_angle_wraparound) @@ -261,9 +264,9 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg } } - std::vector permutation_vector = internal::permutation(msg.joint_names,joint_names); + std::vector mapping_vector = internal::mapping(msg.joint_names,joint_names); - if (permutation_vector.empty()) + if (mapping_vector.empty()) { ROS_ERROR("Cannot create trajectory from message. It does not contain the expected joints."); return Trajectory(); @@ -331,18 +334,18 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg else result_traj.resize(joint_names.size()); - std::vector permutation_vector_per_joint(1,0); //refactor this and remove it as it is not needed + std::vector mapping_vector_per_joint(1,0); //refactor this and remove it as it is not needed - //Iterate through the joints that are in the message, in the order of the permutation vector + //Iterate through the joints that are in the message, in the order of the mapping vector //for (unsigned int joint_id=0; joint_id < joint_names.size();joint_id++) - for (unsigned int msg_joint_it=0; msg_joint_it < permutation_vector.size();msg_joint_it++) + for (unsigned int msg_joint_it=0; msg_joint_it < mapping_vector.size();msg_joint_it++) { std::vector::const_iterator it = msg_it; if (!isValid(*it, it->positions.size())) throw(std::invalid_argument("Size mismatch in trajectory point position, velocity or acceleration data.")); TrajectoryPerJoint result_traj_per_joint; // Currently empty - unsigned int joint_id = permutation_vector[msg_joint_it]; + unsigned int joint_id = mapping_vector[msg_joint_it]; // Initialize offsets due to wrapping joints to zero std::vector position_offset(1, 0.0); @@ -371,7 +374,7 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg point_per_joint.time_from_start = it->time_from_start; const typename Segment::Time first_new_time = o_msg_start_time.toSec() + (it->time_from_start).toSec(); - typename Segment::State first_new_state(point_per_joint, permutation_vector_per_joint); // Here offsets are not yet applied + typename Segment::State first_new_state(point_per_joint, mapping_vector_per_joint); // Here offsets are not yet applied // Compute offsets due to wrapping joints if (has_angle_wraparound) @@ -382,7 +385,7 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg } // Apply offset to first state that will be executed from the new trajectory - first_new_state = typename Segment::State(point_per_joint, permutation_vector_per_joint, position_offset); // Now offsets are applied + first_new_state = typename Segment::State(point_per_joint, mapping_vector_per_joint, position_offset); // Now offsets are applied // Add useful segments of current trajectory to result { @@ -432,7 +435,7 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg if (!next_it->accelerations.empty()) {next_it_point_per_joint.accelerations.resize(1, next_it->accelerations[msg_joint_it]);} next_it_point_per_joint.time_from_start = next_it->time_from_start; - Segment segment(o_msg_start_time, it_point_per_joint, next_it_point_per_joint, permutation_vector_per_joint, position_offset); + Segment segment(o_msg_start_time, it_point_per_joint, next_it_point_per_joint, mapping_vector_per_joint, position_offset); segment.setGoalHandle(options.rt_goal_handle); if (has_rt_goal_handle) {segment.setTolerances(tolerances_per_joint);} result_traj_per_joint.push_back(segment); @@ -458,7 +461,7 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg } // If the trajectory for all joints is empty, empty the trajectory vector - typename Trajectory::const_iterator trajIter = std::find_if (result_traj.begin(), result_traj.end(), IsNotEmpty); + typename Trajectory::const_iterator trajIter = std::find_if (result_traj.begin(), result_traj.end(), isNotEmpty); if (trajIter == result_traj.end()) { result_traj.clear(); diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h index 03850d095..581b42598 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h @@ -196,9 +196,11 @@ class JointTrajectoryController : public controller_interface::Controller time_data_; @@ -206,6 +208,7 @@ class JointTrajectoryController : public controller_interface::Controller successful_joint_traj_; // ROS API ros::NodeHandle controller_nh_; diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h index 8fe6cc326..8ffc2ab72 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h @@ -290,9 +290,13 @@ bool JointTrajectoryController::init(HardwareInt this); // Preeallocate resources - current_state_ = typename Segment::State(n_joints); - desired_state_ = typename Segment::State(n_joints); - state_error_ = typename Segment::State(n_joints); + current_state_ = typename Segment::State(n_joints); + desired_state_ = typename Segment::State(n_joints); + state_error_ = typename Segment::State(n_joints); + desired_joint_state_ = typename Segment::State(1); + state_joint_error_ = typename Segment::State(1); + + successful_joint_traj_.resize(joints_.size(), false); // Initialize trajectory with all joints typename Segment::State current_joint_state_ = typename Segment::State(1); @@ -347,15 +351,9 @@ update(const ros::Time& time, const ros::Duration& period) // fetch the currently followed trajectory, it has been updated by the non-rt thread with something that starts in the // next control cycle, leaving the current cycle without a valid trajectory. - std::vector successful_joint_traj; - successful_joint_traj.resize(joints_.size(), false); - // Update current state and state error for (unsigned int i = 0; i < joints_.size(); ++i) { - typename Segment::State desired_joint_state_ = typename Segment::State(1); - typename Segment::State state_joint_error_= typename Segment::State(1); - current_state_.position[i] = joints_[i].getPosition(); current_state_.velocity[i] = joints_[i].getVelocity(); // There's no acceleration data available in a joint handle @@ -391,6 +389,7 @@ update(const ros::Time& time, const ros::Duration& period) const SegmentTolerancesPerJoint& joint_tolerances = segment_it->getTolerances(); if (!checkStateTolerancePerJoint(state_joint_error_, joint_tolerances.state_tolerance)) { + ROS_ERROR_STREAM_NAMED(name_,"Path tolerances failed for joint: " << joint_names_[i]); rt_segment_goal->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::PATH_TOLERANCE_VIOLATED; rt_segment_goal->setAborted(rt_segment_goal->preallocated_result_); @@ -411,7 +410,7 @@ update(const ros::Time& time, const ros::Duration& period) if (inside_goal_tolerances) { - successful_joint_traj[i] = true; + successful_joint_traj_[i] = true; } else if (uptime.toSec() < segment_it->endTime() + tolerances.goal_time_tolerance) @@ -436,7 +435,7 @@ update(const ros::Time& time, const ros::Duration& period) } //If all segments finished successfully then set goal as succeeded - if (std::find(successful_joint_traj.begin(), successful_joint_traj.end(), false) == successful_joint_traj.end()) + if (std::find(successful_joint_traj_.begin(), successful_joint_traj_.end(), false) == successful_joint_traj_.end()) { rt_active_goal_->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::SUCCESSFUL; rt_active_goal_->setSucceeded(rt_active_goal_->preallocated_result_); @@ -544,11 +543,11 @@ goalCB(GoalHandle gh) return; } - // Goal should specify all controller joints (they can be ordered differently). Reject if this is not the case - using internal::permutation; - std::vector permutation_vector = permutation(gh.getGoal()->trajectory.joint_names, joint_names_); + // Goal should specify valid controller joints (they can be ordered differently). Reject if this is not the case + using internal::mapping; + std::vector mapping_vector = mapping(gh.getGoal()->trajectory.joint_names, joint_names_); - if (permutation_vector.empty()) + if (mapping_vector.empty()) { ROS_ERROR_NAMED(name_, "Joints on incoming goal don't match the controller joints."); control_msgs::FollowJointTrajectoryResult result; diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_segment.h b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_segment.h index ae09a8e33..bdaa27f2b 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_segment.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_segment.h @@ -85,21 +85,12 @@ class JointTrajectorySegment : public Segment /** * \param point Trajectory point. * - * \param permutation (Should be remove it) Permutation vector for mapping the joint order of a \p point to a desired order. - * For instance, if \p point contains data associated to joints "{B, D, A, C}", and we are interested in - * constructing a segment with joints ordered as "{A, B, C, D}", the permutation vector should - * be set to "{2, 0, 3, 1}". - * If unspecified (empty), the joint order of \p point is preserved; if specified, its size must coincide with that - * of \p point. - * This vector can be computed using the \ref trajectory_interface::internal::permutation() - * "permutation" utility function. + * \param permutation (Should be removed) * - * \param position_offset Position offset to applpy to the data in \p point. This parameter is useful for handling + * \param position_offset Position offset to apply to the data in \p point. This parameter is useful for handling * joints that wrap around (ie. continuous), to compensate for multi-turn offsets. * If unspecified (empty), zero offsets are applied; if specified, its size must coincide with that of \p point. * - * \note The offsets in \p position_offsets correspond to joints not ordered according to \p point, but to joints - * in the expected order, that is \p point with \p permutation applied to it. */ State(const trajectory_msgs::JointTrajectoryPoint& point, const std::vector& permutation = std::vector(), diff --git a/joint_trajectory_controller/test/init_joint_trajectory_test.cpp b/joint_trajectory_controller/test/init_joint_trajectory_test.cpp index fe1ea3884..c72035c1d 100644 --- a/joint_trajectory_controller/test/init_joint_trajectory_test.cpp +++ b/joint_trajectory_controller/test/init_joint_trajectory_test.cpp @@ -48,59 +48,70 @@ typedef JointTrajectorySegment TrajectoryPerJoint; typedef vector Trajectory; -TEST(PermutationTest, Permutation) +TEST(MappingTest, Mapping) { vector t1(4); - t1[0] = "A"; - t1[1] = "B"; - t1[2] = "C"; - t1[3] = "D"; + t1[0] = "B"; + t1[1] = "D"; + t1[2] = "A"; + t1[3] = "C"; vector t2(4); - t2[0] = "B"; - t2[1] = "D"; - t2[2] = "A"; - t2[3] = "C"; + t2[0] = "A"; + t2[1] = "B"; + t2[2] = "C"; + t2[3] = "D"; - typedef vector PermutationType; - PermutationType ground_truth(4); + typedef vector MappingType; + MappingType ground_truth(4); ground_truth[0] = 2; ground_truth[1] = 0; ground_truth[2] = 3; ground_truth[3] = 1; - // Mismatching sizes: Return empty permutation vector + // Mismatching sizes (t2 smaller than t1): Return empty mapping vector { vector t2_bad(1,"A"); - PermutationType perm = internal::permutation(t1, t2_bad); - EXPECT_TRUE(perm.empty()); + MappingType mapping = internal::mapping(t1, t2_bad); + EXPECT_TRUE(mapping.empty()); } - // Mismatching contents: Return empty permutation vector + // Mismatching contents: Return empty mapping vector { vector t2_bad(3,"A"); - PermutationType perm = internal::permutation(t1, t2_bad); - EXPECT_TRUE(perm.empty()); + MappingType mapping = internal::mapping(t1, t2_bad); + EXPECT_TRUE(mapping.empty()); } // Valid parameters { - PermutationType perm = internal::permutation(t1, t2); - EXPECT_EQ(ground_truth.size(), perm.size()); - EXPECT_EQ(ground_truth[0], perm[0]); - EXPECT_EQ(ground_truth[1], perm[1]); - EXPECT_EQ(ground_truth[2], perm[2]); - EXPECT_EQ(ground_truth[3], perm[3]); + MappingType mapping = internal::mapping(t1, t2); + EXPECT_EQ(ground_truth.size(), mapping.size()); + EXPECT_EQ(ground_truth[3], mapping[0]); + EXPECT_EQ(ground_truth[2], mapping[1]); + EXPECT_EQ(ground_truth[1], mapping[2]); + EXPECT_EQ(ground_truth[0], mapping[3]); } - // Valid parameters, inverse parameter order yields inverse permutation vector + // Valid parameters, inverse parameter order yields inverse mapping vector { - PermutationType perm = internal::permutation(t2, t1); - EXPECT_EQ(ground_truth.size(), perm.size()); - EXPECT_EQ(ground_truth[3], perm[0]); - EXPECT_EQ(ground_truth[2], perm[1]); - EXPECT_EQ(ground_truth[1], perm[2]); - EXPECT_EQ(ground_truth[0], perm[3]); + MappingType mapping = internal::mapping(t2, t1); + EXPECT_EQ(ground_truth.size(), mapping.size()); + EXPECT_EQ(ground_truth[0], mapping[0]); + EXPECT_EQ(ground_truth[1], mapping[1]); + EXPECT_EQ(ground_truth[2], mapping[2]); + EXPECT_EQ(ground_truth[3], mapping[3]); + } + + // Valid parameters, partial trajectory + { + vector t1_partial(2); + t1_partial[0] = "B"; + t1_partial[1] = "D"; + MappingType mapping = internal::mapping(t1_partial, t2); + EXPECT_EQ(t1_partial.size(), mapping.size()); + EXPECT_EQ(1, mapping[0]); + EXPECT_EQ(3, mapping[1]); } } @@ -475,7 +486,7 @@ TEST_F(InitTrajectoryTest, InitValuesCombine) } } -TEST_F(InitTrajectoryTest, JointPermutation) +TEST_F(InitTrajectoryTest, JointMapping) { // Add an extra joint to the trajectory message created in the fixture trajectory_msg.points[0].positions.push_back(-2.0); diff --git a/joint_trajectory_controller/test/joint_trajectory_segment_test.cpp b/joint_trajectory_controller/test/joint_trajectory_segment_test.cpp index 5ee9e25f0..c38209a6a 100644 --- a/joint_trajectory_controller/test/joint_trajectory_segment_test.cpp +++ b/joint_trajectory_controller/test/joint_trajectory_segment_test.cpp @@ -41,7 +41,7 @@ using namespace trajectory_msgs; const double EPS = 1e-9; typedef JointTrajectorySegment > Segment; -typedef std::vector PermutationType; +typedef std::vector MappingType; TEST(WraparoundOffsetTest, WrappingPositions) { @@ -237,28 +237,28 @@ TEST_F(JointTrajectorySegmentTest, InvalidSegmentConstruction) catch (const std::invalid_argument& ex) {ROS_ERROR_STREAM(ex.what());} } - // Invalid permutation vector size + // Invalid mapping vector size { - PermutationType permutation(2, 1); - EXPECT_THROW(Segment(traj_start_time, p_start, p_end, permutation), std::invalid_argument); - try {Segment(traj_start_time, p_start, p_end, permutation);} + MappingType mapping(2, 1); + EXPECT_THROW(Segment(traj_start_time, p_start, p_end, mapping), std::invalid_argument); + try {Segment(traj_start_time, p_start, p_end, mapping);} catch (const std::invalid_argument& ex) {ROS_ERROR_STREAM(ex.what());} } - // Invalid permutation vector indices + // Invalid mapping vector indices { - PermutationType permutation(1, 1); - EXPECT_THROW(Segment(traj_start_time, p_start, p_end, permutation), std::invalid_argument); - try {Segment(traj_start_time, p_start, p_end, permutation);} + MappingType mapping(1, 1); + EXPECT_THROW(Segment(traj_start_time, p_start, p_end, mapping), std::invalid_argument); + try {Segment(traj_start_time, p_start, p_end, mapping);} catch (const std::invalid_argument& ex) {ROS_ERROR_STREAM(ex.what());} } // Invalid joint wraparound specification { - PermutationType permutation; + MappingType mapping; std::vector pos_offset(2); - EXPECT_THROW(Segment(traj_start_time, p_start, p_end, permutation, pos_offset), std::invalid_argument); - try {Segment(traj_start_time, p_start, p_end, permutation, pos_offset);} + EXPECT_THROW(Segment(traj_start_time, p_start, p_end, mapping, pos_offset), std::invalid_argument); + try {Segment(traj_start_time, p_start, p_end, mapping, pos_offset);} catch (const std::invalid_argument& ex) {ROS_ERROR_STREAM(ex.what());} } } @@ -370,7 +370,7 @@ TEST_F(JointTrajectorySegmentTest, CrossValidateSegmentConstruction) } // TODO: Test with dimension four data -TEST_F(JointTrajectorySegmentTest, PermutationTest) +TEST_F(JointTrajectorySegmentTest, MappingTest) { // Add an extra joint to the trajectory point messages created in the fixture p_start.positions.push_back(-p_start.positions[0]); @@ -379,7 +379,7 @@ TEST_F(JointTrajectorySegmentTest, PermutationTest) p_end.positions.push_back(-p_end.positions[0]); p_end.velocities.push_back(-p_end.velocities[0]); - // No permutation vector + // No mapping vector { // Construct segment from ROS message EXPECT_NO_THROW(Segment(traj_start_time, p_start, p_end)); @@ -392,15 +392,15 @@ TEST_F(JointTrajectorySegmentTest, PermutationTest) EXPECT_EQ(p_start.positions[1], state.position[1]); } - // Permutation vector preserving trajectory message joint order + // Mapping vector preserving trajectory message joint order { - PermutationType permutation(2); - permutation[0] = 0; - permutation[1] = 1; + MappingType mapping(2); + mapping[0] = 0; + mapping[1] = 1; // Construct segment from ROS message - EXPECT_NO_THROW(Segment(traj_start_time, p_start, p_end, permutation)); - Segment segment(traj_start_time, p_start, p_end, permutation); + EXPECT_NO_THROW(Segment(traj_start_time, p_start, p_end, mapping)); + Segment segment(traj_start_time, p_start, p_end, mapping); // Check position values of start state only typename Segment::State state; @@ -409,15 +409,15 @@ TEST_F(JointTrajectorySegmentTest, PermutationTest) EXPECT_EQ(p_start.positions[1], state.position[1]); } - // Permutation vector reversing trajectory message joint order + // Mapping vector reversing trajectory message joint order { - PermutationType permutation(2); - permutation[0] = 1; - permutation[1] = 0; + MappingType mapping(2); + mapping[0] = 1; + mapping[1] = 0; // Construct segment from ROS message - EXPECT_NO_THROW(Segment(traj_start_time, p_start, p_end, permutation)); - Segment segment(traj_start_time, p_start, p_end, permutation); + EXPECT_NO_THROW(Segment(traj_start_time, p_start, p_end, mapping)); + Segment segment(traj_start_time, p_start, p_end, mapping); // Check position values of start state only typename Segment::State state; From eebbc96b221295f3456ee06f5584cc9baccdb062 Mon Sep 17 00:00:00 2001 From: Beatriz Leon Date: Tue, 18 Aug 2015 12:17:30 +0100 Subject: [PATCH 14/18] Removed permutation from vector class and fixed tests --- .../init_joint_trajectory.h | 9 +-- .../joint_trajectory_controller.h | 3 +- .../joint_trajectory_controller_impl.h | 14 +++-- .../joint_trajectory_segment.h | 32 ++--------- .../test/joint_trajectory_segment_test.cpp | 57 +------------------ 5 files changed, 21 insertions(+), 94 deletions(-) diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h index 3a42b10f9..98b988328 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h @@ -172,7 +172,6 @@ bool isNotEmpty(typename Trajectory::value_type trajPerJoint) * Segment(const ros::Time& traj_start_time, * const trajectory_msgs::JointTrajectoryPoint& start_point, * const trajectory_msgs::JointTrajectoryPoint& end_point, - * const std::vector& permutation, * const std::vector& position_offset) * \endcode * The following function must also be defined to properly handle continuous joints: @@ -334,8 +333,6 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg else result_traj.resize(joint_names.size()); - std::vector mapping_vector_per_joint(1,0); //refactor this and remove it as it is not needed - //Iterate through the joints that are in the message, in the order of the mapping vector //for (unsigned int joint_id=0; joint_id < joint_names.size();joint_id++) for (unsigned int msg_joint_it=0; msg_joint_it < mapping_vector.size();msg_joint_it++) @@ -374,7 +371,7 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg point_per_joint.time_from_start = it->time_from_start; const typename Segment::Time first_new_time = o_msg_start_time.toSec() + (it->time_from_start).toSec(); - typename Segment::State first_new_state(point_per_joint, mapping_vector_per_joint); // Here offsets are not yet applied + typename Segment::State first_new_state(point_per_joint); // Here offsets are not yet applied // Compute offsets due to wrapping joints if (has_angle_wraparound) @@ -385,7 +382,7 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg } // Apply offset to first state that will be executed from the new trajectory - first_new_state = typename Segment::State(point_per_joint, mapping_vector_per_joint, position_offset); // Now offsets are applied + first_new_state = typename Segment::State(point_per_joint, position_offset); // Now offsets are applied // Add useful segments of current trajectory to result { @@ -435,7 +432,7 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg if (!next_it->accelerations.empty()) {next_it_point_per_joint.accelerations.resize(1, next_it->accelerations[msg_joint_it]);} next_it_point_per_joint.time_from_start = next_it->time_from_start; - Segment segment(o_msg_start_time, it_point_per_joint, next_it_point_per_joint, mapping_vector_per_joint, position_offset); + Segment segment(o_msg_start_time, it_point_per_joint, next_it_point_per_joint, position_offset); segment.setGoalHandle(options.rt_goal_handle); if (has_rt_goal_handle) {segment.setTolerances(tolerances_per_joint);} result_traj_per_joint.push_back(segment); diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h index 581b42598..e5f88fb8b 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h @@ -40,6 +40,7 @@ // Boost #include #include +#include // ROS #include @@ -208,7 +209,7 @@ class JointTrajectoryController : public controller_interface::Controller successful_joint_traj_; + boost::dynamic_bitset<> successful_joint_traj_; // ROS API ros::NodeHandle controller_nh_; diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h index 8ffc2ab72..28304fa86 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h @@ -296,7 +296,7 @@ bool JointTrajectoryController::init(HardwareInt desired_joint_state_ = typename Segment::State(1); state_joint_error_ = typename Segment::State(1); - successful_joint_traj_.resize(joints_.size(), false); + successful_joint_traj_ = boost::dynamic_bitset<>(joints_.size()); // Initialize trajectory with all joints typename Segment::State current_joint_state_ = typename Segment::State(1); @@ -394,6 +394,7 @@ update(const ros::Time& time, const ros::Duration& period) control_msgs::FollowJointTrajectoryResult::PATH_TOLERANCE_VIOLATED; rt_segment_goal->setAborted(rt_segment_goal->preallocated_result_); rt_active_goal_.reset(); + successful_joint_traj_.reset(); } } else if (segment_it == --curr_traj[i].end()) @@ -410,8 +411,7 @@ update(const ros::Time& time, const ros::Duration& period) if (inside_goal_tolerances) { - successful_joint_traj_[i] = true; - + successful_joint_traj_[i] = 1; } else if (uptime.toSec() < segment_it->endTime() + tolerances.goal_time_tolerance) { @@ -429,17 +429,20 @@ update(const ros::Time& time, const ros::Duration& period) rt_segment_goal->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::GOAL_TOLERANCE_VIOLATED; rt_segment_goal->setAborted(rt_segment_goal->preallocated_result_); rt_active_goal_.reset(); + successful_joint_traj_.reset(); } } } } - //If all segments finished successfully then set goal as succeeded - if (std::find(successful_joint_traj_.begin(), successful_joint_traj_.end(), false) == successful_joint_traj_.end()) + //If there is an active goal and all segments finished successfully then set goal as succeeded + RealtimeGoalHandlePtr current_active_goal(rt_active_goal_); + if (current_active_goal and successful_joint_traj_.count() == joints_.size()) { rt_active_goal_->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::SUCCESSFUL; rt_active_goal_->setSucceeded(rt_active_goal_->preallocated_result_); rt_active_goal_.reset(); + successful_joint_traj_.reset(); } // Hardware interface adapter: Generate and send commands @@ -692,7 +695,6 @@ setHoldPosition(const ros::Time& time) assert(joint_names_.size() == hold_trajectory_ptr_->size()); - ROS_INFO_STREAM_NAMED(name_, "Setting hold position"); typename Segment::State hold_start_state_ = typename Segment::State(1); typename Segment::State hold_end_state_ = typename Segment::State(1); diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_segment.h b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_segment.h index bdaa27f2b..6915a7950 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_segment.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_segment.h @@ -85,22 +85,18 @@ class JointTrajectorySegment : public Segment /** * \param point Trajectory point. * - * \param permutation (Should be removed) - * * \param position_offset Position offset to apply to the data in \p point. This parameter is useful for handling * joints that wrap around (ie. continuous), to compensate for multi-turn offsets. * If unspecified (empty), zero offsets are applied; if specified, its size must coincide with that of \p point. * */ State(const trajectory_msgs::JointTrajectoryPoint& point, - const std::vector& permutation = std::vector(), const std::vector& position_offset = std::vector()) { - init(point, permutation, position_offset); + init(point, position_offset); } void init(const trajectory_msgs::JointTrajectoryPoint& point, - const std::vector& permutation = std::vector(), const std::vector& position_offset = std::vector()) { using std::invalid_argument; @@ -112,17 +108,6 @@ class JointTrajectorySegment : public Segment { throw(invalid_argument("Size mismatch in trajectory point position, velocity or acceleration data.")); } - if (!permutation.empty() && joint_dim != permutation.size()) - { - throw(invalid_argument("Size mismatch between trajectory point and permutation vector.")); - } - for (unsigned int i = 0; i < permutation.size(); ++i) - { - if (permutation[i] >= joint_dim) - { - throw(invalid_argument("Permutation vector contains out-of-range indices.")); - } - } if (!position_offset.empty() && joint_dim != position_offset.size()) { throw(invalid_argument("Size mismatch between trajectory point " @@ -136,15 +121,12 @@ class JointTrajectorySegment : public Segment for (unsigned int i = 0; i < joint_dim; ++i) { - // Apply permutation only if it was specified, otherwise preserve original message order - const unsigned int id = permutation.empty() ? i : permutation[i]; - // Apply position offset only if it was specified const Scalar offset = position_offset.empty() ? 0.0 : position_offset[i]; - if (!point.positions.empty()) {this->position[i] = point.positions[id] + offset;} - if (!point.velocities.empty()) {this->velocity[i] = point.velocities[id];} - if (!point.accelerations.empty()) {this->acceleration[i] = point.accelerations[id];} + if (!point.positions.empty()) {this->position[i] = point.positions[i] + offset;} + if (!point.velocities.empty()) {this->velocity[i] = point.velocities[i];} + if (!point.accelerations.empty()) {this->acceleration[i] = point.accelerations[i];} } } }; @@ -174,7 +156,6 @@ class JointTrajectorySegment : public Segment * segment start time. * \param start_point Start state in ROS message format. * \param end_point End state in ROS message format. - * \param permutation See \ref JointTrajectorySegment::State. * \param position_offset See \ref JointTrajectorySegment::State. * * \throw std::invalid_argument If input parameters are inconsistent and a valid segment can't be constructed. @@ -182,7 +163,6 @@ class JointTrajectorySegment : public Segment JointTrajectorySegment(const ros::Time& traj_start_time, const trajectory_msgs::JointTrajectoryPoint& start_point, const trajectory_msgs::JointTrajectoryPoint& end_point, - const std::vector& permutation = std::vector(), const std::vector& position_offset = std::vector()) : rt_goal_handle_(), tolerances_() @@ -198,8 +178,8 @@ class JointTrajectorySegment : public Segment try { - const State start_state(start_point, permutation, position_offset); - const State end_state(end_point, permutation, position_offset); + const State start_state(start_point, position_offset); + const State end_state(end_point, position_offset); this->init(start_time, start_state, end_time, end_state); diff --git a/joint_trajectory_controller/test/joint_trajectory_segment_test.cpp b/joint_trajectory_controller/test/joint_trajectory_segment_test.cpp index c38209a6a..14ac71cd0 100644 --- a/joint_trajectory_controller/test/joint_trajectory_segment_test.cpp +++ b/joint_trajectory_controller/test/joint_trajectory_segment_test.cpp @@ -237,28 +237,11 @@ TEST_F(JointTrajectorySegmentTest, InvalidSegmentConstruction) catch (const std::invalid_argument& ex) {ROS_ERROR_STREAM(ex.what());} } - // Invalid mapping vector size - { - MappingType mapping(2, 1); - EXPECT_THROW(Segment(traj_start_time, p_start, p_end, mapping), std::invalid_argument); - try {Segment(traj_start_time, p_start, p_end, mapping);} - catch (const std::invalid_argument& ex) {ROS_ERROR_STREAM(ex.what());} - } - - // Invalid mapping vector indices - { - MappingType mapping(1, 1); - EXPECT_THROW(Segment(traj_start_time, p_start, p_end, mapping), std::invalid_argument); - try {Segment(traj_start_time, p_start, p_end, mapping);} - catch (const std::invalid_argument& ex) {ROS_ERROR_STREAM(ex.what());} - } - // Invalid joint wraparound specification { - MappingType mapping; std::vector pos_offset(2); - EXPECT_THROW(Segment(traj_start_time, p_start, p_end, mapping, pos_offset), std::invalid_argument); - try {Segment(traj_start_time, p_start, p_end, mapping, pos_offset);} + EXPECT_THROW(Segment(traj_start_time, p_start, p_end, pos_offset), std::invalid_argument); + try {Segment(traj_start_time, p_start, p_end, pos_offset);} catch (const std::invalid_argument& ex) {ROS_ERROR_STREAM(ex.what());} } } @@ -391,42 +374,6 @@ TEST_F(JointTrajectorySegmentTest, MappingTest) EXPECT_EQ(p_start.positions[0], state.position[0]); EXPECT_EQ(p_start.positions[1], state.position[1]); } - - // Mapping vector preserving trajectory message joint order - { - MappingType mapping(2); - mapping[0] = 0; - mapping[1] = 1; - - // Construct segment from ROS message - EXPECT_NO_THROW(Segment(traj_start_time, p_start, p_end, mapping)); - Segment segment(traj_start_time, p_start, p_end, mapping); - - // Check position values of start state only - typename Segment::State state; - segment.sample(segment.startTime(), state); - EXPECT_EQ(p_start.positions[0], state.position[0]); - EXPECT_EQ(p_start.positions[1], state.position[1]); - } - - // Mapping vector reversing trajectory message joint order - { - MappingType mapping(2); - mapping[0] = 1; - mapping[1] = 0; - - // Construct segment from ROS message - EXPECT_NO_THROW(Segment(traj_start_time, p_start, p_end, mapping)); - Segment segment(traj_start_time, p_start, p_end, mapping); - - // Check position values of start state only - typename Segment::State state; - segment.sample(segment.startTime(), state); - EXPECT_NE(p_start.positions[0], state.position[0]); - EXPECT_NE(p_start.positions[1], state.position[1]); - EXPECT_EQ(p_start.positions[0], state.position[1]); - EXPECT_EQ(p_start.positions[1], state.position[0]); - } } int main(int argc, char** argv) From 967d316ef240c8c75247b00cd9e6f290a06bdc57 Mon Sep 17 00:00:00 2001 From: Beatriz Leon Date: Tue, 18 Aug 2015 16:02:42 +0100 Subject: [PATCH 15/18] Fixed comment about assigning goal handle only to segments after current time --- .../init_joint_trajectory.h | 13 +++++++++---- 1 file changed, 9 insertions(+), 4 deletions(-) diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h index 98b988328..6df4697dd 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h @@ -194,6 +194,7 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg typedef typename Trajectory::value_type TrajectoryPerJoint; typedef typename TrajectoryPerJoint::value_type Segment; typedef typename Segment::Scalar Scalar; + typedef typename TrajectoryPerJoint::const_iterator TrajIter; const unsigned int n_joints = msg.joint_names.size(); @@ -317,16 +318,21 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg // - Useful segments of new trajectory (contained in ROS message) Trajectory result_traj; + // Set active goal to segments after current time if (has_current_trajectory) { result_traj = *(options.current_trajectory); - //Iterate to all segments to set the new goal handler + //Iterate to all segments after current time to set the new goal handler for (unsigned int joint_id=0; joint_id < joint_names.size();joint_id++) { - for (unsigned int segment_id=0; segment_id < result_traj[joint_id].size(); segment_id++) + const TrajectoryPerJoint& curr_joint_traj = result_traj[joint_id]; + TrajIter active_seg = findSegment(curr_joint_traj, o_time.toSec()); // Currently active segment + + while (std::distance(active_seg, curr_joint_traj.end())>=1) { - (result_traj[joint_id])[segment_id].setGoalHandle(options.rt_goal_handle); + (result_traj[joint_id])[std::distance(curr_joint_traj.begin(),active_seg)].setGoalHandle(options.rt_goal_handle); + ++active_seg; } } } @@ -386,7 +392,6 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg // Add useful segments of current trajectory to result { - typedef typename TrajectoryPerJoint::const_iterator TrajIter; TrajIter first = findSegment(curr_joint_traj, o_time.toSec()); // Currently active segment TrajIter last = findSegment(curr_joint_traj, last_curr_time); // Segment active when new trajectory starts if (first == curr_joint_traj.end() || last == curr_joint_traj.end()) From a44284920eef164db4c1cb709d883744b60d4c7b Mon Sep 17 00:00:00 2001 From: Beatriz Leon Date: Wed, 19 Aug 2015 13:23:56 +0100 Subject: [PATCH 16/18] Fixed setting of success to goal handle when all joints succeeded --- .../joint_trajectory_controller_impl.h | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h index 28304fa86..bb15f0dce 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h @@ -439,8 +439,8 @@ update(const ros::Time& time, const ros::Duration& period) RealtimeGoalHandlePtr current_active_goal(rt_active_goal_); if (current_active_goal and successful_joint_traj_.count() == joints_.size()) { - rt_active_goal_->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::SUCCESSFUL; - rt_active_goal_->setSucceeded(rt_active_goal_->preallocated_result_); + current_active_goal->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::SUCCESSFUL; + current_active_goal->setSucceeded(current_active_goal->preallocated_result_); rt_active_goal_.reset(); successful_joint_traj_.reset(); } From b82cf4ed478cb619e27c2938a3859814af794491 Mon Sep 17 00:00:00 2001 From: Beatriz Leon Date: Wed, 19 Aug 2015 16:33:18 +0100 Subject: [PATCH 17/18] Added rosparam to allow partial joints goals --- .../init_joint_trajectory.h | 18 ++++++++-- .../joint_trajectory_controller.h | 1 + .../joint_trajectory_controller_impl.h | 33 +++++++++++++++---- 3 files changed, 43 insertions(+), 9 deletions(-) diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h index 6df4697dd..2a9e76589 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/init_joint_trajectory.h @@ -105,7 +105,8 @@ struct InitJointTrajectoryOptions angle_wraparound(0), rt_goal_handle(), default_tolerances(0), - other_time_base(0) + other_time_base(0), + allow_partial_joints_goal(false) {} Trajectory* current_trajectory; @@ -114,6 +115,7 @@ struct InitJointTrajectoryOptions RealtimeGoalHandlePtr rt_goal_handle; SegmentTolerances* default_tolerances; ros::Time* other_time_base; + bool allow_partial_joints_goal; }; template @@ -248,8 +250,6 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg o_msg_start_time = msg_start_time; } - // Mapping vector contains the map between the message joint order and the expected joint order - // If unspecified, a trivial map is computed const std::vector joint_names = has_joint_names ? *(options.joint_names) : msg.joint_names; if (has_angle_wraparound) @@ -264,6 +264,18 @@ Trajectory initJointTrajectory(const trajectory_msgs::JointTrajectory& msg } } + // If partial joints goals are not allowed, goal should specify all controller joints + if (!options.allow_partial_joints_goal) + { + if (msg.joint_names.size() != joint_names.size()) + { + ROS_ERROR("Cannot create trajectory from message. It does not contain the expected joints."); + return Trajectory(); + } + } + + // Mapping vector contains the map between the message joint order and the expected joint order + // If unspecified, a trivial map is computed std::vector mapping_vector = internal::mapping(msg.joint_names,joint_names); if (mapping_vector.empty()) diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h index e5f88fb8b..147b6e3a6 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller.h @@ -210,6 +210,7 @@ class JointTrajectoryController : public controller_interface::Controller successful_joint_traj_; + bool allow_partial_joints_goal_; // ROS API ros::NodeHandle controller_nh_; diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h index bb15f0dce..02837a4e9 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h @@ -223,6 +223,13 @@ bool JointTrajectoryController::init(HardwareInt } ROS_DEBUG_STREAM_NAMED(name_, "Stop trajectory has a duration of " << stop_trajectory_duration_ << "s."); + // Checking if partial trajectories are allowed + controller_nh_.param("allow_partial_joints_goal", allow_partial_joints_goal_, false); + if (allow_partial_joints_goal_) + { + ROS_ERROR_NAMED(name_, "Goals with partial set of joints are allowed"); + } + // List of controlled joints joint_names_ = getStrings(controller_nh_, "joints"); if (joint_names_.empty()) {return false;} @@ -494,12 +501,13 @@ updateTrajectoryCommand(const JointTrajectoryConstPtr& msg, RealtimeGoalHandlePt curr_trajectory_box_.get(curr_traj_ptr); Options options; - options.other_time_base = &next_update_uptime; - options.current_trajectory = curr_traj_ptr.get(); - options.joint_names = &joint_names_; - options.angle_wraparound = &angle_wraparound_; - options.rt_goal_handle = gh; - options.default_tolerances = &default_tolerances_; + options.other_time_base = &next_update_uptime; + options.current_trajectory = curr_traj_ptr.get(); + options.joint_names = &joint_names_; + options.angle_wraparound = &angle_wraparound_; + options.rt_goal_handle = gh; + options.default_tolerances = &default_tolerances_; + options.allow_partial_joints_goal = allow_partial_joints_goal_; // Update currently executing trajectory try @@ -546,6 +554,19 @@ goalCB(GoalHandle gh) return; } + // If partial joints goals are not allowed, goal should specify all controller joints + if (!allow_partial_joints_goal_) + { + if (gh.getGoal()->trajectory.joint_names.size() != joint_names_.size()) + { + ROS_ERROR_NAMED(name_, "Joints on incoming goal don't match the controller joints."); + control_msgs::FollowJointTrajectoryResult result; + result.error_code = control_msgs::FollowJointTrajectoryResult::INVALID_JOINTS; + gh.setRejected(result); + return; + } + } + // Goal should specify valid controller joints (they can be ordered differently). Reject if this is not the case using internal::mapping; std::vector mapping_vector = mapping(gh.getGoal()->trajectory.joint_names, joint_names_); From e19db70717cd605b816c42200e8178fdef925711 Mon Sep 17 00:00:00 2001 From: Beatriz Leon Date: Thu, 20 Aug 2015 17:14:03 +0100 Subject: [PATCH 18/18] added tests for partial joint goals --- joint_trajectory_controller/CMakeLists.txt | 5 + .../joint_trajectory_controller_impl.h | 8 +- .../joint_partial_trajectory_controller.test | 35 + ...int_partial_trajectory_controller_test.cpp | 637 ++++++++++++++++++ .../test/joint_trajectory_controller_test.cpp | 5 +- .../test/rrbot_partial_controllers.yaml | 15 + 6 files changed, 701 insertions(+), 4 deletions(-) create mode 100644 joint_trajectory_controller/test/joint_partial_trajectory_controller.test create mode 100644 joint_trajectory_controller/test/joint_partial_trajectory_controller_test.cpp create mode 100644 joint_trajectory_controller/test/rrbot_partial_controllers.yaml diff --git a/joint_trajectory_controller/CMakeLists.txt b/joint_trajectory_controller/CMakeLists.txt index 985e04280..98862e610 100644 --- a/joint_trajectory_controller/CMakeLists.txt +++ b/joint_trajectory_controller/CMakeLists.txt @@ -88,6 +88,11 @@ if(CATKIN_ENABLE_TESTING) test/joint_trajectory_controller.test test/joint_trajectory_controller_test.cpp) target_link_libraries(joint_trajectory_controller_test ${catkin_LIBRARIES}) + + add_rostest_gtest(joint_partial_trajectory_controller_test + test/joint_partial_trajectory_controller.test + test/joint_partial_trajectory_controller_test.cpp) + target_link_libraries(joint_partial_trajectory_controller_test ${catkin_LIBRARIES}) endif() # Install diff --git a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h index 02837a4e9..e2082b1f0 100644 --- a/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h +++ b/joint_trajectory_controller/include/joint_trajectory_controller/joint_trajectory_controller_impl.h @@ -227,7 +227,7 @@ bool JointTrajectoryController::init(HardwareInt controller_nh_.param("allow_partial_joints_goal", allow_partial_joints_goal_, false); if (allow_partial_joints_goal_) { - ROS_ERROR_NAMED(name_, "Goals with partial set of joints are allowed"); + ROS_DEBUG_NAMED(name_, "Goals with partial set of joints are allowed"); } // List of controlled joints @@ -396,7 +396,11 @@ update(const ros::Time& time, const ros::Duration& period) const SegmentTolerancesPerJoint& joint_tolerances = segment_it->getTolerances(); if (!checkStateTolerancePerJoint(state_joint_error_, joint_tolerances.state_tolerance)) { - ROS_ERROR_STREAM_NAMED(name_,"Path tolerances failed for joint: " << joint_names_[i]); + if (verbose_) + { + ROS_ERROR_STREAM_NAMED(name_,"Path tolerances failed for joint: " << joint_names_[i]); + checkStateTolerancePerJoint(state_joint_error_, joint_tolerances.state_tolerance, true); + } rt_segment_goal->preallocated_result_->error_code = control_msgs::FollowJointTrajectoryResult::PATH_TOLERANCE_VIOLATED; rt_segment_goal->setAborted(rt_segment_goal->preallocated_result_); diff --git a/joint_trajectory_controller/test/joint_partial_trajectory_controller.test b/joint_trajectory_controller/test/joint_partial_trajectory_controller.test new file mode 100644 index 000000000..7962df69c --- /dev/null +++ b/joint_trajectory_controller/test/joint_partial_trajectory_controller.test @@ -0,0 +1,35 @@ + + + + + + + + + + + + + + + + + + + + + diff --git a/joint_trajectory_controller/test/joint_partial_trajectory_controller_test.cpp b/joint_trajectory_controller/test/joint_partial_trajectory_controller_test.cpp new file mode 100644 index 000000000..270cfb48a --- /dev/null +++ b/joint_trajectory_controller/test/joint_partial_trajectory_controller_test.cpp @@ -0,0 +1,637 @@ +/////////////////////////////////////////////////////////////////////////////// +// Copyright (C) 2013, PAL Robotics S.L. +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are met: +// * Redistributions of source code must retain the above copyright notice, +// this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above copyright +// notice, this list of conditions and the following disclaimer in the +// documentation and/or other materials provided with the distribution. +// * Neither the name of PAL Robotics S.L. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" +// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE +// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE +// ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE +// LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR +// CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF +// SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS +// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN +// CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) +// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE +// POSSIBILITY OF SUCH DAMAGE. +////////////////////////////////////////////////////////////////////////////// + +/// \author Adolfo Rodriguez Tsouroukdissian + +#include +#include + +#include +#include + +#include + +#include +#include + +#include +#include +#include +#include + +// Floating-point value comparison threshold +const double EPS = 0.01; + +using actionlib::SimpleClientGoalState; + +class JointPartialTrajectoryControllerTest : public ::testing::Test +{ +public: + JointPartialTrajectoryControllerTest() + : nh("rrbot_controller"), + short_timeout(1.0), + long_timeout(10.0), + controller_state() + { + n_joints = (2); + n_partial_joints = (2); + joint_names.resize(n_joints); + joint_names[0] = "joint1"; + joint_names[1] = "joint2"; + + trajectory_msgs::JointTrajectoryPoint point; + point.positions.resize(n_joints, 0.0); + point.velocities.resize(n_joints, 0.0); + point.accelerations.resize(n_joints, 0.0); + + // Go home trajectory + traj_home.joint_names = joint_names; + traj_home.points.resize(1, point); + traj_home.points[0].time_from_start = ros::Duration(1.0); + + // Three-point trajectory + points.resize(3, point); + points[0].positions[0] = M_PI / 4.0; + points[0].positions[1] = 0.0; + points[0].time_from_start = ros::Duration(1.0); + + points[1].positions[0] = 0.0; + points[1].positions[1] = -M_PI / 4.0; + points[1].time_from_start = ros::Duration(2.0); + + points[2].positions[0] = -M_PI / 4.0; + points[2].positions[1] = M_PI / 4.0; + points[2].time_from_start = ros::Duration(4.0); + + traj.joint_names = joint_names; + traj.points = points; + + // Partial trajectory + traj_partial.joint_names.resize(1, "joint2"); + points.resize(3, point); + points[0].positions[0] = M_PI / 4.0; + points[0].time_from_start = ros::Duration(1.0); + + points[1].positions[0] = M_PI / 2.0; + points[1].time_from_start = ros::Duration(2.0); + + points[2].positions[0] = -M_PI / 2.0; + points[2].time_from_start = ros::Duration(4.0); + traj_partial.points = points; + + // Action goals + traj_home_goal.trajectory = traj_home; + traj_goal.trajectory = traj; + traj_partial_goal.trajectory = traj_partial; + + // Smoothing publisher (determines how well the robot follows a trajectory) + smoothing_pub = ros::NodeHandle().advertise("smoothing", 1); + + // Trajectory publisher + traj_pub = nh.advertise("command", 1); + + // State subscriber + state_sub = nh.subscribe("state", + 1, + &JointPartialTrajectoryControllerTest::stateCB, + this); + + // Query state service client + query_state_service = nh.serviceClient("query_state"); + + // Action client + const std::string action_server_name = nh.getNamespace() + "/follow_joint_trajectory"; + action_client.reset(new ActionClient(action_server_name)); + action_client2.reset(new ActionClient(action_server_name)); + } + + ~JointPartialTrajectoryControllerTest() + { + state_sub.shutdown(); // This is important, to make sure that the callback is not woken up later in the destructor + } + +protected: + typedef actionlib::SimpleActionClient ActionClient; + typedef boost::shared_ptr ActionClientPtr; + typedef control_msgs::FollowJointTrajectoryGoal ActionGoal; + typedef control_msgs::JointTrajectoryControllerStateConstPtr StateConstPtr; + + boost::mutex mutex; + ros::NodeHandle nh; + + unsigned int n_joints; + unsigned int n_partial_joints; + std::vector joint_names; + std::vector points; + + trajectory_msgs::JointTrajectory traj_home; + trajectory_msgs::JointTrajectory traj; + trajectory_msgs::JointTrajectory traj_partial; + ActionGoal traj_home_goal; + ActionGoal traj_goal; + ActionGoal traj_partial_goal; + + ros::Duration short_timeout; + ros::Duration long_timeout; + + ros::Publisher smoothing_pub; + ros::Publisher traj_pub; + ros::Subscriber state_sub; + ros::ServiceClient query_state_service; + ActionClientPtr action_client; + ActionClientPtr action_client2; + + + StateConstPtr controller_state; + + void stateCB(const StateConstPtr& state) + { + boost::mutex::scoped_lock lock(mutex); + controller_state = state; + } + + StateConstPtr getState() + { + boost::mutex::scoped_lock lock(mutex); + return controller_state; + } + + bool initState(const ros::Duration& timeout = ros::Duration(5.0)) + { + bool init_ok = false; + ros::Time start_time = ros::Time::now(); + while (!init_ok && (ros::Time::now() - start_time) < timeout) + { + { + boost::mutex::scoped_lock lock(mutex); + init_ok = controller_state && !controller_state->joint_names.empty(); + } + ros::Duration(0.1).sleep(); + } + return init_ok; + } + + static bool waitForState(const ActionClientPtr& action_client, + const actionlib::SimpleClientGoalState& state, + const ros::Duration& timeout) + { + using ros::Time; + using ros::Duration; + + Time start_time = Time::now(); + while (action_client->getState() != state && ros::ok()) + { + if (timeout >= Duration(0.0) && (Time::now() - start_time) > timeout) {return false;} // Timed-out + ros::Duration(0.01).sleep(); + } + return true; + } +}; + +// Invalid messages //////////////////////////////////////////////////////////////////////////////////////////////////// + +TEST_F(JointPartialTrajectoryControllerTest, invalidMessages) +{ + ASSERT_TRUE(initState()); + ASSERT_TRUE(action_client->waitForServer(long_timeout)); + + // Incompatible joint names + { + ActionGoal bad_goal = traj_home_goal; + bad_goal.trajectory.joint_names[0] = "bad_name"; + + bad_goal.trajectory.header.stamp = ros::Time(0); // Start immediately + action_client->sendGoal(bad_goal); + ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::REJECTED, long_timeout)); + EXPECT_EQ(action_client->getResult()->error_code, control_msgs::FollowJointTrajectoryResult::INVALID_JOINTS); + } + + // No position data + { + ActionGoal bad_goal = traj_home_goal; + bad_goal.trajectory.points[0].positions.clear(); + + bad_goal.trajectory.header.stamp = ros::Time(0); // Start immediately + action_client->sendGoal(bad_goal); + ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::REJECTED, long_timeout)); + EXPECT_EQ(action_client->getResult()->error_code, control_msgs::FollowJointTrajectoryResult::INVALID_GOAL); + } + + // Incompatible data sizes + { + ActionGoal bad_goal = traj_home_goal; + bad_goal.trajectory.points[0].positions.pop_back(); + + bad_goal.trajectory.header.stamp = ros::Time(0); // Start immediately + action_client->sendGoal(bad_goal); + ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::REJECTED, long_timeout)); + EXPECT_EQ(action_client->getResult()->error_code, control_msgs::FollowJointTrajectoryResult::INVALID_GOAL); + } + + // Non-strictly increasing waypoint times + { + ActionGoal bad_goal = traj_goal; + bad_goal.trajectory.points[2].time_from_start = bad_goal.trajectory.points[1].time_from_start; + + action_client->sendGoal(bad_goal); + ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::REJECTED, long_timeout)); + EXPECT_EQ(action_client->getResult()->error_code, control_msgs::FollowJointTrajectoryResult::INVALID_GOAL); + } + + // Empty trajectory through action interface + // NOTE: Sending an empty trajectory through the topic interface cancels execution of all queued segments, but + // an empty trajectory is interpreted by the action interface as not having the correct joint names. + { + ActionGoal empty_goal; + action_client->sendGoal(empty_goal); + ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::REJECTED, long_timeout)); + EXPECT_EQ(action_client->getResult()->error_code, control_msgs::FollowJointTrajectoryResult::INVALID_JOINTS); + } +} + +// Partial trajectory execution ////////////////////////////////////////////////////////////////////////////////////// + +TEST_F(JointPartialTrajectoryControllerTest, allowPartialTrajIfSpecified) +{ + ASSERT_TRUE(initState()); + ASSERT_TRUE(action_client->waitForServer(long_timeout)); + + // Send partial trajectory. + action_client->sendGoal(traj_partial_goal); + ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::ACTIVE, long_timeout)); +} + +// Uninterrupted Partial trajectory execution ////////////////////////////////////////////////////////////////////////////////// + +TEST_F(JointPartialTrajectoryControllerTest, topicSinglePartialTraj) +{ + ASSERT_TRUE(initState()); + + // Send partial trajectory + traj_partial.header.stamp = ros::Time(0); // Start immediately + traj_pub.publish(traj_partial); + ros::Duration wait_duration = traj_partial.points.back().time_from_start + ros::Duration(0.5); + wait_duration.sleep(); // Wait until done + + // Validate state topic values + StateConstPtr state = getState(); + + EXPECT_NEAR(traj_partial.points.back().positions[0], state->desired.positions[1], EPS); + EXPECT_NEAR(traj_partial.points.back().velocities[0], state->desired.velocities[1], EPS); + EXPECT_NEAR(traj_partial.points.back().accelerations[0], state->desired.accelerations[1], EPS); + + EXPECT_NEAR(traj_partial.points.back().positions[0], state->actual.positions[1], EPS); + EXPECT_NEAR(traj_partial.points.back().velocities[0], state->actual.velocities[1], EPS); + + EXPECT_NEAR(0.0, state->error.positions[1], EPS); + EXPECT_NEAR(0.0, state->error.velocities[1], EPS); + + // Validate query state service + control_msgs::QueryTrajectoryState srv; + srv.request.time = ros::Time::now(); + ASSERT_TRUE(query_state_service.call(srv)); + for (unsigned int i = 0; i < n_joints; ++i) + { + EXPECT_NEAR(state->desired.positions[i], srv.response.position[i], EPS); + EXPECT_NEAR(state->desired.velocities[i], srv.response.velocity[i], EPS); + EXPECT_NEAR(state->desired.accelerations[i], srv.response.acceleration[i], EPS); + } +} + +TEST_F(JointPartialTrajectoryControllerTest, actionSinglePartialTraj) +{ + ASSERT_TRUE(initState()); + ASSERT_TRUE(action_client->waitForServer(long_timeout)); + + // Send partial trajectory + traj_partial_goal.trajectory.header.stamp = ros::Time(0); // Start immediately + action_client->sendGoal(traj_partial_goal); + + // Wait until done + ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::SUCCEEDED, long_timeout)); + EXPECT_EQ(action_client->getResult()->error_code, control_msgs::FollowJointTrajectoryResult::SUCCESSFUL); + ros::Duration(0.5).sleep(); // Allows values to settle to within EPS, especially accelerations + + // Validate state topic values + StateConstPtr state = getState(); + + EXPECT_NEAR(traj_partial.points.back().positions[0], state->desired.positions[1], EPS); + EXPECT_NEAR(traj_partial.points.back().velocities[0], state->desired.velocities[1], EPS); + EXPECT_NEAR(traj_partial.points.back().accelerations[0], state->desired.accelerations[1], EPS); + + EXPECT_NEAR(traj_partial.points.back().positions[0], state->actual.positions[1], EPS); + EXPECT_NEAR(traj_partial.points.back().velocities[0], state->actual.velocities[1], EPS); + + EXPECT_NEAR(0.0, state->error.positions[1], EPS); + EXPECT_NEAR(0.0, state->error.velocities[1], EPS); +} + +// Trajectory replacement ////////////////////////////////////////////////////////////////////////////////////////////// + +TEST_F(JointPartialTrajectoryControllerTest, topicReplacesTopicTraj) +{ + ASSERT_TRUE(initState()); + ASSERT_TRUE(action_client->waitForServer(long_timeout)); + + // Send trajectory + traj.header.stamp = ros::Time(0); // Start immediately + traj_pub.publish(traj); + ros::Duration wait_duration = traj.points.front().time_from_start; + wait_duration.sleep(); // Wait until ~first waypoint + + // Send partial trajectory that preempts the previous one + traj_partial.header.stamp = ros::Time(0); // Start immediately + traj_pub.publish(traj_partial); + wait_duration = traj_partial.points.back().time_from_start + ros::Duration(0.5); + wait_duration.sleep(); // Wait until done + + // Check that we're at the original trajectory for joint1 and on the partial trajectory for joint2 + StateConstPtr state = getState(); + EXPECT_NEAR(traj.points.back().positions[0], state->desired.positions[0], EPS); + EXPECT_NEAR(traj.points.back().velocities[0], state->desired.velocities[0], EPS); + EXPECT_NEAR(traj.points.back().accelerations[0], state->desired.accelerations[0], EPS); + + EXPECT_NEAR(traj_partial.points.back().positions[0], state->desired.positions[1], EPS); + EXPECT_NEAR(traj_partial.points.back().velocities[0], state->desired.velocities[1], EPS); + EXPECT_NEAR(traj_partial.points.back().accelerations[0], state->desired.accelerations[1], EPS); +} + +TEST_F(JointPartialTrajectoryControllerTest, actionReplacesActionTraj) +{ + ASSERT_TRUE(initState()); + ASSERT_TRUE(action_client->waitForServer(long_timeout)); + + // Send trajectory + traj_goal.trajectory.header.stamp = ros::Time(0); // Start immediately + action_client->sendGoal(traj_goal); + ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::ACTIVE, short_timeout)); + + ros::Duration wait_duration = traj.points.front().time_from_start; + wait_duration.sleep(); // Wait until ~first waypoint + + // Send partial trajectory that preempts the previous one + traj_partial_goal.trajectory.header.stamp = ros::Time(0); // Start immediately + action_client2->sendGoal(traj_partial_goal); + ASSERT_TRUE(waitForState(action_client2, SimpleClientGoalState::ACTIVE, short_timeout)); + ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::PREEMPTED, short_timeout)); + //ROS_ERROR_STREAM("state: "<< action_client->getState().toString()); + + // Wait until done + ASSERT_TRUE(waitForState(action_client2, SimpleClientGoalState::SUCCEEDED, long_timeout)); + ros::Duration(0.5).sleep(); // Allows values to settle to within EPS, especially accelerations + + // Check that we're at the original trajectory for joint1 and on the partial trajectory for joint2 + StateConstPtr state = getState(); + EXPECT_NEAR(traj.points.back().positions[0], state->desired.positions[0], EPS); + EXPECT_NEAR(traj.points.back().velocities[0], state->desired.velocities[0], EPS); + EXPECT_NEAR(traj.points.back().accelerations[0], state->desired.accelerations[0], EPS); + + EXPECT_NEAR(traj_partial.points.back().positions[0], state->desired.positions[1], EPS); + EXPECT_NEAR(traj_partial.points.back().velocities[0], state->desired.velocities[1], EPS); + EXPECT_NEAR(traj_partial.points.back().accelerations[0], state->desired.accelerations[1], EPS); +} + +TEST_F(JointPartialTrajectoryControllerTest, actionReplacesTopicTraj) +{ + ASSERT_TRUE(initState()); + ASSERT_TRUE(action_client->waitForServer(long_timeout)); + + // Send trajectory + traj.header.stamp = ros::Time(0); // Start immediately + traj_pub.publish(traj); + ros::Duration wait_duration = traj.points.front().time_from_start; + wait_duration.sleep(); // Wait until ~first waypoint + + // Send partial trajectory that preempts the previous one + traj_partial_goal.trajectory.header.stamp = ros::Time(0); // Start immediately + action_client->sendGoal(traj_partial_goal); + ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::ACTIVE, short_timeout)); + + // Wait until done + ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::SUCCEEDED, long_timeout)); + ros::Duration(0.5).sleep(); // Allows values to settle to within EPS, especially accelerations + + // Check that we're at the original trajectory for joint1 and on the partial trajectory for joint2 + StateConstPtr state = getState(); + EXPECT_NEAR(traj.points.back().positions[0], state->desired.positions[0], EPS); + EXPECT_NEAR(traj.points.back().velocities[0], state->desired.velocities[0], EPS); + EXPECT_NEAR(traj.points.back().accelerations[0], state->desired.accelerations[0], EPS); + + EXPECT_NEAR(traj_partial.points.back().positions[0], state->desired.positions[1], EPS); + EXPECT_NEAR(traj_partial.points.back().velocities[0], state->desired.velocities[1], EPS); + EXPECT_NEAR(traj_partial.points.back().accelerations[0], state->desired.accelerations[1], EPS); +} + +TEST_F(JointPartialTrajectoryControllerTest, topicReplacesActionTraj) +{ + ASSERT_TRUE(initState()); + ASSERT_TRUE(action_client->waitForServer(long_timeout)); + + // Send trajectory + traj_goal.trajectory.header.stamp = ros::Time(0); // Start immediately + action_client->sendGoal(traj_goal); + ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::ACTIVE, short_timeout)); + + ros::Duration wait_duration = traj.points.front().time_from_start; + wait_duration.sleep(); // Wait until ~first waypoint + + // Send partial trajectory that preempts the previous one + traj_partial.header.stamp = ros::Time(0); // Start immediately + traj_pub.publish(traj_partial); + + ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::PREEMPTED, short_timeout)); + + wait_duration = traj_partial.points.back().time_from_start + ros::Duration(0.5); + wait_duration.sleep(); // Wait until done + + // Check that we're at the original trajectory for joint1 and on the partial trajectory for joint2 + StateConstPtr state = getState(); + EXPECT_NEAR(traj.points.back().positions[0], state->desired.positions[0], EPS); + EXPECT_NEAR(traj.points.back().velocities[0], state->desired.velocities[0], EPS); + EXPECT_NEAR(traj.points.back().accelerations[0], state->desired.accelerations[0], EPS); + + EXPECT_NEAR(traj_partial.points.back().positions[0], state->desired.positions[1], EPS); + EXPECT_NEAR(traj_partial.points.back().velocities[0], state->desired.velocities[1], EPS); + EXPECT_NEAR(traj_partial.points.back().accelerations[0], state->desired.accelerations[1], EPS); +} + +// Ignore old trajectory points //////////////////////////////////////////////////////////////////////////////////////// + +TEST_F(JointPartialTrajectoryControllerTest, ignoreOldTopicTraj) +{ + ASSERT_TRUE(initState()); + + // Send trajectory + traj.header.stamp = ros::Time(0); // Start immediately + traj_pub.publish(traj); + ros::Duration wait_duration = traj.points.front().time_from_start; + wait_duration.sleep(); // Wait until ~first waypoint + + // Send partial trajectory with all points occuring in the past. Should not preempts the previous one + traj_partial.header.stamp = ros::Time::now() - traj_partial.points.back().time_from_start - ros::Duration(0.5); + traj_pub.publish(traj_partial); + + wait_duration = traj.points.back().time_from_start - traj.points.front().time_from_start + ros::Duration(0.5); + wait_duration.sleep(); // Wait until first trajectory is done + + // Check that we're at the original trajectory end (Joint2 has not moved according to traj_partial) + StateConstPtr state = getState(); + for (unsigned int i = 0; i < n_joints; ++i) + { + EXPECT_NEAR(traj.points.back().positions[i], state->desired.positions[i], EPS); + EXPECT_NEAR(traj.points.back().velocities[i], state->desired.velocities[i], EPS); + EXPECT_NEAR(traj.points.back().accelerations[i], state->desired.accelerations[i], EPS); + } +} + +TEST_F(JointPartialTrajectoryControllerTest, ignoreOldActionTraj) +{ + ASSERT_TRUE(initState()); + ASSERT_TRUE(action_client->waitForServer(long_timeout)); + + // Send trajectory + traj_goal.trajectory.header.stamp = ros::Time(0); // Start immediately + action_client->sendGoal(traj_goal); + ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::ACTIVE, short_timeout)); + + ros::Duration wait_duration = traj.points.front().time_from_start; + wait_duration.sleep(); // Wait until ~first waypoint + + // Send partial trajectory with all points occuring in the past. Should not preempts the previous one + traj_partial_goal.trajectory.header.stamp = ros::Time::now() - traj_partial.points.back().time_from_start - ros::Duration(0.5); + action_client2->sendGoal(traj_partial_goal); + ASSERT_TRUE(waitForState(action_client2, SimpleClientGoalState::REJECTED, short_timeout)); + + // Wait until done + ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::SUCCEEDED, long_timeout)); + ros::Duration(0.5).sleep(); // Allows values to settle to within EPS, especially accelerations + + // Check that we're at the original trajectory end (Joint2 has not moved according to traj_partial) + StateConstPtr state = getState(); + for (unsigned int i = 0; i < n_joints; ++i) + { + EXPECT_NEAR(traj.points.back().positions[i], state->desired.positions[i], EPS); + EXPECT_NEAR(traj.points.back().velocities[i], state->desired.velocities[i], EPS); + EXPECT_NEAR(traj.points.back().accelerations[i], state->desired.accelerations[i], EPS); + } +} + +TEST_F(JointPartialTrajectoryControllerTest, ignorePartiallyOldActionTraj) +{ + ASSERT_TRUE(initState()); + ASSERT_TRUE(action_client->waitForServer(long_timeout)); + + // Send trajectory + ActionGoal first_goal = traj_goal; + first_goal.trajectory.header.stamp = ros::Time(0); // Start immediately + action_client->sendGoal(first_goal); + ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::ACTIVE, short_timeout)); + + ros::Duration wait_duration = first_goal.trajectory.points.front().time_from_start; + wait_duration.sleep(); // Wait until ~first waypoint + + // Send partial trajectory with only the last point not occuring in the past. Only the last point should be executed + traj_partial_goal.trajectory.header.stamp = ros::Time::now() - traj.points.back().time_from_start + ros::Duration(1.0); + action_client2->sendGoal(traj_partial_goal); + ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::PREEMPTED, short_timeout)); + ASSERT_TRUE(waitForState(action_client2, SimpleClientGoalState::ACTIVE, short_timeout)); + + // Wait until done + ASSERT_TRUE(waitForState(action_client2, SimpleClientGoalState::SUCCEEDED, long_timeout)); + ros::Duration(0.5).sleep(); // Allows values to settle to within EPS, especially accelerations + + // Check that we're at the original trajectory for joint1 and on the partial trajectory for joint2 + StateConstPtr state = getState(); + EXPECT_NEAR(traj.points.back().positions[0], state->desired.positions[0], EPS); + EXPECT_NEAR(traj.points.back().velocities[0], state->desired.velocities[0], EPS); + EXPECT_NEAR(traj.points.back().accelerations[0], state->desired.accelerations[0], EPS); + + EXPECT_NEAR(traj_partial.points.back().positions[0], state->desired.positions[1], EPS); + EXPECT_NEAR(traj_partial.points.back().velocities[0], state->desired.velocities[1], EPS); + EXPECT_NEAR(traj_partial.points.back().accelerations[0], state->desired.accelerations[1], EPS); +} + +TEST_F(JointPartialTrajectoryControllerTest, executeParitalActionTrajInFuture) +{ + ASSERT_TRUE(initState()); + ASSERT_TRUE(action_client->waitForServer(long_timeout)); + + // Send trajectory + ActionGoal first_goal = traj_goal; + first_goal.trajectory.header.stamp = ros::Time(0); // Start immediately + action_client->sendGoal(first_goal); + ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::ACTIVE, short_timeout)); + + ros::Duration wait_duration = first_goal.trajectory.points.front().time_from_start; + wait_duration.sleep(); // Wait until ~first waypoint + + // Send partial trajectory with only the last point not occuring in the past. Only the last point should be executed + traj_partial_goal.trajectory.header.stamp = ros::Time::now() + traj.points.back().time_from_start + ros::Duration(2.0); + action_client2->sendGoal(traj_partial_goal); + ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::PREEMPTED, short_timeout)); + ASSERT_TRUE(waitForState(action_client2, SimpleClientGoalState::ACTIVE, short_timeout)); + + // Wait until first traj is done + wait_duration = first_goal.trajectory.points.back().time_from_start; + wait_duration.sleep(); // Wait until ~first waypoint + + StateConstPtr state = getState(); + for (unsigned int i = 0; i < n_joints; ++i) + { + EXPECT_NEAR(traj.points.back().positions[i], state->desired.positions[i], EPS); + EXPECT_NEAR(traj.points.back().velocities[i], state->desired.velocities[i], EPS); + EXPECT_NEAR(traj.points.back().accelerations[i], state->desired.accelerations[i], EPS); + } + + // Wait until done + ASSERT_TRUE(waitForState(action_client2, SimpleClientGoalState::SUCCEEDED, long_timeout)); + ros::Duration(0.5).sleep(); // Allows values to settle to within EPS, especially accelerations + + // Check that we're at the original trajectory for joint1 and on the partial trajectory for joint2 + state = getState(); + EXPECT_NEAR(traj.points.back().positions[0], state->desired.positions[0], EPS); + EXPECT_NEAR(traj.points.back().velocities[0], state->desired.velocities[0], EPS); + EXPECT_NEAR(traj.points.back().accelerations[0], state->desired.accelerations[0], EPS); + + EXPECT_NEAR(traj_partial.points.back().positions[0], state->desired.positions[1], EPS); + EXPECT_NEAR(traj_partial.points.back().velocities[0], state->desired.velocities[1], EPS); + EXPECT_NEAR(traj_partial.points.back().accelerations[0], state->desired.accelerations[1], EPS); +} + + +int main(int argc, char** argv) +{ + testing::InitGoogleTest(&argc, argv); + ros::init(argc, argv, "joint_partial_trajectory_controller_test"); + + ros::AsyncSpinner spinner(1); + spinner.start(); + int ret = RUN_ALL_TESTS(); + spinner.stop(); + ros::shutdown(); + return ret; +} diff --git a/joint_trajectory_controller/test/joint_trajectory_controller_test.cpp b/joint_trajectory_controller/test/joint_trajectory_controller_test.cpp index 688741ac1..7dfd1ea48 100644 --- a/joint_trajectory_controller/test/joint_trajectory_controller_test.cpp +++ b/joint_trajectory_controller/test/joint_trajectory_controller_test.cpp @@ -248,7 +248,7 @@ TEST_F(JointTrajectoryControllerTest, invalidMessages) ASSERT_TRUE(initState()); ASSERT_TRUE(action_client->waitForServer(long_timeout)); - // Invalid size => Now this is possible. Partial trajectory + // Invalid size (No partial joints goals allowed) { trajectory_msgs::JointTrajectoryPoint point; point.positions.resize(1, 0.0); @@ -264,7 +264,8 @@ TEST_F(JointTrajectoryControllerTest, invalidMessages) bad_goal.trajectory.header.stamp = ros::Time(0); // Start immediately action_client->sendGoal(bad_goal); - ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::ACTIVE, long_timeout)); + ASSERT_TRUE(waitForState(action_client, SimpleClientGoalState::REJECTED, long_timeout)); + EXPECT_EQ(action_client->getResult()->error_code, control_msgs::FollowJointTrajectoryResult::INVALID_JOINTS); } // Incompatible joint names diff --git a/joint_trajectory_controller/test/rrbot_partial_controllers.yaml b/joint_trajectory_controller/test/rrbot_partial_controllers.yaml new file mode 100644 index 000000000..e7a319544 --- /dev/null +++ b/joint_trajectory_controller/test/rrbot_partial_controllers.yaml @@ -0,0 +1,15 @@ +rrbot_controller: + type: "position_controllers/JointTrajectoryController" + joints: + - joint1 + - joint2 + + constraints: + goal_time: 0.5 + joint1: + goal: 0.01 + trajectory: 0.05 + joint2: + goal: 0.01 + trajectory: 0.05 + allow_partial_joints_goal: true