LeRobot documentation

Asynchronous Inference

Hugging Face's logo
Join the Hugging Face community

and get access to the augmented documentation experience

to get started

Asynchronous Inference

With our SmolVLA we introduced a new way to run inference on real-world robots, decoupling action prediction from action execution. In this tutorial, we’ll show how to use asynchronous inference (async inference) using a finetuned version of SmolVLA, and all the policies supported by LeRobot. Try async inference with all the policies supported by LeRobot!

What you’ll learn:

  1. Why asynchronous inference matters and how it compares to, more traditional, sequential inference.
  2. How to spin-up a PolicyServer and connect a RobotClient from the same machine, and even over the network.
  3. How to tune key parameters (actions_per_chunk, chunk_size_threshold) for your robot and policy.

If you get stuck, hop into our Discord community!

In a nutshell: with async inference, your robot keeps acting while the policy server is already busy computing the next chunk of actions---eliminating “wait-for-inference” lags and unlocking smoother, more reactive behaviours. This is fundamentally different from synchronous inference (sync), where the robot stays idle while the policy computes the next chunk of actions.


Getting started with async inference

You can read more information on asynchronous inference in our blogpost. This guide is designed to help you quickly set up and run asynchronous inference in your environment.

First, install lerobot with the async tag, to install the extra dependencies required to run async inference.

pip install -e ".[async]"

Then, spin up a policy server (in one terminal, or in a separate machine) specifying the host address and port for the client to connect to. You can spin up a policy server running:

python src/lerobot/scripts/server/policy_server.py \
    --host=127.0.0.1 \
    --port=8080 \

This will start a policy server listening on 127.0.0.1:8080 (localhost, port 8080). At this stage, the policy server is empty, as all information related to which policy to run and with which parameters are specified during the first handshake with the client. Spin up a client with:

python src/lerobot/scripts/server/robot_client.py \
    --server_address=127.0.0.1:8080 \ # SERVER: the host address and port of the policy server
    --robot.type=so100_follower \ # ROBOT: your robot type
    --robot.port=/dev/tty.usbmodem585A0076841 \ # ROBOT: your robot port
    --robot.id=follower_so100 \ # ROBOT: your robot id, to load calibration file
    --robot.cameras="{ laptop: {type: opencv, index_or_path: 0, width: 1920, height: 1080, fps: 30}, phone: {type: opencv, index_or_path: 0, width: 1920, height: 1080, fps: 30}}" \ # POLICY: the cameras used to acquire frames, with keys matching the keys expected by the policy
    --task="dummy" \ # POLICY: The task to run the policy on (`Fold my t-shirt`). Not necessarily defined for all policies, such as `act`
    --policy_type=your_policy_type \ # POLICY: the type of policy to run (smolvla, act, etc)
    --pretrained_name_or_path=user/model \ # POLICY: the model name/path on server to the checkpoint to run (e.g., lerobot/smolvla_base)
    --policy_device=mps \ # POLICY: the device to run the policy on, on the server
    --actions_per_chunk=50 \ # POLICY: the number of actions to output at once
    --chunk_size_threshold=0.5 \ # CLIENT: the threshold for the chunk size before sending a new observation to the server
    --aggregate_fn_name=weighted_average \ # CLIENT: the function to aggregate actions on overlapping portions
    --debug_visualize_queue_size=True # CLIENT: whether to visualize the queue size at runtime

In summary, you need to specify instructions for:

  • SERVER: the address and port of the policy server
  • ROBOT: the type of robot to connect to, the port to connect to, and the local id of the robot
  • POLICY: the type of policy to run, and the model name/path on server to the checkpoint to run. You also need to specify which device should the sever be using, and how many actions to output at once (capped at the policy max actions value).
  • CLIENT: the threshold for the chunk size before sending a new observation to the server, and the function to aggregate actions on overlapping portions. Optionally, you can also visualize the queue size at runtime, to help you tune the CLIENT parameters.

Importantly,

  • actions_per_chunk and chunk_size_threshold are key parameters to tune for your setup.
  • aggregate_fn_name is the function to aggregate actions on overlapping portions. You can either add a new one to a registry of functions, or add your own in robot_client.py (see here)
  • debug_visualize_queue_size is a useful tool to tune the CLIENT parameters.

Done! You should see your robot moving around by now 😉

Async vs. synchronous inference

Synchronous inference relies on interleaving action chunk prediction and action execution. This inherently results in idle frames, frames where the robot awaits idle the policy’s output: a new action chunk. In turn, inference is plagued by evident real-time lags, where the robot simply stops acting due to the lack of available actions. With robotics models increasing in size, this problem risks becoming only more severe.

Synchronous inference makes the robot idle while the policy is computing the next chunk of actions.

To overcome this, we design async inference, a paradigm where action planning and execution are decoupled, resulting in (1) higher adaptability and, most importantly, (2) no idle frames. Crucially, with async inference, the next action chunk is computed before the current one is exhausted, resulting in no idleness. Higher adaptability is ensured by aggregating the different action chunks on overlapping portions, obtaining an up-to-date plan and a tighter control loop.

Asynchronous inference results in no idleness because the next chunk is computed before the current chunk is exhausted.


Start the Policy Server

Policy servers are wrappers around a PreTrainedPolicy interfacing them with observations coming from a robot client. Policy servers are initialized as empty containers which are populated with the requested policy specified in the initial handshake between the robot client and the policy server. As such, spinning up a policy server is as easy as specifying the host address and port. If you’re running the policy server on the same machine as the robot client, you can use localhost as the host address.

Command
API example
python -m lerobot.scripts.server.policy_server \
    --host="localhost" \
    --port=8080

This listens on localhost:8080 for an incoming connection from the associatedRobotClient, which will communicate which policy to run during the first client-server handshake.


Launch the Robot Client

RobotClient is a wrapper around a Robot instance, which RobotClient connects to the (possibly remote) PolicyServer. The RobotClient streams observations to the PolicyServer, and receives action chunks obtained running inference on the server (which we assume to have better computational resources than the robot controller).

Command
API example
python src/lerobot/scripts/server/robot_client.py \
    --server_address=127.0.0.1:8080 \ # SERVER: the host address and port of the policy server
    --robot.type=so100_follower \ # ROBOT: your robot type
    --robot.port=/dev/tty.usbmodem585A0076841 \ # ROBOT: your robot port
    --robot.id=follower_so100 \ # ROBOT: your robot id, to load calibration file
    --robot.cameras="{ laptop: {type: opencv, index_or_path: 0, width: 1920, height: 1080, fps: 30}, phone: {type: opencv, index_or_path: 0, width: 1920, height: 1080, fps: 30}}" \ # POLICY: the cameras used to acquire frames, with keys matching the keys expected by the policy
    --task="dummy" \ # POLICY: The task to run the policy on (`Fold my t-shirt`). Not necessarily defined for all policies, such as `act`
    --policy_type=your_policy_type \ # POLICY: the type of policy to run (smolvla, act, etc)
    --pretrained_name_or_path=user/model \ # POLICY: the model name/path on server to the checkpoint to run (e.g., lerobot/smolvla_base)
    --policy_device=mps \ # POLICY: the device to run the policy on, on the server
    --actions_per_chunk=50 \ # POLICY: the number of actions to output at once
    --chunk_size_threshold=0.5 \ # CLIENT: the threshold for the chunk size before sending a new observation to the server
    --aggregate_fn_name=weighted_average \ # CLIENT: the function to aggregate actions on overlapping portions
    --debug_visualize_queue_size=True # CLIENT: whether to visualize the queue size at runtime

The following two parameters are key in every setup:

Hyperparameter Default What it does
actions_per_chunk 50 How many actions the policy outputs at once. Typical values: 10-50.
chunk_size_threshold 0.7 When the queue is ≤ 50% full, the client sends a fresh observation. Value in [0, 1].
Different values of `actions_per_chunk` and `chunk_size_threshold` do result in different behaviours.

On the one hand, increasing the value of actions_per_chunk will result in reducing the likelihood of ending up with no actions to execute, as more actions will be available when the new chunk is computed. However, larger values of actions_per_chunk might also result in less precise actions, due to the compounding errors consequent to predicting actions over longer timespans.

On the other hand, increasing the value of chunk_size_threshold will result in sending out to the PolicyServer observations for inference more often, resulting in a larger number of updates action chunks, overlapping on significant portions. This results in high adaptability, in the limit predicting one action chunk for each observation, which is in turn only marginally consumed while a new one is produced. This option does also put more pressure on the inference pipeline, as a consequence of the many requests. Conversely, values of chunk_size_threshold close to 0.0 collapse to the synchronous edge case, whereby new observations are only sent out whenever the current chunk is exhausted.

We found the default values of actions_per_chunk and chunk_size_threshold to work well in the experiments we developed for the SmolVLA paper, but recommend experimenting with different values to find the best fit for your setup.

Tuning async inference for your setup

  1. Choose your computational resources carefully. PI0 occupies 14GB of memory at inference time, while SmolVLA requires only ~2GB. You should identify the best computational resource for your use case keeping in mind smaller policies require less computational resources. The combination of policy and device used (CPU-intensive, using MPS, or the number of CUDA cores on a given NVIDIA GPU) directly impacts the average inference latency you should expect.
  2. Adjust your fps based on inference latency. While the server generates a new action chunk, the client is not idle and is stepping through its current action queue. If the two processes happen at fundamentally different speeds, the client might end up with an empty queue. As such, you should reduce your fps if you consistently run out of actions in queue.
  3. Adjust chunk_size_threshold.
    • Values closer to 0.0 result in almost sequential behavior. Values closer to 1.0 → send observation every step (more bandwidth, relies on good world-model).
    • We found values around 0.5-0.6 to work well. If you want to tweak this, spin up a RobotClient setting the --debug-visualize-queue-size to True. This will plot the action queue size evolution at runtime, and you can use it to find the value of chunk_size_threshold that works best for your setup.

The action queue size is plotted at runtime when the `--debug-visualize-queue-size` flag is passed, for various levels of `chunk_size_threshold` (`g` in the SmolVLA paper).


Conclusion

Asynchronous inference represents a significant advancement in real-time robotics control, addressing the fundamental challenge of inference latency that has long plagued robotics applications. Through this tutorial, you’ve learned how to implement a complete async inference pipeline that eliminates idle frames and enables smoother, more reactive robot behaviors.

Key Takeaways:

  • Paradigm Shift: Async inference decouples action prediction from execution, allowing robots to continue acting while new action chunks are computed in parallel
  • Performance Benefits: Eliminates “wait-for-inference” lags that are inherent in synchronous approaches, becoming increasingly important as policy models grow larger
  • Flexible Architecture: The server-client design enables distributed computing, where inference can run on powerful remote hardware while maintaining real-time robot control
  • Tunable Parameters: Success depends on properly configuring actions_per_chunk and chunk_size_threshold for your specific hardware, policy, and task requirements
  • Universal Compatibility: Works with all LeRobot-supported policies, from lightweight ACT models to vision-language models like SmolVLA

Start experimenting with the default parameters, monitor your action queue sizes, and iteratively refine your setup to achieve optimal performance for your specific use case. If you want to discuss this further, hop into our Discord community, or open an issue on our GitHub repository.

< > Update on GitHub