NavigateToLocation.cpp
Go to the documentation of this file.
1 #include "NavigateToLocation.h"
2 
3 #include <chrono>
4 #include <future>
5 #include <mutex>
6 #include <optional>
7 #include <sstream>
8 #include <string>
9 
10 #include <Eigen/Core>
11 #include <Eigen/Geometry>
12 
18 
22 
30 
32 {
33 
34  template <typename R>
35  bool
36  is_ready(std::future<R> const& f)
37  {
38  if (!f.valid())
39  {
40  return true;
41  }
42  std::future_status status = f.wait_for(std::chrono::seconds(0));
43  if (status == std::future_status::ready)
44  {
45  return true;
46  }
47  else if (status == std::future_status::deferred)
48  {
49  // if the task has not been started --> start the task
50  f.wait();
51  return true;
52  }
53  else
54  {
55  return false;
56  }
57  }
58 
61  {
62  ParamType defaultParameters;
63  defaultParameters.location = "";
64  defaultParameters.enableSafetyGuard = false;
65  defaultParameters.velocityLimitAngular = std::nullopt;
66  defaultParameters.velocityLimitLinear = std::nullopt;
67 
68  std::stringstream description;
70  << "Retrieve the location `location` from the memory and navigate to it."
71  << "\n\nThe `location` is specified by its provider segment name and entity name"
72  " in the format `providerSegmentName/entityName` .";
73 
76  .description = description.str(),
77  .rootProfileDefaults = defaultParameters.toAron(),
79  .parametersType = Params::ToAronType(),
80  };
81  }
82 
84  Base(DefaultSkillDescription()), properties_(properties)
85  {
86  srv_.emplace(srv);
87  }
88 
90  NavigateToLocation::init(const Base::SpecializedInitInput& in)
91  {
93  trajControllerParams{};
95 
96  client::GeneralConfig generalParams{};
97 
98  if (in.parameters.velocityLimitLinear.has_value())
99  {
100  trajControllerParams.limits.linear = in.parameters.velocityLimitLinear.value();
101  globalPlannerParams.linearVelocity = in.parameters.velocityLimitLinear.value();
102  generalParams.maxVel.linear = in.parameters.velocityLimitLinear.value();
103  }
104 
105  if (in.parameters.velocityLimitAngular.has_value())
106  {
107  trajControllerParams.limits.angular = in.parameters.velocityLimitAngular.value();
108  globalPlannerParams.angularVelocity = in.parameters.velocityLimitAngular.value();
109  generalParams.maxVel.angular = in.parameters.velocityLimitAngular.value();
110  }
111 
112  ARMARX_INFO << "Configuring navigator to respect the velocity limits:";
113  ARMARX_INFO << "linear: " << trajControllerParams.limits.linear;
114  ARMARX_INFO << "angular: " << trajControllerParams.limits.angular;
115 
116  // FIXME set
118  {
120  std::lock_guard g{*properties_.safetyGuardParamsMutex};
121  if (properties_.safetyGuardParams != nullptr)
122  {
123  safetyGuardParams = *properties_.safetyGuardParams;
124  }
125  }
126 
127  // parameterize the navigation stack
128  client::NavigationStackConfig cfg;
129  cfg.general(generalParams);
130  cfg.globalPlanner(globalPlannerParams);
131  cfg.trajectoryController(trajControllerParams);
132  if (in.parameters.enableSafetyGuard)
133  {
134  cfg.safetyGuard(safetyGuardParams);
135  }
136 
137  const std::string configId = DefaultSkillDescription().skillId.skillName;
138 
139  // configure the `navigator` which provides a simplified and typed interface to the navigation server
140  ARMARX_TRACE;
141  memorySubscriber.reset();
142 
143  ARMARX_TRACE;
144  memorySubscriber.emplace(configId, srv_->memoryNameSystem);
145 
146  // register our config
147  ARMARX_INFO << "Registering config";
148  srv_->iceNavigator.createConfig(cfg, configId);
149 
150  navigator.emplace(client::Navigator::InjectedServices{
151  .navigator = &srv_->iceNavigator, .subscriber = &memorySubscriber.value()});
152 
153  return ::armarx::skills::Skill::InitResult{
154  .status = ::armarx::skills::TerminatedSkillStatus::Succeeded};
155  }
156 
158  NavigateToLocation::main(const Base::SpecializedMainInput& in)
159  {
160  const std::string& location = in.parameters.location;
161 
162  ARMARX_INFO << "moving to target " << VAROUT(location);
163 
164  // execute
165  ARMARX_INFO << "Sending navigation request";
166 
167  // client::PathBuilder builder;
168  // builder.add(location, client::GlobalPlanningStrategy::Free);
169 
170  navigator->moveToLocation(location);
171 
172  // Wait until goal is reached
173  ARMARX_INFO << "Waiting until goal is reached.";
174  auto future = std::async(std::launch::async, [this]() { return navigator->waitForStop(); });
175  while (not is_ready(future))
176  {
177  if (shouldSkillTerminate())
178  {
179  ARMARX_INFO << "Skill was aborted by user";
180  navigator->stop();
181  break;
182  }
183  }
184 
185  auto se = future.get();
186  if (se)
187  {
188  ARMARX_INFO << "Goal `" << location << "`reached.";
189  return ::armarx::skills::Skill::MainResult{
190  .status = ::armarx::skills::TerminatedSkillStatus::Succeeded};
191  }
192  else
193  {
194  if (se.isSafetyStopTriggeredEvent())
195  {
196  ARMARX_INFO << "Safety stop was triggered!";
197 
198  return ::armarx::skills::Skill::MainResult{
199  .status = ::armarx::skills::TerminatedSkillStatus::Failed};
200  }
201  if (se.isUserAbortTriggeredEvent())
202  {
203  ARMARX_INFO << "Aborted by user!";
204 
205  return ::armarx::skills::Skill::MainResult{
206  .status = ::armarx::skills::TerminatedSkillStatus::Aborted};
207  }
208  if (se.isInternalErrorEvent())
209  {
210  ARMARX_INFO << "Unknown internal error occured! "
211  << se.toInternalErrorEvent().message;
212 
213  return ::armarx::skills::Skill::MainResult{
214  .status = ::armarx::skills::TerminatedSkillStatus::Failed};
215  }
216  }
217 
218  return ::armarx::skills::Skill::MainResult{
219  .status = ::armarx::skills::TerminatedSkillStatus::Failed};
220  }
221 
222  void
223  NavigateToLocation::onStopRequested()
224  {
225  if (navigator.has_value())
226  {
227  ARMARX_CHECK(navigator.has_value());
228  navigator->stop();
229  }
230  }
231 
232 } // namespace armarx::navigation::skills
armarx::skills::SimpleSpecializedSkill< arondto::NavigateToLocationParams >::init
Skill::InitResult init() final
Definition: SimpleSpecializedSkill.h:62
armarx::navigation::safety_guard::LaserBasedProximityParams
Definition: LaserBasedProximity.h:57
SPFA.h
Navigator.h
armarx::skills::SimpleSpecializedSkill< arondto::NavigateToLocationParams >::ParamType
arondto::NavigateToLocationParams ParamType
Definition: SimpleSpecializedSkill.h:14
SkillStatusUpdate.h
armarx::navigation::traj_ctrl::local::TrajectoryFollowingControllerParams
Definition: TrajectoryFollowingController.h:40
armarx::skills::SkillID::skillName
std::string skillName
Definition: SkillID.h:41
armarx::skills::SkillDescription
Definition: SkillDescription.h:17
ARMARX_CHECK_NOT_NULL
#define ARMARX_CHECK_NOT_NULL(ptr)
This macro evaluates whether ptr is not null and if it turns out to be false it will throw an Express...
Definition: ExpressionException.h:206
trace.h
LaserBasedProximity.h
Duration.h
StringHelpers.h
ARMARX_CHECK
#define ARMARX_CHECK(expression)
Shortcut for ARMARX_CHECK_EXPRESSION.
Definition: ExpressionException.h:82
armarx::navigation::skills::NavigateToLocation::NavigateToLocation
NavigateToLocation(const Properties &properties, const Services &srv)
Definition: NavigateToLocation.cpp:83
armarx::navigation::skills::NavigateToLocation::Properties::safetyGuardParamsMutex
std::experimental::observer_ptr< std::mutex > safetyGuardParamsMutex
Definition: NavigateToLocation.h:56
armarx::core::time::Duration::Hours
static Duration Hours(std::int64_t hours)
Constructs a duration in hours.
Definition: Duration.cpp:120
NavigateToLocation.h
ARMARX_TRACE
#define ARMARX_TRACE
Definition: trace.h:77
armarx::status
status
Definition: FiniteStateMachine.h:244
NavigationStackConfig.h
Skill.h
armarx::navigation::skills::NavigateToLocation::DefaultSkillDescription
static armarx::skills::SkillDescription DefaultSkillDescription()
Definition: NavigateToLocation.cpp:60
IceNavigator.h
armarx::navigation::skills::NavigateToLocation::Properties::safetyGuardParams
std::experimental::observer_ptr< safety_guard::LaserBasedProximityParams > safetyGuardParams
Definition: NavigateToLocation.h:58
armarx::skills::Skill::description
SkillDescription description
Definition: Skill.h:300
TrajectoryFollowingController.h
armarx::skills::SkillDescription::skillId
SkillID skillId
Definition: SkillDescription.h:19
armarx::skills::SimpleSpecializedSkill
Definition: SimpleSpecializedSkill.h:10
armarx::navigation::skills::skill_ids::NavigateToLocation
const armarx::skills::SkillID NavigateToLocation
Definition: skill_ids.cpp:61
armarx::skills::Skill::MainResult
A result struct for th main method of a skill.
Definition: Skill.h:39
armarx::skills::SkillDescription::rootProfileDefaults
aron::data::DictPtr rootProfileDefaults
Definition: SkillDescription.h:21
ExpressionException.h
ARMARX_INFO
#define ARMARX_INFO
Definition: Logging.h:181
armarx::skills::SimpleSpecializedSkill< arondto::NavigateToLocationParams >::main
Skill::MainResult main() final
Definition: SimpleSpecializedSkill.h:71
skill_ids.h
VAROUT
#define VAROUT(x)
Definition: StringHelpers.h:198
armarx::navigation::skills::NavigateToLocation::Services
Definition: NavigateToLocation.h:47
armarx::skills::Skill::shouldSkillTerminate
bool shouldSkillTerminate() const
Returns whether the skill should terminate as soon as possible.
Definition: Skill.cpp:385
armarx::skills::Skill::InitResult
A result struct for skill initialization.
Definition: Skill.h:27
armarx::navigation::global_planning::SPFAParams::linearVelocity
float linearVelocity
Definition: SPFA.h:43
armarx::navigation::skills::is_ready
bool is_ready(std::future< R > const &f)
Definition: NavigateRelativeSkill.h:29
Logging.h
armarx::navigation::skills
Definition: constants.cpp:25
armarx::navigation::global_planning::SPFAParams
Parameters for AStar.
Definition: SPFA.h:41
armarx::navigation::skills::NavigateToLocation::Properties
Definition: NavigateToLocation.h:54
SkillDescription.h