Skip to content

sam-xl/py_trees_parser

Repository files navigation

py_trees Parser

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.

Dependencies

  • 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.

XML Parser

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.

Examples

For examples see test/data/.

Python Interpreter

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

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.

Subtrees

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.

Basic Usage

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()

Using Your Own Behaviors

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.

Sub-Trees

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>

Arguments

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>

Conditionals

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'"/>

Logging

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.

Logging Level

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)
  • For details on ROS2 log levels see here.
  • For details on python log levels see here

About

An xml parser for py_trees

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Contributors 3

  •  
  •  
  •