15#include "auto_apms_behavior_tree_core/tree/tree_document.hpp"
20#include "ament_index_cpp/get_package_share_directory.hpp"
21#include "auto_apms_behavior_tree_core/builder.hpp"
22#include "auto_apms_behavior_tree_core/exceptions.hpp"
23#include "auto_apms_behavior_tree_core/node/node_model_type.hpp"
24#include "auto_apms_behavior_tree_core/tree/tree_resource.hpp"
25#include "auto_apms_util/container.hpp"
26#include "auto_apms_util/logging.hpp"
27#include "auto_apms_util/string.hpp"
28#include "behaviortree_cpp/xml_parsing.h"
33std::vector<std::string> getAllTreeNamesImpl(
const tinyxml2::XMLDocument & doc)
35 std::vector<std::string> names;
36 if (
const tinyxml2::XMLElement * root = doc.RootElement()) {
37 if (strcmp(root->Name(), TreeDocument::ROOT_ELEMENT_NAME) == 0) {
39 for (
const tinyxml2::XMLElement * child = root->FirstChildElement(); child !=
nullptr;
40 child = child->NextSiblingElement()) {
41 if (strcmp(TreeDocument::TREE_ELEMENT_NAME, child->Name()) == 0) {
42 if (
const char * name = child->Attribute(TreeDocument::TREE_NAME_ATTRIBUTE_NAME)) {
43 names.push_back(name);
45 throw exceptions::TreeDocumentError(
46 "Cannot get tree name, because required attribute '" +
47 std::string(TreeDocument::TREE_NAME_ATTRIBUTE_NAME) +
"' is missing.");
51 }
else if (strcmp(root->Name(), TreeDocument::TREE_ELEMENT_NAME) == 0) {
53 if (
const char * name = root->Attribute(TreeDocument::TREE_NAME_ATTRIBUTE_NAME)) names.push_back(name);
63 throw exceptions::TreeDocumentError(
"Cannot create an instance of NodeElement with doc_ptr=nullptr.");
66 throw exceptions::TreeDocumentError(
"Cannot create an instance of NodeElement with ele_ptr=nullptr.");
69 NodeModelMap::const_iterator it =
model.find(
ele_ptr_->Name());
70 std::vector<NodePortInfo> port_infos_str_vec;
71 if (it !=
model.end()) port_infos_str_vec = it->second.port_infos;
78 if (!port_infos_str_vec.empty()) {
79 for (const NodePortInfo & port_info : port_infos_str_vec) {
80 port_names_.push_back(port_info.port_name);
81 if (!port_info.port_default.empty()) port_default_values_[port_info.port_name] = port_info.port_default;
87 for (
const auto & [name, val] : port_default_values_) {
88 if (existing_port_values.find(name) == existing_port_values.end()) {
89 ele_ptr_->SetAttribute(name.c_str(), val.c_str());
97 XMLElement * other_ele = other.
ele_ptr_;
100 other_ele = other_ele->DeepClone(
doc_ptr_)->ToElement();
102 XMLElement * prev =
ele_ptr_->PreviousSiblingElement();
104 ele_ptr_->Parent()->InsertAfterChild(
ele_ptr_->PreviousSibling(), other_ele);
106 ele_ptr_->Parent()->InsertFirstChild(other_ele);
113 port_names_ = other.port_names_;
114 port_default_values_ = other.port_default_values_;
120bool TreeDocument::NodeElement::operator!=(
const NodeElement & other)
const {
return !this->operator==(other); }
123 const std::string & name,
const NodeElement * before_this)
125 if (
const std::set<std::string> names =
doc_ptr_->getRegisteredNodeNames(
true); names.find(name) == names.end()) {
126 throw exceptions::TreeDocumentError(
127 "Cannot insert unknown node <" + name +
128 ">. Before inserting a new node, the associated document must register the corresponding behavior tree "
129 "node. Consider using a signature of insertNode() that does this automatically.");
131 XMLElement * ele =
doc_ptr_->NewElement(name.c_str());
132 return insertBeforeImpl(before_this, ele);
140 if (
const std::set<std::string> names =
doc_ptr_->getRegisteredNodeNames(
false); names.find(name) == names.end()) {
147 const std::string & tree_name,
const NodeElement * before_this)
149 if (!
doc_ptr_->hasTreeName(tree_name)) {
150 throw exceptions::TreeDocumentError(
151 "Cannot insert subtree node for tree '" + tree_name +
"' because no tree with that name exists.");
154 ele.
ele_ptr_->SetAttribute(TREE_NAME_ATTRIBUTE_NAME, tree_name.c_str());
160 const std::string tree_name = tree.
getName();
161 if (
doc_ptr_->hasTreeName(tree_name)) {
168 if (existing_tree != tree) {
169 throw exceptions::TreeDocumentError(
170 "Cannot insert subtree node using the provided tree element, because another tree with name '" + tree_name +
171 "' already exists.");
175 const std::string existing_xml = existing_tree.
writeToString();
177 if (existing_xml != incoming_xml) {
178 throw exceptions::TreeDocumentError(
179 "Cannot insert subtree node using the provided tree element, because another tree with name '" + tree_name +
180 "' already exists with different content.");
194 const XMLElement * root_child = tree.
ele_ptr_->FirstChildElement();
196 throw exceptions::TreeDocumentError(
199 if (root_child->NextSibling()) {
200 throw exceptions::TreeDocumentError(
201 "Cannot insert tree element '" + tree.
getFullyQualifiedName() +
"' because it has more than one child node.");
210 const std::vector<std::string> other_tree_names = doc.
getAllTreeNames();
211 if (other_tree_names.empty()) {
212 throw exceptions::TreeDocumentError(
213 "Cannot insert tree '" + tree_name +
"' because document has no <" + TREE_ELEMENT_NAME +
"> elements.");
216 throw exceptions::TreeDocumentError(
217 "Cannot insert tree '" + tree_name +
"' because document doesn't specify a tree with that name.");
221 std::set<std::string> required_tree_names = {tree_name};
227 doc.DeepCopy(&temp_doc);
228 const TreeElement other_tree(&temp_doc, temp_doc.getXMLElementForTreeWithName(tree_name));
230 collect_dependency_tree_names = [&required_tree_names, &collect_dependency_tree_names](
const NodeElement & ele) {
231 if (ele.getRegistrationName() == SUBTREE_ELEMENT_NAME) {
232 if (
const char * name = ele.ele_ptr_->Attribute(TREE_NAME_ATTRIBUTE_NAME)) {
234 required_tree_names.insert(name);
237 ele.doc_ptr_->getTree(name).deepApplyConst(collect_dependency_tree_names);
239 throw exceptions::TreeDocumentError(
"Subtree element has no name attribute.");
249 return available_names.find(ele.getRegistrationName()) == available_names.end();
251 for (
const std::string & name : required_tree_names) {
252 const NodeElement ele(&temp_doc, temp_doc.getXMLElementForTreeWithName(name)->FirstChildElement());
253 if (
const std::vector<NodeElement> found = ele.
deepApplyConst(apply); !found.empty()) {
254 std::vector<std::string> names;
256 throw exceptions::TreeDocumentError(
257 "Cannot insert tree '" + tree_name +
"' because the following nodes found in tree '" + name +
258 "' are unknown to the builder:\n\t- " + auto_apms_util::join(names,
"\n\t- "));
264 required_tree_names.erase(tree_name);
265 for (
const std::string & name : other_tree_names) {
266 if (required_tree_names.find(name) == required_tree_names.end()) {
271 doc_ptr_->mergeTreeDocument(
static_cast<const XMLDocument &
>(temp_doc),
false);
274 return insertBeforeImpl(
275 before_this, doc.getXMLElementForTreeWithName(tree_name)->FirstChildElement()->DeepClone(
doc_ptr_)->ToElement());
285 const std::string & tree_str,
const std::string & tree_name,
const NodeElement * before_this)
293 const std::string & tree_str,
const NodeElement * before_this)
301 const std::string & path,
const std::string & tree_name,
const NodeElement * before_this)
309 const std::string & path,
const NodeElement * before_this)
320 insert_doc.
mergeFile(resource.build_request_file_path_);
339 const std::string & registration_name,
const std::string & instance_name,
bool deep_search)
const
344 if (registration_name.empty())
return ele.getName() == instance_name;
345 if (instance_name.empty())
return ele.getRegistrationName() == registration_name;
346 return ele.getRegistrationName() == registration_name && ele.getName() == instance_name;
350 if (
const std::vector<NodeElement> found =
deepApplyConst(apply); !found.empty())
return found[0];
352 for (
auto child : *
this) {
353 if (apply(child))
return child;
358 throw exceptions::TreeDocumentError(
359 "Cannot find a child node that matches the search arguments (registration_name: '" + registration_name +
364 const std::string & registration_name,
const std::string & instance_name,
bool deep_search)
366 XMLElement * found =
getFirstNode(registration_name, instance_name, deep_search).ele_ptr_;
367 found->Parent()->DeleteChild(found);
382 for (
const tinyxml2::XMLAttribute * attr =
ele_ptr_->FirstAttribute(); attr !=
nullptr; attr = attr->Next()) {
384 values[attr_name] = attr->Value();
393 std::vector<std::string> unknown_keys;
394 for (
const auto & [key, _] : port_values) {
397 if (!unknown_keys.empty()) {
398 throw exceptions::TreeDocumentError(
399 "Cannot set ports. According to the node model, the following ports are not implemented by '" +
400 std::string(
ele_ptr_->Name()) +
"': [ " + auto_apms_util::join(unknown_keys,
", ") +
" ].");
404 for (
const auto & [key, val] : port_values) {
405 ele_ptr_->SetAttribute(key.c_str(), val.c_str());
412 for (
const std::string & name : port_names_) {
413 PortValues::const_iterator it = port_default_values_.find(name);
414 if (it != port_default_values_.end()) {
415 ele_ptr_->SetAttribute(it->first.c_str(), it->second.c_str());
417 ele_ptr_->DeleteAttribute(name.c_str());
425 ele_ptr_->SetAttribute(BT::toStr(type).c_str(), script.
str().c_str());
431 ele_ptr_->SetAttribute(BT::toStr(type).c_str(), script.
str().c_str());
437 ele_ptr_->SetAttribute(NODE_INSTANCE_NAME_ATTRIBUTE_NAME, instance_name.c_str());
445 if (
const char * name =
ele_ptr_->Attribute(NODE_INSTANCE_NAME_ATTRIBUTE_NAME))
return name;
452 std::string instance_name =
getName();
453 if (registration_name == instance_name)
return registration_name;
454 return instance_name +
" (" + registration_name +
")";
464 std::vector<NodeElement> found;
465 deepApplyImpl(*
this, apply_callback, found);
471 std::vector<NodeElement> found;
472 deepApplyImpl(*
this, apply_callback, found);
477 const NodeElement * before_this, XMLElement * add_this)
480 XMLElement * prev =
nullptr;
481 XMLElement * curr = ele_ptr_->FirstChildElement();
487 if (curr == before_this->
ele_ptr_) {
492 curr = curr->NextSiblingElement();
496 ele_ptr_->InsertAfterChild(prev, add_this);
498 throw exceptions::TreeDocumentError(
500 getFullyQualifiedName() +
".");
503 ele_ptr_->InsertFirstChild(add_this);
506 ele_ptr_->InsertEndChild(add_this);
511void TreeDocument::NodeElement::deepApplyImpl(
512 const NodeElement & parent, ConstDeepApplyCallback apply_callback, std::vector<NodeElement> & vec)
514 for (XMLElement * child = parent.ele_ptr_->FirstChildElement(); child !=
nullptr;
515 child = child->NextSiblingElement()) {
516 const NodeElement child_ele(parent.doc_ptr_, child);
519 if (apply_callback(child_ele)) vec.push_back(child_ele);
522 deepApplyImpl(child_ele, apply_callback, vec);
526void TreeDocument::NodeElement::deepApplyImpl(
527 NodeElement & parent, DeepApplyCallback apply_callback, std::vector<NodeElement> & vec)
529 for (XMLElement * child = parent.ele_ptr_->FirstChildElement(); child !=
nullptr;
530 child = child->NextSiblingElement()) {
534 if (apply_callback(child_ele)) vec.push_back(child_ele);
537 deepApplyImpl(child_ele, apply_callback, vec);
544:
doc_ptr_(doc_ptr), current_(current)
555 if (current_) current_ = current_->NextSiblingElement();
566bool TreeDocument::NodeElement::ChildIterator::operator==(
const ChildIterator & other)
const
568 return current_ == other.current_;
571bool TreeDocument::NodeElement::ChildIterator::operator!=(
const ChildIterator & other)
const
573 return current_ != other.current_;
585 if (!ele_ptr->Attribute(TREE_NAME_ATTRIBUTE_NAME)) {
586 throw exceptions::TreeDocumentError(
587 "Cannot create tree element without a '" + std::string(TREE_NAME_ATTRIBUTE_NAME) +
"' attribute.");
593 const std::string other_tree_name = other.
getName();
597 throw exceptions::TreeDocumentError(
598 "Cannot copy tree '" + other.
getName() +
"' because another tree with this name already exists.");
606 ele_ptr_->SetAttribute(TREE_NAME_ATTRIBUTE_NAME, tree_name.c_str());
612 if (
const char * name =
ele_ptr_->Attribute(TREE_NAME_ATTRIBUTE_NAME))
return name;
627 const bool is_native_node =
doc_ptr_->native_node_names_.find(name) !=
doc_ptr_->native_node_names_.end();
628 if (!is_native_node && !m.
contains(name)) {
629 if (!
doc_ptr_->registered_nodes_manifest_.contains(name)) {
630 throw exceptions::NodeManifestError(
631 "Cannot assemble the required node manifest for tree '" +
getName() +
632 "' since there are no registration options for node '" + name +
"'.");
634 m.
add(name,
doc_ptr_->registered_nodes_manifest_[name]);
643 TreeDocument doc(
doc_ptr_->format_version_,
doc_ptr_->tree_node_loader_ptr_);
649 XMLDocument tree_doc;
650 tree_doc.InsertEndChild(
ele_ptr_->DeepClone(&tree_doc));
651 tinyxml2::XMLPrinter printer;
652 tree_doc.Print(&printer);
653 return printer.CStr();
657 const std::string & registration_name,
const std::string & instance_name,
bool deep_search)
672: XMLDocument(true, tinyxml2::PRESERVE_WHITESPACE),
674 native_node_names_(BT::BehaviorTreeFactory().builtinNodes()),
675 format_version_(format_version),
676 tree_node_loader_ptr_(tree_node_loader),
677 registered_nodes_manifest_(),
679 logger_(rclcpp::get_logger(LOGGER_NAME))
686 std::set<std::string> include_stack;
687 return mergeTreeDocumentImpl(other, adopt_root_tree, include_stack);
691 const XMLDocument & other,
bool adopt_root_tree, std::set<std::string> & include_stack)
693 const XMLElement * other_root = other.RootElement();
695 throw exceptions::TreeDocumentError(
"Cannot merge tree documents: other_root is nullptr.");
698 auto verify_tree_structure = [](
const XMLElement * tree_ele) {
699 const char * tree_id = tree_ele->Attribute(TREE_NAME_ATTRIBUTE_NAME);
701 throw exceptions::TreeDocumentError(
702 "Cannot merge tree document: Found a <" + std::string(TREE_ELEMENT_NAME) +
703 "> element that doesn't specify the required attribute '" + TREE_NAME_ATTRIBUTE_NAME +
"'.");
705 const XMLElement * tree_root_child = tree_ele->FirstChildElement();
706 if (!tree_root_child) {
707 throw exceptions::TreeDocumentError(
708 "Cannot merge tree document: Tree '" + std::string(tree_id) +
"' has no child nodes.");
710 if (tree_root_child->NextSibling()) {
711 throw exceptions::TreeDocumentError(
712 "Cannot merge tree document: Tree '" + std::string(tree_id) +
"' has more than one child node.");
717 auto check_duplicates = [
this](
const std::vector<std::string> & new_tree_names,
const std::string & source) {
719 if (!common.empty()) {
720 throw exceptions::TreeDocumentError(
721 "Cannot merge tree document: The following trees from " + source +
" are already defined: [ " +
722 auto_apms_util::join(common,
", ") +
" ].");
727 const std::vector<std::string> other_tree_names = getAllTreeNamesImpl(other);
729 if (strcmp(other_root->Name(), ROOT_ELEMENT_NAME) == 0) {
731 if (
const char * ver = other_root->Attribute(BTCPP_FORMAT_ATTRIBUTE_NAME)) {
732 if (std::string(ver) != format_version_) {
733 throw exceptions::TreeDocumentError(
734 "Cannot merge tree document: Format of other document (" + std::string(BTCPP_FORMAT_ATTRIBUTE_NAME) +
": " +
735 ver +
") is not compatible with this document (" + std::string(BTCPP_FORMAT_ATTRIBUTE_NAME) +
": " +
736 format_version_ +
").");
739 throw exceptions::TreeDocumentError(
740 "Cannot merge tree document: Root element of other document doesn't have required attribute '" +
741 std::string(BTCPP_FORMAT_ATTRIBUTE_NAME) +
"'.");
745 TreeDocument include_doc(format_version_, tree_node_loader_ptr_);
746 for (
const XMLElement * include_ele = other_root->FirstChildElement(INCLUDE_ELEMENT_NAME); include_ele !=
nullptr;
747 include_ele = include_ele->NextSiblingElement(INCLUDE_ELEMENT_NAME)) {
748 if (
const char * path = include_ele->Attribute(INCLUDE_PATH_ATTRIBUTE_NAME)) {
749 if (std::string(path).empty()) {
750 throw exceptions::TreeDocumentError(
751 "Cannot merge tree document: Found an <" + std::string(INCLUDE_ELEMENT_NAME) +
752 "> element that specifies an empty path in attribute '" + INCLUDE_PATH_ATTRIBUTE_NAME +
"'.");
754 std::string absolute_path = path;
755 if (
const char * ros_pkg = include_ele->Attribute(INCLUDE_ROS_PKG_ATTRIBUTE_NAME)) {
756 if (std::string(ros_pkg).empty()) {
757 throw exceptions::TreeDocumentError(
758 "Cannot merge tree document: Found an <" + std::string(INCLUDE_ELEMENT_NAME) +
759 "> element that specifies an empty ROS package name in attribute '" + INCLUDE_ROS_PKG_ATTRIBUTE_NAME +
762 if (std::filesystem::path(path).is_absolute()) {
763 throw exceptions::TreeDocumentError(
764 "Cannot merge tree document: Found an <" + std::string(INCLUDE_ELEMENT_NAME) +
765 "> element that specifies an absolute path '" + std::string(path) +
"' in attribute '" +
766 INCLUDE_PATH_ATTRIBUTE_NAME +
"' together with a ROS package name in attribute '" +
767 INCLUDE_ROS_PKG_ATTRIBUTE_NAME +
"'. Please remove the attribute " + INCLUDE_ROS_PKG_ATTRIBUTE_NAME +
768 " or use a relative path if you want to refer to a location "
769 "relative to the package's share directory.");
773 (std::filesystem::path(ament_index_cpp::get_package_share_directory(ros_pkg)) / path).
string();
774 }
catch (
const ament_index_cpp::PackageNotFoundError & e) {
775 throw exceptions::TreeDocumentError(
776 "Cannot merge tree document: Found an <" + std::string(INCLUDE_ELEMENT_NAME) +
777 "> element that specifies a non-existing ROS package '" + std::string(ros_pkg) +
"' in attribute '" +
778 INCLUDE_ROS_PKG_ATTRIBUTE_NAME +
"'.");
784 include_doc.mergeFileImpl(absolute_path,
false, include_stack);
785 }
catch (
const std::exception & e) {
786 throw exceptions::TreeDocumentError(
787 "Cannot merge tree document: Failed to include file '" + absolute_path +
"': " + e.what());
790 throw exceptions::TreeDocumentError(
791 "Cannot merge tree document: Found an <" + std::string(INCLUDE_ELEMENT_NAME) +
792 "> element that doesn't specify the required attribute '" + INCLUDE_PATH_ATTRIBUTE_NAME +
"'.");
797 check_duplicates(include_doc.getAllTreeNames(),
"included files");
800 for (
const XMLElement * child = include_doc.RootElement()->FirstChildElement(TREE_ELEMENT_NAME); child !=
nullptr;
801 child = child->NextSiblingElement(TREE_ELEMENT_NAME)) {
806 RootElement()->InsertEndChild(child->DeepClone(
this));
810 check_duplicates(other_tree_names,
"the merged document");
813 for (
const XMLElement * child = other_root->FirstChildElement(TREE_ELEMENT_NAME); child !=
nullptr;
814 child = child->NextSiblingElement(TREE_ELEMENT_NAME)) {
815 verify_tree_structure(child);
818 RootElement()->InsertEndChild(child->DeepClone(
this));
821 if (adopt_root_tree) {
824 if (
const char * name = other_root->Attribute(ROOT_TREE_ATTRIBUTE_NAME))
setRootTreeName(name);
826 }
else if (strcmp(other_root->Name(), TREE_ELEMENT_NAME) == 0) {
829 verify_tree_structure(other_root);
832 check_duplicates(other_tree_names,
"the merged document");
835 RootElement()->InsertEndChild(other_root->DeepClone(
this));
837 throw exceptions::TreeDocumentError(
838 "Cannot merge tree document: Root element of other document must either be <" + std::string(ROOT_ELEMENT_NAME) +
839 "> or <" + TREE_ELEMENT_NAME +
">.");
842 if (adopt_root_tree && other_tree_names.size() == 1) {
854 std::set<std::string> include_stack;
855 return mergeTreeDocumentImpl(
static_cast<const XMLDocument &
>(other), adopt_root_tree, include_stack);
860 XMLDocument other_doc;
861 if (other_doc.Parse(tree_str.c_str()) != tinyxml2::XMLError::XML_SUCCESS) {
862 throw exceptions::TreeDocumentError(
"Cannot merge tree document from string: " + std::string(other_doc.ErrorStr()));
864 std::set<std::string> include_stack;
865 return mergeTreeDocumentImpl(other_doc, adopt_root_tree, include_stack);
870 std::set<std::string> include_stack;
871 return mergeFileImpl(path, adopt_root_tree, include_stack);
875 const std::string & path,
bool adopt_root_tree, std::set<std::string> & include_stack)
878 std::string canonical_path;
880 canonical_path = std::filesystem::canonical(path).string();
881 }
catch (
const std::filesystem::filesystem_error &) {
884 canonical_path = path;
888 if (include_stack.count(canonical_path) > 0) {
889 throw exceptions::TreeDocumentError(
890 "Cannot merge tree document: Circular include detected for file '" + path +
"'.");
894 include_stack.insert(canonical_path);
896 XMLDocument other_doc;
897 if (other_doc.LoadFile(path.c_str()) != tinyxml2::XMLError::XML_SUCCESS) {
898 throw exceptions::TreeDocumentError(
"Cannot create tree document from file " + path +
": " + other_doc.ErrorStr());
901 mergeTreeDocumentImpl(other_doc, adopt_root_tree, include_stack);
904 include_stack.erase(canonical_path);
911 return mergeFile(resource.build_request_file_path_, adopt_root_tree);
916 XMLDocument tree_doc;
917 tree_doc.InsertEndChild(tree.
ele_ptr_->DeepClone(&tree_doc));
925 if (tree_name.empty()) {
926 throw exceptions::TreeDocumentError(
"Cannot create a new tree with an empty name");
929 throw exceptions::TreeDocumentError(
930 "Cannot create a new tree with name '" + tree_name +
"' because it already exists.");
932 TreeDocument::XMLElement * new_ele = RootElement()->InsertNewChildElement(TreeDocument::TREE_ELEMENT_NAME);
933 new_ele->SetAttribute(TreeDocument::TREE_NAME_ATTRIBUTE_NAME, tree_name.c_str());
944 std::string name(tree_name);
948 }
else if (
const std::vector<std::string> names = other.
getAllTreeNames(); names.size() == 1) {
951 throw exceptions::TreeDocumentError(
952 "Failed to create new tree element from another document because argument tree_name was omitted and it was not "
953 "possible to determine the root tree automatically.");
963 TreeDocument new_doc(format_version_, tree_node_loader_ptr_);
970 TreeDocument new_doc(format_version_, tree_node_loader_ptr_);
976 const TreeResource & resource,
const std::string & tree_name)
978 TreeDocument new_doc(format_version_, tree_node_loader_ptr_);
979 new_doc.
mergeFile(resource.build_request_file_path_);
993 return TreeElement(
this, getXMLElementForTreeWithName(tree_name));
998 if (tree_name.empty()) {
999 throw exceptions::TreeDocumentError(
"Cannot set root tree name with empty string.");
1002 throw exceptions::TreeDocumentError(
1003 "Cannot make tree with name '" + tree_name +
"' the root tree because it doesn't exist.");
1005 RootElement()->SetAttribute(ROOT_TREE_ATTRIBUTE_NAME, tree_name.c_str());
1011 if (RootElement()->Attribute(ROOT_TREE_ATTRIBUTE_NAME))
return true;
1017 if (
const auto tree_name = RootElement()->Attribute(ROOT_TREE_ATTRIBUTE_NAME))
return tree_name;
1018 throw exceptions::TreeDocumentError(
1019 "Cannot get root tree name because the document's root element has no attribute '" +
1020 std::string(ROOT_TREE_ATTRIBUTE_NAME) +
"'.");
1027 RootElement()->DeleteChild(getXMLElementForTreeWithName(tree_name));
1038 std::set<std::string> old_node_names;
1039 for (
const auto & [name, _] : registered_nodes_manifest_.map()) {
1040 old_node_names.insert(name);
1047 auto rename_recursive = [&old_node_names, &node_namespace, &sep](XMLElement * parent,
auto & self) ->
void {
1048 for (XMLElement * child = parent->FirstChildElement(); child !=
nullptr; child = child->NextSiblingElement()) {
1049 const char * name = child->Name();
1051 if (name && old_node_names.count(name) > 0) {
1052 child->SetName((node_namespace + sep + name).c_str());
1059 for (XMLElement * tree_ele = RootElement()->FirstChildElement(TREE_ELEMENT_NAME); tree_ele !=
nullptr;
1060 tree_ele = tree_ele->NextSiblingElement(TREE_ELEMENT_NAME)) {
1061 rename_recursive(tree_ele, rename_recursive);
1068 std::vector<std::pair<std::string, std::pair<BT::TreeNodeManifest, BT::NodeBuilder>>> registrations_to_update;
1069 const auto & factory_manifests = factory_.manifests();
1070 const auto & factory_builders = factory_.builders();
1071 for (
const auto & old_name : old_node_names) {
1072 const auto manifest_it = factory_manifests.find(old_name);
1073 const auto builder_it = factory_builders.find(old_name);
1074 if (manifest_it == factory_manifests.end() || builder_it == factory_builders.end()) {
1076 throw exceptions::TreeDocumentError(
1077 "Internal error while applying node namespace: Node '" + old_name +
1078 "' not found in BehaviorTreeFactory registrations.");
1081 registrations_to_update.emplace_back(old_name, std::make_pair(manifest_it->second, builder_it->second));
1085 for (
const auto & [old_name, manifest_builder_pair] : registrations_to_update) {
1086 factory_.unregisterBuilder(old_name);
1087 BT::TreeNodeManifest new_manifest = manifest_builder_pair.first;
1088 new_manifest.registration_ID = node_namespace + sep + old_name;
1089 factory_.registerBuilder(new_manifest, manifest_builder_pair.second);
1098 std::set<std::string> all_registration_names;
1099 for (
const auto & [name, _] : tree_node_manifest.
map()) all_registration_names.insert(name);
1100 if (
const std::set<std::string> common =
1103 throw exceptions::TreeDocumentError(
1104 "Found reserved node registration names in the node manifest. The following names are not allowed, because "
1105 "they refer to native behavior tree nodes: [ " +
1106 auto_apms_util::join(std::vector<std::string>(common.begin(), common.end()),
", ") +
" ].");
1109 for (
const auto & [node_name, params] : tree_node_manifest.
map()) {
1114 if (registered_nodes_manifest_.contains(node_name)) {
1117 factory_.unregisterBuilder(node_name);
1118 registered_nodes_manifest_.remove(node_name);
1121 throw exceptions::TreeDocumentError(
1122 "Tried to register node '" + node_name +
"' (Class: " + params.class_name +
1123 ") which is already known. You must make sure that the registration names are unique or explicitly allow "
1124 "overriding previously registered nodes with the same name by setting override=true.");
1129 if (!tree_node_loader_ptr_->isClassAvailable(params.class_name)) {
1130 if (all_node_classes_package_map_.find(params.class_name) == all_node_classes_package_map_.end()) {
1131 throw exceptions::TreeDocumentError(
1132 "Node '" + node_name +
" (" + params.class_name +
1133 ")' cannot be registered, because the class name is not known to the class loader. "
1134 "Make sure that it's spelled correctly and registered by calling "
1135 "auto_apms_behavior_tree_register_nodes() in the CMakeLists.txt of the "
1136 "corresponding package.");
1138 throw exceptions::TreeDocumentError(
1139 "Node '" + node_name +
" (" + params.class_name +
1140 ")' cannot be registered, because the corresponding resource belongs to excluded package '" +
1141 all_node_classes_package_map_.at(params.class_name) +
"'.");
1144 pluginlib::UniquePtr<NodeRegistrationInterface> plugin_instance;
1146 plugin_instance = tree_node_loader_ptr_->createUniqueInstance(params.class_name);
1147 }
catch (
const pluginlib::CreateClassException & e) {
1148 throw pluginlib::CreateClassException(
1149 "Failed to create an instance of node '" + node_name +
" (" + params.class_name +
1150 ")'. Remember that the AUTO_APMS_BEHAVIOR_TREE_REGISTER_NODE "
1151 "macro must be called in the source file for the node class to be discoverable. "
1157 if (plugin_instance->requiresRosNodeContext()) {
1158 if (only_non_ros_nodes_) {
1159 throw exceptions::NodeRegistrationError(
1160 "Node '" + node_name +
1161 "' relies on ROS 2 functionality but this instance only allows to use non-ROS nodes.");
1164 ros_node_wptr_.lock(), tree_node_waitables_callback_group_wptr_.lock(),
1165 tree_node_waitables_executor_wptr_.lock(), params);
1166 plugin_instance->registerWithBehaviorTreeFactory(factory_, node_name, &ros_node_context);
1169 RosNodeContext ros_node_context(
nullptr,
nullptr,
nullptr, params);
1170 plugin_instance->registerWithBehaviorTreeFactory(factory_, node_name, &ros_node_context);
1172 }
catch (
const std::exception & e) {
1173 throw exceptions::NodeRegistrationError(
1174 "Cannot register node '" + node_name +
" (" + params.class_name +
")': " + e.what() +
".");
1176 registered_nodes_manifest_.add(node_name, params);
1183 std::set<std::string> names;
1184 if (include_native) names = native_node_names_;
1185 for (
const auto & [name, _] : registered_nodes_manifest_.map()) names.insert(name);
1194 XMLElement * ptr =
const_cast<XMLElement *
>(getXMLElementForTreeWithName(tree_name));
1204 tinyxml2::XMLElement * model_root = RootElement()->FirstChildElement(TREE_NODE_MODEL_ELEMENT_NAME);
1208 model_root = NewElement(TREE_NODE_MODEL_ELEMENT_NAME);
1209 RootElement()->InsertEndChild(model_root);
1213 for (
const auto & [node_name,
model] : model_map) {
1215 tinyxml2::XMLElement * node_element = NewElement(BT::toStr(
model.type).c_str());
1216 node_element->SetAttribute(
"ID", node_name.c_str());
1219 for (
const auto & port_info :
model.port_infos) {
1220 tinyxml2::XMLElement * port_element =
nullptr;
1223 switch (port_info.port_direction) {
1224 case BT::PortDirection::INPUT:
1225 port_element = NewElement(
"input_port");
1227 case BT::PortDirection::OUTPUT:
1228 port_element = NewElement(
"output_port");
1230 case BT::PortDirection::INOUT:
1231 port_element = NewElement(
"inout_port");
1236 port_element->SetAttribute(
"name", port_info.port_name.c_str());
1238 if (!port_info.port_type.empty()) {
1239 port_element->SetAttribute(
"type", port_info.port_type.c_str());
1242 if (port_info.port_has_default) {
1243 port_element->SetAttribute(
"default", port_info.port_default.c_str());
1247 if (!port_info.port_description.empty()) {
1248 port_element->SetText(port_info.port_description.c_str());
1251 node_element->InsertEndChild(port_element);
1254 model_root->InsertEndChild(node_element);
1262 const tinyxml2::XMLElement * root = doc.RootElement();
1264 throw exceptions::TreeDocumentError(
"Node model document has no root element.");
1266 if (
const char * ver = root->Attribute(TreeDocument::BTCPP_FORMAT_ATTRIBUTE_NAME)) {
1267 const std::string expected_format = TreeDocument::BTCPP_FORMAT_DEFAULT_VERSION;
1268 if (std::string(ver) != expected_format) {
1269 throw exceptions::TreeDocumentError(
1270 "Cannot parse node model document: Format of model document (" +
1271 std::string(TreeDocument::BTCPP_FORMAT_ATTRIBUTE_NAME) +
": " + ver +
1272 ") doesn't comply with the expected format (" + std::string(TreeDocument::BTCPP_FORMAT_ATTRIBUTE_NAME) +
": " +
1273 expected_format +
").");
1276 throw exceptions::TreeDocumentError(
1277 "Cannot parse node model document: Root element of model document doesn't have required attribute '" +
1278 std::string(TreeDocument::BTCPP_FORMAT_ATTRIBUTE_NAME) +
"'.");
1280 tinyxml2::XMLElement * model_ele = doc.RootElement()->FirstChildElement(TreeDocument::TREE_NODE_MODEL_ELEMENT_NAME);
1282 throw exceptions::TreeDocumentError(
1283 "Element <" + std::string(TreeDocument::TREE_NODE_MODEL_ELEMENT_NAME) +
1284 "> doesn't exist in node model document.");
1288 for (tinyxml2::XMLElement * ele = model_ele->FirstChildElement(); ele !=
nullptr; ele = ele->NextSiblingElement()) {
1289 const char * node_name = ele->Attribute(
"ID");
1291 throw exceptions::TreeDocumentError(
1292 "Element '" + std::string(ele->Name()) +
"' in node model document is missing the required attribute 'ID'");
1295 model.type = BT::convertFromString<BT::NodeType>(ele->Name());
1296 for (
const tinyxml2::XMLElement * port_ele = ele->FirstChildElement(); port_ele !=
nullptr;
1297 port_ele = port_ele->NextSiblingElement()) {
1298 const std::string direction = port_ele->Name();
1300 if (direction ==
"input_port") {
1302 }
else if (direction ==
"output_port") {
1304 }
else if (direction ==
"inout_port") {
1307 throw exceptions::TreeDocumentError(
1308 "Unknown port direction in node model for '" + std::string(node_name) +
"': " + direction);
1310 if (
const char * c = port_ele->Attribute(
"name")) {
1313 if (
const char * c = port_ele->Attribute(
"type")) {
1316 if (
const char * c = port_ele->Attribute(
"default")) {
1322 if (
const char * c = port_ele->GetText()) {
1325 model.port_infos.push_back(std::move(port_info));
1331 std::map<std::string, std::vector<std::string>> hidden_ports;
1332 for (
const auto & [node_name, registration_options] : manifest.
map()) {
1334 for (
const std::string & port_name : registration_options.hidden_ports) {
1335 hidden_ports[node_name].push_back(port_name);
1339 for (
const auto & [port_name, _] : registration_options.port_alias) {
1340 hidden_ports[node_name].push_back(port_name);
1345 for (
const auto & [node_name, ports_to_hide] : hidden_ports) {
1346 auto it = model_map.find(node_name);
1347 if (it == model_map.end()) {
1351 model.port_infos.erase(
1353 model.port_infos.begin(),
model.port_infos.end(),
1355 return std::find(ports_to_hide.begin(), ports_to_hide.end(), port_info.port_name) != ports_to_hide.end();
1357 model.port_infos.end());
1366 std::string model_xml;
1368 model_xml = BT::writeTreeNodesModelXML(factory_, include_native);
1369 }
catch (
const std::exception & e) {
1370 throw exceptions::TreeDocumentError(
"Error generating node model XML from factory: " + std::string(e.what()));
1374 tinyxml2::XMLDocument model_doc;
1375 if (model_doc.Parse(model_xml.c_str()) != tinyxml2::XMLError::XML_SUCCESS) {
1376 throw exceptions::TreeDocumentError(
1377 "Error parsing the model of the currently registered nodes: " + std::string(model_doc.ErrorStr()));
1381 return getNodeModel(model_doc, registered_nodes_manifest_);
1387 std::unordered_map<std::string, BT::NodeType> registered_nodes;
1388 for (
const auto & [node_name,
model] : model_map) {
1389 registered_nodes[node_name] =
model.type;
1393 }
catch (
const BT::RuntimeError & e) {
1394 return nonstd::make_unexpected(e.what());
1401 tinyxml2::XMLPrinter printer;
1403 return printer.CStr();
1410 tinyxml2::XMLError result = doc.SaveFile(path.c_str());
1411 if (result != tinyxml2::XML_SUCCESS) {
1412 throw exceptions::TreeDocumentError(
1413 "Failed to write tree document to file. Error ID: " + std::string(doc.ErrorIDToName(result)));
1421 tinyxml2::XMLElement * root_ele = NewElement(TreeDocument::ROOT_ELEMENT_NAME);
1422 root_ele->SetAttribute(TreeDocument::BTCPP_FORMAT_ATTRIBUTE_NAME, format_version_.c_str());
1423 InsertFirstChild(root_ele);
1427 factory_.unregisterBuilder(node_name);
1433const TreeDocument::XMLElement * TreeDocument::getXMLElementForTreeWithName(
const std::string & tree_name)
const
1435 return getXMLElementForTreeWithNameImpl<const XMLElement *>(*
this, tree_name);
1438TreeDocument::XMLElement * TreeDocument::getXMLElementForTreeWithName(
const std::string & tree_name)
1440 return getXMLElementForTreeWithNameImpl<XMLElement *>(*
this, tree_name);
1443template <
typename ReturnT,
typename DocumentT>
1444ReturnT TreeDocument::getXMLElementForTreeWithNameImpl(DocumentT & doc,
const std::string & tree_name)
1446 if (tree_name.empty()) {
1447 throw exceptions::TreeDocumentError(
"Cannot get tree with an empty name.");
1449 if (!doc.hasTreeName(tree_name)) {
1450 throw exceptions::TreeDocumentError(
"Cannot get tree with name '" + tree_name +
"' because it doesn't exist.");
1452 auto child = doc.RootElement()->FirstChildElement(TreeDocument::TREE_ELEMENT_NAME);
1453 while (child && !child->Attribute(TreeDocument::TREE_NAME_ATTRIBUTE_NAME, tree_name.c_str())) {
1454 child = child->NextSiblingElement();
1457 throw std::logic_error(
1458 "Unexpected error trying to get tree element with name '" + tree_name +
1459 "'. Since hasTreeName() returned true, there MUST be a corresponding element.");
const NodeManifest & getNodeManifest() const
Get the node manifest associated with this resource.
Data structure for information about which behavior tree node plugin to load and how to configure the...
NodeManifest & add(const std::string &node_name, const RegistrationOptions &opt)
Add registration options for a behavior tree node to the manifest.
bool contains(const std::string &node_name) const
Determine if a behavior tree node has been added to the manifest.
NodeManifest & merge(const NodeManifest &other, bool replace=false)
Merges another NodeManifest with this one.
const Map & map() const
Get a view of the internal map.
A pluginlib::ClassLoader specifically for loading installed behavior tree node plugins.
Additional parameters specific to ROS 2 determined at runtime by TreeBuilder.
Class that encapsulates behavior tree script expressions.
std::string str() const
Concatenate all expressions of this instance to a single string.
Forward iterator for traversing the first-level children of a node.
ChildIterator & operator++()
Pre-increment: advance to the next sibling element.
ChildIterator()
Create a past-the-end iterator.
value_type operator*() const
Dereference the iterator to obtain a NodeElement handle for the current child.
Handle for a single node of a TreeDocument.
const TreeDocument & getParentDocument() const
Get a const view of this node's parent tree document.
NodeElement insertTreeFromResource(const TreeResource &resource, const std::string &tree_name, const NodeElement *before_this=nullptr)
Concatenate a tree from one of the installed package's behavior tree resources and add its first chil...
const std::vector< std::string > & getPortNames() const
Get the names of the data ports implemented by the node represented by this element.
virtual std::string getRegistrationName() const
Get the name of this node given during registration representing its dynamic type.
XMLElement * getXMLElement()
Get a pointer to the underlying tinyxml2::XMLElement of this node.
std::function< bool(const NodeElement &)> ConstDeepApplyCallback
Callback invoked for every node found under another node. It cannot modify the current node.
NodeElement & resetPorts()
Delete all currently specified port values and reset with the defaults.
std::function< bool(NodeElement &)> DeepApplyCallback
Callback invoked for every node found under another node. It may modify the current node.
NodeElement & setPorts(const PortValues &port_values)
Populate the the node's data ports.
virtual NodeElement & setName(const std::string &instance_name)
Assign a name for this specific node instance.
NodeElement(TreeDocument *doc_ptr, XMLElement *ele_ptr)
Protected constructor intended for internal use only.
std::map< std::string, std::string > PortValues
Mapping of port names and its respective value encoded as string.
const std::vector< NodeElement > deepApplyConst(ConstDeepApplyCallback apply_callback) const
Recursively apply a callback to this node's children.
TreeDocument * doc_ptr_
Pointer to the tree document that created the tree this node belongs to.
ChildIterator end() const
Get a past-the-end iterator for the children of this node.
virtual std::string getName() const
Get the name of this node given to this specific instance by the developer.
PortValues getPorts() const
Assemble the values given to each data port implemented by this node.
NodeElement & setConditionalScript(BT::PreCond type, const Script &script)
Specify a script that is evaluated before this node is ticked.
NodeElement insertTreeFromString(const std::string &tree_str, const std::string &tree_name, const NodeElement *before_this=nullptr)
Concatenate a tree from a document created from a string and add its first child node to the children...
tinyxml2::XMLElement * ele_ptr_
Pointer to the corresponding XMLElement of the base document.
NodeElement insertTree(const TreeElement &tree, const NodeElement *before_this=nullptr)
Concatenate a tree and add its first child node to the children of this node.
std::string getFullyQualifiedName() const
Create a string that uniquely identifies this node considering its registration and its instance name...
NodeElement insertTreeFromDocument(const TreeDocument &doc, const std::string &tree_name, const NodeElement *before_this=nullptr)
Concatenate a tree from a document and add its first child node to the children of this node.
model::SubTree insertSubTreeNode(const std::string &tree_name, const NodeElement *before_this=nullptr)
Add a subtree node for a specific tree element to the children of this node.
bool hasChildren() const
Determine whether any children have been given to this node.
bool operator==(const NodeElement &other) const
Determine if two node elements refer to the same node.
NodeElement getFirstNode(const std::string ®istration_name="", const std::string &instance_name="", bool deep_search=false) const
Get the first node with a particular registration and instance name.
ChildIterator begin() const
Get an iterator to the first child of this node.
NodeElement insertNode(const std::string &name, const NodeElement *before_this=nullptr)
Add a new node to the children of this node.
NodeElement & removeFirstChild(const std::string ®istration_name="", const std::string &instance_name="", bool deep_search=false)
Remove the first node with a particular registration and instance name.
NodeElement insertTreeFromFile(const std::string &path, const std::string &tree_name, const NodeElement *before_this=nullptr)
Concatenate a tree from a document created from a file and add its first child node to the children o...
NodeElement & removeChildren()
Recursively remove all children of this node element.
NodeElement & operator=(const NodeElement &other)
Replace this node with another.
std::vector< NodeElement > deepApply(DeepApplyCallback apply_callback)
Recursively apply a callback to this node's children.
Handle for a single behavior tree of a TreeDocument.
TreeElement & removeFirstChild(const std::string ®istration_name="", const std::string &instance_name="", bool deep_search=false)
TreeElement & removeChildren()
TreeElement(TreeDocument *doc_ptr, XMLElement *ele_ptr)
Protected constructor intended to be used only by certain factory methods of TreeDocument.
std::string getName() const override
Get the name of the behavior tree.
BT::Result verify() const
Verify that this behavior tree is structured correctly and can be created successfully.
NodeManifest getRequiredNodeManifest() const
Assemble the node manifest that is required for successfully creating an instance of this tree.
TreeElement & makeRoot()
Set this behavior tree as the root tree of the parent document.
TreeElement & operator=(const TreeElement &other)
Replace the behavior tree represented by this element with another.
TreeElement & setName(const std::string &tree_name) override
Set the name of the behavior tree.
std::string writeToString() const
Write this behavior tree to an XML encoded in a string.
Document Object Model (DOM) for the behavior tree XML schema. This class offers a programmatic approa...
TreeElement getRootTree()
Get the corresponding behavior tree element for the root tree of this document.
TreeDocument & mergeResource(const TreeResource &resource, bool adopt_root_tree=false)
Merge the behavior trees from one of the installed package's behavior tree resources.
std::string getRootTreeName() const
Get the name of this document's root tree.
TreeDocument & mergeTreeDocument(const XMLDocument &other, bool adopt_root_tree=false)
Merge another tree document with this one.
TreeDocument & mergeString(const std::string &tree_str, bool adopt_root_tree=false)
Create a tree document from a string and merge it with this one.
TreeElement newTreeFromDocument(const TreeDocument &other, const std::string &tree_name="")
Create a new behavior tree inside this document with the content of one found inside another tree doc...
void writeToFile(const std::string &path) const
Write the XML of this tree document to a file.
TreeElement newTreeFromFile(const std::string &path, const std::string &tree_name="")
Create a new behavior tree inside this document with the content of one found inside the XML file.
TreeDocument & applyNodeNamespace(const std::string &node_namespace, const std::string &sep=_AUTO_APMS_BEHAVIOR_TREE_CORE__NODE_NAMESPACE_DEFAULT_SEP)
Prepend a namespace to all nodes associated with this document.
BT::Result verify() const
Verify that all behavior trees of this document are structured correctly and can be created successfu...
NodeManifest getRequiredNodeManifest() const
Assemble the node manifest that is required for successfully creating an instance of any of the docum...
std::set< std::string > getRegisteredNodeNames(bool include_native=true) const
Get the names of all nodes that are known to this document.
TreeElement newTreeFromString(const std::string &tree_str, const std::string &tree_name="")
Create a new behavior tree inside this document with the content of one found inside the XML string.
TreeElement newTreeFromResource(const TreeResource &resource, const std::string &tree_name="")
Create a new behavior tree inside this document with the content of one the trees found inside a part...
TreeDocument & reset()
Clear this document and reset it to its initial state.
TreeElement getTree(const std::string &tree_name)
Get the corresponding behavior tree element for a tree inside this document.
virtual TreeDocument & registerNodes(const NodeManifest &tree_node_manifest, bool override=false)
Load behavior tree node plugins and register them with the internal behavior tree factory.
TreeDocument & setRootTreeName(const std::string &tree_name)
Define the root tree of this document.
bool hasTreeName(const std::string &tree_name) const
Determine if this document contains a behavior tree with a particular name.
TreeDocument & mergeFile(const std::string &path, bool adopt_root_tree=false)
Create a tree document from a file and merge it with this one.
std::vector< std::string > getAllTreeNames() const
Get the names of all behavior trees inside this document.
TreeDocument & addNodeModel(NodeModelMap model_map)
Add a behavior tree node model element to the document by parsing the contents of model_map.
TreeDocument & mergeTree(const TreeElement &tree, bool make_root_tree=false)
Merge an existing behavior tree with this tree document.
std::string writeToString() const
Write the XML of this tree document to a string.
bool hasRootTreeName() const
Determine if this document specifies which of its trees is the root tree.
TreeDocument & removeTree(const std::string &tree_name)
Remove a particular behavior tree from this document.
TreeDocument(const std::string &format_version=BTCPP_FORMAT_DEFAULT_VERSION, NodeRegistrationLoader::SharedPtr tree_node_loader=NodeRegistrationLoader::make_shared())
Create a an empty tree document.
static NodeModelMap getNodeModel(tinyxml2::XMLDocument &doc, const NodeManifest &manifest)
Convert a behavior tree node model document to the corresponding data structure.
TreeElement newTree(const std::string &tree_name)
Create a new behavior tree inside this document.
Class containing behavior tree resource data.
std::string getRootTreeName() const
Get the name of the root tree of this behavior tree resource.
bool hasRootTreeName() const
Determine if this behavior tree resource specifies a root tree.
Subtree behavior tree node model.
bool contains(const ContainerT< ValueT, AllocatorT > &c, const ValueT &val)
Check whether a particular container structure contains a value.
std::set< KeyT, CompareT, AllocatorT > getCommonElements(std::set< KeyT, CompareT, AllocatorT > c1, std::set< KeyT, CompareT, AllocatorT > c2)
Assemble common elements of two sets.
Core API for AutoAPMS's behavior tree implementation.
Models for all available behavior tree nodes.
Powerful tooling for incorporating behavior trees for task development.
std::map< std::string, NodeModel > NodeModelMap
Mapping of node registration names and their implementation details.
Data structure encapsulating the information of all ports implemented by a behavior tree node.
Implementation details of a single data port.
std::string port_default
Default value of the port encoded as string.
bool port_has_default
Flag whether the port implements a default value or not.
std::string port_description
Description of the port.
BT::PortDirection port_direction
Direction of the port.
std::string port_type
String representation of the C++ type given to the port.
std::string port_name
Name of the port.
Parameters for loading and registering a behavior tree node class from a shared library using e....