The example programs demonstrate how to create a data acquisition plugin service and run the service such that it can communicate with the data acquisition service on-robot. A data acquisition plugin service is used to communicate with external payloads and hardware, retrieve the data from these sensors, and save the data in the data acquisition store service.
The DataAcquisitionPluginService base class (defined in data_acquisition_plugin_service.py
) can be used to create the data acquisition plugin services. The example directory contains multiple different example plugins, each contained in their own folder:
- battery_service: A very simple plugin which reads and saves the battery information out of the robot state as json-formatted metadata. This plugin is used as part of the Data collection tutorial.
- pointcloud_plugin: Reads from a point cloud service (such as a Velodyne) and saves the point cloud data as a binary protobuf.
- network_compute_bridge_plugin: Used to call a Network Compute Bridge model to process data, and save the results. Customize this example for when the built-in network compute support in the Data Acquisition Service is not expressive enough for your case.
- gps_metadata_plugin and piksi_gps_plugin: Used to store GPS coordinates as json-formatted metadata. gps_metadata_plugin just inserts random data and is meant to be customized for individual hardware. piksi_gps_plugin reads from a Piksi unit.
- save_file_plugin: Loads a file from disk and saves the file as binary. For example, this plugin can save an mp4 video file to the robot.
- signals_coreio_modem_plugin: Reads modem data from the CORE IO and exposes a new GetLiveData rpc to display signals data during Spot teleoperation.
Each example plugin requires the Spot SDK to be installed, and must be run using python3. Within the example plugin's directory and using pip, these dependencies can be installed using:
python3 -m pip install -r requirements.txt
Note, this command must be run from within the directory containing the specific plugin service file that will be run.
Each data acquisition plugin example will run the service locally and register it with the robot's directory service using a directory keep alive. After running the plugin service, the data acquisition service on the robot will detect the registration and adopt the plugin capabilities. This will allow your data acquisition plugin's capabilities to be available in the tablet or other applications communicating with the data acquisition service.
To run a plugin service from this example directory, issue the command:
python3 {PLUGIN_FILE_NAME} --guid {GUID} --secret {SECRET} --host-ip {IP_WHERE_PLUGIN_WILL_RUN} --port {PORT_THE_PLUGIN_WILL_MONITOR} {ROBOT_IP}
This example takes two different IP addresses as arguments. The --host-ip
argument describes the IP address for the computer that will be running the data acquisition plugin service. A helper exists to try to determine the correct IP address. This command must be run on the same computer that will be running the plugin service:
python3 -m bosdyn.client {ROBOT_IP} self-ip
The other IP address is the traditional robot hostname ("ROBOT_IP") argument, which describes the IP address of the robot hosting the directory service.
Since the example is created to run off of a payload computer, it requires the input arguments --guid
(uniquely generated payload specifier) and --secret
(private string associated with a payload) for the registered payload computer that will be running the example plugins. See documentation on configuration of payload software for more information.
For the CORE I/O, this information by default will be located in the file /opt/payload_credentials/payload_guid_and_secret
.
Note, you can run the example plugins locally on your PC by registering it as a weightless payload using the payloads example and creating a GUID and secret for your computer.
Lastly, port numbers for the example plugin services can be specified using the --port
("PORT_THE_PLUGIN_WILL_MONITOR") argument. It is possible to bypass the port argument and allow a random port number to be selected, but it is discouraged since restarts may result in unexpected changes to a services listening port. The port numbers will be used with the --host-ip
("IP_WHERE_PLUGIN_WILL_RUN") to fully specify where the two services are running on the payload computer. The port numbers of different plugins cannot be the same, they must be open, and they must not be blocked by a local firewall, otherwise the service will be unreachable from the robot and other applications.
The network ports used by the services can be opened by running this command on the host computer (Linux):
sudo ufw allow {PORT_NUMBER}
The pointcloud and network compute bridge plugin will require you to pass the associated registered service name. You can find this by running:
python -m bosdyn.client {ROBOT_IP} dir list`
This will list all the services registered to the robot. Use --pointcloud-service
to provide the registered service name for pointcloud plugin. Use ---worker-name
to provide the registered service name for network compute bridge plugin.
For example, if a network compute bridge plugin is needed for a fire-extinguisher-server
, the registered service name would show up in the directory list like the following:
name type authority tokens
----------------------------------------------------------------------------------------------------------------------------------------------------
fire-extinguisher-server bosdyn.api.NetworkComputeBridgeWorker auth.spot.robot user
The command to start the network compute bridge plugin would be:
python3 {PLUGIN_FILE_NAME} --guid {GUID} --secret {SECRET} --host-ip {IP_WHERE_PLUGIN_WILL_RUN} --port {PORT_THE_PLUGIN_WILL_MONITOR} --worker-name fire-extinguisher-server {ROBOT_IP}
To run the plugin that saves a file to the robot:
export BOSDYN_CLIENT_USERNAME={ROBOT_USERNAME}
export BOSDYN_CLIENT_PASSWORD={ROBOT_PASSWORD}
python3 save_file_plugin_service.py --host-ip {IP_WHERE_PLUGIN_WILL_RUN} --port {PORT_THE_PLUGIN_WILL_MONITOR} --file-path {FULL_PATH_TO_A_FILE} {ROBOT_IP}
There is a plugin tester script that can be used while developing a new data acquisition plugin service to help ensure that the service can be communicated with and behaves as expected. The script runs through a series of tests checking the networking and the functionality of each of the plugin's RPCs.
The data acquisition example shows a program which communicates with the data acquisition service on robot and runs through the different data acquisition service RPCs. The data acquisition service will aggregate responses from each plugin service and will direct communication and requests to the specific plugins.
Run the example that communicates with the data acquisition service:
python3 data_acquisition_example.py {ROBOT_IP}
The data acquisition download script allows users to download data from the DataAcquisitionStore service.
There are two different methods in which the data is downloaded from the DataAcquisitionStore service:
- REST: Uses REST calls to download a zip file.
- gRPC Streaming: Uses DataAcquisitionStoreClient.query_stored_captures() call to download & save capture data directly.
The script supports only time-based queries for filtering which data is downloaded and saved locally, though the "data" command can be modified to filter based on other fields listed in QueryParameters in data_acquisition_store.proto. The following command is an example of a time-based query and download. The timestamps are specified in RFC 3339 date string format (YYYY-MM-DDTHH:MM::SS.SZ, Y:year, M:month, D:day, H:hours, M:minutes, S:seconds as double, Z:zulu timezone). Zulu clock time is the same as UTC time (UTC+0).
python3 data_acquisition_download.py --query-from-timestamp 2020-09-01T00:00:00.0Z --query-to-timestamp 2020-09-04T00:00:00.0Z {ROBOT_IP} {rest,grpc}
Note, by default, the download script will save the data to the current directory, however the --destination-folder
argument can be used to change where the downloaded data is saved.
Please refer to this document for general instructions on how to run software applications on computation payloads as docker containers.
With docker installed and setup, any of the data acquisition plugin services can be created into a docker container, saved as a tar file, and then run on the CORE I/O as an Extension. There are Dockerfiles for each plugin service within the plugin's specific directory. These files will create a docker container with all necessary dependencies installed, and will start the plugin service.
Follow the instructions on how to build and use the docker image from this section on.
Once built, all examples can be run with the following command: sudo docker run -it {IMAGE_NAME} {ROBOT_IP} {APPLICATION ARGUMENTS}
. Application arguments generally include payload authentication parameters such as a guid/secret
or payload-credentials-file
, and service endpoint parameters such as port
and host-ip
. For a detailed list of application arguments for any given example, you can run sudo docker run -it {IMAGE_NAME} -h
.