| Humble | Jazzy | Kilted | |
|---|---|---|---|
| Pixi | |||
| colcon |
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 viarclcpp - 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.
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.
- 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
For detailed architecture documentation, see docs/ARCHITECTURE.md.
All dependencies (IXWebSocket, spdlog, nlohmann_json, ZSTD) are provided by the dependency manager — nothing is vendored except tl::expected.
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 testStandard 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 --verbosecd 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)# Via pixi (154 unit tests)
pixi run -e humble test
# Or manually after building
colcon test --packages-select pj_bridge && colcon test-result --verbose# 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.0pj_bridge_fastdds --domains 0 1 --port 8080 --publish-rate 50 --session-timeout 10| 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 |
# 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 /topic2For the full API protocol documentation (commands, responses, binary wire format), see docs/API.md.
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- Verify server is running:
ps aux | grep pj_bridge - Check topics are being published:
ros2 topic list - Verify heartbeat is being sent (required every 1 second)
- Check server logs:
ros2 run pj_bridge pj_bridge_node --ros-args --log-level debug
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>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.0pj_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.
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
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
You must share modifications to pj_bridge only if you:
- Distribute modified versions to others (e.g., shipping a modified binary), OR
- 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
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:
- Modify the software, AND
- Provide it as a service to external users
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.