Skip to content

PlotJuggler/plotjuggler_bridge

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

114 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

pj_bridge

Humble Jazzy Kilted
Pixi Pixi: Humble Pixi: Jazzy Pixi: Kilted
colcon ROS: Humble ROS: Jazzy ROS: Kilted

A high-performance bridge server that forwards middleware topic content over WebSocket to PlotJuggler clients. Three backends share a common core:

  • ROS2 (pj_bridge_ros2) — ROS2 Humble / Jazzy via rclcpp
  • FastDDS (pj_bridge_fastdds) — eProsima Fast DDS 3.4 (standalone, no ROS2 required)
  • RTI (pj_bridge_rti) — RTI Connext DDS (build disabled, code preserved)

Even if primarily created for PlotJuggler, this can be considered a general purpose DDS-to-Websocket bridge and be used independently.

Overview

pj_bridge enables clients to subscribe to topics and receive aggregated messages at 50 Hz without needing a full middleware installation. This is useful for visualization tools like PlotJuggler, remote monitoring, and lightweight clients.

Key Features

  • No DDS Required: Clients connect via WebSocket (single port) without needing ROS2/DDS installed
  • High Performance: 50 Hz message aggregation with ZSTD compression
  • Multi-Client Support: Multiple clients can connect simultaneously with shared subscriptions
  • Session Management: Automatic cleanup of disconnected clients via heartbeat monitoring and WebSocket close events
  • Runtime Schema Discovery: Automatic extraction of message schemas from installed ROS2 packages
  • Zero-Copy Design: Efficient message handling using shared pointers and move semantics
  • Large Message Stripping: Automatic stripping of large array fields (Image, PointCloud2, LaserScan, OccupancyGrid) to reduce bandwidth while preserving metadata
  • Type-Safe Error Handling: Comprehensive error reporting using tl::expected

Architecture

For detailed architecture documentation, see docs/ARCHITECTURE.md.

Building

All dependencies (IXWebSocket, spdlog, nlohmann_json, ZSTD) are provided by the dependency manager — nothing is vendored except tl::expected.

ROS2 backend — Pixi (recommended)

Pixi manages the full toolchain including ROS2 via RoboStack.

git clone <repository_url> pj_bridge && cd pj_bridge

# Humble
pixi run -e humble build
pixi run -e humble test

# Jazzy
pixi run -e jazzy build
pixi run -e jazzy test

ROS2 backend — colcon

Standard ROS2 build using colcon. Dependencies are installed via rosdep; only IXWebSocket is fetched automatically via CMake FetchContent.

# Set up workspace
mkdir -p ~/ws_plotjuggler/src && cd ~/ws_plotjuggler/src
git clone <repository_url> pj_bridge

# Install dependencies
source /opt/ros/${ROS_DISTRO}/setup.bash
rosdep install --from-paths pj_bridge --ignore-src -y

# Build and test
cd ~/ws_plotjuggler
colcon build --packages-select pj_bridge --cmake-args -DCMAKE_BUILD_TYPE=Release
colcon test --packages-select pj_bridge && colcon test-result --verbose

FastDDS backend — Conan

cd pj_bridge
conan install . --output-folder=build_fastdds --build=missing -s build_type=Release
cd build_fastdds
cmake .. -DCMAKE_BUILD_TYPE=Release -DENABLE_FASTDDS=ON \
         -DCMAKE_TOOLCHAIN_FILE=conan_toolchain.cmake
make -j$(nproc)

Running Tests

# Via pixi (154 unit tests)
pixi run -e humble test

# Or manually after building
colcon test --packages-select pj_bridge && colcon test-result --verbose

Usage

Starting the Server

ROS2 backend

# Default (port 8080, 50 Hz, 10 s timeout)
ros2 run pj_bridge pj_bridge_ros2

# Custom
ros2 run pj_bridge pj_bridge_ros2 --ros-args \
  -p port:=9090 -p publish_rate:=50.0 -p session_timeout:=10.0

FastDDS backend

pj_bridge_fastdds --domains 0 1 --port 8080 --publish-rate 50 --session-timeout 10

Configuration Parameters

Parameter Type Default Description
port int 9090 WebSocket server port
publish_rate double 50.0 Aggregation publish rate in Hz
session_timeout double 10.0 Client timeout duration in seconds
strip_large_messages bool true Strip large arrays from Image, PointCloud2, LaserScan, OccupancyGrid messages

Testing with Sample Data

# Terminal 1: Play rosbag
source /opt/ros/humble/setup.bash
ros2 bag play /path/to/sample.mcap --loop

# Terminal 2: Run server
source /opt/ros/humble/setup.bash
source ~/ws_plotjuggler/install/setup.bash
ros2 run pj_bridge pj_bridge_node

# Terminal 3: Run Python test client
cd ~/ws_plotjuggler/src/pj_bridge
python3 tests/integration/test_client.py --subscribe /topic1 /topic2

API Protocol

For the full API protocol documentation (commands, responses, binary wire format), see docs/API.md.

Troubleshooting

Server fails to start with "Failed to listen on port"

Another process is using the port. Either kill the conflicting process or use a custom port:

ros2 run pj_bridge pj_bridge_node --ros-args -p port:=9090

Client receives no data

  1. Verify server is running: ps aux | grep pj_bridge
  2. Check topics are being published: ros2 topic list
  3. Verify heartbeat is being sent (required every 1 second)
  4. Check server logs: ros2 run pj_bridge pj_bridge_node --ros-args --log-level debug

"Failed to get schema for topic" error

The message type's .msg file was not found. Ensure the ROS2 package containing the message type is installed and sourced:

ros2 interface show <package_name>/msg/<MessageType>

Session timeout / Automatic unsubscription

The client stopped sending heartbeats. Ensure the client sends a heartbeat every 1 second. The default timeout is 10 seconds. Increase if needed:

ros2 run pj_bridge pj_bridge_node --ros-args -p session_timeout:=20.0

License

pj_bridge is licensed under the GNU Affero General Public License v3.0 (AGPL-3.0).

Copyright (C) 2026 Davide Faconti

This program is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version.

See the LICENSE file for the full license text.


License FAQ

Can I use this software commercially?

Yes, absolutely. The AGPL does not restrict commercial use. You can:

  • Use pj_bridge in commercial products and services
  • Deploy it in production environments for profit

Does using pj_bridge affect my proprietary software?

No, it does not. Because pj_bridge is a standalone application that communicates via inter-process communication (WebSocket), it does not impose license restrictions on:

  • Your ROS2 nodes and packages
  • Client applications connecting to the bridge
  • Other software running on the same system
  • Proprietary code that publishes to or subscribes from ROS2 topics

When do I need to share my code?

You must share modifications to pj_bridge only if you:

  1. Distribute modified versions to others (e.g., shipping a modified binary), OR
  2. Provide the modified software as a network service to external users

You do NOT need to share code if you:

  • Use pj_bridge unmodified (even commercially)
  • Modify it for internal use only within your organization
  • Connect proprietary clients or ROS2 nodes to the bridge

What about the AGPL "network" clause?

The AGPL's network provision states that users who interact with the software over a network should have access to the source code. However, this only applies if you:

  1. Modify the software, AND
  2. Provide it as a service to external users

I'm still concerned about licensing. What should I do?

If you're using pj_bridge unmodified, you have nothing to worry about - there are zero licensing obligations. If still concerned, contact me for alternative licensing options.

About

No description, website, or topics provided.

Resources

License

Stars

Watchers

Forks

Packages

 
 
 

Contributors