16 #if defined(__linux) || defined(__linux__) 17 #pragma GCC diagnostic push 18 #pragma GCC diagnostic ignored "-Wattributes" 22 #pragma warning(disable : 4996) // do not complain about sprintf 35 #include <ament_index_cpp/get_package_share_directory.hpp> 45 auto StrEqual = [](
const char* str1,
const char* str2) ->
bool {
46 return strcmp(str1, str2) == 0;
55 void recursivelyCreateTree(
const std::string& tree_ID,
Tree& output_tree,
59 void getPortsRecursively(
const XMLElement* element,
60 std::vector<std::string>& output_ports);
74 factory(fact), current_path(filesystem::
path::getcwd()), suffix_count(0)
80 current_path = filesystem::path::getcwd();
81 opened_documents.clear();
86 #if defined(__linux) || defined(__linux__) 87 #pragma GCC diagnostic pop 105 filesystem::path file_path(filename);
116 doc->
Parse(xml_text.c_str(), xml_text.size());
123 std::vector<std::string> out;
126 out.push_back(it.first);
136 sprintf(buffer,
"Error parsing the XML: %s", doc->
ErrorName());
152 filesystem::path file_path(include_node->Attribute(
"path"));
153 const char* ros_pkg_relative_path = include_node->
Attribute(
"ros_pkg");
155 if (ros_pkg_relative_path)
157 if (file_path.is_absolute())
159 std::cout <<
"WARNING: <include path=\"...\"> contains an absolute " 161 <<
"Attribute [ros_pkg] will be ignored." << std::endl;
165 std::string ros_pkg_path;
168 file_path = filesystem::path(ros_pkg_path) / file_path;
169 #elif defined USING_ROS2 171 ament_index_cpp::get_package_share_directory(ros_pkg_relative_path);
172 file_path = filesystem::path(ros_pkg_path) / file_path;
174 throw RuntimeError(
"Using attribute [ros_pkg] in <include>, but this " 177 "without ROS support. Recompile the BehaviorTree.CPP " 184 if (!file_path.is_absolute())
186 file_path = current_path / file_path;
193 const filesystem::path previous_path = current_path;
194 current_path = file_path.parent_path().make_absolute();
196 next_doc->
LoadFile(file_path.str().c_str());
197 loadDocImpl(next_doc, add_includes);
200 current_path = previous_path;
204 std::unordered_map<std::string, BT::NodeType> registered_nodes;
205 for (
const auto& it : factory.manifests())
207 registered_nodes.insert({it.first, it.second.type});
211 doc->
Print(&printer);
218 for (
auto bt_node = xml_root->
FirstChildElement(
"BehaviorTree"); bt_node !=
nullptr;
221 std::string tree_name;
222 if (bt_node->Attribute(
"ID"))
224 tree_name = bt_node->Attribute(
"ID");
231 tree_roots.insert({tree_name, bt_node});
236 const std::unordered_map<std::string, BT::NodeType>& registered_nodes)
239 auto xml_error = doc.
Parse(xml_text.c_str(), xml_text.size());
243 sprintf(buffer,
"Error parsing the XML: %s", doc.
ErrorName());
248 auto ThrowError = [&](
int line_num,
const std::string& text) {
250 sprintf(buffer,
"Error at line %d: -> %s", line_num, text.c_str());
254 auto ChildrenCount = [](
const XMLElement* parent_node) {
256 for (
auto node = parent_node->FirstChildElement(); node !=
nullptr;
257 node = node->NextSiblingElement())
269 throw RuntimeError(
"The XML must have a root node called <root>");
278 ThrowError(meta_sibling->GetLineNum(),
" Only a single node <TreeNodesModel> is " 288 const char* name = node->
Name();
293 const char* ID = node->Attribute(
"ID");
296 ThrowError(node->GetLineNum(),
"Error at line %d: -> The attribute " 306 std::function<void(const XMLElement*)> recursiveStep;
309 const int children_count = ChildrenCount(node);
310 const char* name = node->Name();
313 if (children_count != 1)
315 ThrowError(node->GetLineNum(),
"The node <Decorator> must have exactly 1 " 318 if (!node->Attribute(
"ID"))
320 ThrowError(node->GetLineNum(),
"The node <Decorator> must have the " 326 if (children_count != 0)
328 ThrowError(node->GetLineNum(),
"The node <Action> must not have any " 331 if (!node->Attribute(
"ID"))
333 ThrowError(node->GetLineNum(),
"The node <Action> must have the " 337 else if (
StrEqual(name,
"Condition"))
339 if (children_count != 0)
341 ThrowError(node->GetLineNum(),
"The node <Condition> must not have any " 344 if (!node->Attribute(
"ID"))
346 ThrowError(node->GetLineNum(),
"The node <Condition> must have the " 352 if (children_count == 0)
354 ThrowError(node->GetLineNum(),
"The node <Control> must have at least 1 " 357 if (!node->Attribute(
"ID"))
359 ThrowError(node->GetLineNum(),
"The node <Control> must have the " 366 if (children_count == 0)
368 ThrowError(node->GetLineNum(),
"A Control node must have at least 1 " 374 auto child = node->FirstChildElement();
378 if (
StrEqual(child->Name(),
"remap"))
380 ThrowError(node->GetLineNum(),
"<remap> was deprecated");
384 ThrowError(node->GetLineNum(),
"<SubTree> should not have any child");
388 if (!node->Attribute(
"ID"))
390 ThrowError(node->GetLineNum(),
"The node <SubTree> must have the " 394 else if (
StrEqual(name,
"BehaviorTree"))
396 if (children_count != 1)
398 ThrowError(node->GetLineNum(),
"The node <BehaviorTree> must have exactly 1 " 405 const auto search = registered_nodes.find(name);
406 bool found = (search != registered_nodes.end());
409 ThrowError(node->GetLineNum(), std::string(
"Node not recognized: ") + name);
414 if (children_count != 1)
416 ThrowError(node->GetLineNum(),
417 std::string(
"The node <") + name +
"> must have exactly 1 child");
422 if (
StrEqual(name,
"SubTree") ==
false)
424 for (
auto child = node->FirstChildElement(); child !=
nullptr;
425 child = child->NextSiblingElement())
427 recursiveStep(child);
432 for (
auto bt_root = xml_root->
FirstChildElement(
"BehaviorTree"); bt_root !=
nullptr;
435 recursiveStep(bt_root);
440 std::string main_tree_to_execute)
443 std::string main_tree_ID = main_tree_to_execute;
447 if (main_tree_ID.empty())
451 if (
auto main_tree_attribute = first_xml_root->
Attribute(
"main_tree_to_execute"))
453 main_tree_ID = main_tree_attribute;
462 throw RuntimeError(
"[main_tree_to_execute] was not specified correctly");
467 if (!root_blackboard)
469 throw RuntimeError(
"XMLParser::instantiateTree needs a non-empty " 489 const std::string element_name = element->
Name();
491 std::string instance_name;
494 if (element_name ==
"Action" || element_name ==
"Decorator" ||
495 element_name ==
"Condition" || element_name ==
"Control")
504 const char* attr_alias = element->
Attribute(
"name");
507 instance_name = attr_alias;
516 if (element_name ==
"SubTree" || element_name ==
"SubTreePlus")
518 instance_name = element->
Attribute(
"ID");
525 const std::string attribute_name = att->Name();
528 port_remap[attribute_name] = att->Value();
538 if (factory.builders().count(ID) != 0)
540 const auto&
manifest = factory.manifests().at(ID);
543 for (
const auto& remap_it : port_remap)
545 if (
manifest.ports.count(remap_it.first) == 0)
547 throw RuntimeError(
"Possible typo? In the XML, you tried to remap port " 549 remap_it.first,
"\" in node [", ID,
" / ", instance_name,
550 "], but the manifest of this node does not contain a " 557 for (
const auto& port_it :
manifest.ports)
559 const std::string& port_name = port_it.first;
560 const auto& port_info = port_it.second;
562 auto remap_it = port_remap.find(port_name);
563 if (remap_it == port_remap.end())
571 const auto port_key =
static_cast<std::string
>(param_res.value());
573 auto prev_info = blackboard->portInfo(port_key);
577 blackboard->createEntry(port_key, port_info);
582 if (prev_info->type() &&
584 *prev_info->type() != *port_info.type())
586 blackboard->debugMessage();
588 throw RuntimeError(
"The creation of the tree failed because the port [",
589 port_key,
"] was initially created with type [",
590 demangle(prev_info->type()),
"] and, later type [",
591 demangle(port_info.type()),
"] was used somewhere else.");
598 for (
const auto& remap_it : port_remap)
600 const auto& port_name = remap_it.first;
601 auto port_it =
manifest.ports.find(port_name);
602 if (port_it !=
manifest.ports.end())
604 auto direction = port_it->second.direction();
617 for (
const auto& port_it :
manifest.ports)
619 const std::string& port_name = port_it.first;
620 const PortInfo& port_info = port_it.second;
631 child_node = factory.instantiateTreeNode(instance_name, ID, config);
633 else if (tree_roots.count(ID) != 0)
635 child_node = std::make_unique<SubtreeNode>(instance_name);
639 throw RuntimeError(ID,
" is not a registered node, nor a Subtree");
644 if (
auto control_parent = dynamic_cast<ControlNode*>(node_parent.get()))
646 control_parent->addChild(child_node.get());
648 if (
auto decorator_parent = dynamic_cast<DecoratorNode*>(node_parent.get()))
650 decorator_parent->setChild(child_node.get());
661 std::function<void(const TreeNode::Ptr&, const XMLElement*)> recursiveStep;
665 auto node = createNodeFromXML(element, blackboard, parent);
666 output_tree.
nodes.push_back(node);
670 if (dynamic_cast<const SubtreeNode*>(node.get()))
672 bool is_isolated =
true;
674 for (
const XMLAttribute* attr = element->FirstAttribute(); attr !=
nullptr;
677 if (
StrEqual(attr->Name(),
"__shared_blackboard"))
686 recursivelyCreateTree(node->name(), output_tree, blackboard, node);
693 for (
const XMLAttribute* attr = element->FirstAttribute(); attr !=
nullptr;
698 new_bb->addSubtreeRemapping(attr->Name(), attr->Value());
702 recursivelyCreateTree(node->name(), output_tree, new_bb, node);
705 else if (dynamic_cast<const SubtreePlusNode*>(node.get()))
709 std::set<StringView> mapped_keys;
711 for (
const XMLAttribute* attr = element->FirstAttribute(); attr !=
nullptr;
714 const char* attr_name = attr->Name();
715 const char* attr_value = attr->Value();
721 if (
StrEqual(attr_name,
"__autoremap"))
724 new_bb->enableAutoRemapping(do_autoremap);
732 new_bb->addSubtreeRemapping(attr_name, port_name);
733 mapped_keys.insert(attr_name);
738 new_bb->set(attr_name, static_cast<std::string>(attr_value));
739 mapped_keys.insert(attr_name);
743 recursivelyCreateTree(node->name(), output_tree, new_bb, node);
748 for (
auto child_element = element->FirstChildElement(); child_element;
749 child_element = child_element->NextSiblingElement())
751 recursiveStep(node, child_element);
756 auto it = tree_roots.find(tree_ID);
757 if (it == tree_roots.end())
759 throw std::runtime_error(std::string(
"Can't find a tree with name: ") + tree_ID);
762 auto root_element = it->second->FirstChildElement();
765 recursiveStep(root_parent, root_element);
769 std::vector<std::string>& output_ports)
774 const char* attr_name = attr->Name();
775 const char* attr_value = attr->Value();
780 output_ports.push_back(static_cast<std::string>(port_name));
787 getPortsRecursively(child_element, output_ports);
792 bool include_builtin)
802 std::set<std::string> ordered_names;
804 for (
auto& model_it : factory.
manifests())
806 const auto& registration_ID = model_it.first;
807 if (!include_builtin && factory.
builtinNodes().count(registration_ID) != 0)
811 ordered_names.insert(registration_ID);
814 for (
auto& registration_ID : ordered_names)
816 const auto& model = factory.
manifests().at(registration_ID);
819 element->
SetAttribute(
"ID", model.registration_ID.c_str());
821 std::vector<std::string> ordered_ports;
824 for (
int d = 0; d < 3; d++)
826 std::set<std::string> port_names;
827 for (
auto& port : model.ports)
829 const auto& port_name = port.first;
830 const auto& port_info = port.second;
831 if (port_info.direction() == directions[d])
833 port_names.insert(port_name);
836 for (
auto& port : port_names)
838 ordered_ports.push_back(port);
842 for (
const auto& port_name : ordered_ports)
844 const auto& port_info = model.ports.at(port_name);
847 switch (port_info.direction())
861 if (port_info.type())
865 if (!port_info.defaultValue().empty())
867 port_element->
SetAttribute(
"default", port_info.defaultValue().c_str());
870 if (!port_info.description().empty())
872 port_element->
SetText(port_info.description().c_str());
877 if (!model.description.empty())
879 element->
SetAttribute(
"description", model.registration_ID.c_str());
887 return std::string(printer.
CStr(), size_t(printer.
CStrSize() - 1));
Tree buildTreeFromText(const BehaviorTreeFactory &factory, const std::string &text, const Blackboard::Ptr &blackboard)
Pimpl(const BehaviorTreeFactory &fact)
void SetText(const char *inText)
static Blackboard::Ptr create(Blackboard::Ptr parent={})
const char * Name() const
Get the name of an element (which is the Value() of the node.)
XMLError LoadFile(const char *filename)
void recursivelyCreateTree(const std::string &tree_ID, Tree &output_tree, Blackboard::Ptr blackboard, const TreeNode::Ptr &root_parent)
static raw_event_t * buffer
std::string toStr(T value)
const XMLAttribute * FirstAttribute() const
Return the first attribute in the list.
const std::set< std::string > & builtinNodes() const
List of builtin IDs.
bool Error() const
Return true if there was an error parsing the document.
void loadFromFile(const std::string &filename, bool add_includes=true) override
void Print(XMLPrinter *streamer=0) const
std::vector< Blackboard::Ptr > blackboard_stack
filesystem::path current_path
void VerifyXML(const std::string &xml_text, const std::unordered_map< std::string, NodeType > ®istered_nodes)
std::list< std::unique_ptr< tinyxml2::XMLDocument > > opened_documents
const std::string & defaultValue() const
std::shared_ptr< Blackboard > Ptr
const char * CStr() const
static const char * xml_text
void clearInternalState() override
Blackboard::Ptr blackboard
const XMLElement * NextSiblingElement(const char *name=0) const
Get the next (right) sibling element of this node, with an optionally supplied name.
void SetAttribute(const char *name, const char *value)
Sets the named attribute to value.
const std::unordered_set< std::string > ReservedPortNames
XMLElement * NewElement(const char *name)
Tree instantiateTree(const Blackboard::Ptr &root_blackboard, std::string main_tree_to_execute={}) override
XMLElement * RootElement()
std::string demangle(char const *name)
The BehaviorTreeFactory is used to create instances of a TreeNode at run-time.
nonstd::string_view StringView
const BehaviorTreeFactory & factory
void loadDocImpl(tinyxml2::XMLDocument *doc, bool add_includes)
XMLNode * InsertFirstChild(XMLNode *addThis)
const char * ErrorName() const
void loadFromText(const std::string &xml_text, bool add_includes=true) override
std::map< std::string, const XMLElement * > tree_roots
bool convertFromString< bool >(StringView str)
PortsRemapping output_ports
Struct used to store a tree. If this object goes out of scope, the tree is destroyed.
Simple class for manipulating paths on Linux/Windows/Mac OS.
ROSLIB_DECL std::string getPath(const std::string &package_name)
const char * Attribute(const char *name, const char *value=0) const
const XMLElement * FirstChildElement(const char *name=0) const
std::vector< std::string > registeredBehaviorTrees() const override
const XMLAttribute * Next() const
The next attribute in the list.
std::vector< TreeNode::Ptr > nodes
static volatile int count
static StringView stripBlackboardPointer(StringView str)
void getPortsRecursively(const XMLElement *element, std::vector< std::string > &output_ports)
The XMLParser is a class used to read the model of a BehaviorTree from file or text and instantiate t...
const std::unordered_map< std::string, TreeNodeManifest > & manifests() const
Manifests of all the registered TreeNodes.
static bool isBlackboardPointer(StringView str)
XMLParser(const BehaviorTreeFactory &factory)
std::shared_ptr< TreeNode > Ptr
PortDirection direction() const
XMLError Parse(const char *xml, size_t nBytes=static_cast< size_t >(-1))
TreeNode::Ptr createNodeFromXML(const XMLElement *element, const Blackboard::Ptr &blackboard, const TreeNode::Ptr &node_parent)
std::basic_string< CharT, Traits > to_string(basic_string_view< CharT, Traits > v)
XMLNode * InsertEndChild(XMLNode *addThis)
PortsRemapping input_ports
Tree buildTreeFromFile(const BehaviorTreeFactory &factory, const std::string &filename, const Blackboard::Ptr &blackboard)
std::string writeTreeNodesModelXML(const BehaviorTreeFactory &factory, bool include_builtin=false)
static Optional< StringView > getRemappedKey(StringView port_name, StringView remapping_value)
std::unordered_map< std::string, std::string > PortsRemapping