Entity.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 MemoryX::WorkingMemory
17 * @author ALexey Kozlov ( kozlov at kit dot edu)
18 * @date 2012
19 * @copyright http://www.gnu.org/licenses/gpl-2.0.txt
20 * GNU General Public License
21 */
22 
23 #include "Entity.h"
24 
28 
29 #include "ProbabilityMeasures.h"
30 
31 namespace memoryx
32 {
33 
35  IceUtil::Shared(source), ::armarx::Serializable(source), EntityBase()
36  {
37  deepCopy(source);
38  }
39 
40  EntityPtr
42  {
43  return EntityPtr(new Entity());
44  }
45 
46  void
47  Entity::deepCopy(const Entity& source)
48  {
49  {
50  // boost::shared_lock<boost::shared_mutex> lock(source.entityMutex);
51  std::shared_lock lock2(entityMutex);
52  id = source.id;
53  name = source.name;
54  }
55  // ARMARX_INFO_S << "Entity copy const - " << VAROUT(source.wrappers.size());
56  // clone attributes
57  {
58  // armarx::ScopedLock lock(source.attributesMutex);
59  std::scoped_lock lock2(attributesMutex);
60  attributes.clear();
61 
62  for (AttributeMap::const_iterator iter = source.attributes.begin();
63  iter != source.attributes.end();
64  iter++)
65  {
66  EntityAttributeBasePtr attrBase = iter->second;
67  EntityAttributeBasePtr attr =
68  EntityAttributeBasePtr::dynamicCast(attrBase->ice_clone());
69  attributes[iter->first] = attr;
70  }
71  }
72  // clone wrappers
73  {
74  // RecursiveMutex::scoped_lock lock(source.wrappersMutex);
75  std::scoped_lock lock2(wrappersMutex);
76  wrappers.clear();
77 
78  for (AbstractEntityWrapperBaseList::const_iterator iter = source.wrappers.begin();
79  iter != source.wrappers.end();
80  iter++)
81  {
82  if ((*iter))
83  {
85  EntityWrappers::AbstractEntityWrapperPtr::dynamicCast((*iter)->ice_clone());
86 
87  if (wrapper)
88  {
89  wrapper->setEntity(this);
90  wrappers.push_back(wrapper);
91  }
92  }
93  }
94  }
95  }
96 
97  void
99  {
100  int wrapperCount;
101  {
102  std::scoped_lock lock(wrappersMutex);
103  wrapperCount = wrappers.size();
104  }
105  int remainingRefs = __getRef() - 1;
106  if (remainingRefs <= wrapperCount && remainingRefs > 0 && wrapperCount > 0 &&
107 
108 #if ICE_INT_VERSION >= 30603
109  !__hasFlag(NoDelete)
110 #else
111  !_noDelete
112 #endif
113  )
114  {
115  std::scoped_lock lock(wrappersMutex);
116  int counter = 0;
117  for (const AbstractEntityWrapperBasePtr& wrapper : wrappers)
118  {
120  EntityWrappers::AbstractEntityWrapperPtr::dynamicCast(wrapper);
121  if (p)
122  {
123  if (p->getEntity().get() == this)
124  {
125  counter++;
126  }
127  }
128  }
129  if (counter == remainingRefs)
130  {
131  // only the wrappers point on this object
132  // first make a temporal copy of the wrapper list, then delete the wrappers list
133  // so that when the temp wrapper list is deleted and the ref count of the entity decreased, the wrappers.size() check is zero and this part is not run again
134  auto tmpWrappers = wrappers;
135  wrappers.clear();
136  for (const AbstractEntityWrapperBasePtr& wrapper : tmpWrappers)
137  {
139  EntityWrappers::AbstractEntityWrapperPtr::dynamicCast(wrapper);
140  p->setEntity(NULL);
141  }
142  }
143  }
144  Shared::__decRef();
145  }
146 
148  {
149  // ARMARX_INFO_S << "Deleting entity with id " << id;
150  __setNoDelete(true);
151  // ARMARX_INFO_S << "~Entity() " << getName();
152  // Remove entity ptr in wrapper, because it becomes invalid
153  // it is not a shared ptr because it would introduce a cycle of shared ptrs
154  // auto tmpWrapper = wrappers;
155  // wrappers.clear();
156  // for (const AbstractEntityWrapperBasePtr& wrapper : tmpWrapper)
157  // {
158  // auto p = EntityWrappers::AbstractEntityWrapperPtr::dynamicCast(wrapper);
159  // if (p)
160  // {
161  // p->setEntity(NULL);
162  // }
163  // }
164  }
165 
166  ::std::string
167  Entity::getId(const ::Ice::Current&) const
168  {
169  std::shared_lock lock(entityMutex);
170  return id;
171  }
172 
173  void
174  Entity::setId(const ::std::string& id, const ::Ice::Current&)
175  {
176  std::unique_lock lock(entityMutex);
177  this->id = id;
178  }
179 
180  ::std::string
181  Entity::getName(const ::Ice::Current&) const
182  {
183  std::shared_lock lock(entityMutex);
184  return name;
185  }
186 
187  void
188  Entity::setName(const ::std::string& name, const ::Ice::Current&)
189  {
190  std::unique_lock lock(entityMutex);
191  this->name = name;
192  }
193 
194  bool
195  Entity::isMetaEntity(const Ice::Current&) const
196  {
197  if (hasAttribute("isMetaEntity"))
198  {
199  auto attr = getAttribute("isMetaEntity");
200  return attr->getValue()->getBool();
201  }
202 
203  return false;
204  }
205 
206  void
207  Entity::setMetaEntity(bool isMetaEntity, const Ice::Current&)
208  {
209  if (hasAttribute("isMetaEntity"))
210  {
211  removeAttribute("isMetaEntity");
212  }
213 
214  EntityAttributePtr attr = new EntityAttribute("isMetaEntity");
215  attr->addValue(armarx::VariantPtr(new armarx::Variant(isMetaEntity)));
216  putAttribute(attr);
217  }
218 
219  EntityRefBaseList
221  {
222  EntityRefBaseList result;
223 
224  if (!hasAttribute("parentEntityRefs"))
225  {
226  return result;
227  }
228 
229  auto attr = getAttribute("parentEntityRefs");
230 
231  for (int i = 0; i < attr->size(); i++)
232  {
233  result.push_back(
234  armarx::VariantPtr::dynamicCast(attr->getValueAt(i))->get<EntityRefBase>());
235  }
236 
237  return result;
238  }
239 
240  EntityRefBaseList
241  Entity::getAllParentRefs(bool includeMetaEntities) const
242  {
243  EntityRefBaseList result;
244  EntityRefBaseList explored;
245  EntityRefBaseList unexplored = getDirectParentRefs();
246 
247  while (!unexplored.empty())
248  {
249  EntityRefBasePtr entityRef = unexplored.back();
250  EntityPtr next = EntityPtr::dynamicCast(entityRef->getEntity());
251  unexplored.pop_back();
252 
253  bool isExplored = std::find_if(explored.cbegin(),
254  explored.cend(),
255  [&](const EntityRefBasePtr& entityRef) {
256  return entityRef->entityId == next->getId();
257  }) != explored.cend();
258 
259  if (isExplored)
260  {
261  continue;
262  }
263 
264  EntityRefBaseList newParents = next->getDirectParentRefs();
265  explored.push_back(entityRef);
266  unexplored.insert(unexplored.end(), newParents.begin(), newParents.end());
267 
268  if (next->isMetaEntity() && !includeMetaEntities)
269  {
270  continue;
271  }
272 
273  result.push_back(entityRef);
274  }
275 
276  return result;
277  }
278 
279  std::vector<std::string>
281  {
282  Ice::StringSeq result;
283  const auto& parents = getAllParentRefs();
284  std::transform(parents.cbegin(),
285  parents.cend(),
286  std::back_inserter(result),
287  [](const EntityRefBasePtr& e) { return e->entityName; });
288 
289  return result;
290  }
291 
292  void
293  Entity::setDirectParentRefs(const EntityRefBaseList& entityRefs)
294  {
295  if (hasAttribute("parentEntityRefs"))
296  {
297  removeAttribute("parentEntityRefs");
298  }
299 
300  EntityAttributePtr attr = new EntityAttribute("parentEntityRefs");
301 
302  for (const auto& entity : entityRefs)
303  {
304  attr->addValue(new armarx::Variant(entity));
305  }
306 
307  putAttribute(attr);
308  }
309 
310  EntityAttributeBasePtr
311  Entity::getAttribute(const ::std::string& attrName, const ::Ice::Current&) const
312  {
313  std::scoped_lock lock(attributesMutex);
314  AttributeMap::const_iterator it = attributes.find(attrName);
315 
316  if (it != attributes.end())
317  {
318  return it->second;
319  }
320  else
321  {
322  return EntityAttributeBasePtr();
323  }
324  }
325 
327  Entity::getAttributeValue(const ::std::string& attrName) const
328  {
329  EntityAttributeBasePtr attr = getAttribute(attrName);
330 
331  // check if attribute exists
332  if (!attr)
333  {
334  return armarx::VariantPtr();
335  }
336 
337  // check if attr has exactly one value (ie not empty and not multimodal)
338  if (attr->size() != 1)
339  {
340  return armarx::VariantPtr();
341  }
342 
343  return armarx::VariantPtr::dynamicCast(attr->getValue());
344  }
345 
346  void
347  Entity::putAttribute(const ::memoryx::EntityAttributeBasePtr& attr, const ::Ice::Current&)
348  {
349  std::scoped_lock lock(attributesMutex);
350  attributes[attr->getName()] = attr;
351  }
352 
353  void
354  Entity::removeAttribute(const ::std::string& attrName, const ::Ice::Current&)
355  {
356  std::scoped_lock lock(attributesMutex);
357  attributes.erase(attrName);
358  }
359 
360  bool
361  Entity::hasAttribute(const std::string& attrName, const Ice::Current& c) const
362  {
363  ARMARX_TRACE;
364  std::scoped_lock lock(attributesMutex);
365  return attributes.find(attrName) != attributes.end();
366  }
367 
368  memoryx::NameList
369  Entity::getAttributeNames(const Ice::Current&) const
370  {
371  memoryx::NameList result;
372  std::scoped_lock lock(attributesMutex);
373 
374  for (AttributeMap::const_iterator it = attributes.begin(); it != attributes.end(); ++it)
375  {
376  result.push_back(it->first);
377  }
378 
379  return result;
380  }
381 
382  bool
383  Entity::equals(const EntityBasePtr& otherEntity, const Ice::Current&) const
384  {
385  if (getId() != otherEntity->getId())
386  {
387  return false;
388  }
389  return equalsAttributes(otherEntity);
390  }
391 
392  bool
393  Entity::equalsAttributes(const EntityBasePtr& otherEntity, const Ice::Current&) const
394  {
395  if (getName() != otherEntity->getName())
396  {
397  return false;
398  }
399  NameList attributeNames = getAttributeNames();
400  if (otherEntity->getAttributeNames().size() != attributeNames.size())
401  {
402  return false;
403  }
404 
405  for (std::string& attributeName : attributeNames)
406  {
407  // check if attribute values are equal
408  armarx::JSONObjectPtr thisSerializer = new armarx::JSONObject();
409  armarx::JSONObjectPtr otherSerializer = new armarx::JSONObject();
410  getAttribute(attributeName)->serialize(thisSerializer);
411  otherEntity->getAttribute(attributeName)->serialize(otherSerializer);
412  if (otherSerializer->toString() != thisSerializer->toString())
413  {
414  return false;
415  }
416  }
417 
418  return true;
419  }
420 
423  {
424  return this->clone();
425  }
426 
427  EntityPtr
428  Entity::clone(const Ice::Current& c) const
429  {
430  EntityPtr ret = new Entity();
431  ret->deepCopy(*this);
432  return ret;
433  }
434 
435  void
436  Entity::output(std::ostream& stream) const
437  {
438  std::scoped_lock lock(attributesMutex);
439  stream << "[" << id << "] " << name << std::endl;
440  stream << "Attributes: (" << attributes.size() << ")" << std::endl;
441 
442  for (AttributeMap::const_iterator it = attributes.begin(); it != attributes.end(); ++it)
443  {
444  stream << it->first << ": " << EntityAttributePtr::dynamicCast(it->second) << std::endl;
445  }
446  }
447 
448  void
450  {
451  // wrappers are only locally available -> clear them before sending (they would be broken on the other side)
452  // armarx::RecursiveMutex::scoped_lock lock(wrappersMutex);
453  // wrappers.clear();
454  // for(auto& elem : wrappers)
455  // {
456  // elem = NULL;
457  // }
458  // ARMARX_INFO_S << getName() << " size: " << wrappers.size();
459  }
460 
461  void
463  {
464  // ARMARX_INFO_S << getName() << " size: " << wrappers.size();
465  std::scoped_lock lock(wrappersMutex);
466  wrappers.clear();
467  }
468 
469  void
470  Entity::serialize(const armarx::ObjectSerializerBasePtr& serializer,
471  const ::Ice::Current& c) const
472  {
474  armarx::AbstractObjectSerializerPtr::dynamicCast(serializer);
475  {
476  std::shared_lock lock(entityMutex);
477 
478  // empty ids shouldn't be added to allow them to be auto-generated
479  if (!id.empty())
480  {
481  obj->setId(id);
482  }
483 
484  obj->setString("name", name);
485  }
486 
487  armarx::AbstractObjectSerializerPtr objAttrs = obj->createElement();
488  {
489  std::scoped_lock lock(attributesMutex);
490 
491  for (AttributeMap::const_iterator it = attributes.begin(); it != attributes.end(); ++it)
492  {
493  armarx::AbstractObjectSerializerPtr attr = obj->createElement();
494  it->second->serialize(attr, c);
495  objAttrs->setElement(it->first, attr);
496  }
497  }
498  obj->setElement("attrs", objAttrs);
499  }
500 
501  void
502  Entity::deserialize(const armarx::ObjectSerializerBasePtr& serializer, const ::Ice::Current& c)
503  {
505  armarx::AbstractObjectSerializerPtr::dynamicCast(serializer);
506  {
507  std::unique_lock lock(entityMutex);
508 
509  if (!obj->getIdField().empty() && obj->hasElement(obj->getIdField()))
510  {
511  id = obj->getStringId();
512  }
513 
514  name = obj->getString("name");
515  }
516 
517  const armarx::AbstractObjectSerializerPtr objAttrs = obj->getElement("attrs");
518  const std::vector<std::string> attrNames = objAttrs->getElementNames();
519  {
520  // putAttribute() also locks the datastructure!
521  std::scoped_lock lock(attributesMutex);
522  attributes.clear();
523  }
524 
525  for (std::vector<std::string>::const_iterator it = attrNames.begin(); it != attrNames.end();
526  ++it)
527  {
528  EntityAttributePtr attr = new EntityAttribute(*it);
529  attr->deserialize(objAttrs->getElement(*it));
530  putAttribute(attr);
531  }
532  }
533 
534 } // namespace memoryx
memoryx::Entity::ice_preMarshal
void ice_preMarshal() override
Definition: Entity.cpp:449
memoryx::Entity::setMetaEntity
void setMetaEntity(bool isMetaEntity, const ::Ice::Current &=Ice::emptyCurrent) override
Mark this entity as meta (i.e.
Definition: Entity.cpp:207
armarx::Variant
The Variant class is described here: Variants.
Definition: Variant.h:223
armarx::aron::ret
ReaderT::InputType T & ret
Definition: rw.h:13
memoryx::Entity::putAttribute
void putAttribute(const ::memoryx::EntityAttributeBasePtr &attr, const ::Ice::Current &=Ice::emptyCurrent) override
Store attribute in entity.
Definition: Entity.cpp:347
JSONObject.h
memoryx::Entity::getName
::std::string getName(const ::Ice::Current &=Ice::emptyCurrent) const override
Retrieve name of this entity.
Definition: Entity.cpp:181
memoryx::Entity::getId
::std::string getId(const ::Ice::Current &=Ice::emptyCurrent) const override
Retrieve id of this entity which is an integer in string representation.
Definition: Entity.cpp:167
memoryx::Entity::deserialize
void deserialize(const armarx::ObjectSerializerBasePtr &serializer, const ::Ice::Current &=Ice::emptyCurrent) override
Subclasses should use serializable attributes and consider these methods final.
Definition: Entity.cpp:502
memoryx::Entity::getAttribute
EntityAttributeBasePtr getAttribute(const ::std::string &attrName, const ::Ice::Current &=Ice::emptyCurrent) const override
Retrieve attribute from entity.
Definition: Entity.cpp:311
memoryx::Entity::Entity
Entity()
Definition: Entity.h:515
memoryx::Entity::removeAttribute
void removeAttribute(const ::std::string &attrName, const ::Ice::Current &=Ice::emptyCurrent) override
Remove attribute with given name from entity.
Definition: Entity.cpp:354
memoryx::Entity::~Entity
~Entity() override
Definition: Entity.cpp:147
armarx::JSONObject
The JSONObject class is used to represent and (de)serialize JSON objects.
Definition: JSONObject.h:43
memoryx::Entity::clone
EntityPtr clone(const Ice::Current &c=Ice::emptyCurrent) const
Definition: Entity.cpp:428
trace.h
AbstractObjectSerializer.h
memoryx::Entity::serialize
void serialize(const armarx::ObjectSerializerBasePtr &serializer, const ::Ice::Current &=Ice::emptyCurrent) const override
Subclasses should use serializable attributes and consider these methods final.
Definition: Entity.cpp:470
memoryx
VirtualRobot headers.
Definition: CommonPlacesTester.cpp:48
c
constexpr T c
Definition: UnscentedKalmanFilterTest.cpp:46
memoryx::Entity::getAttributeNames
memoryx::NameList getAttributeNames(const ::Ice::Current &=Ice::emptyCurrent) const override
Retrieve list of all attribute names.
Definition: Entity.cpp:369
IceUtil
Definition: Instance.h:21
memoryx::Entity::CreateGenericEntity
static EntityPtr CreateGenericEntity()
Creates an entity without any convenience getter/setter functions.
Definition: Entity.cpp:41
memoryx::Entity::ice_postUnmarshal
void ice_postUnmarshal() override
Definition: Entity.cpp:462
memoryx::Entity::output
void output(std::ostream &stream) const
Definition: Entity.cpp:436
memoryx::Entity::getAttributeValue
virtual armarx::VariantPtr getAttributeValue(const ::std::string &attrName) const
Retrieve value of an attribute from entity.
Definition: Entity.cpp:327
cxxopts::empty
bool empty(const std::string &s)
Definition: cxxopts.hpp:234
IceInternal::Handle
Definition: forward_declarations.h:8
memoryx::Entity::setDirectParentRefs
virtual void setDirectParentRefs(const EntityRefBaseList &entityRefs)
Replace parent entity references.
Definition: Entity.cpp:293
ARMARX_TRACE
#define ARMARX_TRACE
Definition: trace.h:77
armarx::VariantPtr
IceInternal::Handle< Variant > VariantPtr
Definition: Variant.h:41
memoryx::Entity::ice_clone
Ice::ObjectPtr ice_clone() const override
Definition: Entity.cpp:422
memoryx::Entity::getDirectParentRefs
virtual EntityRefBaseList getDirectParentRefs() const
Retrieve parent entity references.
Definition: Entity.cpp:220
memoryx::Entity::attributesMutex
std::mutex attributesMutex
Definition: Entity.h:524
memoryx::Entity::__decRef
void __decRef() override
Definition: Entity.cpp:98
memoryx::Entity::getAllParentsAsStringList
virtual std::vector< std::string > getAllParentsAsStringList() const
Definition: Entity.cpp:280
memoryx::Entity::getAllParentRefs
virtual EntityRefBaseList getAllParentRefs(bool includeMetaEntities=true) const
Retrieve all parents by traversing the whole hierarchy.
Definition: Entity.cpp:241
memoryx::EntityWrappers::AbstractEntityWrapperPtr
IceInternal::Handle< AbstractEntityWrapper > AbstractEntityWrapperPtr
Typedef of AbstractEntityWrapperPtr as IceUtil::Handle<AbstractEntityWrapper> for convenience.
Definition: AbstractEntityWrapper.h:112
boost::source
Vertex source(const detail::edge_base< Directed, Vertex > &e, const PCG &)
Definition: point_cloud_graph.h:661
memoryx::Entity::setId
void setId(const ::std::string &id, const ::Ice::Current &=Ice::emptyCurrent) override
Set id of this entity.
Definition: Entity.cpp:174
memoryx::Entity::wrappers
AbstractEntityWrapperBaseList wrappers
Definition: Entity.h:522
memoryx::EntityPtr
IceInternal::Handle< Entity > EntityPtr
Typedef of EntityPtr as IceInternal::Handle<Entity> for convenience.
Definition: Entity.h:45
memoryx::Entity::equalsAttributes
bool equalsAttributes(const EntityBasePtr &otherEntity, const ::Ice::Current &=Ice::emptyCurrent) const override
equalsAttributes computes if two Entity instances are equal.
Definition: Entity.cpp:393
armarx::transform
auto transform(const Container< InputT, Alloc > &in, OutputT(*func)(InputT const &)) -> Container< OutputT, typename std::allocator_traits< Alloc >::template rebind_alloc< OutputT >>
Convenience function (with less typing) to transform a container of type InputT into the same contain...
Definition: algorithm.h:351
ProbabilityMeasures.h
memoryx::Entity::entityMutex
std::shared_mutex entityMutex
Definition: Entity.h:525
memoryx::Entity::hasAttribute
bool hasAttribute(const ::std::string &attrName, const ::Ice::Current &=Ice::emptyCurrent) const override
Check whether this entity has an attribute with the given name.
Definition: Entity.cpp:361
Entity.h
memoryx::EntityAttribute
Attribute of MemoryX entities.
Definition: EntityAttribute.h:49
armarx::aron::type::ObjectPtr
std::shared_ptr< Object > ObjectPtr
Definition: Object.h:36
memoryx::Entity
Definition: Entity.h:245
memoryx::Entity::equals
bool equals(const EntityBasePtr &otherEntity, const ::Ice::Current &=Ice::emptyCurrent) const override
equals computes if two Entity instances are equal.
Definition: Entity.cpp:383
endif
endif() find_package(Simox QUIET) armarx_build_if(Simox_FOUND "Simox not available") find_package(PythonLibs 3.6 QUIET) armarx_build_if(PYTHONLIBS_FOUND "Python libs not available") if(PYTHONLIBS_FOUND) include_directories($
Definition: CMakeLists.txt:10
memoryx::Entity::setName
void setName(const ::std::string &name, const ::Ice::Current &=Ice::emptyCurrent) override
Set name of this entity.
Definition: Entity.cpp:188
memoryx::Entity::wrappersMutex
std::recursive_mutex wrappersMutex
Definition: Entity.h:526
armarx
This file offers overloads of toIce() and fromIce() functions for STL container types.
Definition: ArmarXTimeserver.cpp:27
memoryx::Entity::isMetaEntity
bool isMetaEntity(const ::Ice::Current &=Ice::emptyCurrent) const override
Indicates whether this entity only contains meta information.
Definition: Entity.cpp:195