MP.cpp
Go to the documentation of this file.
1 #include "MP.h"
2 
3 #include <boost/archive/text_iarchive.hpp>
4 #include <boost/archive/text_oarchive.hpp>
5 #include <boost/archive/xml_iarchive.hpp>
6 #include <boost/archive/xml_oarchive.hpp>
7 
9 
10 #include "../utils.h"
11 #include <mplib/core/SystemState.h>
12 
14 {
15 
16  //MP::MP(const std::string &name, nlohmann::json &userConfig, nlohmann::json &defaultConfig)
17  MP::MP(const MPConfig& c)
18  {
19 
20  cfg = c;
21  mplib::factories::VMPFactory mpfactory;
22  mpfactory.addConfig("kernelSize", c.kernelSize);
23 
24  resetVMPType(c.mpTypeString);
25  // ARMARX_INFO << "Creating mp with name: " << c.name << "\n" << VAROUT(c.className) << "\n" << VAROUT(c.mpTypeString) << "\n" << VAROUT(c.mpMode) << "\n" << VAROUT(c.mpStyle);
26  std::shared_ptr<mplib::representation::AbstractMovementPrimitive> mp =
27  mpfactory.createMP(vmpType);
28  vmp = std::dynamic_pointer_cast<mplib::representation::vmp::PrincipalComponentVMP>(mp);
29 
30  if (cfg.viaPoints.size() > 0)
31  {
32  ARMARX_INFO << "-- adding via points: " << cfg.viaPoints.size();
33  int index = 0;
34  for (const auto& vp : cfg.viaPoints)
35  {
36  ARMARX_INFO << "---- (" << index << ") canonical value: " << vp.canonicalValue
37  << " --- " << common::dVecToString(vp.viaPointValue);
38  ARMARX_CHECK_GREATER_EQUAL(vp.canonicalValue, 0);
39  ARMARX_CHECK_LESS_EQUAL(vp.canonicalValue, 1);
40  userDefinedViaPoints.push_back({vp.canonicalValue, vp.viaPointValue});
41  index++;
42  }
43  }
44  }
45 
46  const std::string&
47  MP::getMPName() const
48  {
49  return cfg.name;
50  }
51 
52  const std::string&
54  {
55  return cfg.className;
56  }
57 
58  const std::string&
60  {
61  return cfg.nodeSetName;
62  }
63 
64  const std::string&
65  MP::getRole() const
66  {
67  return cfg.role;
68  }
69 
70  void
72  {
73  start(std::vector<double>());
74  }
75 
76  void
77  MP::start(const DVec& goals)
78  {
79  start(goals, std::vector<double>());
80  }
81 
82  void
83  MP::start(const DVec& goals, Ice::Double timeDuration)
84  {
85  cfg.durationSec = timeDuration;
86  start(goals);
87  }
88 
89  void
91  {
92  cfg.durationSec = timeDuration;
93  }
94 
95  void
96  MP::start(const DVec& goals, const DVec& starts, Ice::Double timeDuration)
97  {
98  cfg.durationSec = timeDuration;
99  start(goals, starts);
100  }
101 
102  void
103  MP::start(const DVec& goals, const DVec& starts)
104  {
105  ARMARX_INFO << "---- start " << cfg.name;
106  canonicalValue = 1.0;
107  /// parse user defined viapoints from configuration file
108  for (const auto& vp : userDefinedViaPoints)
109  {
110  setViaPoint(vp.first, vp.second);
111  // ARMARX_INFO << "viapoint at " << vp.first << " is " << common::dVecToString(vp.second);
112  }
113 
114  /// overwrite goals and starts if available
115  setStartAndGoal(starts, goals);
116 
117  /// prepare execution
118  LockGuardType guard{mpMutex};
119  vmp->prepareExecution(targetPoseVec, currentState);
120  auto vps = vmp->getViaPoints();
121  ARMARX_INFO << "---- Your MP have: ";
122  int index = 0;
123  for (const auto& v : vps)
124  {
125  ARMARX_INFO << "------ (" << index << ") viapoint at " << v.first << " --- "
127  mplib::core::SystemState::convertStatesToArray(v.second, 0));
128  index++;
129  }
130 
131  /// set flag
132  running.store(true);
133  resume();
134  ARMARX_INFO << "---- MP started: " << cfg.name;
135  }
136 
137  void
139  {
140  running.store(false);
141  }
142 
143  void
145  {
146  paused.store(true);
147  }
148 
149  void
151  {
152  paused.store(false);
153  }
154 
155  void
157  {
158  if (running.load())
159  {
160  ARMARX_INFO << "cannot reset a running mp, please stop it first.";
161  }
162  else
163  {
164  userDefinedViaPoints.clear();
165  targetPoseVec.clear();
166  targetPoseVecInTraj.clear();
167  currentState.clear();
169  goalSetByUser.store(false);
170  mpTrained.store(false);
171  prevViapoint.clear();
172  }
173  }
174 
175  bool
177  {
178  return !running.load();
179  }
180 
181  bool
183  {
184  return running.load();
185  }
186 
187  bool
189  {
190  return firstRun.load();
191  }
192 
193  MPConfig
195  {
196  return cfg;
197  }
198 
199  void
200  MP::learnFromTraj(const MPTrajs& mpTrajs)
201  {
202  ARMARX_INFO << "-- Train MP '" << cfg.name << "' from trajectories";
203  MPTrajs trajList;
204  if (mpTrajs.empty())
205  {
206  ARMARX_INFO << "---- using default trajectories from the MP configuration";
207  trajList = cfg.trajectoryList;
208  }
209  else
210  {
211  ARMARX_INFO << "---- use user-provided traj";
212  trajList = mpTrajs;
213  }
214 
215  std::vector<mplib::core::SampledTrajectory> trajs;
216  for (const auto& mpTraj : trajList)
217  {
218  std::map<double, std::vector<double>> trajMap;
219  for (int i = 0; i < mpTraj.time.size(); ++i)
220  {
221  auto& row_vector = mpTraj.traj.row(i);
222  std::vector<double> vec(row_vector.data(), row_vector.data() + row_vector.size());
223  trajMap.emplace(mpTraj.time(i), vec);
224  }
225  mplib::core::SampledTrajectory traj(trajMap);
226  trajs.push_back(traj);
227  }
228  ARMARX_TRACE;
229  trainMPFromTraj(trajs);
230  }
231 
232  void
233  MP::learnFromCSV(const Ice::StringSeq& fileNames)
234  {
235  ARMARX_INFO << "-- Train MP '" << cfg.name << "' from csv files";
236  std::vector<std::string> fileList;
237  if (fileNames.empty())
238  {
239  ARMARX_INFO << "---- using default trajectory list: " << cfg.fileList[0];
240  fileList = cfg.fileList;
241  }
242  else
243  {
244  fileList = fileNames;
245  }
246 
247  std::vector<mplib::core::SampledTrajectory> trajs;
248  for (auto& file : fileList)
249  {
250  mplib::core::SampledTrajectory traj;
251  traj.readFromCSVFile(file);
252  trajs.push_back(traj);
253  }
254  trainMPFromTraj(trajs);
255  }
256 
257  void
258  MP::trainMPFromTraj(std::vector<mplib::core::SampledTrajectory>& trajs)
259  {
260  if (cfg.regressionModel == "gpr")
261  {
262  vmp->setBaseFunctionApproximator(
263  std::make_unique<mplib::math::function_approximation::GaussianProcessRegression<
264  mplib::math::kernel::SquaredExponentialCovarianceFunction>>(
265  0.01, 0.1, 0.000001));
266  for (auto& traj : trajs)
267  {
268  traj = mplib::core::SampledTrajectory::downSample(traj, 100);
269  }
270  }
271  ARMARX_TRACE;
272  vmp->learnFromTrajectories(trajs);
273  ARMARX_TRACE;
274 
275  currentState.clear();
276  targetPoseVecInTraj.clear();
277 
278  ARMARX_INFO << "---- Your trajectory has " << VAROUT(trajs[0].dim());
279  for (size_t i = 0; i < trajs[0].dim(); i++)
280  {
281  mplib::representation::MPState state(trajs[0].begin()->getPosition(i),
282  trajs[0].begin()->getDeriv(i, 1));
283  currentState.push_back(state);
284  targetPoseVecInTraj.push_back(trajs[0].rbegin()->getPosition(i));
285  }
286  ARMARX_TRACE;
287  mpTrained.store(true);
288  ARMARX_INFO << "---- MP trained";
289  }
290 
291  void
293  {
294  if (not cfg.fileList.empty())
295  {
296  learnFromCSV();
297  }
298  else if (not cfg.trajectoryList.empty())
299  {
300  learnFromTraj();
301  }
302  else
303  {
305  << cfg.name
306  << ": You don't provide any trajectory files (path on the robot) nor trajectories "
307  "in "
308  "type std::vector<armarx::control::common::mp::arondto::MPTraj> in your "
309  "configuration file. \n"
310  << "If you intended train MPs by providing files or "
311  "trajectories on the fly, consider using \n"
312  << "ctrl.learnFromCSV(Ice::StringSeq& fileNames) or \n"
313  << "ctrl.learnFromCSVlearnFromTrajs(armarx::aron::data::dto::DictPtr& dict)";
314  }
315  }
316 
317  void
318  MP::setGoal(const DVec& goals)
319  {
320  ARMARX_INFO << "---- Reset MP goal: " << common::dVecToString(goals);
321  if (!goals.empty())
322  {
323  goalSetByUser.store(true);
324  targetPoseVec = goals;
325  }
326  else if (!targetPoseVec.empty())
327  {
329  << "---- user specified empty goal, try to use goals in your configuration file:\n"
331  }
332  else if (!targetPoseVecInTraj.empty())
333  {
335  << "---- user specified empty goal, try to use goals learned from trajectory:\n"
338  }
339  else
340  {
341  ARMARX_WARNING << "---- user specified empty goal";
342  }
343  }
344 
345  void
346  MP::setStart(const DVec& starts)
347  {
348  ARMARX_INFO << "---- Reset MP start: " << common::dVecToString(starts);
349  if (!starts.empty())
350  {
351  ARMARX_CHECK_EQUAL(starts.size(), currentState.size())
352  << ": size of starts and learned mp are not consistent";
353  for (size_t i = 0; i < currentState.size(); i++)
354  {
355  currentState[i].pos = starts[i];
356  currentState[i].vel = 0.0;
357  }
358  }
359  else
360  {
361  std::vector<double> learnedStart;
362  for (size_t i = 0; i < currentState.size(); i++)
363  {
364  learnedStart.push_back(currentState[i].pos);
365  }
366  ARMARX_INFO << "---- user doesn't define start point, fall back to start point "
367  "learned from trajectory: "
368  << common::dVecToString(learnedStart);
369  }
370  }
371 
372  std::vector<double>
374  {
375  const auto& startState = vmp->getStartState();
376  std::vector<double> mpStartVec;
377  for (const auto& state : startState)
378  {
379  mpStartVec.push_back(state.pos);
380  }
381  return mpStartVec;
382  }
383 
384  bool
386  {
387  return cfg.startFromPrevTarget;
388  }
389 
390  void
391  MP::setStartAndGoal(const DVec& starts, const DVec& goals)
392  {
393  setStart(starts);
394  setGoal(goals);
395  }
396 
397  void
398  MP::setViaPoint(Ice::Double canVal, const DVec& viapoint)
399  {
400  if (mpTrained.load())
401  {
402  prevViapoint = validateViaPoint(viapoint);
403  ARMARX_INFO << "====> Adding via point " << common::dVecToString(prevViapoint);
404  LockGuardType guard{mpMutex};
405  if (canVal <= vmp->getUmin())
406  {
408  }
409  else if (canVal >= 1.0 - vmp->getUmin())
410  {
412  }
413  else
414  {
415  vmp->setViaPoint(canVal, prevViapoint);
416  }
417  }
418  else
419  {
420  ARMARX_ERROR << " -- mp is not trained yet, potential memory allocation error! "
421  "Please train mp before setting any via points";
422  }
423  }
424 
425  void
427  {
428  LockGuardType guard{mpMutex};
429  vmp->removeViaPoints();
430  }
431 
432  double
434  {
435  return canonicalValue;
436  }
437 
438  /// serialze
439  void
440  MP::setWeight(const DVecVec& weights)
441  {
442  vmp->setWeights(weights);
443  }
444 
445  DVecVec
447  {
448  return vmp->getWeights();
449  }
450 
451  std::string
452  MP::serialize(const std::string& mode)
453  {
454  // std::stringstream ss;
455  // if (mode == "string")
456  // {
457  // boost::archive::text_oarchive oa{ss};
458  // oa << vmp.get();
459  // }
460  // else if (mode == "xml")
461  // {
462  // boost::archive::xml_oarchive oa(ss);
463  // oa << vmp.get();
464  // }
465  // return ss.str();
466  return "";
467  }
468 
469  DVec
470  MP::deserialize(const std::string& mpString, const std::string& mode)
471  {
472  // std::stringstream ss;
473  // ss.str(mpString);
474  // mplib::representation::vmp::PrincipalComponentVMP* dmpPointer;
475  // if (mode == "string")
476  // {
477  // boost::archive::text_iarchive ia{ss};
478  // ia >> dmpPointer;
479  // }
480  // else if (mode == "xml")
481  // {
482  // boost::archive::xml_iarchive ia(ss);
483  // ia >> dmpPointer;
484  // }
485  // else
486  // {
487  // ARMARX_ERROR << VAROUT(mode) << " unsupported! please use 'string' or 'xml' mode";
488  // }
489  // vmp.reset(dmpPointer);
490  return vmp->getGoals();
491  }
492 
493 } // namespace armarx::control::common::mp
armarx::control::common::mp::MP::reset
void reset()
Definition: MP.cpp:156
armarx::control::common::mp::MP::getMPName
const std::string & getMPName() const
Definition: MP.cpp:47
index
uint8_t index
Definition: EtherCATFrame.h:59
armarx::control::common::mp::MP::getWeight
DVecVec getWeight()
Definition: MP.cpp:446
armarx::control::common::mp::MP::canonicalValue
double canonicalValue
Definition: MP.h:155
armarx::control::common::mp::MP::getRole
const std::string & getRole() const
Definition: MP.cpp:65
armarx::control::common::mp::DVecVec
DoubleSeqSeq DVecVec
Definition: MP.h:48
armarx::control::common::mp::MP::trainMPFromTraj
void trainMPFromTraj(std::vector< mplib::core::SampledTrajectory > &trajs)
Definition: MP.cpp:258
armarx::control::common::mp::MP::trainMP
void trainMP()
Definition: MP.cpp:292
armarx::control::common::mp::MP::learnFromCSV
void learnFromCSV(const Ice::StringSeq &fileNames=std::vector< std::string >())
setting
Definition: MP.cpp:233
armarx::control::common::mp::MP::setDurationSec
void setDurationSec(Ice::Double timeDuration)
Definition: MP.cpp:90
armarx::control::common::mp::MP::stop
void stop()
Definition: MP.cpp:138
c
constexpr T c
Definition: UnscentedKalmanFilterTest.cpp:46
armarx::control::common::mp::MP::pause
void pause()
Definition: MP.cpp:144
armarx::control::common::mp::MP::deserialize
DVec deserialize(const std::string &, const std::string &mode="string")
Definition: MP.cpp:470
armarx::control::common::mp::MP::targetPoseVecInTraj
DVec targetPoseVecInTraj
Definition: MP.h:163
armarx::control::common::mp
This file is part of ArmarX.
Definition: aron_conversions.cpp:331
armarx::control::common::mp::MP::getConfig
MPConfig getConfig()
Definition: MP.cpp:194
armarx::control::common::mp::MP::mpTrained
std::atomic_bool mpTrained
Note: if mp is not trained, the dimension is not initialized, thus mp->getDim() is undefined,...
Definition: MP.h:161
armarx::control::common::mp::MP::setStart
void setStart(const DVec &starts)
Definition: MP.cpp:346
armarx::control::common::mp::MP::firstRun
std::atomic_bool firstRun
Definition: MP.h:156
armarx::control::common::mp::MP::MP
MP(const MPConfig &c)
Definition: MP.cpp:17
armarx::control::common::mp::MP::cfg
MPConfig cfg
Definition: MP.h:144
armarx::control::common::mp::LockGuardType
std::lock_guard< std::recursive_mutex > LockGuardType
Definition: MP.h:45
MP.h
armarx::control::common::mp::MP::userDefinedViaPoints
std::vector< std::pair< double, DVec > > userDefinedViaPoints
Definition: MP.h:165
armarx::VariantType::Double
const VariantTypeId Double
Definition: Variant.h:920
armarx::control::common::dVecToString
std::string dVecToString(const mplib::core::DVec &dvec)
Definition: utils.cpp:301
ARMARX_TRACE
#define ARMARX_TRACE
Definition: trace.h:77
armarx::control::common::mp::MP::getStartFromPrevTarget
bool getStartFromPrevTarget()
Definition: MP.cpp:385
armarx::control::common::mp::MP::getCanonicalValue
double getCanonicalValue()
Definition: MP.cpp:433
armarx::control::common::mp::MP::serialize
std::string serialize(const std::string &mode="string")
Definition: MP.cpp:452
armarx::control::common::mp::MP::setGoal
void setGoal(const DVec &goals)
Definition: MP.cpp:318
armarx::control::common::mp::DVec
Ice::DoubleSeq DVec
Definition: MP.h:47
armarx::control::common::mp::MP::goalSetByUser
std::atomic_bool goalSetByUser
Definition: MP.h:158
armarx::control::common::mp::MP::paused
std::atomic_bool paused
Definition: MP.h:154
armarx::control::common::mp::MP::vmp
VMPPtr vmp
Definition: MP.h:149
armarx::control::common::mp::MP::currentState
std::vector< mplib::representation::MPState > currentState
Definition: MP.h:164
armarx::control::common::mp::MP::removeAllViaPoint
void removeAllViaPoint()
Definition: MP.cpp:426
armarx::control::common::mp::MP::isRunning
bool isRunning()
Definition: MP.cpp:182
ARMARX_ERROR
#define ARMARX_ERROR
Definition: Logging.h:196
armarx::control::common::mp::MP::learnFromTraj
void learnFromTraj(const MPTrajs &trajs=MPTrajs())
Definition: MP.cpp:200
armarx::control::common::mp::MP::prevViapoint
DVec prevViapoint
Definition: MP.h:111
ARMARX_CHECK_GREATER_EQUAL
#define ARMARX_CHECK_GREATER_EQUAL(lhs, rhs)
This macro evaluates whether lhs is greater or equal (>=) rhs and if it turns out to be false it will...
Definition: ExpressionException.h:123
ARMARX_CHECK_LESS_EQUAL
#define ARMARX_CHECK_LESS_EQUAL(lhs, rhs)
This macro evaluates whether lhs is less or equal (<=) rhs and if it turns out to be false it will th...
Definition: ExpressionException.h:109
armarx::control::common::mp::MP::getNodeSetName
const std::string & getNodeSetName() const
Definition: MP.cpp:59
ExpressionException.h
armarx::ctrlutil::v
double v(double t, double v0, double a0, double j)
Definition: CtrlUtil.h:39
armarx::control::common::mp::MP::getStartVec
std::vector< double > getStartVec()
Definition: MP.cpp:373
ARMARX_INFO
#define ARMARX_INFO
Definition: Logging.h:181
VAROUT
#define VAROUT(x)
Definition: StringHelpers.h:198
armarx::control::common::mp::MP::isFirstRun
bool isFirstRun()
Definition: MP.cpp:188
armarx::control::common::mp::MPTrajs
std::vector< MPTraj > MPTrajs
Definition: MP.h:66
armarx::control::common::mp::MP::validateViaPoint
virtual DVec validateViaPoint(const DVec &viapoint)
Definition: MP.h:114
armarx::control::common::mp::MP::start
void start()
control
Definition: MP.cpp:71
armarx::control::common::mp::MP::vmpType
mplib::representation::VMPType vmpType
Definition: MP.h:150
armarx::control::common::mp::MP::setWeight
void setWeight(const DVecVec &weights)
serialze
Definition: MP.cpp:440
ARMARX_CHECK_EQUAL
#define ARMARX_CHECK_EQUAL(lhs, rhs)
This macro evaluates whether lhs is equal (==) rhs and if it turns out to be false it will throw an E...
Definition: ExpressionException.h:130
armarx::control::common::mp::MP::isFinished
bool isFinished()
status
Definition: MP.cpp:176
ARMARX_WARNING
#define ARMARX_WARNING
Definition: Logging.h:193
armarx::control::common::mp::MP::mpMutex
MutexType mpMutex
Definition: MP.h:148
armarx::control::common::mp::MP::setViaPoint
void setViaPoint(Ice::Double u, const DVec &viapoint)
Definition: MP.cpp:398
armarx::control::common::mp::MP::getClassName
const std::string & getClassName() const
Definition: MP.cpp:53
armarx::control::common::mp::MP::resume
void resume()
Definition: MP.cpp:150
armarx::control::common::mp::MP::targetPoseVec
DVec targetPoseVec
Definition: MP.h:162
armarx::control::common::mp::MP::running
std::atomic_bool running
Definition: MP.h:152
armarx::control::common::mp::MP::setStartAndGoal
void setStartAndGoal(const DVec &starts, const DVec &goals)
Definition: MP.cpp:391