PersonMemoryDebugger.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 VisionX::ArmarXObjects::person_memory_debugger
17  * @author Peter Albrecht ( usnlf at student dot kit dot edu )
18  * @date 2024
19  * @copyright http://www.gnu.org/licenses/gpl-2.0.txt
20  * GNU General Public License
21  */
22 
23 
24 #include "PersonMemoryDebugger.h"
25 
26 #include <optional>
27 #include <utility>
28 
29 #include <Eigen/Core>
30 #include <Eigen/Geometry>
31 
32 #include <SimoxUtility/algorithm/get_map_keys_values.h>
33 #include <SimoxUtility/algorithm/string/string_tools.h>
34 
36 
45 
47 #include <VisionX/libraries/armem_human/aron/FaceRecognition.aron.generated.h>
48 #include <VisionX/libraries/armem_human/aron/HumanPose.aron.generated.h>
49 #include <VisionX/libraries/armem_human/aron/Profile.aron.generated.h>
56 
58 {
59 
60  const std::string PersonMemoryDebugger::pose_provider_name = "person_memory_debugger";
61  const std::string PersonMemoryDebugger::profile_provider_name = "person_memory_debugger";
62  const std::string PersonMemoryDebugger::recognition_provider_name = "person_memory_debugger";
63 
66  {
69  def->defineRequiredProperty<std::string>("robotName");
70  return def;
71  }
72 
73  void
75  {
76  }
77 
78  void
80  {
81  idCounter = 0;
82  running = true;
83  mns = memoryNameSystem();
84  properties.robotName = this->getProperty<std::string>("robotName");
85  while (running)
86  {
87  this->run();
88  }
89  }
90 
91  void
93  {
94  }
95 
96  void
98  {
99  }
100 
101  void
102  PersonMemoryDebugger::run()
103  {
104  ARMARX_INFO << "Use 'pose [id] [x] [y] [z]' to commit a pose at location";
105  ARMARX_INFO << "Use 'face [profile entityName] [x] [y] [z]' to commit a face at location";
106  ARMARX_INFO << "Use 'profile [profile entityName]' to commit a profile";
107  ARMARX_INFO << "Use 'exit' to exit";
108  std::string command;
109  std::cin >> command;
110  if (command == "pose")
111  {
112  poseCommand();
113  }
114  else if (command == "face")
115  {
116  faceRecognitionCommand();
117  }
118  else if (command == "profile")
119  {
120  profileCommand();
121  }
122  else if (command == "exit")
123  {
124  running = false;
125  }
126  else
127  {
128  ARMARX_WARNING << "Unrecognized command. Ignoring";
129  }
130 
131  std::cin.clear();
132  }
133 
134  void
135  PersonMemoryDebugger::poseCommand()
136  {
137  std::string id, x_s, y_s, z_s;
138  std::cin >> id >> x_s >> y_s >> z_s;
139  float x = std::stof(x_s);
140  float y = std::stof(y_s);
141  float z = std::stof(z_s);
142  Eigen::Vector3f pos(x, y, z);
143  commitPose(pos, id);
144  }
145 
146  void
147  PersonMemoryDebugger::faceRecognitionCommand()
148  {
149  std::string id, x_s, y_s, z_s;
150  std::cin >> id >> x_s >> y_s >> z_s;
151 
152  auto profileId = findProfileId(id);
153  if (not profileId.has_value())
154  {
155  ARMARX_WARNING << "Could not find profile. Continuing..";
156  }
157 
158  float x = std::stof(x_s);
159  float y = std::stof(y_s);
160  float z = std::stof(z_s);
161 
162  Eigen::Vector3f pos(x, y, z);
163  commitFaceRecognition(pos, profileId);
164  }
165 
166  void
167  PersonMemoryDebugger::profileCommand()
168  {
169  std::string name;
170  std::cin >> name;
171  commitProfile(name);
172  }
173 
174  void
175  PersonMemoryDebugger::commitProfile(std::string& name)
176  {
177  armarx::armem::client::Writer poseWriter =
179 
180  // we only set the firstName. FaceImages are not required,
181  // as the user manually sets the profile in the face recognition.
182  // therefore, this will not work with a real implementation of face recognition
183  // TODO: implement option to add face images
185  dto.firstName = name;
186 
187  armarx::armem::MemoryID entityId =
189  .withEntityName(name);
190 
192  .entityID = entityId,
193  .instancesData = {dto.toAron()},
194  .referencedTime = armarx::armem::Time::Now(),
195  .confidence = 1.0F,
196  .sentTime = armarx::armem::Time::Now(),
197  .arrivedTime = armarx::armem::Time::Now(),
198  };
199 
200  ARMARX_INFO << "Committing profile with name: " << name;
201  poseWriter.commit(update);
202  }
203 
204  std::optional<armarx::armem::MemoryID>
205  PersonMemoryDebugger::findProfileId(std::string& entityId)
206  {
207  armarx::armem::client::Reader profileReader =
209 
212  std::optional<armarx::armem::MemoryID> foundProfileId = std::nullopt;
213 
214  // memory fetch failed
215  if (not queryResult.success)
216  {
217  return std::nullopt;
218  }
219 
220  // find the first matching profile
221  queryResult.memory.forEachInstance(
222  [&entityId, &foundProfileId](armarx::armem::wm::EntityInstance& instance)
223  {
224  if (foundProfileId.has_value())
225  {
226  // we want the first match, and ignore all further matches here.
227  return;
228  }
229 
230  if (instance.id().entityName == entityId)
231  {
232  // match found
233  foundProfileId = instance.id();
234  }
235  });
236 
237  return foundProfileId;
238  }
239 
240  void
241  PersonMemoryDebugger::commitFaceRecognition(Eigen::Vector3f& facePosition,
242  std::optional<armarx::armem::MemoryID> profileId)
243  {
244  armarx::armem::client::Writer poseWriter =
246 
247  // get the id counter
248  std::stringstream ss;
249  ss << idCounter++;
250 
252  bo.framedPosition3D =
253  armarx::FramedPosition(facePosition, armarx::GlobalFrame, properties.robotName);
254  // doesnt make sense, but this isn't set in prod anyways
255  bo.position2D = Eigen::Vector2i::Identity();
256  bo.position3DGlobal = facePosition;
257  // profile id may be nullopt, but this case is communicated to the user
258  bo.profileID = std::move(profileId);
259  bo.extents2D = Eigen::Vector2i::Zero();
260 
261  armarx::human::arondto::FaceRecognition dto;
263 
264  armarx::armem::MemoryID entityId =
266  .withEntityName(ss.str());
267 
269  .entityID = entityId,
270  .instancesData = {dto.toAron()},
271  .referencedTime = armarx::armem::Time::Now(),
272  .confidence = 1.0F,
273  .sentTime = armarx::armem::Time::Now(),
274  .arrivedTime = armarx::armem::Time::Now(),
275  };
276 
277  // hack
279  << "Committing face recognition for profile: "
280  << profileId.value_or(armarx::armem::MemoryID().withEntityName("INVALID")).entityName;
281  poseWriter.commit(update);
282  }
283 
284  void
285  PersonMemoryDebugger::commitPose(Eigen::Vector3f& facePosition, std::string& humanId)
286  {
288 
290  pose.humanTrackingId = humanId;
294  armarx::FramedOrientation defaultOrientation(
295  Eigen::Quaternionf::UnitRandom(), armarx::GlobalFrame, properties.robotName);
296 
298 
299  // iterate over all joints, and set their global positions to the face pos.
300  // this is the only value required to be valid by the PersonInstanceUpdater
301  for (auto& joint : JointNames.names())
302  {
304  .label = joint,
305  .confidence = 1.0F,
306  .positionCamera =
307  armarx::FramedPosition(facePosition, "camera", properties.robotName),
308  .orientationCamera = armarx::FramedOrientation(
309  Eigen::Quaternionf::UnitRandom(), "camera", properties.robotName),
310  .positionRobot =
311  armarx::FramedPosition(facePosition, "robot", properties.robotName),
312  .orientationRobot = armarx::FramedOrientation(
313  Eigen::Quaternionf::UnitRandom(), "robot", properties.robotName),
314  .positionGlobal =
315  armarx::FramedPosition(facePosition, armarx::GlobalFrame, properties.robotName),
316  .orientationGlobal = defaultOrientation,
317  };
318  // set every joint to face position
319  pose.keypoints[joint] = key;
320  }
321 
322  armarx::armem::MemoryID entityId =
324  .withEntityName(humanId);
325 
326  armarx::human::arondto::HumanPose dto;
327  armarx::armem::human::toAron(dto, pose);
328 
330  .entityID = entityId,
331  .instancesData = {dto.toAron()},
332  .referencedTime = armarx::armem::Time::Now(),
333  .confidence = 1.0F,
334  .sentTime = armarx::armem::Time::Now(),
335  .arrivedTime = armarx::armem::Time::Now(),
336  };
337 
338  ARMARX_INFO << "Committing pose with HumanID '" << humanId << "'.";
339  poseWriter.commit(update);
340  }
341 
342  std::string
344  {
345  return "PersonMemoryDebugger";
346  }
347 
348  // ARMARX_REGISTER_COMPONENT_EXECUTABLE(PersonMemoryDebugger,
349  // PersonMemoryDebugger::GetDefaultName());
350 
351 } // namespace VisionX::components::person_memory_debugger
armarx::armem::detail::SuccessHeader::success
bool success
Definition: SuccessHeader.h:19
visionx::components::person_memory_debugger
Definition: PersonMemoryDebugger.cpp:57
armarx::armem::client::Reader
Reads data from a memory server.
Definition: Reader.h:23
Reader.h
armarx::armem::client::QueryResult::memory
wm::Memory memory
The slice of the memory that matched the query.
Definition: Query.h:58
armarx::armem::client::Reader::getLatestSnapshotsIn
QueryResult getLatestSnapshotsIn(const MemoryID &id, armem::query::DataMode dataMode=armem::query::DataMode::WithData) const
Get the latest snapshots under the given memory ID.
Definition: Reader.cpp:340
visionx::components::person_memory_debugger::PersonMemoryDebugger::getDefaultName
std::string getDefaultName() const override
Definition: PersonMemoryDebugger.cpp:343
Writer.h
armarx::human::FaceRecognitionCoreSegmentID
const armem::MemoryID FaceRecognitionCoreSegmentID
Definition: memory_ids.cpp:30
armarx::armem::wm::EntityInstance
Client-side working entity instance.
Definition: memory_definitions.h:32
armarx::armem::human::toAron
void toAron(armarx::human::arondto::HumanPose &dto, const HumanPose &bo)
Definition: aron_conversions.cpp:31
armarx::core::time::DateTime::Now
static DateTime Now()
Definition: DateTime.cpp:51
MemoryID.h
armarx::armem::client::plugins::PluginUser::memoryNameSystem
MemoryNameSystem & memoryNameSystem()
Definition: PluginUser.cpp:20
armarx::human::pose::model::k4a_bt_body_32
This file is part of ArmarX.
Definition: k4a_bt_body_32.h:31
armarx::armem::human::FaceRecognition
Definition: types.h:59
forward_declarations.h
armarx::GlobalFrame
const std::string GlobalFrame
Definition: FramedPose.h:65
armarx::armem::human::PoseKeypoint::label
std::string label
Definition: types.h:19
PersonMemoryDebugger.h
visionx::components::person_memory_debugger::PersonMemoryDebugger::createPropertyDefinitions
armarx::PropertyDefinitionsPtr createPropertyDefinitions() override
Definition: PersonMemoryDebugger.cpp:65
memory_ids.h
armarx::armem::client::QueryResult
Result of a QueryInput.
Definition: Query.h:50
armarx::armem::client::MemoryNameSystem::getWriter
Writer getWriter(const MemoryID &memoryID)
Get a writer to the given memory name.
Definition: MemoryNameSystem.cpp:283
k4a_bt_body_32.h
armarx::armem::human::HumanPose::poseModelId
std::string poseModelId
Definition: types.h:32
GfxTL::Identity
void Identity(MatrixXX< N, N, T > *a)
Definition: MatrixXX.h:570
armarx::armem::MemoryID::withProviderSegmentName
MemoryID withProviderSegmentName(const std::string &name) const
Definition: MemoryID.cpp:417
armarx::armem::human::PoseKeypoint
Definition: types.h:17
types.h
armarx::armem::MemoryID
A memory ID.
Definition: MemoryID.h:47
armarx::human::pose::model::k4a_bt_body_32::JointNames
const simox::meta::EnumNames< Joints > JointNames
Names of the joints as defined in the body model.
Definition: k4a_bt_body_32.h:77
PoseSegment.h
if
if(!yyvaluep)
Definition: Grammar.cpp:645
FramedPose.h
visionx::components::person_memory_debugger::PersonMemoryDebugger::profile_provider_name
static const std::string profile_provider_name
Definition: PersonMemoryDebugger.h:60
armarx::armem::human::HumanPose::timestamp
DateTime timestamp
Definition: types.h:37
armarx::armem::human::HumanPose::cameraFrameName
std::string cameraFrameName
Definition: types.h:38
armarx::armem::server::human::profile::Profile
armarx::human::arondto::Profile Profile
Definition: Segment.cpp:43
armarx::armem::EntityUpdate
An update of an entity for a specific point in time.
Definition: Commit.h:25
armarx::armem::client::Writer
Helps a memory client sending data to a memory.
Definition: Writer.h:22
armarx::armem::client::Writer::commit
CommitResult commit(const Commit &commit) const
Writes a Commit to the memory.
Definition: Writer.cpp:59
armarx::FramedOrientation
The FramedOrientation class.
Definition: FramedPose.h:215
armarx::armem::base::detail::MemoryItem::id
MemoryID & id()
Definition: MemoryItem.h:25
armarx::armem::human::HumanPose
Definition: types.h:30
armarx::armem::MemoryID::entityName
std::string entityName
Definition: MemoryID.h:53
armarx::armem::server::ltm::util::mongodb::detail::update
bool update(mongocxx::collection &coll, const nlohmann::json &query, const nlohmann::json &update)
Definition: mongodb.cpp:68
armarx::armem::human::HumanPose::humanTrackingId
std::optional< std::string > humanTrackingId
Definition: types.h:36
memory_definitions.h
armarx::human::pose::model::k4a_bt_body_32::ModelId
const std::string ModelId
Definition: k4a_bt_body_32.h:33
Commit.h
visionx::components::person_memory_debugger::PersonMemoryDebugger::onConnectComponent
void onConnectComponent() override
Definition: PersonMemoryDebugger.cpp:79
armarx::Component::getConfigIdentifier
std::string getConfigIdentifier()
Retrieve config identifier for this component as set in constructor.
Definition: Component.cpp:79
visionx::components::person_memory_debugger::PersonMemoryDebugger::onDisconnectComponent
void onDisconnectComponent() override
Definition: PersonMemoryDebugger.cpp:92
Decoupled.h
armarx::armem::MemoryID::withEntityName
MemoryID withEntityName(const std::string &name) const
Definition: MemoryID.cpp:425
armarx::VariantType::FramedOrientation
const VariantTypeId FramedOrientation
Definition: FramedPose.h:39
armarx::ComponentPropertyDefinitions
Default component property definition container.
Definition: Component.h:69
ARMARX_INFO
#define ARMARX_INFO
Definition: Logging.h:181
aron_conversions.h
visionx::components::person_memory_debugger::PersonMemoryDebugger::onExitComponent
void onExitComponent() override
Definition: PersonMemoryDebugger.cpp:97
Segment.h
armarx::human::ProfileCoreSegmentID
const armem::MemoryID ProfileCoreSegmentID
Definition: memory_ids.cpp:33
IceUtil::Handle< class PropertyDefinitionContainer >
Segment.h
armarx::armem::base::detail::ForEachEntityInstanceMixin::forEachInstance
bool forEachInstance(InstanceFunctionT &&func)
Definition: iteration_mixins.h:147
armarx::human::PoseCoreSegmentID
const armem::MemoryID PoseCoreSegmentID
Definition: memory_ids.cpp:38
MemoryNameSystem.h
armarx::armem::human::HumanPose::keypoints
KeyPointMap keypoints
Definition: types.h:35
armarx::VariantType::FramedPosition
const VariantTypeId FramedPosition
Definition: FramedPose.h:38
visionx::components::person_memory_debugger::PersonMemoryDebugger::onInitComponent
void onInitComponent() override
Definition: PersonMemoryDebugger.cpp:74
ARMARX_WARNING
#define ARMARX_WARNING
Definition: Logging.h:193
armarx::armem::client::MemoryNameSystem::getReader
Reader getReader(const MemoryID &memoryID)
Get a reader to the given memory name.
Definition: MemoryNameSystem.cpp:191
armarx::aron::bo
const std::optional< BoT > & bo
Definition: aron_conversions.h:174
visionx::components::person_memory_debugger::PersonMemoryDebugger::pose_provider_name
static const std::string pose_provider_name
Definition: PersonMemoryDebugger.h:59
visionx::components::person_memory_debugger::PersonMemoryDebugger::recognition_provider_name
static const std::string recognition_provider_name
Definition: PersonMemoryDebugger.h:61