Skip to main content

Asynchronous Python library for RPLidar C1 laser scanners with robust data handling, error recovery, and minimal dependencies (pyserial only). Designed for efficient robotics and sensing applications.

Project description

RPLidarC1

A Python library for interfacing with the RPLidar C1 360-degree laser scanner. This library provides an asynchronous API for controlling the RPLidar device and processing scan data.

Features

  • Asynchronous API for non-blocking operation
  • Simple interface for connecting to and controlling the RPLidar device
  • Health status checking
  • Device reset functionality
  • Scan data processing with angle and distance measurements
  • Support for both queue-based and dictionary-based data output
  • Robust byte alignment handling for reliable data parsing
  • Improved asynchronous data processing with error recovery

Requirements

  • Python 3.10+ (for asyncio TaskGroup support)
  • pyserial

Installation

# Clone the repository
git clone https://github.com/dsaadatmandi/rplidarc1.git
cd rplidarc1

# Install dependencies
pip install pyserial

Usage

Basic Usage

from scanner import RPLidar
import asyncio

# Initialize the RPLidar with the appropriate port and baudrate
lidar = RPLidar("/dev/ttyUSB0", 460800)

# Perform a simple scan
async def scan_example():
    # Start a scan and get the coroutine
    scan_coroutine = lidar.simple_scan()
    
    # Create a task to process the scan data
    async with asyncio.TaskGroup() as tg:
        tg.create_task(scan_coroutine)
        # Add other tasks as needed
    
    # Reset the device when done
    lidar.reset()

# Run the example
try:
    asyncio.run(scan_example())
except KeyboardInterrupt:
    # Ensure proper shutdown on keyboard interrupt
    lidar.reset()

Processing Scan Data

from scanner import RPLidar
import asyncio

lidar = RPLidar("/dev/ttyUSB0", 460800)

async def process_scan_data():
    # Start a scan with dictionary output
    async with asyncio.TaskGroup() as tg:
        # Create a task to stop scanning after 5 seconds
        tg.create_task(wait_and_stop(5, lidar.stop_event))
        
        # Create a task to process data from the queue
        tg.create_task(process_queue(lidar.output_queue, lidar.stop_event))
        
        # Start the scan with dictionary output
        tg.create_task(lidar.simple_scan(make_return_dict=True))
    
    # Access the scan data dictionary
    print(lidar.output_dict)
    
    # Reset the device
    lidar.reset()

async def wait_and_stop(seconds, event):
    await asyncio.sleep(seconds)
    event.set()

async def process_queue(queue, stop_event):
    while not stop_event.is_set():
        if not queue.empty():
            data = await queue.get()
            # Process the data
            print(f"Angle: {data['a_deg']}°, Distance: {data['d_mm']}mm, Quality: {data['q']}")
        else:
            await asyncio.sleep(0.1)

# Run the example
try:
    asyncio.run(process_scan_data())
except KeyboardInterrupt:
    lidar.reset()

API Reference

RPLidar Class

RPLidar(port="/dev/ttyUSB0", baudrate=460800, timeout=0.2)

Parameters:

  • port (str): Serial port to connect to
  • baudrate (int): Baud rate for the serial connection
  • timeout (float): Timeout for serial operations

Methods:

  • healthcheck(): Check the health status of the device
  • shutdown(): Properly shut down the device
  • reset(): Reset the device
  • simple_scan(make_return_dict=False): Start a scan and return a coroutine for processing scan data
    • make_return_dict (bool): If True, scan data will also be stored in output_dict

Properties:

  • output_queue (asyncio.Queue): Queue containing scan data
  • output_dict (dict): Dictionary containing scan data (angle -> distance)
  • stop_event (asyncio.Event): Event to signal stopping the scan

Data Format

Scan data is provided in the following format:

{
    "a_deg": 45.33,  # Angle in degrees (0-360)
    "d_mm": 1250,    # Distance in millimeters
    "q": 15          # Quality of the measurement (0-63)
}

Advanced Features

Byte Alignment Handling

The library implements robust byte alignment verification to ensure reliable data parsing from the RPLidar device. The protocol requires specific bit patterns for proper data alignment:

  • The S bit (least significant bit of the first byte) and S̄ bit (second least significant bit of the first byte) must be different (one 0, one 1).
  • The C bit (least significant bit of the second byte) must be set to 1.

When misalignment is detected, the library automatically realigns the data stream by shifting one byte at a time until proper alignment is restored. This ensures that scan data is accurately parsed even when communication errors occur.

Improved Asynchronous Processing

The library uses modern asyncio features to provide efficient non-blocking operation:

  • Utilizes asyncio.TaskGroup for structured concurrency
  • Implements error recovery mechanisms to handle communication interruptions
  • Provides both queue-based and dictionary-based data output options
  • Uses asyncio.Event for clean termination of scanning operations

Project Structure

  • scanner.py: Main RPLidar class implementation
  • protocol.py: Implementation of the RPLidar communication protocol
  • serial_handler.py: Serial connection management
  • utils.py: Utility classes and functions
  • examples/: Example scripts demonstrating usage
  • tests/: Unit and integration tests

Testing

The project includes a comprehensive test suite that covers all major components of the library. The tests use pytest and mock the hardware to allow testing without an actual RPLidar device.

Running Tests

To run the tests, first install the testing dependencies:

pip install -r requirements.txt

Then run the tests using pytest:

pytest

For more information about the tests, see the tests README.

License

MIT License

Contributing

Contributions are welcome! Please feel free to submit a Pull Request.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

rplidarc1-0.1.3.tar.gz (11.4 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

rplidarc1-0.1.3-py3-none-any.whl (12.8 kB view details)

Uploaded Python 3

File details

Details for the file rplidarc1-0.1.3.tar.gz.

File metadata

  • Download URL: rplidarc1-0.1.3.tar.gz
  • Upload date:
  • Size: 11.4 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: poetry/2.1.2 CPython/3.13.2 Linux/6.11.0-21-generic

File hashes

Hashes for rplidarc1-0.1.3.tar.gz
Algorithm Hash digest
SHA256 5c69fdf93222e4f6df8ee26fada86824907d52035490e5d90eb44a6c9ffb134e
MD5 b8d368ebc33f8ee9f22586e8a2c84714
BLAKE2b-256 7b0cfe6a45c31dc1675928f05f64018c97daeeb567c1bce6890eeb127800ab94

See more details on using hashes here.

File details

Details for the file rplidarc1-0.1.3-py3-none-any.whl.

File metadata

  • Download URL: rplidarc1-0.1.3-py3-none-any.whl
  • Upload date:
  • Size: 12.8 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: poetry/2.1.2 CPython/3.13.2 Linux/6.11.0-21-generic

File hashes

Hashes for rplidarc1-0.1.3-py3-none-any.whl
Algorithm Hash digest
SHA256 167cca82b412ca942f34ff0812904336cdfa963da8c19d7361ac6d71aed5e022
MD5 9d9fd5e494d4c08265b9727442b7f413
BLAKE2b-256 39e4e027439d0627ff8920d2bd663342f61381d685f25a0b160e618207865dfa

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page