ExampleMemory.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 RobotAPI::ArmarXObjects::ExampleMemory
17  * @author Rainer Kartmann ( rainer dot kartmann at kit dot edu )
18  * @date 2020
19  * @copyright http://www.gnu.org/licenses/gpl-2.0.txt
20  * GNU General Public License
21  */
22 
23 #include "ExampleMemory.h"
24 
25 #include <SimoxUtility/algorithm/string.h>
26 
29 
30 #include <RobotAPI/components/armem/server/ExampleMemory/aron/ExampleData.aron.generated.h>
36 
37 namespace armarx
38 {
41  {
44 
45  defs->topic(debugObserver);
46 
47  setMemoryName("Example");
48 
49  p.core._defaultSegmentsStr = simox::alg::join(p.core.defaultCoreSegments, ", ");
50  defs->optional(p.core._defaultSegmentsStr,
51  "core.DefaultSegments",
52  "Core segments to add on start up (just as example).");
53  defs->optional(
54  p.core.addOnUsage,
55  "core.AddOnUsage",
56  "If enabled, core segments are added when required by a new provider segment."
57  "This will usually be off for most memory servers.");
58 
59  defs->optional(p.enableRemoteGui,
60  "p.enableRemoteGui",
61  "If true, the memory cotent is shown in the remote gui."
62  "Can be very slow for high-frequency updates!");
63  return defs;
64  }
65 
66  std::string
68  {
69  return "ExampleMemory";
70  }
71 
72  void
74  {
75  // Usually, the memory server will specify a number of core segments with a specific aron type.
76  workingMemory().addCoreSegment("ExampleData", armem::example::ExampleData::ToAronType());
77  workingMemory().addCoreSegment("LinkedData", armem::example::LinkedData::ToAronType());
78 
79  // We support the "Latest" prediction engine for the entire memory.
81  [this](const armem::PredictionRequest& request)
82  { return this->predictLatest(request); });
83 
84  // For illustration purposes, we add more segments (without types).
85  bool trim = true;
86  p.core.defaultCoreSegments = simox::alg::split(p.core._defaultSegmentsStr, ",", trim);
87  p.core._defaultSegmentsStr.clear();
88 
89  for (const std::string& name : p.core.defaultCoreSegments)
90  {
91  auto& c = workingMemory().addCoreSegment(name);
92  c.setMaxHistorySize(100);
93  }
94  }
95 
96  void
98  {
99  if (p.enableRemoteGui)
100  {
103  }
104  }
105 
106  void
108  {
109  }
110 
111  void
113  {
114  }
115 
116  // WRITING
117 
118  armem::data::AddSegmentsResult
119  ExampleMemory::addSegments(const armem::data::AddSegmentsInput& input, const Ice::Current&)
120  {
121  // This function is overloaded to trigger the remote gui rebuild.
122  armem::data::AddSegmentsResult result =
123  ReadWritePluginUser::addSegments(input, p.core.addOnUsage);
124  tab.rebuild = true;
125  return result;
126  }
127 
128  armem::data::CommitResult
129  ExampleMemory::commit(const armem::data::Commit& commit, const Ice::Current&)
130  {
131  // This function is overloaded to trigger the remote gui rebuild.
132  armem::data::CommitResult result = ReadWritePluginUser::commit(commit);
133  tab.rebuild = true;
134  return result;
135  }
136 
137  // READING
138 
139  // Inherited from Plugin
140 
141 
142  // ACTIONS
143  armem::actions::GetActionsOutputSeq
144  ExampleMemory::getActions(const armem::actions::GetActionsInputSeq& input)
145  {
146  using namespace armem::actions;
147  Action greeting{"hi", "Say hello to " + input[0].id.entityName};
148  Action failure{"fail", "Fail dramatically"};
149  Action nothing{"null", "Do nothing, but deeply nested"};
150 
151  SubMenu one{"one", "One", {nothing}};
152  SubMenu two{"two", "Two", {one}};
153  SubMenu three{"three", "Three", {two}};
154  SubMenu four{"four", "Four", {three}};
155 
156  Menu menu{greeting,
157  failure,
158  four,
159  SubMenu{"mut", "Mutate", {Action{"copy", "Copy latest instance"}}}};
160 
161  return {{menu.toIce()}};
162  }
163 
164  armem::actions::ExecuteActionOutputSeq
165  ExampleMemory::executeActions(const armem::actions::ExecuteActionInputSeq& input)
166  {
167  using namespace armem::actions;
168 
169  ExecuteActionOutputSeq output;
170  for (const auto& [id, path] : input)
171  {
172  const armem::MemoryID memoryID = armarx::fromIce<armem::MemoryID>(id);
173  if (path == ActionPath{"hi"})
174  {
175  ARMARX_INFO << "Hello, " << memoryID.str() << "!";
176  output.emplace_back(true, "");
177  }
178  else if (path == ActionPath{"fail"})
179  {
180  ARMARX_WARNING << "Alas, I am gravely wounded!";
181  output.emplace_back(false, "Why would you do that to him?");
182  }
183  else if (not path.empty() and path.front() == "four" and path.back() == "null")
184  {
185  // Do nothing.
186  ARMARX_INFO << "Nested action (path: " << path << ")";
187  output.emplace_back(true, "");
188  }
189  else if (path == ActionPath{"mut", "copy"})
190  {
191  auto* instance = workingMemory().findLatestInstance(memoryID);
192  if (instance != nullptr)
193  {
195  armem::MemoryID newID =
199  update.entityID = newID;
200  update.referencedTime = armem::Time::Now();
201  update.instancesData = {instance->data()};
202 
203  armem::Commit newCommit;
204  newCommit.add(update);
205  this->commit(armarx::toIce<armem::data::Commit>(newCommit));
206 
207  tab.rebuild = true;
208  ARMARX_INFO << "Duplicated " << memoryID;
209  output.emplace_back(true, "");
210  }
211  else
212  {
213  output.emplace_back(false, "Couldn't duplicate " + memoryID.str());
214  }
215  }
216  }
217 
218  return output;
219  }
220 
221  // PREDICTING
223  ExampleMemory::predictLatest(const armem::PredictionRequest& request)
224  {
226  auto memID = request.snapshotID;
227  result.snapshotID = memID;
228 
230  builder.latestEntitySnapshot(memID);
231  auto queryResult =
232  query(armarx::toIce<armem::query::data::Input>(builder.buildQueryInput()));
233  if (queryResult.success)
234  {
235  auto readMemory = fromIce<armem::wm::Memory>(queryResult.memory);
236  auto* latest = readMemory.findLatestSnapshot(memID);
237  if (latest != nullptr)
238  {
239  auto instance = memID.hasInstanceIndex()
240  ? latest->getInstance(memID)
241  : latest->getInstance(latest->getInstanceIndices().at(0));
242  result.success = true;
243  result.prediction = instance.data();
244  }
245  else
246  {
247  result.success = false;
248  result.errorMessage =
249  "Could not find entity referenced by MemoryID '" + memID.str() + "'.";
250  }
251  }
252  else
253  {
254  result.success = false;
255  result.errorMessage =
256  "Could not find entity referenced by MemoryID '" + memID.str() + "'.";
257  }
258 
259  return result;
260  }
261 
262  // REMOTE GUI
263 
264  void
266  {
267  using namespace armarx::RemoteGui::Client;
268 
269  {
270  // Core segments are locked by MemoryRemoteGui.
272  }
273 
274  VBoxLayout root = {tab.memoryGroup, VSpacer()};
275  RemoteGui_createTab(getName(), root, &tab);
276  }
277 
278  void
280  {
281  if (tab.rebuild.exchange(false))
282  {
284  }
285  }
286 
287 } // namespace armarx
armarx::armem::base::detail::GetFindSnapshotMixin::findLatestInstance
const auto * findLatestInstance(int instanceIndex=0) const
Find the latest entity instance.
Definition: lookup_mixins.h:372
armarx::armem::PredictionRequest
Definition: Prediction.h:48
armarx::armem::PredictionResult::snapshotID
armem::MemoryID snapshotID
Definition: Prediction.h:62
armarx::armem::PredictionEngine::engineID
std::string engineID
Definition: Prediction.h:34
armarx::armem::server::plugins::ReadWritePluginUser::query
virtual armem::query::data::Result query(const armem::query::data::Input &input, const Ice::Current &=Ice::emptyCurrent) override
Definition: ReadWritePluginUser.cpp:56
armarx::armem::MemoryID::providerSegmentName
std::string providerSegmentName
Definition: MemoryID.h:52
armarx::armem::Commit
A bundle of updates to be sent to the memory.
Definition: Commit.h:84
armarx::ExampleMemory::RemoteGui_update
void RemoteGui_update() override
Definition: ExampleMemory.cpp:279
armarx::RemoteGui::Client::VBoxLayout
Definition: Widgets.h:167
armarx::armem::client::query::Builder::buildQueryInput
QueryInput buildQueryInput() const
Definition: Builder.cpp:12
armarx::armem::server::MemoryRemoteGui::makeGroupBox
GroupBox makeGroupBox(const armem::wm::Memory &memory) const
Definition: MemoryRemoteGui.cpp:122
armarx::core::time::DateTime::Now
static DateTime Now()
Definition: DateTime.cpp:51
armarx::armem::MemoryID::str
std::string str(bool escapeDelimiters=true) const
Get a string representation of this memory ID.
Definition: MemoryID.cpp:102
Prediction.h
armarx::armem::server::plugins::ReadWritePluginUser::setMemoryName
void setMemoryName(const std::string &memoryName)
Definition: ReadWritePluginUser.cpp:25
c
constexpr T c
Definition: UnscentedKalmanFilterTest.cpp:46
armarx::armem::client::query::Builder::latestEntitySnapshot
void latestEntitySnapshot(const MemoryID &entityID)
Definition: Builder.cpp:131
armarx::armem::PredictionResult
Definition: Prediction.h:57
armarx::armem::server::plugins::ReadWritePluginUser::workingMemory
server::wm::Memory & workingMemory()
Definition: ReadWritePluginUser.cpp:106
armarx::status::failure
@ failure
armarx::RemoteGui::Client::VSpacer
Definition: Widgets.h:204
armarx::ExampleMemory::commit
armem::data::CommitResult commit(const armem::data::Commit &commit, const Ice::Current &=Ice::emptyCurrent) override
Definition: ExampleMemory.cpp:129
armarx::ExampleMemory::onInitComponent
void onInitComponent() override
Pure virtual hook for the subclass.
Definition: ExampleMemory.cpp:73
armarx::ExampleMemory::onDisconnectComponent
void onDisconnectComponent() override
Hook for subclass.
Definition: ExampleMemory.cpp:107
armarx::armem::MemoryID::withProviderSegmentName
MemoryID withProviderSegmentName(const std::string &name) const
Definition: MemoryID.cpp:417
armarx::armem::MemoryID
A memory ID.
Definition: MemoryID.h:47
armarx::armem::PredictionResult::prediction
aron::data::DictPtr prediction
Definition: Prediction.h:63
armarx::ExampleMemory::onConnectComponent
void onConnectComponent() override
Pure virtual hook for the subclass.
Definition: ExampleMemory.cpp:97
MemoryRemoteGui.h
armarx::aron::input
ReaderT::InputType & input
Definition: rw.h:12
armarx::ExampleMemory::createPropertyDefinitions
armarx::PropertyDefinitionsPtr createPropertyDefinitions() override
Definition: ExampleMemory.cpp:40
error.h
armarx::armem::EntityUpdate
An update of an entity for a specific point in time.
Definition: Commit.h:25
ice_conversions.h
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
armarx::ExampleMemory::addSegments
armem::data::AddSegmentsResult addSegments(const armem::data::AddSegmentsInput &input, const Ice::Current &) override
Definition: ExampleMemory.cpp:119
armarx::ExampleMemory::createRemoteGuiTab
void createRemoteGuiTab()
Definition: ExampleMemory.cpp:265
armarx::armem::server::wm::detail::Prediction::addPredictor
void addPredictor(const PredictionEngine &engine, Predictor &&predictor)
Definition: Prediction.h:68
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::PredictionEngine
Definition: Prediction.h:32
armarx::LightweightRemoteGuiComponentPluginUser::RemoteGui_startRunningTask
void RemoteGui_startRunningTask()
Definition: LightweightRemoteGuiComponentPlugin.cpp:119
ExpressionException.h
armarx::armem::client::query_fns::latest
std::function< void(query::SnapshotSelector &)> latest()
Definition: query_fns.h:81
armarx::armem::MemoryID::getCoreSegmentID
MemoryID getCoreSegmentID() const
Definition: MemoryID.cpp:294
armarx::Component::getConfigIdentifier
std::string getConfigIdentifier()
Retrieve config identifier for this component as set in constructor.
Definition: Component.cpp:79
armarx::armem::MemoryID::withEntityName
MemoryID withEntityName(const std::string &name) const
Definition: MemoryID.cpp:425
armarx::ExampleMemory::onExitComponent
void onExitComponent() override
Hook for subclass.
Definition: ExampleMemory.cpp:112
armarx::armem::Commit::add
EntityUpdate & add()
Definition: Commit.cpp:80
armarx::ComponentPropertyDefinitions
Default component property definition container.
Definition: Component.h:69
armarx::armem::PredictionResult::errorMessage
std::string errorMessage
Definition: Prediction.h:60
ARMARX_INFO
#define ARMARX_INFO
Definition: Logging.h:181
armarx::ExampleMemory::getActions
armem::actions::GetActionsOutputSeq getActions(const armem::actions::GetActionsInputSeq &input) override
Definition: ExampleMemory.cpp:144
armarx::LightweightRemoteGuiComponentPluginUser::RemoteGui_createTab
void RemoteGui_createTab(std::string const &name, RemoteGui::Client::Widget const &rootWidget, RemoteGui::Client::Tab *tab)
Definition: LightweightRemoteGuiComponentPlugin.cpp:103
armarx::ExampleMemory::getDefaultName
std::string getDefaultName() const override
Definition: ExampleMemory.cpp:67
IceUtil::Handle
Definition: forward_declarations.h:30
armarx::armem::index::memoryID
const MemoryID memoryID
Definition: memory_ids.cpp:28
Builder.h
ExampleMemory.h
armarx::armem::client::query::Builder
The query::Builder class provides a fluent-style specification of hierarchical queries.
Definition: Builder.h:21
ice_conversions_templates.h
armarx::ManagedIceObject::getName
std::string getName() const
Retrieve name of object.
Definition: ManagedIceObject.cpp:108
armarx::ExampleMemory::executeActions
armem::actions::ExecuteActionOutputSeq executeActions(const armem::actions::ExecuteActionInputSeq &input) override
Definition: ExampleMemory.cpp:165
armarx::armem::PredictionResult::success
bool success
Definition: Prediction.h:59
ARMARX_WARNING
#define ARMARX_WARNING
Definition: Logging.h:193
armarx::RemoteGui::Client
Definition: EigenWidgets.cpp:8
armarx::armem::server::MemoryRemoteGui
Utility for memory Remote Guis.
Definition: MemoryRemoteGui.h:13
armarx::armem::PredictionRequest::snapshotID
armem::MemoryID snapshotID
Definition: Prediction.h:50
armarx
This file offers overloads of toIce() and fromIce() functions for STL container types.
Definition: ArmarXTimeserver.cpp:27
armarx::split
std::vector< std::string > split(const std::string &source, const std::string &splitBy, bool trimElements=false, bool removeEmptyElements=false)
Definition: StringHelpers.cpp:38