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_countfor 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
CommandTimeoutErrorwhen the response does not arrive in time. availability_transition_counttracks 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 communicationpydantic>=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
- Protocol documentation: See
s&b_procol.md - BLE GATT specification: https://www.bluetooth.com/specifications/specs/core-specification/
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
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
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
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
8b5966fe493806b2964a318cdae27430b2ea42c8b0225d7ae8d8bae3a2b6f1aa
|
|
| MD5 |
a800c72cef1d42642947374faf8d2feb
|
|
| BLAKE2b-256 |
6dce813201c83e80937d9b1172145c57c3f7ffeeadf626f24f5e875dfb31bb34
|
File details
Details for the file storzandbickel_ble-0.1.5-py3-none-any.whl.
File metadata
- Download URL: storzandbickel_ble-0.1.5-py3-none-any.whl
- Upload date:
- Size: 33.0 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.14.2
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
c5cbfa4a0633d8de8a8599ad1586362c790824a1d91080408630383b047b9b9c
|
|
| MD5 |
93e032be6f4bf78cbbf114c5d28bd803
|
|
| BLAKE2b-256 |
9b25964b2641433ebaa9b657b0b097d1ed81555e09d1368be3a2476eb1025b61
|