Component.cpp
Go to the documentation of this file.
1 /*
2  * This file is part of ArmarX.
3  *
4  * ArmarX is free software; you can redistribute it and/or modify
5  * it under the terms of the GNU General Public License version 2 as
6  * published by the Free Software Foundation.
7  *
8  * ArmarX is distributed in the hope that it will be useful, but
9  * WITHOUT ANY WARRANTY; without even the implied warranty of
10  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11  * GNU General Public License for more details.
12  *
13  * You should have received a copy of the GNU General Public License
14  * along with this program. If not, see <http://www.gnu.org/licenses/>.
15  *
16  * @package Navigation::ArmarXObjects::ControlMemory
17  * @author Fabian Reister ( fabian dot reister at kit dot edu )
18  * @date 2021
19  * @copyright http://www.gnu.org/licenses/gpl-2.0.txt
20  * GNU General Public License
21  */
22 
23 #include "Component.h"
24 
25 // STD / STL
26 #include <filesystem>
27 #include <fstream>
28 #include <iostream>
29 
34 
41 
44 // #include <armarx/navigation/algorithms/Costmap.h>
45 // #include <armarx/navigation/memory/constants.h>
46 // #include <armarx/navigation/algorithms/aron/Costmap.aron.generated.h>
47 // #include <armarx/navigation/core/Graph.h>
48 // #include <armarx/navigation/core/aron/Graph.aron.generated.h>
49 // #include <armarx/navigation/core/aron/Location.aron.generated.h>
50 // #include <armarx/navigation/core/aron/Trajectory.aron.generated.h>
51 // #include <armarx/navigation/core/aron/Twist.aron.generated.h>
52 // #include <armarx/navigation/graph/constants.h>
53 // #include <armarx/navigation/location/constants.h>
54 
55 #include <armarx/control/common/control_law/aron/TaskspaceImpedanceControllerConfig.aron.generated.h>
56 
57 // #include <armarx/control/njoint_qp_controller/aron_conversions.h>
58 
59 // #include <simox/control/method/examples/json_conversions.h>
60 
62 {
63 
64 
67  {
70 
71  // Publish to a topic (passing the TopicListenerPrx).
72  // def->topic(myTopicListener);
73 
74  // Subscribe to a topic (passing the topic name).
75  // def->topic<PlatformUnitListener>("MyTopic");
76 
77  // Use (and depend on) another component (passing the ComponentInterfacePrx).
78  // def->component(myComponentProxy)
79 
80 
82  def->optional(properties.snapshotToLoad,
83  "p.snapshotToLoad",
84  "Memory snapshot to load at start up \n"
85  "(e.g. 'PriorKnowledgeData/navigation-graphs/snapshot').");
86 
87  def->optional(properties.locationGraph.visuFrequency,
88  "p.locationGraph.visuFrequency",
89  "Visualization frequeny of locations and graph edges [Hz].");
90 
91  return def;
92  }
93 
94  void
96  {
97  // Topics and properties defined above are automagically registered.
98 
99  // Keep debug observer data until calling `sendDebugObserverBatch()`.
100  // (Requies the armarx::DebugObserverComponentPluginUser.)
101  // setDebugObserverBatchModeEnabled(true);
102 
105 
108 
109 
110  /*
111  if (not properties.snapshotToLoad.empty())
112  {
113  std::filesystem::path snapshotToLoadPath(
114  ArmarXDataPath::resolvePath(properties.snapshotToLoad));
115  if (std::filesystem::is_directory(snapshotToLoadPath))
116  {
117  // This section loads the snapshot specified by the scenario parameters
118  // resolve the paths for the locations and graphs
119  const std::filesystem::path graph = snapshotToLoadPath / "Graph";
120  const std::filesystem::path location = snapshotToLoadPath / "Location";
121 
122  // remove date from folder name (if present)
123  // Sometimes, we use the date before the snapshotname and use a symlink to the real data (e.g. R003 and 2022-03-01_R003)
124  auto splitted = simox::alg::split(snapshotToLoadPath.filename().string(), "_");
125  ARMARX_CHECK_GREATER(splitted.size(), 0);
126  const std::string providerName = splitted[splitted.size() - 1];
127 
128  // This if statement loads the location. Each location is a single file (without extension). The filename is the name of the location.
129  // The file contains json with the globalRobotPose (4x4 matrix) and relativeToObject information
130  if (std::filesystem::is_directory(location))
131  {
132  armem::Commit c;
133  armem::MemoryID providerID = workingMemory().id();
134  providerID.coreSegmentName = "Location";
135  providerID.providerSegmentName = providerName;
136  for (const auto& subdir : std::filesystem::directory_iterator(
137  location)) // iterate over all files in folder (the locations)
138  {
139  const std::filesystem::path location = subdir.path();
140  if (std::filesystem::is_regular_file(
141  location)) // check if its a file (otherwise skip)
142  {
143  std::ifstream ifs(location);
144  const std::string content((std::istreambuf_iterator<char>(ifs)),
145  (std::istreambuf_iterator<char>()));
146 
147  // parse location as json. All files in Location folder must be valid json objects!
148  nlohmann::json j = nlohmann::json::parse(content);
149 
150  if (j.find("globalRobotPose") == j.end())
151  {
152  ARMARX_WARNING
153  << "The file '" << location.string()
154  << "' has no 'globalRobotPose' member. Skipping this file.";
155  continue;
156  }
157 
158  if (j.find("relativeToObject") == j.end())
159  {
160  ARMARX_WARNING
161  << "The file '" << location.string()
162  << "' has no 'relativeToObject' member. Skipping this file.";
163  continue;
164  }
165 
166  std::vector<float> p = j.at("globalRobotPose");
167  ARMARX_CHECK_EQUAL(p.size(), 16);
168 
169  navigation::location::arondto::Location loc;
170  loc.globalRobotPose << p[0], p[1], p[2], p[3], p[4], p[5], p[6], p[7],
171  p[8], p[9], p[10], p[11], p[12], p[13], p[14],
172  p[15]; // load the 4x4 matrix
173 
174  // TODO: All location I have seen were null.
175  // I don't know how this member should look like (von @Fabian Peller to @Fabian Reister)
176  loc.relativeToObject = std::nullopt;
177 
178  // send commit to memory
179  auto& up = c.add();
180  up.confidence = 1.0;
181  up.referencedTime = armem::Time::Now();
182  up.timeSent = armem::Time::Now();
183  up.timeArrived = armem::Time::Now();
184  up.entityID = providerID.withEntityName(location.filename().string());
185  up.instancesData = {loc.toAron()};
186  }
187  }
188 
189  // commit all locations at once
190  iceAdapter().commit(c);
191  }
192 
193  // Next we load the graphs. The graph folder may contain several graphs, represented by different folders.
194  // Each of those graphs contains a list of files representing the vertices. The filename is the vertice id (ideally starting at 0).
195  // The file contains a json with the corresponding location name (as path) and the adjacent vertives (representing the directed outgoing edges).
196  if (std::filesystem::is_directory(graph))
197  {
198  armem::Commit c;
199  armem::MemoryID providerID = workingMemory().id();
200  providerID.coreSegmentName = memory::constants::GraphCoreSegmentName;
201  providerID.providerSegmentName = providerName;
202 
203  for (const auto& graphdir : std::filesystem::directory_iterator(
204  graph)) // iterate over the different graphs (subfolders)
205  {
206  const std::filesystem::path singleGraph = graphdir.path();
207  if (std::filesystem::is_directory(
208  singleGraph)) // assure that its a folder. otherwise skip
209  {
210  navigation::core::arondto::Graph g;
211 
212  for (const auto& subdir : std::filesystem::directory_iterator(
213  singleGraph)) // iterate over all files in the graph
214  {
215  const std::filesystem::path vertice = subdir.path();
216  if (std::filesystem::is_regular_file(
217  vertice)) // assure its a file. otherwise skip
218  {
219  std::ifstream ifs(vertice);
220  const std::string content((std::istreambuf_iterator<char>(ifs)),
221  (std::istreambuf_iterator<char>()));
222 
223  // parse vertice. Each vertice must be a valid json object
224  nlohmann::json j = nlohmann::json::parse(content);
225  if (j.find("location") == j.end())
226  {
227  ARMARX_WARNING
228  << "The file '" << vertice.string()
229  << "' has no 'location' member. Skipping this file.";
230  continue;
231  }
232 
233  if (j.find("outgoingEdges") == j.end())
234  {
235  ARMARX_WARNING << "The file '" << vertice.string()
236  << "' has no 'outgoingEdges' member. "
237  "Skipping this file.";
238  continue;
239  }
240 
241  std::string location = j.at("location");
242  int id = std::stoi(vertice.filename());
243 
244  auto split = simox::alg::split(location, "/");
245  ARMARX_CHECK_EQUAL(
246  split.size(),
247  4); // the location is always a path like Navigation/Location/XXX/YYY
248 
249  armarx::control::core::arondto::Vertex v;
250  v.vertexID = id;
251  v.locationID.memoryName = split[0];
252  v.locationID.coreSegmentName = split[1];
253  v.locationID.providerSegmentName = split[2];
254  v.locationID.entityName = split[3];
255  toAron(v.locationID.timestamp, armem::Time::Invalid());
256  v.locationID.instanceIndex = 0;
257 
258  g.vertices.push_back(v);
259 
260  // add edges of this vertice to graph
261  std::vector<float> edges = j.at("outgoingEdges");
262 
263  for (const auto edge_id : edges)
264  {
265  armarx::control::core::arondto::Edge e;
266  e.sourceVertexID = id;
267  e.targetVertexID = edge_id;
268  g.edges.push_back(e);
269  }
270  }
271  }
272 
273  auto& up = c.add();
274  up.confidence = 1.0;
275  up.referencedTime = armem::Time::Now();
276  up.timeSent = armem::Time::Now();
277  up.timeArrived = armem::Time::Now();
278  up.entityID =
279  providerID.withEntityName(singleGraph.filename().string());
280  up.instancesData = {g.toAron()};
281  }
282  }
283 
284  // send graph to memory
285  iceAdapter().commit(c);
286  }
287 
288  // LEGACY CODE (Not working anymore since the wm json export changed due to ltm updates and aron updates)
289  // armem::wm::Memory memory = armem::server::ltm::disk::load(path.value());
290  //armem::server::ltm::disk::Memory ltm(path.value());
291  //armem::wm::Memory memory;
292  //ltm.loadAll(memory);
293  //workingMemory().update(armem::toCommit(memory));
294  ARMARX_INFO << "Loaded ControlMemory '" << properties.snapshotToLoad << "'";
295  }
296  else
297  {
298  ARMARX_WARNING << "Could not load ControlMemory '" << properties.snapshotToLoad
299  << "'. Continue with empty memory.";
300  }
301  }
302  */
303 
304  loadDefaultConfigs();
305  }
306 
307  void
308  Component::loadDefaultConfigs()
309  {
310  // std::string filename;
311  // ArmarXDataPath::getAbsolutePath("armarx_control/controller_config/default/"
312  // "njoint_controller/impedance_controller_config.json",
313  // filename);
314 
315  // std::ifstream ifs{filename};
316 
317  // nlohmann::json defaultConfig;
318  // ifs >> defaultConfig;
319 
320  // {
321 
322  // common::control_law::arondto::TaskSpaceImpedanceControllerConfig cfg;
323  // // TODO load
324 
325  // armem::Commit c;
326  // armem::MemoryID providerID = workingMemory().id();
327  // providerID.coreSegmentName = memory::constants::DefaultParameterizationCoreSegmentName;
328  // providerID.providerSegmentName =
329  // common::ControllerTypeNames.to_name(common::ControllerType::TSImp);
330 
331 
332  // auto& up = c.add();
333  // up.confidence = 1.0;
334  // up.referencedTime = up.timeSent = up.timeArrived = armem::Time::Now();
335 
336  // up.entityID = providerID.withEntityName("default");
337  // up.instancesData = {cfg.toAron()};
338 
339  // iceAdapter().commit(c);
340  // }
341 
342  // {
343  // njoint_qp_controller::impedance::arondto::WholeBodyImpedanceControllerConfig cfg;
344 
345  // // "kpImpedance": [300, 300, 300, 3, 3, 3],
346  // // "kdImpedance": [ 50, 50, 50, 1, 1, 1],
347  // // "kpNullspace": [25.0, 15.0, 15.0, 5.0, 5.0, 5.0, 5.0, 5.0],
348  // // "kdNullspace": [4.0, 2.0, 2.0, 1.0, 1.0, 1.0, 1.0, 2.0],
349  // // "desiredNullspaceJointAngles": [],
350  // // "torqueLimit": 20,
351  // // "qvelFilter": 0.9
352 
353 
354  // cfg.leftArmImpedanceConfig.kpImpedance << 300, 300, 300, 3, 3, 3;
355  // cfg.leftArmImpedanceConfig.kdImpedance << 50, 50, 50, 1, 1, 1;
356  // cfg.leftArmImpedanceConfig.kpNullspace << 25.0, 15.0, 15.0, 5.0, 5.0, 5.0, 5.0, 5.0;
357  // cfg.leftArmImpedanceConfig.kdNullspace << 4.0, 2.0, 2.0, 1.0, 1.0, 1.0, 1.0, 2.0;
358  // // cfg.leftArmImpedanceConfig.desiredNullspaceJointAngles;
359  // cfg.leftArmImpedanceConfig.torqueLimit = 20;
360  // cfg.leftArmImpedanceConfig.qvelFilter = 0.9;
361 
362  // cfg.rightArmImpedanceConfig = cfg.leftArmImpedanceConfig;
363 
364 
365  // // TODO load
366 
367  // armem::Commit c;
368  // armem::MemoryID providerID = workingMemory().id();
369  // providerID.coreSegmentName = memory::constants::DefaultParameterizationCoreSegmentName;
370  // providerID.providerSegmentName =
371  // common::ControllerTypeNames.to_name(common::ControllerType::QPWholeBodyImpedance);
372 
373 
374  // auto& up = c.add();
375  // up.confidence = 1.0;
376  // up.referencedTime = up.timeSent = up.timeArrived = armem::Time::Now();
377 
378  // up.entityID = providerID.withEntityName("default");
379  // up.instancesData = {cfg.toAron()};
380 
381  // iceAdapter().commit(c);
382  // }
383 
384  // CMakePackageFinder pkg_finder("armarx_control");
385  // const std::filesystem::path pkgDataDir{pkg_finder.getDataDir()};
386 
387  // const std::filesystem::path configBasePath =
388  // pkgDataDir / "armarx_control/controller_config";
389 
390  // for (auto const& dir_entry : std::filesystem::directory_iterator{
391  // configBasePath /
392  // common::ControllerTypeNames.to_name(common::ControllerType::QPWholeBodyVelocity)})
393  // {
394  // if (not dir_entry.is_regular_file())
395  // {
396  // continue;
397  // }
398 
399  // const std::string filename = dir_entry.path();
400 
401  // ARMARX_INFO << "Loading config from file `" << filename << "`";
402  // std::ifstream ifs{filename};
403 
404  // nlohmann::json jsonConfig;
405  // ifs >> jsonConfig;
406 
407  // njoint_qp_controller::velocity::WholeBodyVelocityControllerConfig cfg;
408  // cfg.params.weights = jsonConfig.at("weights");
409  // cfg.params.kinematicConstraints = jsonConfig.at("kinematicConstraints");
410 
411  // njoint_qp_controller::velocity::arondto::WholeBodyVelocityControllerConfig aronCfg;
412  // toAron(aronCfg, cfg);
413 
414 
415  // armem::Commit c;
416  // armem::MemoryID providerID = workingMemory().id();
417  // providerID.coreSegmentName = memory::constants::DefaultParameterizationCoreSegmentName;
418  // providerID.providerSegmentName =
419  // common::ControllerTypeNames.to_name(common::ControllerType::QPWholeBodyVelocity);
420 
421 
422  // auto& up = c.add();
423  // up.confidence = 1.0;
424  // up.referencedTime = up.timeSent = up.timeArrived = armem::Time::Now();
425 
426  // up.entityID = providerID.withEntityName("default");
427  // up.instancesData = {aronCfg.toAron()};
428 
429  // iceAdapter().commit(c);
430  // }
431  }
432 
433  void
435  {
436  // Do things after connecting to topics and components.
437 
438 
439  /* (Requies the armarx::DebugObserverComponentPluginUser.)
440  // Use the debug observer to log data over time.
441  // The data can be viewed in the ObserverView and the LivePlotter.
442  // (Before starting any threads, we don't need to lock mutexes.)
443  {
444  setDebugObserverDatafield("numBoxes", properties.numBoxes);
445  setDebugObserverDatafield("boxLayerName", properties.boxLayerName);
446  sendDebugObserverBatch();
447  }
448  */
449 
450  // Setup the remote GUI.
451  {
454  }
455 
456  tasks.visuTask = new SimpleRunningTask<>([this]() { this->visuRun(); }, "Visualization");
457  tasks.visuTask->start();
458  }
459 
460  void
462  {
463  }
464 
465  void
467  {
468  }
469 
470  std::string
472  {
473  return GetDefaultName();
474  }
475 
476  std::string
478  {
479  return "ControlMemory";
480  }
481 
482  void
484  {
485  using namespace armarx::RemoteGui::Client;
486 
487  // Setup the widgets.
488  tab.locationGraph.setup(*this);
489 
490 
491  // Setup the layout.
492  VBoxLayout root = {tab.locationGraph.group, VSpacer()};
493  RemoteGui_createTab(getName(), root, &tab);
494  }
495 
496  void
498  {
499  tab.locationGraph.update(*this);
500  }
501 
502  void
504  {
505  using namespace armarx::RemoteGui::Client;
506  GridLayout grid;
507  int row = 0;
508  {
509  visuLocations.setValue(owner.properties.locationGraph.visuLocations);
510  visuGraphEdges.setValue(owner.properties.locationGraph.visuLocations);
511 
512  grid.add(Label("Visualize Locations"), {row, 0}).add(visuLocations, {row, 1});
513  ++row;
514 
515  grid.add(Label("Visualize Graph Edges"), {row, 0}).add(visuGraphEdges, {row, 1});
516  ++row;
517  }
518 
519  group = GroupBox({grid});
520  }
521 
522  void
524  {
525  if (visuLocations.hasValueChanged() or visuGraphEdges.hasValueChanged())
526  {
527  std::scoped_lock lock(owner.propertiesMutex);
528  owner.properties.locationGraph.visuLocations = visuLocations.getValue();
529  owner.properties.locationGraph.visuGraphEdges = visuGraphEdges.getValue();
530  }
531  }
532 
533  void
534  Component::visuRun()
535  {
536  // memory::Visu visu{arviz,
537  // workingMemory().getCoreSegment(navigation::location::coreSegmentID),
538  // workingMemory().getCoreSegment(navigation::graph::coreSegmentID)};
539 
540  // Properties::LocationGraph p;
541  // {
542  // std::scoped_lock lock(propertiesMutex);
543  // p = properties.locationGraph;
544  // }
545 
546  // CycleUtil cycle(static_cast<int>(1000 / p.visuFrequency));
547  // while (tasks.visuTask and not tasks.visuTask->isStopped())
548  // {
549  // {
550  // std::scoped_lock lock(propertiesMutex);
551  // p = properties.locationGraph;
552  // }
553 
554  // std::vector<viz::Layer> layers;
555 
556  // // Locations
557  // visu.drawLocations(layers, p.visuLocations);
558 
559  // // Graph Edges
560  // visu.drawGraphs(layers, p.visuGraphEdges);
561 
562  // arviz.commit(layers);
563 
564  // cycle.waitForCycleDuration();
565  // }
566  }
567 
569 
570 
571 } // namespace armarx::control::components::control_memory
armarx::SimpleRunningTask
Usage:
Definition: TaskUtil.h:85
armarx::control::memory::constants::DefaultParameterizationCoreSegmentName
const std::string DefaultParameterizationCoreSegmentName
Definition: constants.h:29
query.h
armarx::RemoteGui::Client::VBoxLayout
Definition: Widgets.h:167
armarx::control::components::control_memory::Component::createPropertyDefinitions
armarx::PropertyDefinitionsPtr createPropertyDefinitions() override
Definition: Component.cpp:66
armarx::RemoteGui::Client::GridLayout::add
GridLayout & add(Widget const &child, Pos pos, Span span=Span{1, 1})
Definition: Widgets.cpp:438
armarx::control::components::control_memory::Component::onExitComponent
void onExitComponent() override
Definition: Component.cpp:466
armarx::control::components::control_memory::Component::onInitComponent
void onInitComponent() override
Definition: Component.cpp:95
armarx::armem::base::MemoryBase::name
std::string & name()
Definition: MemoryBase.h:92
armarx::control::memory::constants::ParameterizationCoreSegmentName
const std::string ParameterizationCoreSegmentName
Definition: constants.h:30
armarx::control::components::control_memory
Definition: Component.cpp:61
armarx::control::components::control_memory::Component::RemoteGui_update
void RemoteGui_update() override
Definition: Component.cpp:497
armarx::armem::server::plugins::ReadWritePluginUser::workingMemory
server::wm::Memory & workingMemory()
Definition: ReadWritePluginUser.cpp:106
armarx::RemoteGui::Client::VSpacer
Definition: Widgets.h:204
type.h
armarx::control::components::control_memory::ARMARX_REGISTER_COMPONENT_EXECUTABLE
ARMARX_REGISTER_COMPONENT_EXECUTABLE(Component, Component::GetDefaultName())
json_conversions.h
armarx::control::components::control_memory::Component::onDisconnectComponent
void onDisconnectComponent() override
Definition: Component.cpp:461
memory_definitions.h
visionx::voxelgrid::Label
uint32_t Label
Type of an object label.
Definition: types.h:6
armarx::control::memory::constants::MemoryName
const std::string MemoryName
Definition: constants.h:27
armarx::RemoteGui::Client::GridLayout
Definition: Widgets.h:186
armarx::control::components::control_memory::Component::RemoteGuiTab::LocationGraph::update
void update(Component &owner)
Definition: Component.cpp:523
Component.h
armarx::RemoteGui::Client::GroupBox
Definition: Widgets.h:193
armarx::control::components::control_memory::Component::onConnectComponent
void onConnectComponent() override
Definition: Component.cpp:434
armarx::control::components::control_memory::Component
Definition: Component.h:51
armarx::armem::base::MemoryBase::addCoreSegment
CoreSegmentT & addCoreSegment(const std::string &name, aron::type::ObjectPtr coreSegmentType=nullptr, const std::vector< PredictionEngine > &predictionEngines={})
Add an empty core segment with the given name, type and prediction engines.
Definition: MemoryBase.h:259
operations.h
armarx::control::components::control_memory::Component::getDefaultName
std::string getDefaultName() const override
Definition: Component.cpp:471
aron_conversions.h
armarx::LightweightRemoteGuiComponentPluginUser::RemoteGui_startRunningTask
void RemoteGui_startRunningTask()
Definition: LightweightRemoteGuiComponentPlugin.cpp:119
CycleUtil.h
Memory.h
armarx::Component::getConfigIdentifier
std::string getConfigIdentifier()
Retrieve config identifier for this component as set in constructor.
Definition: Component.cpp:79
Decoupled.h
constants.h
armarx::ComponentPropertyDefinitions
Default component property definition container.
Definition: Component.h:69
armarx::control::components::control_memory::Component::RemoteGuiTab::LocationGraph::setup
void setup(Component &owner)
Definition: Component.cpp:503
armarx::LightweightRemoteGuiComponentPluginUser::RemoteGui_createTab
void RemoteGui_createTab(std::string const &name, RemoteGui::Client::Widget const &rootWidget, RemoteGui::Client::Tab *tab)
Definition: LightweightRemoteGuiComponentPlugin.cpp:103
IceUtil::Handle< class PropertyDefinitionContainer >
armarx::control::components::control_memory::Component::GetDefaultName
static std::string GetDefaultName()
Definition: Component.cpp:477
armarx::ManagedIceObject::getName
std::string getName() const
Retrieve name of object.
Definition: ManagedIceObject.cpp:108
armarx::RemoteGui::Client
Definition: EigenWidgets.cpp:8
ArmarXDataPath.h
armarx::control::components::control_memory::Component::createRemoteGuiTab
void createRemoteGuiTab()
Definition: Component.cpp:483
PackagePath.h