27 #include <type_traits>
29 #include <SimoxUtility/json/json.hpp>
36 #include <RobotAPI/interface/units/RobotUnit/RobotUnitInterface.h>
42 #include <armarx/control/core/Controller.h>
43 #include <armarx/control/interface/ConfigurableNJointControllerInterface.h>
44 #include <armarx/control/memory/config/Reader.h>
45 #include <armarx/control/memory/config/Writer.h>
62 detail::ctrl_class_name::CtrlConfigClassName_v<ControllerDescriptionType>;
65 using AronDTO =
typename ControllerDescriptionType::AronDTO;
68 static_assert(::armarx::aron::cpp::isAronGeneratedClass<AronDTO>,
69 "The AronConfigT must be an AronGeneratedClass!");
72 const memory::config::Reader& configProvider,
73 const memory::config::Writer& configWriter,
74 const std::string& controllerNamePrefix) :
75 controllerCreator(controllerCreator),
76 configReader(configProvider),
77 configWriter(configWriter),
78 controllerNamePrefix(controllerNamePrefix)
81 <<
"By convention, the controller name prefix must exist. This ensures that "
82 "in between components, no naming clashes occur.";
88 if (
auto cfg = configReader.getDefaultConfig(ControllerDescriptionType::name, name))
90 dto = AronDTO::FromAron(cfg);
102 const std::string& robotNameSuffix =
"",
103 const std::string& appendix =
"")
108 "controller_config/" + configClassName +
"/" +
109 name + robotNameSuffix + appendix +
".json");
117 auto cfg = configReader.getConfig(memoryId);
118 dto = AronDTO::FromAron(cfg);
126 dto = AronDTO::FromAron(configDict);
138 nlohmann::json jsonConfig;
142 if (not this->dto.has_value())
152 this->dto->read(reader, jsonConfig);
175 this->nodeSetName = nodeSetName;
182 if (namePrefix.empty())
186 this->controllerNamePrefix = this->controllerNamePrefix +
"_" + namePrefix;
193 this->daemonized =
true;
202 std::unique_ptr<ControllerWrapper<T>>
205 const std::string controllerClassName = ControllerDescriptionType::name;
206 return create(controllerClassName);
209 std::unique_ptr<ControllerWrapper<T>>
212 if (not ensureInitialized())
217 ::armarx::control::ConfigurableNJointControllerConfigPtr
config =
218 new ::armarx::control::ConfigurableNJointControllerConfig;
221 config->config = dto.value().toAronDTO();
233 controllerName = controllerNamePrefix +
"_" + name;
234 const std::string controllerClassName = ControllerDescriptionType::name;
239 armarx::NJointControllerInterfacePrx
controller =
240 controllerCreator->getNJointController(controllerName);
241 ARMARX_DEBUG <<
"--> get controller by name from robot unit took "
244 this->reusing_ = [&]() ->
bool
259 if (
controller->getClassName() != controllerClassName)
298 ARMARX_INFO <<
"--- Deleting existing controller `" << controllerName <<
"`";
302 ARMARX_INFO <<
"--- Wait until controller is in-active";
311 ARMARX_INFO <<
"--- Controller is not active anymore, deleting controller";
325 ARMARX_INFO <<
"Reuse of controller would be possible.";
329 if (not this->reusing_)
333 ARMARX_INFO <<
"Creating controller `" << controllerName <<
"`";
335 controller = controllerCreator->createNJointController(
336 controllerClassName, controllerName,
config);
340 ARMARX_INFO <<
"Controller `" << controllerName <<
"` created.";
349 ARMARX_WARNING <<
"Failed to create controller of type `" << controllerClassName
355 armarx::control::ConfigurableNJointControllerInterfacePrx::checkedCast(
controller);
357 std::unique_ptr<ControllerWrapper<T>> ctrlWrapper =
358 std::make_unique<ControllerWrapper<T>>(
359 ctrl, controllerName, configWriter, dto.value());
360 ctrlWrapper->daemonize(daemonized);
365 ctrlWrapper->updateConfig();
371 std::unique_ptr<ControllerWrapper<T>>
374 const std::string& configFilename =
"",
375 bool flagActivate =
true,
376 bool flagAllowReuse =
true,
377 bool flagFromMemory =
false)
379 ARMARX_CHECK(not(configDict.get() ==
nullptr and configFilename.empty()));
388 if (configFilename.empty())
398 <<
"configuration file does not exist\n"
401 "1) if you need to configure a controller by passing configuration "
404 "controllerBuilder (me), you need to make sure the "
406 "component is running "
407 "on the same PC as your application, otherwise\n"
408 "2) use the createControllerFromConfig() interface, which does not "
411 "creator is running";
423 std::ostringstream oss;
424 for (
const auto& pair :
config().limbs)
427 if (&pair != &(*
config().limbs.rbegin()))
432 std::string robotNodeSets = oss.str();
434 if (namePrefix.empty())
447 auto ctrlWrapper =
create();
451 ARMARX_INFO <<
"using controller name " << controllerName;
457 <<
"setting nullspace target: if the default is null use the cfg from the "
458 "existing controller";
461 auto ctrlConfig = ctrlWrapper->getConfig().limbs;
462 for (
const auto& pair :
config().limbs)
464 if (not pair.second.desiredNullspaceJointAngles.has_value())
466 ctrlWrapper->config.limbs.at(pair.first).desiredNullspaceJointAngles =
467 ctrlConfig.at(pair.first).desiredNullspaceJointAngles;
481 ctrlWrapper->activate();
483 while (not ctrlWrapper->isActive())
487 ARMARX_INFO <<
"Controller " << controllerName <<
" is active";
491 ARMARX_INFO <<
"User have to activate " << controllerName
492 <<
" controller in your application";
500 const std::string& configFilename =
"",
501 bool flagActivate =
true,
502 bool flagAllowReuse =
true,
503 bool flagFromMemory =
false)
511 auto ctrl = CtrlProxy::checkedCast(ctrlWrapper->ctrl());
515 std::unique_ptr<ControllerWrapper<T>>
519 const std::string& configFilename =
"",
520 bool flagActivate =
true,
521 bool flagAllowReuse =
true,
522 bool flagFromMemory =
false,
523 bool flagStart =
true)
531 auto ctrl = CtrlProxy::checkedCast(ctrlWrapper->ctrl());
532 ctrl->updateMPConfig(mpConfigDict);
548 const std::string& configFilename =
"",
549 bool flagActivate =
true,
550 bool flagAllowReuse =
true,
551 bool flagFromMemory =
false,
552 bool flagStart =
true)
560 auto ctrl = CtrlProxy::checkedCast(ctrlWrapper->ctrl());
563 ctrl->updateMPConfig(mpConfigDict);
564 ARMARX_DEBUG <<
"--> ctrl->updateMPConfig(mpConfigDict: DictPtr) took "
592 return this->reusing_;
598 return controllerName;
605 if (
auto cfg = configReader.getDefaultConfig(ControllerDescriptionType::name))
607 dto = AronDTO::FromAron(cfg);
618 if (not dto.has_value())
620 return initDefaultConfig();
628 const memory::config::Reader configReader;
629 const memory::config::Writer configWriter;
631 std::optional<AronDTO> dto;
633 std::optional<std::string> nodeSetName;
635 std::string controllerNamePrefix;
636 std::string controllerName;
638 bool daemonized =
false;
639 bool allowReuse_ =
false;
640 bool reusing_ =
false;