It's not ROS... but it is ROS-y!
rosy, inspired by ROS (Robot Operating System), provides simple and fast inter-process message passing for distributed Python processes, or "nodes".
rosy allows sending messages between nodes in two different ways:
- Topics: Unidirectional, "fire and forget" messages that are sent from a node to all nodes listening to that topic.
- Services: Bidirectional, request-response messages that allow a node to get a response from any node hosting the service.
Messages can contain any Python data that is serializable by pickle (default), json, or msgpack. Alternatively, you can even provide your own custom codec.
Nodes can...
- Run on a single machine, or be distributed across multiple machines on a local network.
- Automatically discover each other using the Zeroconf protocol.
- Automatically reconnect to each other if they lose connection.
rosy also has simple load balancing: if multiple nodes of the same name are listening to a topic, then messages will be sent to them in a round-robin fashion. (The load balancing strategy can be changed or disabled if desired.)
Here are some simplified examples. See the linked files for the full code.
import rosy
async def main():
async with await rosy.build_node(name="topic_sender") as node:
await node.send("some-topic", "hello", name="world")import rosy
async def main():
async with await rosy.build_node(name="topic_listener") as node:
await node.listen("some-topic", callback)
await node.forever()
async def callback(topic, message, name=None):
print(f'Received "{message} {name}" on topic={topic}')Terminal:
# Terminal 1
$ python -m rosy.demo.topic_listener
Listening...
# Terminal 2
$ python -m rosy.demo.topic_sender
# Terminal 1
Received "hello world" on topic=some-topicimport rosy
async def main():
async with await rosy.build_node(name="service_caller") as node:
print("Calculating 2 * 2...")
result = await node.call("multiply", 2, 2)
print(f"Result: {result}")rosy/demo/service_provider.py:
import rosy
async def main():
async with await rosy.build_node(name="service_provider") as node:
await node.add_service("multiply", multiply)
await node.forever()
async def multiply(service, a, b):
return a * bTerminal:
# Terminal 1
$ python -m rosy.demo.service_provider
Started service...
# Terminal 2
$ python -m rosy.demo.service_caller
Calculating 2 * 2...
Result: 4pip install rosyThese commands mirror the ros2 ROS commands. Use the --help flag on any command to see all options.
Display help for all commands.
List all nodes in the mesh, what topics they are listening to, and what services they are providing.
List all topics, echo messages from a topic, or send a message to a topic.
List all services, or call a service.
Launch several nodes all at once. config defaults to launch.yaml. Check out the template launch.yaml for all options, or the demo launch.yaml for a runnable example.
Tool for recording and playing back messages. The options are:
record <topics>: Record messages on the given topic(s) to a file. By default, a file namedrecord_<datetime>.bagwill be created in the current directory.play: Play back messages from a bag file, with the same timing between messages as when they were recorded. By default, the most recent bag file in the current directory will be played back.info: Print information about a bag file. By default, the most recent bag file in the current directory will be used.
Performs a speed test sending and receiving topic messages.
Some results sending messages between two nodes on the same machine:
| Hardware | Message size | Messages/s | Latency (ms) | Bandwidth (MB/s) |
|---|---|---|---|---|
| Laptop* | 0 | 52000 | 0.026 | N/A |
| Laptop* | 1 kB | 50000 | 0.027 | 50 |
| Laptop* | 1 MB | 580 | 1.0 | 580 |
| Orin Nano** | 0 | 13000 | 0.11 | N/A |
| Orin Nano** | 1 kB | 13000 | 0.11 | 13 |
| Orin Nano** | 1 MB | 250 | 3.2 | 250 |
* Dell XPS 17 9730 with an Intel Core i9-13900H CPU and 64 GB DDR5 RAM running Ubuntu 24.04 and Python 3.10.
** NVIDIA Jetson Orin Nano
running Ubuntu 22.04 and Python 3.13.
A mesh is a collection of "nodes" that can send messages to each other. A message can be any Python object. There is one node per Python process, with nodes potentially distributed across multiple machines. Each node listens to specific message "topics", and calls listener callbacks when messages are received on those topics. Each node can send messages to any topic, and the message will be sent to all listening nodes.
The nodes use the Zeroconf protocol to discover each other and share their supported connection specifications, topics they are listening to, and services they are providing. Each node maintains the current "mesh topology" of all other nodes in the mesh.
When a node needs to send a message, it uses the mesh topology to find all currently listening nodes, connects to them, and sends the message.
Nodes will only form a mesh and communicate with other nodes sharing the same "domain ID". The default domain ID is "default".
If you want to have multiple, independent meshes on the same network (e.g. you have multiple robots), you can use a different domain ID for each mesh. Here are some examples:
await rosy.build_node(..., domain_id="my-domain")# The rosy CLI has a dedicated argument
$ rosy --domain-id my-domain ...# Or you can set the ROSY_DOMAIN_ID environment variable
$ ROSY_DOMAIN_ID=my-domain rosy ...rosy only guarantees that topic messages and service requests will be received in the order in which they were sent from a single node. It is possible for messages sent from different nodes to be received out of order.
It does not guarantee topic message delivery; there are no delivery confirmations, and if a message fails to be sent to a node (e.g. due to network failure), it will not be retried.
Security is not a primary concern of rosy. Messages are sent unencrypted for speed and simplicity, and there is no authentication of nodes on the mesh.