Skip to main content

Python library for controlling Storz & Bickel devices via Bluetooth Low Energy (BLE)

Project description

Storz & Bickel BLE Python Library

A Python library for controlling Storz & Bickel vaporizers (Volcano Hybrid, Venty/Veazy, and Crafty/Crafty+) via Bluetooth Low Energy (BLE).

Features

  • Device Support: Volcano Hybrid, Venty, Veazy (qvap path), and Crafty/Crafty+
  • Async/Await: Built with modern Python async/await patterns
  • Type Safe: Full type hints and Pydantic models for data validation
  • Home Assistant Ready: Follows Home Assistant best practices
  • Controls: Temperature/heater/pump, boost/superboost, and settings APIs
  • Diagnostics: Local run_analysis() helpers (no cloud upload)
  • Workflows: Volcano preset workflow execution (balloon, flow1, flow2, flow3)

Installation

Using pip

pip install storzandbickel-ble

Using uv

uv pip install storzandbickel-ble

Quick Start

Basic Usage

import asyncio
from storzandbickel_ble import StorzBickelClient

async def main():
    # Create client
    client = StorzBickelClient()
    
    # Scan for devices
    devices = await client.scan(timeout=10.0)
    print(f"Found {len(devices)} devices")
    
    # Connect to first device
    if devices:
        device = await client.connect_device(devices[0])
        
        # Set target temperature
        await device.set_target_temperature(185.0)
        
        # Turn heater on
        await device.turn_heater_on()
        
        # Read current state
        print(f"Current temperature: {device.state.current_temperature}°C")
        print(f"Target temperature: {device.state.target_temperature}°C")
        
        # Disconnect
        await device.disconnect()

asyncio.run(main())

Home Assistant-Oriented Quickstart

The library patterns map cleanly to HA coordinator/entity flows:

  • discover/connect once
  • keep notifications enabled as the primary state path
  • call update_state() for startup reconciliation or recovery
  • use explicit exceptions and availability_transition_count for clean availability handling

Device-Specific Examples

Crafty/Crafty+

import asyncio
from storzandbickel_ble import StorzBickelClient
from storzandbickel_ble.models import DeviceType

async def main():
    client = StorzBickelClient()
    
    # Scan for Crafty devices
    devices = await client.scan(timeout=10.0, device_type=DeviceType.CRAFTY)
    
    if devices:
        # Connect to first Crafty device
        crafty = await client.connect_device(devices[0])
        
        # Update state to get current values
        await crafty.update_state()
        
        # Check temperatures
        print(f"Current temperature: {crafty.state.current_temperature}°C")
        print(f"Target temperature: {crafty.state.target_temperature}°C")
        
        # Set target temperature
        await crafty.set_target_temperature(185.0)
        print(f"Set target temperature to 185°C")
        
        # Verify the change
        await crafty.update_state()
        print(f"New target temperature: {crafty.state.target_temperature}°C")
        
        await crafty.disconnect()

asyncio.run(main())

For additional per-device examples (Venty, Veazy, Volcano), see docs/usage.rst.

Workflow and Analysis APIs

import asyncio
from storzandbickel_ble import StorzBickelClient

async def main():
    client = StorzBickelClient()
    volcano = await client.connect_by_name("S&B VOLCANO")

    # Volcano workflow presets: balloon, flow1, flow2, flow3
    await volcano.run_workflow_preset("flow1")

    # Local diagnostics summary (no cloud upload)
    report = await volcano.run_analysis()
    print(report["ok"], report["warnings"], report["errors"])

    await volcano.disconnect()

asyncio.run(main())

Device Support Matrix

Device Temperature Heater Battery Pump Boost/Superboost Vibration Brightness Workflow Presets Local Analysis
Volcano Hybrid
Venty
Veazy
Crafty/Crafty+

Supported Functions by Family

Function Volcano Venty Veazy Crafty
Discovery and connect
Real-time notification state
Temperature target control
Heater control
Pump control
Brightness and vibration settings
Local diagnostics (run_analysis)
Workflow presets

Data Update Model

  • Notification-driven updates are enabled on connect for supported characteristics.
  • update_state() performs explicit reads/commands and is useful for an immediate refresh.
  • Home Assistant integrations can treat notifications as the primary state stream and call update_state() for recovery or reconciliation.

Concurrency and Command Semantics

  • Device-level BLE I/O is serialized internally to avoid overlapping reads/writes.
  • Commands that require responses (for example, qvap command exchange) raise CommandTimeoutError when the response does not arrive in time.
  • availability_transition_count tracks connect/disconnect transitions to help downstream integrations avoid noisy up/down logging.

Diagnostics Snapshot API

Each device exposes a sanitized diagnostics payload via:

snapshot = device.get_diagnostics_snapshot()

Snapshot payloads intentionally omit serial numbers from state to reduce sensitive data exposure in logs and diagnostics exports.

Known Limitations

  • Firmware update workflows are not implemented in this library.
  • Vendor cloud upload paths are intentionally out of scope.
  • Some frontend-specific maintenance/analysis flows are not yet mirrored as first-class Python APIs.
  • BLE behavior can vary by adapter, OS, and stack implementation.

Troubleshooting

  • Ensure your BLE adapter is enabled and supports Bluetooth Low Energy.
  • If a device is not discovered, retry scanning and confirm it is advertising.
  • For flaky connections, disconnect and reconnect before issuing control commands.
  • If command operations timeout, retry with a higher timeout and reduced command burst frequency.
  • Use get_diagnostics_snapshot() when reporting issues to provide a sanitized runtime context.

Documentation

Full documentation is available at Read the Docs.

Requirements

  • Python 3.14+
  • Bluetooth adapter with BLE support
  • Linux, macOS, or Windows

Dependencies

  • bleak>=0.21.0 - BLE communication
  • pydantic>=2.0.0 - Data validation

Development

Setup

# Clone repository
git clone https://github.com/yourusername/storzandbickel-ble.git
cd storzandbickel-ble

# Install with uv
uv pip install -e ".[dev]"

# Or with pip
pip install -e ".[dev]"

Running Tests

uv run pytest

Linting

uv run ruff check .
uv run ruff format .

Type Checking

uv run mypy src

Docker

Build

docker build -t storzandbickel-ble .

Run

docker run --rm storzandbickel-ble

Development

docker-compose up dev

License

MIT License - see LICENSE file for details

Disclaimer

This library is based on reverse engineering and is not officially supported by Storz & Bickel. Use at your own risk. The authors are not responsible for any damage to devices.

Contributing

Contributions are welcome! Please open an issue or submit a pull request.

References

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

storzandbickel_ble-0.1.5.tar.gz (94.1 kB view details)

Uploaded Source

Built Distribution

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

storzandbickel_ble-0.1.5-py3-none-any.whl (33.0 kB view details)

Uploaded Python 3

File details

Details for the file storzandbickel_ble-0.1.5.tar.gz.

File metadata

  • Download URL: storzandbickel_ble-0.1.5.tar.gz
  • Upload date:
  • Size: 94.1 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.14.2

File hashes

Hashes for storzandbickel_ble-0.1.5.tar.gz
Algorithm Hash digest
SHA256 8b5966fe493806b2964a318cdae27430b2ea42c8b0225d7ae8d8bae3a2b6f1aa
MD5 a800c72cef1d42642947374faf8d2feb
BLAKE2b-256 6dce813201c83e80937d9b1172145c57c3f7ffeeadf626f24f5e875dfb31bb34

See more details on using hashes here.

File details

Details for the file storzandbickel_ble-0.1.5-py3-none-any.whl.

File metadata

File hashes

Hashes for storzandbickel_ble-0.1.5-py3-none-any.whl
Algorithm Hash digest
SHA256 c5cbfa4a0633d8de8a8599ad1586362c790824a1d91080408630383b047b9b9c
MD5 93e032be6f4bf78cbbf114c5d28bd803
BLAKE2b-256 9b25964b2641433ebaa9b657b0b097d1ed81555e09d1368be3a2476eb1025b61

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