This is an xml parser for processing and building a py_trees behavior tree. The hope is that most if not all capabilities of py_trees and py_trees_ros will be available for xml parsing. As such, a py_trees behavior tree can be created by simply creating an xml file.
- ElementTree
- ros-humble
- ros-humble-py-trees
- ros-humble-py-trees-ros
To install dependencies you can run the following commands:
vcs import src < src/py_trees_parser/dependencies.repos
rosdep update
rosdep -q install --from-paths src/ --ignore-src -y --rosdistro "${ROS_DISTRO}"
where ROS_DISTRO
is an environment variable containing the ros2 distribution
name.
The Behavior Tree Parser (BTParser
) is a Python class that allows you to
parse an XML file representing a behavior tree and construct the corresponding
behavior tree using the py_trees
library. It supports composite nodes, behavior
nodes from py_trees
, and custom behavior nodes defined in your local library.
For examples see test/data/
.
For any parameter that is python code, the code must be surrounded by $()
.
This allows the parser know that the following parameter value is in fact code
and should be evaluated as code. For a concrete example see below:
<py_trees_ros.battery.ToBlackboard name="Battery2BB"
topic_name="/battery/state"
qos_profile="$(py_trees_ros.utilities.qos_profile_unlatched())"
threshold="30.0" />
In the above example the qos_profile
is evaluated as python code. Notice to
use any python module you must use the fully qualified name.
Idioms are also now supported. An idiom is a special function that produces a behavior tree, the function is expected to either take no children, takes a list of children via parameters "subtrees", or takes a single child via parameter "behavior". The xml parser will treat children in the same way that it treats children for all other behaviors. That is the children should be a subnode of the idiom node.
Additionally, it is possible to create a subtree, where a subtree is an xml containing a complete behavior tree. This xml file can be included in other xml files and therefore allows for complete modularity of trees.
To use the Behavior Tree Parser, follow these steps:
Create an XML file that represents your behavior tree. The XML structure should define the nodes and their attributes. It should be similar to the following:
<py_trees.composites.Parallel name="TutorialOne" synchronise="False">
<py_trees.composites.Sequence name="Topics2BB" memory="False">
<py_trees_ros.battery.ToBlackboard name="Battery2BB"
topic_name="/battery/state"
qos_profile="$(py_trees_ros.utilities.qos_profile_unlatched())"
threshold="30.0" />
</py_trees.composites.Sequence>
<py_trees.composites.Selector name="Tasks" memory="False">
<py_trees.behaviours.Running name="Idle" />
<py_trees.behaviours.Periodic name="Flip Eggs" n="2" />
</py_trees.composites.Selector>
</py_trees.composites.Parallel>
Assuming the above is saved in behavior_tree.xml
it can be imported via the
following code:
from py_trees_parser import BTParser
import py_trees
# Parse the XML file and create the behavior tree:
xml_file = "behavior_tree.xml"
parser = BTParser(xml_file)
behavior_tree = parser.parse()
The xml parser can use any behavior, whether it is part of py_trees
, py_trees_ros
,
or your own python module. The way the parser knows the existence of
the behavior is via the behavior tag in the xml. The behavior tag should be the fully
qualified python path of the behavior, so if you have the following structure of your
python module
my_behavior_tree
├── my_behavior_tree
│ ├── __init__.py
│ ├── behaviors
│ │ ├── __init__.py
│ │ └── my_behavior.py
├── package.xml
├── resource
│ └── my_behavior_tree
├── setup.cfg
├── setup.py
└── trees
├── my_tree.xml
├── subtree2.xml
└── subtree3.xml
then you would include the behaviors in my_behavior.py
in the following way:
<my_bhavior_tree.behaviors.my_behavior.MyBehavior name="MyFancyBehavior">
The path to this can be shortened by including the class in __init__.py
.
It is possible to include sub-trees in the xml file containing a behavior tree. This is made possible via the following:
<subtree
name="my_subtree"
include="/location/of/subtree.xml" />
The included subtree should be a complete tree, but can only contain one root. It is possible to include multiple sub-trees and a sub-tree can also include another subtree. However, be aware that the all directories are absolute, but it is possible to use python to determine the path like so:
<py_trees.composites.Parallel name="Subtree Tutorial" synchronise="False">
<subtree
name="my_subtree"
include="$(os.path.join(ament_index_python.packages.get_package_share_directory('my_package'), 'tree', 'subtree.xml'))" />
</py_trees.composites.Parallel>
It is also possible to use arguments for subtrees. The syntax of which looks like
<subtree
name="my_subtree"
include="/location/of/subtree.xml" >
<arg name="foo" value="bar" />
</subtree>
and then inside the subtree
<py_trees.composites.Sequence name="Arg Tutorial">
<py_trees.behaviors.Success name="${foo}" />
</py_trees.composites.Sequence>
Additionally, arguments can be embedded within strings for partial substitution using the same syntax:
<py_trees.composites.Sequence name="Embedded Arg Tutorial">
<py_trees.behaviors.Success name="prefix_${foo}_suffix" />
</py_trees.composites.Sequence>
Furthermore, one can cascade arguments down subtrees using the following syntax:
<subtree
name="my_subtree1"
include="/location/of/subtree1.xml" >
<arg name="foo" value="bar" />
</subtree>
and in subtree1
<subtree
name="my_subtree2"
include="/location/of/subtree2.xml" >
<arg name="baz" value="${foo}" />
</subtree>
and finally in subtree2
<py_trees.composites.Sequence name="Cascading Arg Tutorial">
<py_trees.behaviors.Success name="${baz}" />
</py_trees.composites.Sequence>
A conditionals allows one to choose which xml nodes should be included in the
final tree. If a condition evaluates to true then the element is included in the
final tree otherwise it is not. Conditionals can be used with any element of a
behavior tree and is represented by the if
keyword and are written
like
<py_trees.behaviours.Success name="Feature 1" if="True"/>
<py_trees.behaviours.Success name="Feature 2" if="False"/>
<py_trees.behaviours.Success name="Feature 3" if="${my_arg1} < 5"/>
<py_trees.behaviours.Success name="Feature 4" if="'${my_arg2}' == 'simulation'"/>
By default the parser uses rclpy.logging
module to log messages. However, if this is not available
it falls back to the python logging
module. The user need not intervene to change the logger as the
parser will determine which logger to use on its own.
If you would like to set the logging level you will need to know which logger is available. If you are
using ROS2 then the logger will be rclpy.logging
and the log level can be set in the following way
import rclpy
from py_trees_parser import BTParser
xml_file = "behavior_tree.xml"
parser = BTParser(xml_file, log_level=rclpy.logging.LoggingSeverity.DEBUG)
On the other hand if you are not using ROS2 then the log level can be set in the following way
import logging
from py_trees_parser import BTParser
xml_file = "behavior_tree.xml"
parser = BTParser(xml_file, log_level=logging.DEBUG)