Graph.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  * @author Rainer Kartmann ( rainer dot kartmann at kit dot edu )
17  * @date 2021
18  * @copyright http://www.gnu.org/licenses/gpl-2.0.txt
19  * GNU General Public License
20  */
21 
22 #include "Graph.h"
23 
24 #include <algorithm>
25 #include <set>
26 #include <sstream>
27 #include <string>
28 #include <vector>
29 
31 
32 #include <RobotAPI/libraries/armem_locations/aron/Location.aron.generated.h>
36 
37 #include <armarx/navigation/core/aron/Graph.aron.generated.h>
39 
40 #include <SemanticObjectRelations/RelationGraph/RelationGraph.h>
41 #include <SemanticObjectRelations/Shapes/Shape.h>
42 
44 {
45 
46  std::string
48  {
49  if (not hasName())
50  {
51  return "anonymous";
52  }
53 
54  return aron.locationID.providerSegmentName + "/" + aron.locationID.entityName;
55  }
56 
57  bool
59  {
60  return not(aron.locationID.providerSegmentName.empty() and
61  aron.locationID.entityName.empty());
62  }
63 
64  bool
66  {
67  return _pose.has_value();
68  }
69 
72  {
73  ARMARX_CHECK(hasPose()) << "Pose for vertex `" << getName() << "` not set.";
74  return _pose.value();
75  }
76 
77  void
79  {
80  this->_pose = pose;
81  }
82 
83  Pose
85  {
86  ARMARX_CHECK(hasPose()) << "Pose for vertex '" << getName() << "' not set.";
87  ARMARX_CHECK_EMPTY(_pose->agent)
88  << "Pose for vertex '" << getName() << "' must be in the global frame. Agent is '"
89  << _pose->agent << "'.";
91  << "Pose for vertex '" << getName() << "' must be in the global frame. Frame is '"
92  << _pose->frame << "'.";
93  return Pose(_pose->toEigen());
94  }
95 
96  std::string
97  Graph::strEdge(ConstEdge edge) const
98  {
99  std::stringstream ss;
100  ss << semrel::RelationGraph<VertexAttribs, EdgeAttribs, GraphAttribs>::strEdge(edge);
101 
102  ss << " cost " << edge.attrib().cost()
103  << ", type: " << static_cast<int>(edge.attrib().strategy) << " , has traj "
104  << edge.attrib().trajectory.has_value();
105  // << ", aron: " << nlohmann::json(edge.attrib().aron);
106 
107  return ss.str();
108  }
109 } // namespace armarx::navigation::core
110 
111 namespace armarx::navigation
112 {
113 
114 
115  namespace detail
116  {
117 
118  std::vector<core::GraphPath>
119  findPathsTo(core::Graph::ConstVertex vertex,
120  const core::Graph& graph,
121  std::set<core::Graph::VertexDescriptor> visited)
122  {
123  using namespace core;
124 
125  // Stop recursion on loops.
126  if (visited.count(vertex.descriptor()) > 0)
127  {
128  return {};
129  }
130  // Mark vertex as visited for this branch.
131  visited.insert(vertex.descriptor());
132 
133  std::vector<GraphPath> paths;
134  // In any case, vertex is reachable from itself.
135  paths.push_back({vertex}); // Add [C]
136 
137  /**
138  * If not a leaf, e.g:
139  * A \
140  * C == vertex
141  * B /
142  * => [[C], [A, C], [B, C]}
143  */
144  for (const auto edge : vertex.inEdges())
145  {
146  std::vector<GraphPath> pathsToSource = findPathsTo(edge.source(), graph, visited);
147  for (auto path : pathsToSource)
148  {
149  // As we mark vertices as visited, it must not occur that we already have
150  // vertex in the path.
151  ARMARX_CHECK(std::find(path.begin(), path.end(), vertex) == path.end());
152 
153  // e.g., path = [A] => expand to [A, C]
154  path.push_back(vertex);
155  // Add to result paths
156  paths.push_back(path);
157  }
158  }
159 
160  return paths;
161  }
162  } // namespace detail
163 
164  std::vector<core::GraphPath>
165  core::findPathsTo(Graph::ConstVertex vertex, const Graph& graph)
166  {
167  std::set<Graph::VertexDescriptor> visited; // empty
168  return detail::findPathsTo(vertex, graph, visited);
169  }
170 
171  void
173  {
174  dto = {};
175  for (auto vertex : bo.vertices())
176  {
177  auto& v = dto.vertices.emplace_back(vertex.attrib().aron);
178  v.vertexID = static_cast<long>(vertex.objectID());
179  }
180  ARMARX_CHECK_EQUAL(dto.vertices.size(), bo.numVertices());
181 
182  for (auto edge : bo.edges())
183  {
184  auto& e = dto.edges.emplace_back(edge.attrib().aron);
185  e.sourceVertexID = static_cast<long>(edge.sourceObjectID());
186  e.targetVertexID = static_cast<long>(edge.targetObjectID());
187  }
188  ARMARX_CHECK_EQUAL(dto.edges.size(), bo.numEdges());
189  }
190 
191  core::Graph::ConstVertex
192  core::getVertexByName(const std::string& vertexName, const core::Graph& graph)
193  {
194  auto vertices = graph.vertices(); // begin() and end() are not const ...
195  const auto vertexIt =
196  std::find_if(vertices.begin(),
197  vertices.end(),
198  [&vertexName](const core::Graph::ConstVertex& vertex) -> bool
199  { return vertex.attrib().getName() == vertexName; });
200 
201  ARMARX_CHECK(vertexIt != vertices.end())
202  << "No vertex found with id `" << vertexName << "`";
203  return *vertexIt;
204  }
205 
206  bool
207  core::hasVertex(const std::string& vertexName, const core::Graph& graph)
208  {
209  auto vertices = graph.vertices(); // begin() and end() are not const ...
210  return std::any_of(vertices.begin(),
211  vertices.end(),
212  [&vertexName](const core::Graph::ConstVertex& vertex) -> bool
213  { return vertex.attrib().getName() == vertexName; });
214  }
215 
216  const core::Graph&
217  core::getSubgraph(const std::string& vertexName, const core::Graphs& graphs)
218  {
219  auto graphIt = std::find_if(graphs.begin(),
220  graphs.end(),
221  [&vertexName](const core::Graph& graph) -> bool
222  { return hasVertex(vertexName, graph); });
223 
224  ARMARX_CHECK(graphIt != graphs.end())
225  << "No subgraph found for vertex `" << vertexName << "`";
226  return *graphIt;
227  }
228 
229  void
231  {
232  bo = {};
233  for (const arondto::Vertex& vertex : dto.vertices)
234  {
235  auto v = bo.addVertex(semrel::ShapeID(vertex.vertexID));
236  v.attrib().aron = vertex;
237  }
238  ARMARX_CHECK_EQUAL(bo.numVertices(), dto.vertices.size());
239 
240  for (const arondto::Edge& edge : dto.edges)
241  {
242  auto e = bo.addEdge(semrel::ShapeID(edge.sourceVertexID),
243  semrel::ShapeID(edge.targetVertexID));
244  e.attrib().aron = edge;
245  }
246  ARMARX_CHECK_EQUAL(bo.numEdges(), dto.edges.size());
247  }
248 
249  void
250  core::resolveLocation(Graph::Vertex& vertex, const aron::data::DictPtr& locationData)
251  {
252  navigation::location::arondto::Location dto;
253  dto.fromAron(locationData);
254  FramedPose pose;
255  fromAron(dto.framedPose, pose);
256 
257  vertex.attrib().setPose(pose);
258  }
259 
260 
261 } // namespace armarx::navigation
armarx::navigation::core::resolveLocation
void resolveLocation(Graph::Vertex &vertex, const aron::data::DictPtr &locationData)
Definition: Graph.cpp:250
armarx::navigation::core::VertexAttribs::requireGlobal
Pose requireGlobal() const
Definition: Graph.cpp:84
armarx::navigation::core::Pose
Eigen::Isometry3f Pose
Definition: basic_types.h:31
framed.h
basic_types.h
armarx::FramedPose
The FramedPose class.
Definition: FramedPose.h:280
armarx::navigation::core::fromAron
void fromAron(const arondto::GlobalTrajectoryPoint &dto, GlobalTrajectoryPoint &bo)
Definition: aron_conversions.cpp:31
ARMARX_CHECK_EMPTY
#define ARMARX_CHECK_EMPTY(c)
Definition: ExpressionException.h:218
armarx::navigation::core::VertexAttribs::getName
std::string getName() const
Definition: Graph.cpp:47
armarx::GlobalFrame
const std::string GlobalFrame
Definition: FramedPose.h:65
armarx::navigation::core::Graphs
std::vector< Graph > Graphs
Definition: Graph.h:105
armarx::navigation::core
This file is part of ArmarX.
Definition: aron_conversions.cpp:19
detail
Definition: OpenCVUtil.cpp:128
armarx::navigation::core::toAron
void toAron(arondto::GlobalTrajectoryPoint &dto, const GlobalTrajectoryPoint &bo)
Definition: aron_conversions.cpp:24
armarx::navigation::core::hasVertex
bool hasVertex(const std::string &vertexName, const Graph &graph)
Definition: Graph.cpp:207
armarx::navigation::core::Graph::strEdge
std::string strEdge(ConstEdge edge) const override
Get a string representation of the given vertex for usage in str().
Definition: Graph.cpp:97
Dict.h
armarx::navigation
This file is part of ArmarX.
Definition: aron_conversions.cpp:25
ARMARX_CHECK
#define ARMARX_CHECK(expression)
Shortcut for ARMARX_CHECK_EXPRESSION.
Definition: ExpressionException.h:82
armarx::navigation::core::VertexAttribs::hasPose
bool hasPose() const
Definition: Graph.cpp:65
FramedPose.h
armarx::navigation::core::Graph
Definition: Graph.h:89
armarx::navigation::core::findPathsTo
std::vector< GraphPath > findPathsTo(Graph::ConstVertex vertex, const Graph &graph)
Definition: Graph.cpp:165
armarx::Graph
boost::subgraph< CloudGraph > Graph
Definition: Common.h:58
armarx::navigation::core::VertexAttribs::getPose
FramedPose getPose() const
Definition: Graph.cpp:71
armarx::navigation::core::VertexAttribs::hasName
bool hasName() const
Definition: Graph.cpp:58
ExpressionException.h
armarx::aron::data::DictPtr
std::shared_ptr< Dict > DictPtr
Definition: Dict.h:41
Graph.h
armarx::ctrlutil::v
double v(double t, double v0, double a0, double j)
Definition: CtrlUtil.h:39
armarx::fromAron
void fromAron(const arondto::PackagePath &dto, PackageFileLocation &bo)
armarx::navigation::detail::findPathsTo
std::vector< core::GraphPath > findPathsTo(core::Graph::ConstVertex vertex, const core::Graph &graph, std::set< core::Graph::VertexDescriptor > visited)
Definition: Graph.cpp:119
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::navigation::core::VertexAttribs::aron
armarx::navigation::core::arondto::Vertex aron
Definition: Graph.h:46
armarx::navigation::core::VertexAttribs::setPose
void setPose(const FramedPose &pose)
Definition: Graph.cpp:78
armarx::aron::bo
const std::optional< BoT > & bo
Definition: aron_conversions.h:174
armarx::navigation::core::getSubgraph
const core::Graph & getSubgraph(const std::string &vertexName, const Graphs &graphs)
Definition: Graph.cpp:217
armarx::navigation::core::getVertexByName
Graph::ConstVertex getVertexByName(const std::string &vertexName, const Graph &graph)
Definition: Graph.cpp:192