A lightweight, open-source communication bridge between the Basilisk astrodynamics simulation framework and ROS 2, enabling real-time, bidirectional data exchange.
This repository provides a ROS 2 interface for the Basilisk astrodynamics framework that enables spacecraft simulations to interact directly with ROS 2 nodes without requiring any modifications to Basilisk's core. The bridge connects Basilisk's high-fidelity spacecraft dynamics and simulation timing with ROS 2's distributed middleware.
Spacecraft states generated in Basilisk are published as ROS 2 topics, while commands such as forces, torques, or thruster states can be sent from ROS 2 back into the simulation. This allows standard ROS 2 tools and workflows (e.g., rosbag, RViz, PlotJuggler) to be applied directly to spacecraft simulations, and supports modular autonomy, estimation, control, and monitoring software running externally in ROS 2.
The bridge supports single- and multi-spacecraft scenarios using namespace-aware topic conventions, enabling scalable setups such as formation flying and coordinated control.
If you use this package in academic work, please cite:
E. Krantz, N. N. Chan, G. Tibert, H. Mao, C. Fuglesang, Bridging the Basilisk Astrodynamics Framework with ROS 2 for Modular Spacecraft Simulation and Hardware Integration, arXiv preprint arXiv:2512.09833, 2025. https://arxiv.org/abs/2512.09833
BibTeX
@article{krantz2025bridging,
title={Bridging the Basilisk Astrodynamics Framework with ROS 2 for Modular Spacecraft Simulation and Hardware Integration},
author={Krantz, Elias and Chan, Ngai Nam and Tibert, Gunnar and Mao, Huina and Fuglesang, Christer},
journal={arXiv preprint arXiv:2512.09833},
year={2025}
}| Environment | ROS 2 Distro | Notes |
|---|---|---|
| Ubuntu 22.04 LTS | Humble | Full build, CLI tools, and launch files tested |
| Ubuntu 24.04 LTS | Jazzy | Full build, CLI tools, and launch files tested |
| Windows (WSL, Ubuntu 24.04) | Rolling | Build and topic communication verified |
WSL note: Locale settings may need correction (see ROS 2 docs). Minor GUI latency (e.g., BSK-Vizard) is expected; CLI tools work normally.
After installation of Basilisk, it is recommended to export the user path of Basilisk to .bashrc as:
export BSK_PATH=your_BSK_pathcd your_ros2_workspace/src
git clone https://github.com/DISCOWER/bsk-msgs.git
git clone https://github.com/DISCOWER/bsk-ros2-bridge.git
cd ..
colcon build --packages-select bsk-msgs bsk-ros2-bridge
source install/setup.bash
pip install -r src/bsk-ros2-bridge/requirements.txt# Terminal 1: start the bridge
ros2 launch bsk-ros2-bridge bridge.launch.py
# Terminal 2: start a Basilisk scenario (requires the BSK environment)
source $BSK_PATH/.venv/bin/activate
python examples/scenarioRosBasic_wrench.py
# Terminal 3: start a controller (e.g., BSK-ROS2-MPC)
ros2 launch bsk-ros2-mpc mpc.launch.py namespace:=bskSat0 type:=wrench use_hill:=FalseNote: In the basic example scenario, no orbit dynamics are present and the example MPC needs the launch argument use_hill:=False. Set to True for one of the orbit examples.
For closed-loop control, see the BSK-ROS 2 MPC Controller.
| Scenario | Description | Control Mode |
|---|---|---|
scenarioRosBasic_da.py |
Single spacecraft (no Earth) | da |
scenarioRosBasic_wrench.py |
Single spacecraft (no Earth) | wrench |
scenarioRosOrbit_da.py |
Spacecraft orbiting Earth | da |
scenarioRosOrbit_wrench.py |
Spacecraft orbiting Earth | wrench |
scenarioRosLeaderFollowerBasic_wrench.py |
1 leader + 2 followers (no Earth) | wrench |
scenarioRosLeaderFollowerOrbit_wrench.py |
1 leader + 2 followers (orbit) | wrench |
Control modes: da (direct allocation) commands individual thrusters via /<ns>/bsk/in/thr_array_cmd_force. wrench commands 3D forces and torques via /<ns>/bsk/in/cmd_force and /<ns>/bsk/in/cmd_torque, mapped to thrusters by Basilisk.
The orbit scenarios support any number of spacecraft. Launch one controller per namespace (e.g., /bskSat0, /bskSat1).
from bsk_module.rosBridgeHandler import RosBridgeHandler
ros_bridge = RosBridgeHandler()
ros_bridge.ModelTag = "ros_bridge"
# Add publishers (Basilisk → ROS 2)
ros_bridge.add_ros_publisher('SCStatesMsgPayload', 'SCStatesMsgIn', 'sc_states', 'bskSat', max_rate=100.0)
# Add subscribers (ROS 2 → Basilisk)
ros_bridge.add_ros_subscriber('THRArrayCmdForceMsgPayload', 'THRArrayCmdForceMsgOut', 'thr_array_cmd_force', 'bskSat')
# Connect to Basilisk messages
ros_bridge.bskSat.SCStatesMsgIn.subscribeTo(scObject.scStateOutMsg)
thrFiringSchmittObj.thrForceInMsg.subscribeTo(ros_bridge.bskSat.THRArrayCmdForceMsgOut)
# Add the module to a simulation task
scSim.AddModelToTask(simTaskName, ros_bridge)# Publisher: Sends Basilisk data to ROS 2
ros_bridge.add_ros_publisher(msg_type_name, handler_name, topic_name, namespace, max_rate=None)
# Subscriber: Receives ROS 2 commands in Basilisk
ros_bridge.add_ros_subscriber(msg_type_name, handler_name, topic_name, namespace)Parameters:
msg_type_name- Basilisk message type (e.g.,'SCStatesMsgPayload','CmdForceBodyMsgPayload')handler_name- Internal message handler (e.g.,'SCStatesMsgIn','CmdForceBodyMsgOut')topic_name- ROS 2 topic name (e.g.,'sc_states','cmd_force')namespace- Spacecraft identifier (e.g.,'bskSat','bskSat0')max_rate- (Publishers only) optional maximum publishing rate (Hz). If not specified, publishes at the task rate.
Topics follow the pattern: /<namespace>/bsk/<in|out>/<topic_name>
Common Topics:
/clock- Simulation time synchronization (only in non-realtime mode)/<namespace>/bsk/out/sc_states- Spacecraft states/<namespace>/bsk/in/thr_array_cmd_force- Thruster commands/<namespace>/bsk/in/cmd_force- Force commands/<namespace>/bsk/in/cmd_torque- Torque commands
In non-realtime mode (accelFactor is not 1), the bridge publishes simulation time to /clock. For time synchronization, ROS 2 nodes must use use_sim_time:=true in this case. In realtime mode, /clock is not published and nodes use system time.
The clock update interval can be configured via the clock_timestep:
ros2 launch bsk-ros2-bridge bridge.launch.py clock_timestep:=0.1All topics use the following ROS 2 Quality of Service (QoS):
- Reliability: BEST_EFFORT
- Durability: VOLATILE
- History: KEEP_LAST
- Queue depth: 1
This configuration prioritizes low-latency communication and ensures that only the most recent message is delivered, suitable for high-rate spacecraft simulation data.
Bridge launch arguments:
| Argument | Default | Description |
|---|---|---|
sub_port |
5550 | ZMQ port: Basilisk → Bridge |
pub_port |
5551 | ZMQ port: Bridge → Basilisk |
heartbeat_port |
5552 | ZMQ heartbeat port |
clock_timestep |
0.01 | Clock update interval (s) |
namespace |
'' |
Bridge namespace |
To set custom ports:
ros2 launch bsk-ros2-bridge bridge.launch.py sub_port:=6550 pub_port:=6551 heartbeat_port:=6552ros_bridge = RosBridgeHandler(send_port=6550, receive_port=6551, heartbeat_port=6552)RosBridgeHandler arguments:
| Parameter | Default | Description |
|---|---|---|
ModelTag |
'ros_bridge' |
Module identifier for logging |
send_port |
5550 | ZMQ port for sending data to bridge (BSK → ROS 2) |
receive_port |
5551 | ZMQ port for receiving data from bridge (ROS 2 → BSK) |
heartbeat_port |
5552 | ZMQ port for heartbeat monitoring |
accelFactor |
NaN | Simulation speed factor; set to enable /clock publishing |
Missing message types: ensure bsk_msgs is built and sourced.
Port conflicts: check with lsof -i :5550 and kill occupied ports if needed.
Connection issues: ensure bridge is running, verify BSK environment is activated.
