mirror of
https://gitlab.com/obbart/universal_robots_ros_driver.git
synced 2026-04-10 01:50:46 +02:00
Action server improvements
This commit is contained in:
@@ -1,24 +1,23 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#include <actionlib/server/action_server.h>
|
||||||
|
#include <actionlib/server/server_goal_handle.h>
|
||||||
|
#include <control_msgs/FollowJointTrajectoryAction.h>
|
||||||
|
#include <ros/ros.h>
|
||||||
#include <atomic>
|
#include <atomic>
|
||||||
#include <chrono>
|
#include <chrono>
|
||||||
#include <condition_variable>
|
#include <condition_variable>
|
||||||
#include <mutex>
|
#include <mutex>
|
||||||
#include <set>
|
#include <set>
|
||||||
#include <thread>
|
#include <thread>
|
||||||
#include <ros/ros.h>
|
|
||||||
#include <actionlib/server/action_server.h>
|
|
||||||
#include <actionlib/server/server_goal_handle.h>
|
|
||||||
#include <control_msgs/FollowJointTrajectoryAction.h>
|
|
||||||
#include "ur_modern_driver/log.h"
|
#include "ur_modern_driver/log.h"
|
||||||
|
#include "ur_modern_driver/ros/service_stopper.h"
|
||||||
|
#include "ur_modern_driver/ros/trajectory_follower.h"
|
||||||
#include "ur_modern_driver/ur/consumer.h"
|
#include "ur_modern_driver/ur/consumer.h"
|
||||||
#include "ur_modern_driver/ur/master_board.h"
|
#include "ur_modern_driver/ur/master_board.h"
|
||||||
#include "ur_modern_driver/ur/state.h"
|
#include "ur_modern_driver/ur/state.h"
|
||||||
#include "ur_modern_driver/ros/service_stopper.h"
|
|
||||||
#include "ur_modern_driver/ros/trajectory_follower.h"
|
|
||||||
|
|
||||||
|
class ActionServer : public Service, public IConsumer<RTPacket>
|
||||||
class ActionServer : public Service //,public URRTPacketConsumer
|
|
||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
typedef control_msgs::FollowJointTrajectoryAction Action;
|
typedef control_msgs::FollowJointTrajectoryAction Action;
|
||||||
@@ -32,8 +31,9 @@ private:
|
|||||||
std::vector<std::string> joint_names_;
|
std::vector<std::string> joint_names_;
|
||||||
std::set<std::string> joint_set_;
|
std::set<std::string> joint_set_;
|
||||||
double max_velocity_;
|
double max_velocity_;
|
||||||
RobotState state_;
|
|
||||||
|
|
||||||
|
RobotState state_;
|
||||||
|
std::array<double, 6> q_actual_, qd_actual_;
|
||||||
|
|
||||||
GoalHandle curr_gh_;
|
GoalHandle curr_gh_;
|
||||||
std::atomic<bool> interrupt_traj_;
|
std::atomic<bool> interrupt_traj_;
|
||||||
@@ -58,11 +58,16 @@ private:
|
|||||||
std::vector<size_t> reorderMap(std::vector<std::string> goal_joints);
|
std::vector<size_t> reorderMap(std::vector<std::string> goal_joints);
|
||||||
|
|
||||||
void trajectoryThread();
|
void trajectoryThread();
|
||||||
|
bool updateState(RTShared& data);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
ActionServer(TrajectoryFollower& follower, std::vector<std::string>& joint_names, double max_velocity);
|
ActionServer(TrajectoryFollower& follower, std::vector<std::string>& joint_names, double max_velocity);
|
||||||
|
|
||||||
void start();
|
void start();
|
||||||
virtual void onRobotStateChange(RobotState state);
|
virtual void onRobotStateChange(RobotState state);
|
||||||
|
|
||||||
|
virtual bool consume(RTState_V1_6__7& state);
|
||||||
|
virtual bool consume(RTState_V1_8& state);
|
||||||
|
virtual bool consume(RTState_V3_0__1& state);
|
||||||
|
virtual bool consume(RTState_V3_2__3& state);
|
||||||
};
|
};
|
||||||
@@ -12,6 +12,9 @@ ActionServer::ActionServer(TrajectoryFollower& follower, std::vector<std::string
|
|||||||
, joint_names_(joint_names)
|
, joint_names_(joint_names)
|
||||||
, joint_set_(joint_names.begin(), joint_names.end())
|
, joint_set_(joint_names.begin(), joint_names.end())
|
||||||
, max_velocity_(max_velocity)
|
, max_velocity_(max_velocity)
|
||||||
|
, interrupt_traj_(false)
|
||||||
|
, running_(false)
|
||||||
|
, has_goal_(false)
|
||||||
, state_(RobotState::Error)
|
, state_(RobotState::Error)
|
||||||
, follower_(follower)
|
, follower_(follower)
|
||||||
{
|
{
|
||||||
@@ -21,23 +24,82 @@ void ActionServer::start()
|
|||||||
{
|
{
|
||||||
if(running_)
|
if(running_)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
LOG_INFO("Starting ActionServer");
|
||||||
running_ = true;
|
running_ = true;
|
||||||
tj_thread_ = thread(&ActionServer::trajectoryThread, this);
|
tj_thread_ = thread(&ActionServer::trajectoryThread, this);
|
||||||
|
as_.start();
|
||||||
}
|
}
|
||||||
|
|
||||||
void ActionServer::onRobotStateChange(RobotState state)
|
void ActionServer::onRobotStateChange(RobotState state)
|
||||||
{
|
{
|
||||||
state_ = state;
|
state_ = state;
|
||||||
|
|
||||||
|
//don't interrupt if everything is fine
|
||||||
|
if(state == RobotState::Running)
|
||||||
|
return;
|
||||||
|
|
||||||
|
//don't retry interrupts
|
||||||
|
if(interrupt_traj_ || !has_goal_)
|
||||||
|
return;
|
||||||
|
|
||||||
|
//on successful lock we're not executing a goal so don't interrupt
|
||||||
|
if(tj_mutex_.try_lock())
|
||||||
|
{
|
||||||
|
tj_mutex_.unlock();
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
interrupt_traj_ = true;
|
||||||
|
//wait for goal to be interrupted
|
||||||
|
std::lock_guard<std::mutex> lock(tj_mutex_);
|
||||||
|
|
||||||
|
Result res;
|
||||||
|
res.error_code = -100;
|
||||||
|
res.error_string = "Received another trajectory";
|
||||||
|
curr_gh_.setAborted(res, res.error_string);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool ActionServer::updateState(RTShared& data)
|
||||||
|
{
|
||||||
|
q_actual_ = data.q_actual;
|
||||||
|
qd_actual_ = data.qd_actual;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool ActionServer::consume(RTState_V1_6__7& state)
|
||||||
|
{
|
||||||
|
return updateState(state);
|
||||||
|
}
|
||||||
|
bool ActionServer::consume(RTState_V1_8& state)
|
||||||
|
{
|
||||||
|
return updateState(state);
|
||||||
|
}
|
||||||
|
bool ActionServer::consume(RTState_V3_0__1& state)
|
||||||
|
{
|
||||||
|
return updateState(state);
|
||||||
|
}
|
||||||
|
bool ActionServer::consume(RTState_V3_2__3& state)
|
||||||
|
{
|
||||||
|
return updateState(state);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
void ActionServer::onGoal(GoalHandle gh)
|
void ActionServer::onGoal(GoalHandle gh)
|
||||||
{
|
{
|
||||||
Result res;
|
Result res;
|
||||||
res.error_code = -100;
|
res.error_code = -100;
|
||||||
|
|
||||||
|
LOG_INFO("Received new goal");
|
||||||
|
|
||||||
if(!validate(gh, res) || !try_execute(gh, res))
|
if(!validate(gh, res) || !try_execute(gh, res))
|
||||||
|
{
|
||||||
|
LOG_WARN("Goal error: %s", res.error_string.c_str());
|
||||||
gh.setRejected(res, res.error_string);
|
gh.setRejected(res, res.error_string);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void ActionServer::onCancel(GoalHandle gh)
|
void ActionServer::onCancel(GoalHandle gh)
|
||||||
{
|
{
|
||||||
@@ -53,7 +115,7 @@ void ActionServer::onCancel(GoalHandle gh)
|
|||||||
|
|
||||||
bool ActionServer::validate(GoalHandle& gh, Result& res)
|
bool ActionServer::validate(GoalHandle& gh, Result& res)
|
||||||
{
|
{
|
||||||
return !validateState(gh, res) || !validateJoints(gh, res) || !validateTrajectory(gh, res);
|
return validateState(gh, res) && validateJoints(gh, res) && validateTrajectory(gh, res);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ActionServer::validateState(GoalHandle& gh, Result& res)
|
bool ActionServer::validateState(GoalHandle& gh, Result& res)
|
||||||
@@ -100,6 +162,10 @@ bool ActionServer::validateTrajectory(GoalHandle& gh, Result& res)
|
|||||||
auto goal = gh.getGoal();
|
auto goal = gh.getGoal();
|
||||||
res.error_code = Result::INVALID_GOAL;
|
res.error_code = Result::INVALID_GOAL;
|
||||||
|
|
||||||
|
//must at least have one point
|
||||||
|
if(goal->trajectory.points.size() < 1)
|
||||||
|
return false;
|
||||||
|
|
||||||
for(auto const& point : goal->trajectory.points)
|
for(auto const& point : goal->trajectory.points)
|
||||||
{
|
{
|
||||||
if(point.velocities.size() != joint_names_.size())
|
if(point.velocities.size() != joint_names_.size())
|
||||||
@@ -194,28 +260,41 @@ std::vector<size_t> ActionServer::reorderMap(std::vector<std::string> goal_joint
|
|||||||
|
|
||||||
void ActionServer::trajectoryThread()
|
void ActionServer::trajectoryThread()
|
||||||
{
|
{
|
||||||
|
LOG_INFO("Trajectory thread started");
|
||||||
follower_.start(); //todo check error
|
follower_.start(); //todo check error
|
||||||
//as_.start();
|
|
||||||
while(running_)
|
while(running_)
|
||||||
{
|
{
|
||||||
std::unique_lock<std::mutex> lk(tj_mutex_);
|
std::unique_lock<std::mutex> lk(tj_mutex_);
|
||||||
if(!tj_cv_.wait_for(lk, std::chrono::milliseconds(100), [&]{return running_ && has_goal_;}))
|
if(!tj_cv_.wait_for(lk, std::chrono::milliseconds(100), [&]{return running_ && has_goal_;}))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
LOG_DEBUG("Trajectory received and accepted");
|
LOG_INFO("Trajectory received and accepted");
|
||||||
curr_gh_.setAccepted();
|
curr_gh_.setAccepted();
|
||||||
|
|
||||||
auto goal = curr_gh_.getGoal();
|
auto goal = curr_gh_.getGoal();
|
||||||
auto mapping = reorderMap(goal->trajectory.joint_names);
|
|
||||||
std::vector<TrajectoryPoint> trajectory(goal->trajectory.points.size());
|
std::vector<TrajectoryPoint> trajectory(goal->trajectory.points.size());
|
||||||
|
|
||||||
|
//joint names of the goal might have a different ordering compared
|
||||||
|
//to what URScript expects so need to map between the two
|
||||||
|
auto mapping = reorderMap(goal->trajectory.joint_names);
|
||||||
|
|
||||||
|
LOG_INFO("Translating trajectory");
|
||||||
|
|
||||||
|
auto const& fp = goal->trajectory.points[0];
|
||||||
|
auto fpt = convert(fp.time_from_start)
|
||||||
|
|
||||||
|
//make sure we have a proper t0 position
|
||||||
|
if(fpt > std::chrono::microseconds(0))
|
||||||
|
{
|
||||||
|
LOG_INFO("Trajectory without t0 recieved, inserting t0 at currrent position");
|
||||||
|
trajectory.push_back(TrajectoryPoint(q_actual_, qd_actual_, std::chrono::microseconds(0)));
|
||||||
|
}
|
||||||
|
|
||||||
for(auto const& point : goal->trajectory.points)
|
for(auto const& point : goal->trajectory.points)
|
||||||
{
|
{
|
||||||
std::array<double, 6> pos, vel;
|
std::array<double, 6> pos, vel;
|
||||||
for(size_t i = 0; i < 6; i++)
|
for(size_t i = 0; i < 6; i++)
|
||||||
{
|
{
|
||||||
//joint names of the goal might have a different ordering compared
|
|
||||||
//to what URScript expects so need to map between the two
|
|
||||||
size_t idx = mapping[i];
|
size_t idx = mapping[i];
|
||||||
pos[idx] = point.positions[i];
|
pos[idx] = point.positions[i];
|
||||||
vel[idx] = point.velocities[i];
|
vel[idx] = point.velocities[i];
|
||||||
@@ -223,20 +302,25 @@ void ActionServer::trajectoryThread()
|
|||||||
trajectory.push_back(TrajectoryPoint(pos, vel, convert(point.time_from_start)));
|
trajectory.push_back(TrajectoryPoint(pos, vel, convert(point.time_from_start)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
double t = std::chrono::duration_cast<std::chrono::duration<double>>(trajectory[trajectory.size()-1].time_from_start).count();
|
||||||
|
LOG_INFO("Trajectory with %d points and duration of %f constructed, executing now", trajectory.size(), t);
|
||||||
|
|
||||||
Result res;
|
Result res;
|
||||||
if(follower_.execute(trajectory, interrupt_traj_))
|
if(follower_.execute(trajectory, interrupt_traj_))
|
||||||
{
|
{
|
||||||
//interrupted goals must be handled by interrupt trigger
|
//interrupted goals must be handled by interrupt trigger
|
||||||
if(!interrupt_traj_)
|
if(!interrupt_traj_)
|
||||||
{
|
{
|
||||||
LOG_DEBUG("Trajectory executed successfully");
|
LOG_INFO("Trajectory executed successfully");
|
||||||
res.error_code = Result::SUCCESSFUL;
|
res.error_code = Result::SUCCESSFUL;
|
||||||
curr_gh_.setSucceeded(res);
|
curr_gh_.setSucceeded(res);
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
LOG_INFO("Trajectory interrupted");
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
LOG_DEBUG("Trajectory failed");
|
LOG_INFO("Trajectory failed");
|
||||||
res.error_code = -100;
|
res.error_code = -100;
|
||||||
res.error_string = "Connection to robot was lost";
|
res.error_string = "Connection to robot was lost";
|
||||||
curr_gh_.setAborted(res, res.error_string);
|
curr_gh_.setAborted(res, res.error_string);
|
||||||
|
|||||||
Reference in New Issue
Block a user