Isaac Mission Dispatch is a cloud service that enables the communication between edge robots and other cloud services responsible for managing a fleet of robots. The communication between Mission Dispatch and robots is designed per VDA5050 protocol and uses MQTT, as MQTT is the industry standard for a highly efficient, scalable protocol for connecting devices over the internet. VDA 5050 is an open standard for communication between fleets of AGVs/AMRs and a central fleet service.
The Mission Dispatch system is composed of two main components:
-
The mission database microservice
This component hosts REST APIs used to create, update, watch, and get the state of API mission/robot objects. It also manages persistence for objects, allowing them to rebuild their internal state if they crash or are restarted.
-
The mission dispatch microservice
This component handles the communication with robots and manages mission state transitions. A mission is defined as a series of tasks needed to be completed by a specific robot. See the Mission section for more details of how a mission maps to VDA5050 orders.
There are several types of communication between a control service and the fleet of robots, including large data transfers such as map updates to the robot, events recordings from the robot, and high throughput, low latency teleoperation of the robot. Each type is better serviced by other side-channel communication protocols. When a side channel for communication is needed, Mission Dispatch should establish and provide the connection details as it maintains a database for the current state of the system and available robots.
In order for a robot to execute missions, it needs to be a mission client. A lightweight mission client simulator is provided to test a simple simulated robot that implements a VDA5050 client (see the Robot Clients section for more details). A mission client node is also available in ROS2.
In summary, Isaac Mission Dispatch microservices enable fleet management software to submit missions to multiple robots and monitor the robot and mission states. It provides a connection between the robots and the fleet management system but does not handle logistics such as task allocation or conflict resolution, e.g., robots with intersecting paths. The implementation relies VDA5050 protocol as an industry standard between a cloud control service and mobile robots, and uses MQTT as a lightweight, publish-subscribe, machine to machine network protocol designed for devices with resource constraints and limited network bandwidth.
There are other approaches to distributing tasks to and monitoring a fleet of robots which are available and have their respective benefits and limitations. This solution is one approach to the distributing, and tracking tasks and state from a control service to multiple concurrent robots efficiently with robustness to limited network bandwidth and communication interruption. This solution enables heterogenous robotics stacks in the fleet of robots with VDA5050 protocol for consistent communication. This package provides a part of a fleet management system.
Note: Mission dispatch is intended for VDA5050, and no other protocols are supported for now.
- Isaac Mission Dispatch
Update 2022-10-19: Initial release
This package is designed and tested to be compatible with laptops or desktops with Ubuntu 20.04+ for development, as well as cloud service providers (CSPs) and EGX certified platforms for deployment.
Note: This package does not require GPU acceleration.
A mission is a series of tasks to be completed by a given robot. It is represented as a behavior tree generated from the mission_tree
, which is a list of task nodes that may be performed on the robot. The advantage of using a mission tree instead of an array of steps is that it allows the robot to react to all sorts of situations. The behavior tree has an implicit sequence node as its root, which is also named “root”.
Each mission tree node has four possible states: IDLE
, RUNNING
, SUCCESS
, and FAILURE
. Currently supported mission tree nodes are: sequence, selector, route, and action:
Field | Type | Parameters | Description |
---|---|---|---|
name | Optional[str] |
None | A unique name to give the node. If not specified, it will be automatically set to the index of the node in the list |
parent | Optional[str] |
None | The parent of the node. If not specified, it will be the “root” node. |
sequence | Optional[object] |
None | Executes children nodes in order. If the child node currently running completes with SUCCESS , then the next child node is started. Otherwise, the sequence node completes with FAILURE . If all children nodes complete with SUCCESS , then the sequence node completes with SUCCESS . |
selector | Optional[object] |
None | Executes children nodes in order. If the child node currently running completes with SUCCESS , then the selector node completes with SUCCESS . Otherwise, the next child node is started. If all children nodes fail, then the selector node completes with FAILURE . |
action | Optional[object] |
name(string) : The name of the action to trigger on the robot params(json) : An arbitrary, action-specific JSON payload to send as parameters to the action |
Performs some generic, named action on the robot. |
route | Optional[object] |
waypoints(List[Pose2D]) : A list of 2D poses for the robot to visit |
Instructs the robot to travel a given route. The robot may or may not visit intermediate waypoints, but the final waypoint must be visited. Will return either SUCCESS or FAILURE , depending on whether the robot can successfully navigate to the final waypoint. |
The difference between the sequence and selector node: a sequence node will attempt to run all the child nodes as long as SUCCESS
is being returned and will instantly return FAILURE
upon a node failing, whereas "selector" will attempt to get only a single SUCCESS
, and upon failure, will keep trying child nodes until it gets either gets SUCCESS
or exhausts all child nodes.
Note: All mission tree nodes are mutually exclusive -- a sequence node cannot also be a selector node, an action node cannot also be a route node, etc.
Each route or action mission tree node will be translated into a separate VDA5050 Order message. Each of the waypoints in the route mission tree node is an Order node. They are appended together to form an Order message. For the action mission tree node, the action will be attached to the first node of the Order, which is the one corresponding to the current pose of the robot. The Order messages will then be sent out sequentially based on the progression of the behavior tree. An example is shown in the figure below.
In this example, the mission is shown on the left. The robot is asked to go to a pick-up point and execute the action of picking up a book. After the action is executed successfully, the robot jumps to the next route_fallback
selector node: if it fails to go to the drop-off point, then it goes back to the previous point and drops off the book; otherwise, it drops off the book at the final goal point. The corresponding VDA5050 Order messages are shown on the right.
We provide three deployment options here for using Mission Dispatch services: deploy with official Docker containers, deploy with a docker-compose
file, and deploy with Kubernetes.
Download the repository:
git clone https://github.com/NVIDIA-ISAAC/isaac_mission_dispatch
Continue here to run Mission Dispatch microservices directly on a computer, CSP, or EGX. Skip to section Getting Started with Local Development to develop services locally on your computer.
An interactive documentation page that can be used to submit missions will be launched after the deployment. If you used the default parameters, this can be found at http://localhost:5000/docs
or http:<your_ip_address>:5000/docs
.
-
Launch Dependencies.
Mission Dispatch needs a few services to be running to function properly.
-
MQTT Broker
The MQTT broker is used for communication between the Mission Dispatch and the robots. There are many ways to run an MQTT broker, including as a system daemon, a stand alone application, or a docker container. Here we use mosquitto as our MQTT broker. Start the mosquitto broker by running the following:
cd mission_dispatch docker run -it --network host -v ${PWD}/packages/utils/test_utils/mosquitto.sh:/mosquitto.sh -d eclipse-mosquitto:latest sh mosquitto.sh 1883 9001
-
Postgres database
Set the following environment variable:
export POSTGRES_PASSWORD=<Any password>
Start the Postgres database by running the following:
docker run --rm --name postgres \ --network host \ -p 5432:5432 \ -e POSTGRES_USER=postgres \ -e POSTGRES_PASSWORD \ -e POSTGRES_DB=mission \ -d postgres:14.5
-
Launch the Mission Database microservice:
Start the API and database server with the official docker container.
docker run -it --network host nvcr.io/nvidia/isaac/mission-database:2022.10.17_de4892b # To see what configuration options are, run # docker run -it --network host nvcr.io/nvidia/isaac/mission-database:2022.10.17_de4892b --help # For example, if you want to change the port for the user API from the default 5000 to 5002, add `--port 5002` configuration option in the command.
-
Launch the Mission Dispatch microservice:
Start the mission dispatch server with the official docker container.
docker run -it --network host nvcr.io/nvidia/isaac/mission-dispatch:2022.10.17_de4892b # To see what configuration options are, add --help option after the command.
To simplify the steps in the Deploy with Official Docker Containers section, the two dependencies (MQTT broker/Postgres database) and the Mission Dispatch microservices (database/dispatch) are packaged into one docker-compose file. You can simply run the steps below to achieve the bring up all the microservices:
cd mission_dispatch/docker_compose
docker-compose -f mission_dispatch_services.yaml up
# run `docker-compose -f mission_dispatch_services.yaml down` if you want to bring down all the services.
-
Install Kubernetes using these instructions.
-
Set up the Postgres database:
a. Set up the Kubernetes Persistent Volume (PV) for the Postgres database:
kubectl apply -f https://k8s.io/examples/pods/storage/pv-volume.yaml
b. Install the chart for the Postgres database and pass in the primary arguments:
helm install postgres-db bitnami/postgresql \ --set containerPorts.postgresql=5432 \ --set auth.postgresPassword=postgres \ --set auth.database=mission \ --set primary.persistence.storageClass=manual \ --set volumePermissions.enabled=True
-
Set up Mission Dispatch services:
cd mission_dispatch helm install mission-dispatch charts/
Continue here to develop Mission Dispatch services locally. Skip to the Run Mission Dispatch section to try our provided services.
All building and running of applications through bazel should be done within this container to ensure that the correct dependencies are present.
cd mission_dispatch
docker build --network host -t isaac-misssion-dispatch "${PWD}/docker"
-
Launch Dependencies by following steps in the Deploy with Official Docker Containers section.
-
Launch the developer Docker container.
docker run -it --rm \ --network host \ --workdir "$PWD" \ -e USER="$(id -u)" \ -v "$PWD:$PWD" \ -v /etc/passwd:/etc/passwd:ro \ -v /etc/group:/etc/group:ro \ -v "$HOME/.docker:$HOME/.docker:ro" \ -v "/etc/timezone:/etc/timezone:ro" \ -v "$HOME/.cache/bazel:$HOME/.cache/bazel" \ -v /var/run/docker.sock:/var/run/docker.sock \ -u $(id -u) \ --group-add $(getent group docker | cut -d: -f3) \ isaac-misssion-dispatch /bin/bash
You may run this command in as many terminals as you want to get more terminals in the same developer environment. A new docker container will be launched for each instance, but they will share the same Bazel cache and source code.
-
Verify with the provided test cases.
Make sure that you are able to build and run the applications in the repo by running the unit tests within the docker container:
bazel test ...
-
Launch Mission Dispatch microservices.
Mission Dispatch is comprised of two microservices that are launched independently.
a. Launch mission database
Start the API and database server by using Bazel directly.
# Make sure to run step 2 (launch the developer docker container) first # in your terminal before running the command below bazel run packages/database:postgres -- --db_name mission --db_username postgres \ --db_password <add_postgres_password> --db_host localhost --db_port 5432 # Change <add_postgres_password> to a valid postgre password, such as `postgres` or $POSTGRES_PASSWORD if you have set the environment variable before. # To see what configuration options are, run # bazel run packages/database:postgres -- --help
An interactive documentation page that can be used to submit missions will be launched with the Mission Database microservice. If you used the default parameters, this can be found at
http://localhost:5000/docs
, orhttp:<your_ip_address>:5000/docs
.b. Launch Mission Dispatch
Start the Mission Dispatch server, which will execute missions and populate the robot status in the database.
# Make sure to run step 2 (launch the developer docker container) first # in your terminal before running the command below bazel run packages/controllers/mission # To see what configuration options are, run # bazel run packages/controllers/mission -- --help
Robot clients are needed to execute the missions. You have three options, depending on your preference:
The client simulator will simulate a set of robots that will do the following:
- Connect to the MQTT broker.
- Accept missions through VDA5050 and follow the waypoints by moving in an L-shaped path (first
$X$ , then$Y$ ). - Report the updated mission status and robot position back to the server through VDA5050 feedback messages.
- Optionally fail every
$N^\text{th}$ mission where$N$ is configurable.
To run with bazel (self-build container):
# Runs a simulation with two robots
# First robot:
# name = carter01
# x position = 4
# y position = 5
# theta = 0 (default)
# failure_period = 0 (default)
# (A failure period of 0 means the robot will successfully complete all missions)
#
# Second robot:
# name = carter02
# x position = 9
# y position = 9
# theta = 3.14
# failure_period = 3
# (A failure period of 3 means that the 3rd, 6th, 9th... missions will fail,
# and 1st, 2nd, 4th, 5th... will pass)
bazel run packages/controllers/mission/tests:client -- --robots \
carter01,4,5 \
carter02,9,9,3.14,3
To run with docker (official image):
docker run -it --network host nvcr.io/nvidia/isaac/mission-simulator:2022.10.17_de4892b --robots \
carter01,4,5 \
carter02,9,9,3.14,3
A ROS2 Humble package that receives tasks and actions from the fleet management service through Mission Dispatch and updates its progress, state, and errors. It also performs navigation actions with Nav2 and can be integrated with other ROS actions.
See the tutorials given in the ROS Mission Client for how to use Mission Dispatch services with ROS Mission Client and NVIDIA Isaac Sim.
A ROS2 Galatic package that implements a connector for VDA5050 and works as a bridge between a Mission Dispatch and a ROS2 robot. It also supports the Nav2 robot navigation stack with other user-defined actions.
Follow the first two steps in the Running the TB3 adapter to launch the VDA5050_connector with Nav2. Next step is to set the initial pose for Nav2 by clicking the 2D Pose Estimate button in RViz, and then down clicking on the map in where the robot is in the Gazebo world.
Note: if you choose to use VDA5050_connector as a mission client:
- Run mission_dispatch first and then vda5050_connector to avoid MQTT connection issue.
- Set
--mqtt_prefix uagv/v1/{manufacturer_name}
option when running mission_dispatch. This applies to all the deployment and development approaches. - Set the robot object's name to {serial_number} when post robot and mission. (See the Add Robots with REST API section for more details)
- {manufacturer_name} and {serial_number} can be found in
vda5050_adapter_examples/vda5050_tb3_adapter/config/connector_tb3.yaml
The interactive REST API page is used to set up the robot.
Use the POST /robot
endpoint to create robot objects to represent the robots that will be connected to the mission dispatch. Make sure the names of the robots match the names given to the simulated/real robots.
Use the GET /robot
endpoint to query the status of the robots once they are created. If the robots are connected, the state should reflect the actual position of the robots.
Note: When using the API page, the default value for the robot object's
name
in the spec
is "string"
, so make sure to change it from "string"
to another name that has
more meaning (e.g. "carter01"
). Set the robot object's
name
to {serial_number}
if you choose to use VDA5050_connector as a mission client.
A request body example is shown below:
{
"labels": [],
"battery": {
"critical_level": 0.1
},
"heartbeat_timeout": 30,
"name": "carter01"
}
The video below shows the step-by-step process of how to create and query a robot.
managing_robots.mp4
For more details about the robot object, specifications, and status definition, please refer to the schemas at the end of the interactive REST API documentation page.
Missions can be created using the POST /mission
endpoint and monitored via the
GET /mission
endpoint. Users can also monitor the state of a robot through GET /robot
and watch
its position update as it completes the mission.
Note: By default, the value for the robot
field in spec
is "string"
, so make sure to change it to
the name
you used for one of the robot objects you created earlier. If you set the name
of the robot
object to "carter01"
, use that to fill in the robot
field for the mission.
Note: A mission_tree
includes multiple mission nodes. A mission node can only be one of the four node types mentioned in the Mission section. When sending a mission, a user should choose one and remove the other three in the request body.
This video shows the step-by-step process to create and query multiple missions with multiple robots given in the Examples section.
managing_missions.mp4
Users are allowed to DELETE
and CANCEL
a queued mission (submitted but waiting to be executed) at any time. However, if users want to DELETE
a currently running mission, Mission Dispatch will label the life cycle of the current mission as PENDING_DELETE
, wait for the current mission to finish, and then delete it.
Similarly, if users want to CANCEL
a currently running mission, Mission Dispatch will set needs_canceled
in mission specifications to True
, wait for the current mission to finish, and then cancel the mission.
All missions are expected to be idempotent and instead of updating a mission: developers are expected to delete and re-submit.
For more details on the mission object, specifications, and status definition, refer to the schemas at the end of the interactive REST API documentation page.
We provide two POST /mission
request body examples here as a reference for how to construct a mission tree.
- Example 1 -- Simple mission
Consider a simple mission where the robot "carter01"
is asked to go to the pickup point and then move to the drop-off point with certain predefined routes. The sample request body for posting a mission is shown below:
{
"robot": "carter01",
"mission_tree": [
{
"name": "goto_pickup",
"parent": "root",
"route": {
"waypoints": [
{
"x": 1.5,
"y": 1.5,
"theta": 0,
"map_id": "map"
},
{
"x": 3.3,
"y": 2.1,
"theta": 0,
"map_id": "map"
}
]
}
},
{
"name": "goto_dropoff",
"parent": "root",
"route": {
"waypoints": [
{
"x": 4.0,
"y": 3.2,
"theta": 0,
"map_id": "map"
},
{
"x": 5.0,
"y": 5.0,
"theta": 0,
"map_id": "map"
}
]
}
}
],
"timeout": 300,
"deadline": "2022-09-28T04:04:24.013Z",
"needs_canceled": false,
"name": "simple_mission_example"
}
- Example 2 -- Complex mission
A relatively complex mission will include behavior control nodes such as sequence
and selector
, as shown in the mission given in the mission section. We provide the request body example for that mission below.
In this example, the robot completes the goto_pickup
and pickup_book
nodes, then completes the route_fallback
selector node as the robot can reach the waypoints set in the goto_dropoff
node. Thus, it would skip the reset_pose
and dropoff_book
nodes under the same route_fallback
selector node. However, the mission fails at the last dropoff_book_at_goal
node as the should_fail
parameter for the action is set to True
. Logging of the mission progression and error messages are displayed in the terminal output as well as on the API page.
Watch the Submitting Missions video tutorial given in the Submitting Missions section for how to use the example given here.
{
"robot": "carter01",
"mission_tree": [
{
"name": "goto_pickup",
"parent": "root",
"route": {
"waypoints": [
{
"x": 1.5,
"y": 0.8,
"theta": 0,
"map_id": "map"
}
]
}
},
{
"name": "pickup_book",
"parent": "root",
"action": {
"action_type": "dummy_action",
"action_parameters": {
"should_fail": 0,
"time": 1
}
}
},
{
"name": "route_fallback",
"parent": "root",
"selector": {}
},
{
"name": "goto_dropoff",
"parent": "route_fallback",
"route": {
"waypoints": [
{
"x": 3,
"y": 2.2,
"theta": 0,
"map_id": "map"
},
{
"x": 0.8,
"y": 1.8,
"theta": 0,
"map_id": "map"
}
]
}
},
{
"name": "goto_dropoff_seq",
"parent": "route_fallback",
"sequence": {}
},
{
"name": "reset_pose",
"parent": "goto_dropoff_seq",
"route": {
"waypoints": [
{
"x": 1.5,
"y": 0.8,
"theta": 0,
"map_id": "map"
}
]
}
},
{
"name": "dropoff_book",
"parent": "goto_dropoff_seq",
"action": {
"action_type": "dummy_action",
"action_parameters": {
"should_fail": 1,
"time": 1
}
}
},
{
"name": "dropoff_book_at_goal",
"parent": "root",
"action": {
"action_type": "dummy_action",
"action_parameters": {
"should_fail": 1,
"time": 1
}
}
}
],
"timeout": 300,
"deadline": "2022-09-28T04:04:24.013Z",
"needs_canceled": false,
"name": "complex_mission_example"
}
Note: We provide a dummy action server in our mission client simulator for the user to test the above complex mission. The action parameters include the expected behavior should_fail
(0: success; 1: failure) and the execution time (in seconds) time
should be given to simulate a real action server.
A ROS mission client package that allows the mission dispatch to communicate with robots through the MQTT protocol. Visit this resource for more video tutorials on using Mission Dispatch with Isaac ROS Mission Client, as well as for NVIDIA Isaac Sim running on local and cloud.
The vda5050_connector package is another mission client (in ROS Galatic) that provides a set of ROS2 nodes for connecting a ROS2-based robot to the Mission Dispatch.
Check here for solutions to problems with Isaac ROS.
Date | Changes |
---|---|
2022-10-19 | Initial release |
Isaac Mission Dispatch is under Apache 2.0 license.