Reader.cpp
Go to the documentation of this file.
1 #include "Reader.h"
2 
3 #include <map>
4 #include <ostream>
5 #include <sstream>
6 #include <string>
7 #include <vector>
8 
9 #include <SimoxUtility/algorithm/get_map_keys_values.h>
10 #include <SimoxUtility/algorithm/string/fuzzy_multi_match.hpp>
11 
15 
23 #include <RobotAPI/libraries/armem_locations/aron/Location.aron.generated.h>
26 
28 #include <armarx/navigation/core/aron/Graph.aron.generated.h>
32 
34 {
35  std::string
37  {
38  return "mem.nav.graph.";
39  }
40 
43  {
46  }
47 
48  std::map<std::string, core::Location>
50  {
51  std::map<std::string, core::Location> locations;
52 
53  const auto locs = queryLocations();
54  for (const auto& [id, location] : locs)
55  {
56  const std::string locationName = id.providerSegmentName + "/" + id.entityName;
57  ARMARX_DEBUG << "Location: " << locationName;
58 
60  armarx::fromAron(location.framedPose, fp);
61 
62  core::Location& loc = locations[locationName];
63  loc.name = locationName;
64  loc.framedPose = fp;
65  }
66 
67  return locations;
68  }
69 
70  std::vector<armarx::navigation::core::Graph>
72  {
74 
75  std::map<armem::MemoryID, armarx::navigation::core::Graph> graphs;
76 
77  const auto graphSegment = allGraphs();
78  graphSegment.forEachEntity(
79  [&](const armem::wm::Entity& entity)
80  {
81  navigation::core::Graph& graph = graphs[entity.id()];
82  if (const armem::wm::EntityInstance* instance = entity.findLatestInstance())
83  {
84  navigation::core::arondto::Graph aron;
85  aron.fromAron(instance->data());
86  fromAron(aron, graph);
87  }
88  // else: empty layer
89  });
90 
91 
92  const auto locSegment = allLocations();
93 
94  for (auto& [_, graph] : graphs)
95  {
97  }
98 
99  return simox::get_values(graphs);
100  }
101 
104  {
105  ARMARX_TRACE;
106  // Query all entities from provider.
108 
109  // clang-format off
110  qb
112  .providerSegments().all()
113  .entities().all()
114  .snapshots().latest();
115  // clang-format on
116 
117  const armem::client::QueryResult qResult =
118  memoryReaderLocations.query(qb.buildQueryInput());
119  ARMARX_DEBUG << "Lookup result in reader: " << qResult;
120 
121  if (not qResult.success)
122  {
123  ARMARX_WARNING << "Failure ...";
124  return {};
125  }
126 
127  const auto& locSegment =
128  qResult.memory.getCoreSegment(navigation::location::coreSegmentID.coreSegmentName);
129  return locSegment;
130  }
131 
134  {
135  ARMARX_TRACE;
136 
137  // Query all entities from provider.
139 
140  // clang-format off
141  qb
143  .providerSegments().all()
144  .entities().all()
145  .snapshots().latest();
146  // clang-format on
147 
148  const armem::client::QueryResult qResult = memoryReaderGraphs.query(qb.buildQueryInput());
149  ARMARX_DEBUG << "Lookup result in reader: " << qResult;
150 
151  if (not qResult.success)
152  {
153  ARMARX_WARNING << "Failure ...";
154  return {};
155  }
156 
157  const auto& graphSegment =
158  qResult.memory.getCoreSegment(navigation::graph::coreSegmentID.coreSegmentName);
159  return graphSegment;
160  }
161 
162  std::map<armem::MemoryID, location::arondto::Location>
164  {
165  // Query all entities from provider.
167 
168  // clang-format off
169  qb
171  .providerSegments().all()
172  .entities().all()
173  .snapshots().latest();
174  // clang-format on
175 
176  const armem::client::QueryResult qResult =
177  memoryReaderLocations.query(qb.buildQueryInput());
178  ARMARX_DEBUG << "Lookup result in reader: " << qResult;
179 
180  if (not qResult.success)
181  {
182  return {};
183  }
184 
185  std::map<armem::MemoryID, location::arondto::Location> locations;
186 
187  const auto& locSegment =
188  qResult.memory.getCoreSegment(navigation::location::coreSegmentID.coreSegmentName);
189  locSegment.forEachEntity(
190  [&](const armem::wm::Entity& entity)
191  {
192  if (const armem::wm::EntityInstance* instance = entity.findLatestInstance())
193  {
194  locations[entity.id()].fromAron(instance->data());
195  }
196  });
197 
198  ARMARX_VERBOSE << "Retrieved " << locations.size() << " locations.";
199  return locations;
200  }
201 
203  Reader::resolveLocationId(const std::string& locationId)
204  {
205  const auto locs = locations();
206 
207  ARMARX_CHECK(locs.count(locationId) > 0) << "Location `" << locationId << "` unknown!";
208 
209  return locs.at(locationId);
210  }
211 
213  Reader::resolveLocationName(const std::string& locationName,
214  double minMatchRatio,
215  std::ostream* log)
216  {
217  const std::map<armem::MemoryID, location::arondto::Location> locations = queryLocations();
218  return ResolveLocationName(locations, locationName, minMatchRatio, log);
219  }
220 
223  const std::map<armem::MemoryID, location::arondto::Location>& locations,
224  const std::string& locationName,
225  double minMatchRatio,
226  std::ostream* log)
227  {
228  std::map<armarx::armem::MemoryID, std::vector<std::string>> locationIdsToWords;
229  for (const auto& [id, location] : locations)
230  {
231  if (location.names.has_value())
232  {
233  locationIdsToWords[id] = location.names->recognized;
234  }
235  locationIdsToWords[id].push_back(id.entityName);
236  // TODO: Split entityName by "/" and add components?
237  }
238 
239  ResolveLocationNameResult resolved;
240 
241  if (locationIdsToWords.empty())
242  {
243  std::stringstream msg;
244  msg << "Found no locations in core segment "
245  << armarx::navigation::location::coreSegmentID << " the name '" << locationName
246  << "' could be matched against.";
247 
248  resolved.errorMessage = msg.str();
249  return resolved;
250  }
251 
252  auto match = log ? simox::alg::fuzzy_multi_match(locationName, locationIdsToWords, *log)
253  : simox::alg::fuzzy_multi_match(locationName, locationIdsToWords);
254  ARMARX_CHECK(match.has_value());
255 
256  if (match->ratio < minMatchRatio)
257  {
258  std::stringstream msg;
259  msg << "Match ratio " << match->ratio << " of best match '" << *match->word
260  << "' (location ID " << match->option
261  << ") is below the requested minimum match ratio (" << minMatchRatio << ").";
262 
263  resolved.errorMessage = msg.str();
264  return resolved;
265  }
266 
267  resolved.resolved = ResolvedLocationName{
268  .locationId = *match->option,
269  .locationName = *match->word,
270  .matchRatio = match->ratio,
271  };
272 
273  return resolved;
274  }
275 
276  void
278  {
279  // for the sake of completion ...
280  // armem::client::util::SimpleReaderBase::connect();
281 
282  // Wait for the memory to become available and add it as dependency.
283  ARMARX_INFO << "Reader: Waiting for memory '" << navigation::graph::coreSegmentID.memoryName
284  << "' ...";
285 
286  try
287  {
288  ARMARX_INFO << "graphs ...";
289  memoryReaderGraphs = mns.useReader(
291 
292  ARMARX_INFO << "locations ...";
293  memoryReaderLocations = mns.useReader(
295 
296  ARMARX_INFO << "Reader: Connected to memory '"
298  }
300  {
301  ARMARX_ERROR << e.what();
302  return;
303  }
304  }
305 
306 } // namespace armarx::navigation::memory::client::graph
armarx::armem::client::query::EntitySelector::all
EntitySelector & all() override
Definition: selectors.cpp:104
armarx::navigation::graph::coreSegmentID
const armem::MemoryID coreSegmentID
Definition: constants.h:30
ARMARX_VERBOSE
#define ARMARX_VERBOSE
Definition: Logging.h:187
armarx::armem::client::query::SnapshotSelector::latest
SnapshotSelector & latest()
Definition: selectors.cpp:30
armarx::navigation::core::resolveLocations
void resolveLocations(Graph &graph, const MemoryContainerT &locationContainer)
Definition: Graph.h:141
armarx::armem::client::query::ProviderSegmentSelector::entities
EntitySelector & entities()
Start specifying entities.
Definition: selectors.cpp:135
armarx::navigation::memory::client::graph::Reader::resolveLocationName
ResolveLocationNameResult resolveLocationName(const std::string &locationName, double minMatchRatio, std::ostream *log=nullptr)
Resolve a location by human-readable name.
Definition: Reader.cpp:213
armarx::navigation::memory::client::graph::ResolvedLocationName
Definition: Reader.h:42
armarx::navigation::memory::client::graph::Reader::allLocations
armem::wm::CoreSegment allLocations()
Definition: Reader.cpp:103
framed.h
armarx::armem::wm::EntityInstance
Client-side working entity instance.
Definition: memory_definitions.h:32
armarx::FramedPose
The FramedPose class.
Definition: FramedPose.h:280
armarx::navigation::memory::client::graph::Reader::ResolveLocationName
static ResolveLocationNameResult ResolveLocationName(const std::map< armem::MemoryID, location::arondto::Location > &locations, const std::string &locationName, double minMatchRatio, std::ostream *log=nullptr)
Resolve a location by human-readable name.
Definition: Reader.cpp:222
armarx::armem::client::query::Builder::buildQueryInput
QueryInput buildQueryInput() const
Definition: Builder.cpp:12
armarx::navigation::core::Location::name
std::string name
Definition: location.h:35
armarx::armem::client::query::EntitySelector::snapshots
SnapshotSelector & snapshots()
Start specifying entity snapshots.
Definition: selectors.cpp:92
MemoryID.h
trace.h
mns.h
armarx::navigation::memory::client::graph::Reader::queryLocations
std::map< armem::MemoryID, location::arondto::Location > queryLocations()
Definition: Reader.cpp:163
armarx::armem::client::QueryResult
Result of a QueryInput.
Definition: Query.h:50
Query.h
armarx::navigation::memory::client::graph::ResolveLocationNameResult
Definition: Reader.h:49
armarx::armem::client::util::SimpleReaderBase::Properties::memoryName
std::string memoryName
Definition: SimpleReaderBase.h:45
armarx::armem::MemoryID::coreSegmentName
std::string coreSegmentName
Definition: MemoryID.h:51
ARMARX_CHECK
#define ARMARX_CHECK(expression)
Shortcut for ARMARX_CHECK_EXPRESSION.
Definition: ExpressionException.h:82
constants.h
ARMARX_TRACE
#define ARMARX_TRACE
Definition: trace.h:77
armarx::armem::MemoryID
A memory ID.
Definition: MemoryID.h:47
FramedPose.h
armarx::armem::client::query::Builder::coreSegments
CoreSegmentSelector & coreSegments()
Start specifying core segments.
Definition: Builder.cpp:42
armarx::navigation::memory::client::graph::ResolvedLocationName::locationId
armarx::armem::MemoryID locationId
Definition: Reader.h:44
ARMARX_DEBUG
#define ARMARX_DEBUG
Definition: Logging.h:184
armarx::navigation::memory::client::graph::Reader::propertyPrefix
std::string propertyPrefix() const override
Definition: Reader.cpp:36
armarx::armem::wm::CoreSegment
Client-side working memory core segment.
Definition: memory_definitions.h:119
armarx::navigation::core::Graph
Definition: Graph.h:89
armarx::armem::client::query::CoreSegmentSelector::withName
CoreSegmentSelector & withName(const std::string &name) override
Definition: selectors.cpp:198
armarx::armem::client::MemoryNameSystem::useReader
Reader useReader(const MemoryID &memoryID)
Use a memory server and get a reader for it.
Definition: MemoryNameSystem.cpp:198
armarx::armem::base::EntityBase::findLatestInstance
auto * findLatestInstance(int instanceIndex=0)
Definition: EntityBase.h:355
ARMARX_ERROR
#define ARMARX_ERROR
Definition: Logging.h:196
armarx::navigation::memory::client::graph::ResolveLocationNameResult::resolved
std::optional< ResolvedLocationName > resolved
Definition: Reader.h:51
armarx::armem::base::detail::MemoryItem::id
MemoryID & id()
Definition: MemoryItem.h:25
armarx::navigation::memory::client::graph
This file is part of ArmarX.
Definition: Reader.cpp:33
location.h
memory_definitions.h
armarx::navigation::memory::client::graph::Reader::connect
void connect(armarx::armem::client::MemoryNameSystem &mns) override
Definition: Reader.cpp:277
armarx::armem::MemoryID::memoryName
std::string memoryName
Definition: MemoryID.h:50
ExpressionException.h
armarx::navigation::memory::client::graph::Reader::locations
std::map< std::string, core::Location > locations()
Definition: Reader.cpp:49
armarx::navigation::memory::client::graph::Reader::graphs
std::vector< armarx::navigation::core::Graph > graphs()
Definition: Reader.cpp:71
Graph.h
armarx::armem::client::util::SimpleReaderBase::Properties
Definition: SimpleReaderBase.h:43
ARMARX_INFO
#define ARMARX_INFO
Definition: Logging.h:181
armarx::navigation::core::Location
Definition: location.h:32
armarx::armem::laser_scans::constants::memoryName
const std::string memoryName
Definition: constants.h:28
armarx::navigation::memory::client::graph::Reader::allGraphs
armem::wm::CoreSegment allGraphs()
Definition: Reader.cpp:133
Builder.h
armarx::fromAron
void fromAron(const arondto::PackagePath &dto, PackageFileLocation &bo)
armarx::armem::client::MemoryNameSystem
The memory name system (MNS) client.
Definition: MemoryNameSystem.h:68
armarx::navigation::memory::client::graph::Reader::defaultProperties
Properties defaultProperties() const override
Definition: Reader.cpp:42
armarx::armem::error::CouldNotResolveMemoryServer
Indicates that a query to the Memory Name System failed.
Definition: mns.h:24
armarx::armem::client::query::Builder
The query::Builder class provides a fluent-style specification of hierarchical queries.
Definition: Builder.h:21
armarx::navigation::location::coreSegmentID
const armem::MemoryID coreSegmentID
Definition: constants.h:30
armarx::armem::client::query::CoreSegmentSelector::providerSegments
ProviderSegmentSelector & providerSegments()
Start specifying provider segments.
Definition: selectors.cpp:178
MemoryNameSystem.h
armarx::armem::wm::Entity
Client-side working memory entity.
Definition: memory_definitions.h:93
armarx::navigation::memory::client::graph::Reader::resolveLocationId
core::Location resolveLocationId(const std::string &locationId)
Definition: Reader.cpp:203
armarx::navigation::memory::client::graph::ResolveLocationNameResult::errorMessage
std::optional< std::string > errorMessage
Definition: Reader.h:52
Logging.h
armarx::navigation::core::Location::framedPose
armarx::FramedPose framedPose
Definition: location.h:36
ARMARX_WARNING
#define ARMARX_WARNING
Definition: Logging.h:193
SimpleReaderBase.h
armarx::armem::client::query::ProviderSegmentSelector::all
ProviderSegmentSelector & all() override
Definition: selectors.cpp:147
Reader.h
armarx::armem::client::Reader::query
QueryResult query(const QueryInput &input) const
Perform a query.
Definition: Reader.cpp:32
constants.h