Developer Guide
This is the EdgeFirst Developer's Guide. In this guide we will walk you through the API behind the EdgeFirst Middleware, before reading this guide you should be familiar with the general architecture of the middleware so you can understand how the services fit together and the topics that bind them.
The examples can all be run on a device running the EdgeFirst Middleware or from a remote platform, such as a PC running Windows, Mac, or Linux. If running these examples remotely make sure you have correctly enabled the zenohd
router on the target device running the EdgeFirst Middleware.
Tip
The zenohd
router allows remote connections to the device so you can follow these tutorials from a PC.
The router is enabled using the sudo systemctl enable --now zenohd
command on the target device.
Warning
The default configuration for the zenohd
service does not enable any authentication or encryption.
Make sure you follow the Zenoh TLS Guide if you're running the
zenohd
on an untrusted network.
The EdgeFirst Middleware examples are provided in Python and Rust (C/C++ Coming Soon!) and you can toggle the documentation for your preferred language throughout the tutorial.
Note
Some of the examples, such as those using DMA buffers, cannot be run remotely; these will be called out with a note.
Installation
Whether running the examples on the target running the EdgeFirst Middleware or a PC connected to one you'll need to download the EdgeFirst Samples to follow along with this guide. This provides sample code in Python and Rust as well as the appropriate setup scripts to install the required dependencies.
git clone https://github.com/EdgeFirstAI/samples.git
You'll now have all the samples cloned locally (or remotely) and can now move onto the various sections outlining the specific services and samples demonstrating how to interface with them.
Listing Topics
This is the closest to the "Hello, World!" example, discovering and printing out the available topics. We'll cover the highlights from the full samples available at the EdgeFirst Samples repository.
Complete Python Sample
https://github.com/EdgeFirstAI/samples/blob/main/edgefirst/samples/list-topics/main.py
Complete Rust Sample
https://github.com/EdgeFirstAI/samples/blob/main/src/list-topics.rs
Imports
For this example the zenoh
import is key as we are simply going to be listing the available topics. The other imports in the full sample are for argument parsing and handling timeouts.
import zenoh
Note
The Rust sample doesn't use any explicit imports for Zenoh.
Subscriber
Open a Zenoh session then declare a subscriber for the rt/**
topic which matches all available topics under the rt/
root topic.
# Create the default Zenoh configuration and if the connect argument is
# provided set the mode to client and add the target to the endpoints.
config = zenoh.Config()
if args.connect is not None:
config.insert_json5("mode", "'client'")
config.insert_json5("connect", '{"endpoints": ["%s"]}' % args.connect)
session = zenoh.open(config)
# Create a subscriber for all topics matching the pattern "rt/**"
subscriber = session.declare_subscriber('rt/**')
// Create the default Zenoh configuration and if the connect argument is
// provided set the mode to client and add the target to the endpoints.
let mut config = Config::default();
if let Some(connect) = args.connect {
config.insert_json5("mode", "client").unwrap();
config.insert_json5("connect/endpoints", &connect).unwrap();
}
let session = zenoh::open(config).await.unwrap();
// Create a subscriber for all topics matching the pattern "rt/**"
let subscriber = session.declare_subscriber("rt/**").await.unwrap();
Receive Messages
In a loop we now receive messages and print the topic name and schema. The complete example includes some noise filtering and an optional timeout.
while True:
# Receive the next available message.
msg = subscriber.recv()
# Capture the message encoding MIME type then split on the first ';'
# to get the schema.
schema = str(msg.encoding).split(';', maxsplit=1)[-1]
print("topic: %s → %s" % (topic, schema))
while let Ok(msg) = subscriber.recv() {
// Capture the message encoding MIME type then split on the first ';' to get the schema
let schema = msg.encoding().to_string();
let schema = schema.splitn(2, ';').last().unwrap_or_default();
println!("topic: {} → {}", msg.key_expr(), schema);
}
Results
Running this sample on the target will list the available topics. The example can list topics on a remote device by providing the address of a device with an available zenohd router.
Remote Targets
If running these examples remotely you will need to provide the remote endpoint, for example if your target device has the address 10.1.1.10 then you would use --connect tcp/10.1.1.10:7447
to connect using TCP on the default Zenoh port.
$ python -m edgefirst.samples.list-topics
topic: rt/imu → sensor_msgs/msg/Imu
topic: rt/camera/h264 → foxglove_msgs/msg/CompressedVideo
topic: rt/camera/dma → edgefirst_msgs/msg/DmaBuffer
topic: rt/camera/info → sensor_msgs/msg/CameraInfo
topic: rt/radar/cube → edgefirst_msgs/msg/RadarCube
topic: rt/radar/targets → sensor_msgs/msg/PointCloud2
topic: rt/radar/clusters → sensor_msgs/msg/PointCloud2
topic: rt/tf_static → geometry_msgs/msg/TransformStamped
topic: rt/gps → sensor_msgs/msg/NavSatFix
topic: rt/radar/info → edgefirst_msgs/msg/RadarInfo
$ cargo run --bin list-targets
topic: rt/imu → sensor_msgs/msg/Imu
topic: rt/camera/h264 → foxglove_msgs/msg/CompressedVideo
topic: rt/camera/dma → edgefirst_msgs/msg/DmaBuffer
topic: rt/camera/info → sensor_msgs/msg/CameraInfo
topic: rt/radar/cube → edgefirst_msgs/msg/RadarCube
topic: rt/radar/targets → sensor_msgs/msg/PointCloud2
topic: rt/radar/clusters → sensor_msgs/msg/PointCloud2
topic: rt/tf_static → geometry_msgs/msg/TransformStamped
topic: rt/gps → sensor_msgs/msg/NavSatFix
topic: rt/radar/info → edgefirst_msgs/msg/RadarInfo
That's it! The next examples will dive into the topic schemas and how to parse the message contents and provide examples on visualizing the results using the Rerun tool.