- Table of Contents
- Project Goal
- Package usage and dependencies
- PDDL
- Behavior Tree diagrams
- Launcher
- Tests
- Team
- Licencia
The aim of this project is to create a ROS2 application that use PlanSys2 to resolve PDDL based planning systems. The project is focus in the resolution of tasks in a 5 roomed-house and the requests that a human ask the robot to do, as it is shown in the above image.
This project requires the ir_robots repositories, as well as the br_navigation for the navigation of Tiago.
To use the repository, you will need to follow this steps:
Note: Due to the problems in the updates of the Tiago, you could find some troubles in the navigation launch and using this project repository if you have updated the packages.
Launch Commands:
ros2 launch ir_robots simulation.launch
ros2 launch br2_navigation tiago_navigation.launch.py
ros2 launch plansys2_gpsr_tayros2 plansys2_gpsr.launch.py
ros2 run plansys2_gpsr_tayros2 gpsr_controller_node
In order to solve this task, we need to represent the "robot world" in such a way that a planning algorithm can find an optimal solution to make the robot tidy up the house and assist the human with the tasks they request (such as opening and closing doors or bringing them something they need). To do this, we have used PDDL.
Some of the things we have improved for this task include the implementation of the "move_without_door" and "move_by_door" functions as durative actions. The problem we had with these functions is that the robot executed both actions simultaneously, causing the robot to be in two different positions in the house at the same time (which is impossible) or not finding a solution to the specified problem.
(:durative-action move_by_door
:parameters (?r - robot ?from ?to - location ?d - door)
:duration ( = ?duration 5)
:condition (and
(at start(robot_at ?r ?from))
(at start(connected_by_door ?from ?to ?d))
)
:effect (and
(at start(not (robot_at ?r ?from)))
(at end(robot_at ?r ?to))
)
)
We have also added the "overall" functionality of PDDL to those conditions we believe they require it.
(:durative-action pick
:parameters (?u - util ?l - location ?r - robot ?g - gripper)
:duration (= ?duration 1)
:condition (and
(at start(gripper_at ?g ?r))
(at start(object_at ?u ?l))
(at start(gripper_free ?g))
(over all(robot_at ?r ?l))
(over all(no_prio_task_remaining))
)
:effect (and
;it's really important to specify that the gripper is not free at the start of the function
;in orther to not grab more than one object.
(at start(not (gripper_free ?g)))
(at end(not (object_at ?u ?l)))
(at end(robot_carry ?r ?g ?u))
)
)
And the use of "high-level" functions that need the execution of some other speficific funtions in orther to execute it.
(:action organize_object
:parameters (?h - human ?u - util ?l - location)
:precondition (and
(human_attended ?h)
(object_at ?u ?l)
)
:effect (and
(move_object ?u ?l)
)
)
You can see the video demonstration here:
- Simple example video #1 using the initial Transport method (organize objects): Alternative Link (Youtube)
PDDL_FINAL_TRANSPORT.mp4
- Complete example video #2 using the Pick & Drop method: Alternative Link (Youtube)
PDDL_Example_Pick_and_Drop.mp4
- Example video #3 (Granny requests): Alternative Link (Youtube)
PDDL_Example_Granny_Requests.mp4
Program diagram:
In order to execute more complex actions such as picking and dropping objects, we employed behavior trees which included the steps of opening the gripper, approaching the object with the arm, and closing the gripper. However, due to the fact that this was a simulation, we omitted the approximation step in order to allocate more attention to other aspects of the program.
In an attempt to add another level of abstraction, we aimed to implement a transport action that would directly move objects. However, we encountered conflicts with the move without door action, so we ultimately opted to simplify the program.
To launch all the action nodes and the bt nodes, a launcher has been implemented which also serves to connect the different nodes with their configuration parameter file (/config/params.yaml) and the bt nodes with their respective trees (.xml).
plansys2_gpsr.launch.py(snippet):
move_without_door_cmd = Node(
package='plansys2_bt_actions',
executable='bt_action_node',
name='move_wod',
namespace=namespace,
output='screen',
parameters=[
example_dir + '/config/params.yaml',
{
'action_name': 'move_without_door',
'publisher_port': 1676,
'server_port': 1677,
'bt_xml_file': example_dir + '/behavior_trees_xml/move.xml'
}
])
config/params.yaml(snippet):
move_wod:
ros__parameters:
plugins:
- plansys2_move_bt_node
waypoints: ["living_room", "bathroom", "bedroom", "kitchen", "garage"]
waypoint_coords:
living_room: [-5.08, -4.93, 0.0]
bathroom: [3.92, 1.03, 0.0]
bedroom: [1.92, -4.97, 0.0]
kitchen: [-2.03, 1.06, 0.0]
garage: [-0.85, -10.17, 0.0]
In order to facilitate the user with the implementation of what would be the typical "problem.pddl" file and using the modularity offered by Plansys2, which allows us to use functions from the classes responsible for the operation of domain and problem processing as well as the correct functioning of the planner and executor; we have created a ros2 node as a "controller" that will be responsible for instantiating "the robot's world", indicating the goal to be achieved and ensuring proper robot operation (feedback). The logic for this node is located in the "gpsr_controller_node.cpp" file.
gpsr_controller_node.cpp:
// Initializing the objects that will allow us to use the functionality of the Plansys2 modules.
domain_expert_ = std::make_shared<plansys2::DomainExpertClient>();
planner_client_ = std::make_shared<plansys2::PlannerClient>();
problem_expert_ = std::make_shared<plansys2::ProblemExpertClient>();
executor_client_ = std::make_shared<plansys2::ExecutorClient>();
// ...
// Initializing the robot's world.
problem_expert_->addInstance(plansys2::Instance{"tay", "robot"});
problem_expert_->addInstance(plansys2::Instance{"garage", "room"});
problem_expert_->addPredicate(plansys2::Predicate("(robot_at tay kitchen)"));
problem_expert_->addPredicate(plansys2::Predicate("(human_at granny bedroom)"));
// ...
// Indicating the goal to be achieved.
problem_expert_->setGoal(plansys2::Goal("(and(object_at tools bedroom))"));
// ...
// Executing the plan
if (executor_client_->start_plan_execution(plan.value())) {
state_ = WORKING;
}
//...
// getting feedback
auto feedback = executor_client_->getFeedBack();
for (const auto & action_feedback : feedback.action_execution_status) {
if (action_feedback.completion) {
std::cout << "[" << action_feedback.action << " " <<
"SUCCESS" << "]";
}
else {
std::cout << "[" << action_feedback.action << " " <<
"RUNNING..." << "]";
}
}
In order to install the necessary repositories for the packages in this assignment, which rely on plansys2, we added the following lines to the container for continuous integration:
- name: Create custom repos
run: wget -O /tmp/all.repos https://raw.githubusercontent.com/IntelligentRoboticsLabs/plansys2_tfd_plan_solver/master/plansys2.repos
...
vcs-repo-file-url: /tmp/all.repos
colcon-mixin-name: coverage-gcc
colcon-mixin-repository: https://raw.githubusercontent.com/colcon/colcon-mixin-repository/master/index.yaml
This significantly increases the time required to run tests as the package needs to be compiled in the container. Then we conducted a simple test using behavior trees, similar to the previous assignment, to verify the functionality of the basic nodes.
(TayROS2)
This work is licensed under a Apache license 2.0