Skip to content

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.

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.