Navigator.cpp
Go to the documentation of this file.
1 #include "Navigator.h"
2 
3 #include <chrono>
4 #include <cstdint>
5 #include <functional>
6 #include <future>
7 #include <mutex>
8 #include <string>
9 #include <vector>
10 
15 
21 
23 {
24 
25  Navigator::Navigator(const InjectedServices& services) : srv{services}
26  {
28  // these checks cannot be used when component is started in property-readout mode
29  ARMARX_CHECK_NOT_NULL(srv.navigator) << "Navigator service must not be null!";
30  ARMARX_CHECK_NOT_NULL(srv.subscriber) << "Subscriber service must not be null!";
31 
32  auto stopped_callback = [this](const auto& e) { stopped(StopEvent(e)); };
33 
34  srv.subscriber->onGoalReached(stopped_callback);
35  srv.subscriber->onSafetyStopTriggered(stopped_callback);
36  srv.subscriber->onUserAbortTriggered(stopped_callback);
37  srv.subscriber->onInternalError(stopped_callback);
38  srv.subscriber->onGlobalPlanningFailed(stopped_callback);
39  }
40 
41  void
43  {
45  moveTo(std::vector<core::Pose>{pose}, frame);
46  }
47 
48  void
49  Navigator::moveTo(const std::vector<core::Pose>& waypoints, core::NavigationFrame frame)
50  {
52  ARMARX_CHECK_NOT_NULL(srv.navigator) << "Navigator service must not be null!";
53  {
54  // TODO: This still leads to a race condition, if extern a stop event is generated before moveTo but arrives
55  // after the event is reset
56  std::scoped_lock const l{stoppedInfo.m};
57  stoppedInfo.event.reset();
58  }
59  srv.navigator->moveTo(waypoints, frame);
60  }
61 
62  void
64  {
66  ARMARX_CHECK_NOT_NULL(srv.navigator) << "Navigator service must not be null!";
67 
68  const std::vector<WaypointTarget>& path = builder.path();
69  validate(path);
70 
71  {
72  // TODO: This still leads to a race condition, if extern a stop event is generated before moveTo but arrives
73  // after the event is reset
74  std::scoped_lock const l{stoppedInfo.m};
75  stoppedInfo.event.reset();
76  }
77  srv.navigator->moveTo(path, frame);
78  }
79 
80  void
81  Navigator::update(const std::vector<core::Pose>& waypoints, core::NavigationFrame frame)
82  {
84  ;
85  ARMARX_CHECK_NOT_NULL(srv.navigator) << "Navigator service must not be null!";
86  srv.navigator->update(waypoints, frame);
87  }
88 
89  void
91  {
93  ARMARX_CHECK_NOT_NULL(srv.navigator) << "Navigator service must not be null!";
94 
95  {
96  // TODO: This still leads to a race condition, if extern a stop event is generated before moveTo but arrives
97  // after the event is reset
98  std::scoped_lock const l{stoppedInfo.m};
99  stoppedInfo.event.reset();
100  }
101  srv.navigator->moveTowards(direction, frame);
102  }
103 
104  void
105  Navigator::moveToLocation(const std::string& location)
106  {
107  ARMARX_TRACE;
108  ARMARX_CHECK_NOT_NULL(srv.navigator) << "Navigator service must not be null!";
109 
110  {
111  // TODO: This still leads to a race condition, if extern a stop event is generated before moveTo but arrives
112  // after the event is reset
113  std::scoped_lock const l{stoppedInfo.m};
114  stoppedInfo.event.reset();
115  }
116  srv.navigator->moveToLocation(location);
117  }
118 
119  void
121  {
122  ARMARX_TRACE;
123  ARMARX_CHECK_NOT_NULL(srv.navigator) << "Navigator service must not be null!";
124  srv.navigator->pause();
125  }
126 
127  void
129  {
130  ARMARX_TRACE;
131  ARMARX_CHECK_NOT_NULL(srv.navigator) << "Navigator service must not be null!";
132  srv.navigator->resume();
133  }
134 
135  void
137  {
138  ARMARX_TRACE;
139  ARMARX_CHECK_NOT_NULL(srv.navigator) << "Navigator service must not be null!";
140  srv.navigator->stop();
141  }
142 
143  void
144  Navigator::onGoalReached(const std::function<void(void)>& callback)
145  {
146  ARMARX_TRACE;
147  onGoalReached([&callback](const core::GoalReachedEvent&) { callback(); });
148  }
149 
150  void
151  Navigator::onGoalReached(const std::function<void(const core::GoalReachedEvent&)>& callback)
152  {
153  }
154 
155  void
156  Navigator::onWaypointReached(const std::function<void(int)>& callback)
157  {
158  }
159 
160  StopEvent
161  Navigator::waitForStop(const std::int64_t timeoutMs)
162  {
163  ARMARX_TRACE;
164 
165  std::future<void> future = std::async(
166  std::launch::async,
167  [&]()
168  {
169  std::unique_lock l{stoppedInfo.m};
170  stoppedInfo.cv.wait(l, [&i = stoppedInfo] { return i.event.has_value(); });
171  });
172 
173 
174  if (timeoutMs > 0)
175  {
176  ARMARX_VERBOSE << "future.wait()";
177  auto status = future.wait_for(std::chrono::milliseconds(timeoutMs));
178  ARMARX_VERBOSE << "done";
179 
180  switch (status)
181  {
182  case std::future_status::ready:
183  ARMARX_INFO << "waitForStop: terminated on goal reached";
184  break;
185  case std::future_status::timeout:
186  ARMARX_INFO << "waitForStop: terminated due to timeout";
187  ARMARX_INFO << "Stopping robot due to timeout";
188  stop();
189 
190  throw LocalException("Navigator::waitForStop: timeout");
191  break;
192  case std::future_status::deferred:
193  ARMARX_INFO << "waitForStop: deferred";
194  break;
195  }
196  }
197  else
198  {
199  ARMARX_VERBOSE << "future.wait()";
200  future.wait();
201  ARMARX_VERBOSE << "done";
202  }
203 
204  // only due to timeout, stoppedInfo.event should be nullopt
205  ARMARX_CHECK(stoppedInfo.event.has_value());
206 
207  StopEvent e = stoppedInfo.event.value();
208  stoppedInfo.event.reset();
209 
210  return e;
211  }
212 
213  void
214  Navigator::stopped(const StopEvent& e)
215  {
216  ARMARX_TRACE;
217  {
218  std::scoped_lock l{stoppedInfo.m};
219  stoppedInfo.event = e;
220  }
221  stoppedInfo.cv.notify_all();
222  }
223 
224 } // namespace armarx::navigation::client
ice_conversions.h
ARMARX_VERBOSE
#define ARMARX_VERBOSE
Definition: Logging.h:187
armarx::navigation::client::Navigator::onGoalReached
void onGoalReached(const std::function< void(void)> &callback)
Definition: Navigator.cpp:144
Navigator.h
armarx::navigation::client::Navigator::waitForStop
StopEvent waitForStop(std::int64_t timeoutMs=-1)
Definition: Navigator.cpp:161
LocalException.h
armarx::navigation::core::NavigationFrame
NavigationFrame
Definition: types.h:42
armarx::navigation::client::Navigator::InjectedServices
Definition: Navigator.h:120
armarx::navigation::core::Pose
Eigen::Isometry3f Pose
Definition: basic_types.h:31
basic_types.h
armarx::navigation::client::PathBuilder
Definition: PathBuilder.h:39
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
armarx::navigation::client
This file is part of ArmarX.
Definition: ComponentPlugin.cpp:21
armarx::navigation::client::Navigator::InjectedServices::navigator
core::NavigatorInterface * navigator
Definition: Navigator.h:122
ARMARX_CHECK
#define ARMARX_CHECK(expression)
Shortcut for ARMARX_CHECK_EXPRESSION.
Definition: ExpressionException.h:82
armarx::navigation::core::NavigatorInterface::moveToLocation
virtual void moveToLocation(const std::string &location)=0
events.h
armarx::navigation::client::Navigator::update
void update(const std::vector< core::Pose > &waypoints, core::NavigationFrame frame)
Definition: Navigator.cpp:81
armarx::navigation::client::Navigator::onWaypointReached
void onWaypointReached(const std::function< void(int)> &callback)
Definition: Navigator.cpp:156
armarx::navigation::client::StopEvent
Definition: Navigator.h:43
ARMARX_TRACE
#define ARMARX_TRACE
Definition: trace.h:77
armarx::status
status
Definition: FiniteStateMachine.h:244
armarx::navigation::client::Navigator::Navigator
Navigator(const InjectedServices &services)
Definition: Navigator.cpp:25
PathBuilder.h
armarx::navigation::client::Navigator::resume
void resume()
Definition: Navigator.cpp:128
armarx::navigation::client::Navigator::moveToLocation
void moveToLocation(const std::string &location)
Definition: Navigator.cpp:105
armarx::navigation::client::Navigator::moveTowards
void moveTowards(const core::Direction &direction, core::NavigationFrame frame)
Definition: Navigator.cpp:90
armarx::navigation::core::NavigatorInterface::moveTo
virtual void moveTo(const std::vector< core::Pose > &waypoints, core::NavigationFrame navigationFrame)=0
armarx::navigation::core::NavigatorInterface::moveTowards
virtual void moveTowards(const core::Direction &direction, core::NavigationFrame navigationFrame)=0
armarx::navigation::core::GoalReachedEvent
Event describing that the targeted goal was successfully reached.
Definition: events.h:52
armarx::navigation::client::Navigator::pause
void pause()
Definition: Navigator.cpp:120
ExpressionException.h
armarx::navigation::client::validate
void validate(const std::vector< WaypointTarget > &path)
Definition: ice_conversions.h:70
armarx::navigation::core::NavigatorInterface::update
virtual void update(const std::vector< core::Pose > &waypoints, core::NavigationFrame navigationFrame)=0
ARMARX_INFO
#define ARMARX_INFO
Definition: Logging.h:181
armarx::navigation::core::Direction
Eigen::Vector3f Direction
Definition: basic_types.h:39
armarx::navigation::client::Navigator::stop
void stop()
Definition: Navigator.cpp:136
armarx::navigation::client::PathBuilder::path
const WaypointTargets & path() const
Definition: PathBuilder.cpp:35
Logging.h
armarx::navigation::core::NavigatorInterface::pause
virtual void pause()=0
armarx::navigation::core::NavigatorInterface::stop
virtual void stop()=0
types.h
armarx::navigation::core::NavigatorInterface::resume
virtual void resume()=0
armarx::navigation::client::Navigator::moveTo
void moveTo(const core::Pose &pose, core::NavigationFrame frame)
Definition: Navigator.cpp:42