A library to communicate with Ooler Sleep System Bluetooth devices.
Project description
ooler-ble-client
A Python library to communicate with Ooler Sleep System Bluetooth devices via BLE GATT characteristics. Built on bleak and bleak-retry-connector.
Designed for use with the Home Assistant Ooler integration, but can be used standalone.
Installation
pip install ooler-ble-client
Usage
import asyncio
from bleak import BleakScanner
from ooler_ble_client import OolerBLEDevice
async def main():
# Discover the device
device = await BleakScanner.find_device_by_name("OOLER")
# Create client and connect
client = OolerBLEDevice(model="OOLER")
client.set_ble_device(device)
await client.connect()
# Read state
print(client.state)
# Control the device
await client.set_power(True)
await client.set_temperature(72)
await client.set_mode("Regular")
# Listen for state changes
def on_state_change(state):
print(f"State changed: {state}")
unsubscribe = client.register_callback(on_state_change)
# Clean up
unsubscribe()
await client.stop()
asyncio.run(main())
API
OolerBLEDevice(model: str)
Main client class.
set_ble_device(device)-- set the BLE device to connect toconnect()-- establish BLE connection, read initial state, subscribe to notificationsstop()-- unsubscribe from notifications and disconnectis_connected-- whether the device is currently connectedstate-- currentOolerBLEStateregister_callback(fn)-- register a state change callback, returns an unsubscribe functionasync_poll()-- read all characteristics from the deviceset_power(bool)-- turn device on/off (re-sends mode and temperature on power-on)set_mode(OolerMode)-- set pump mode:"Silent","Regular", or"Boost"set_temperature(int)-- set target temperature in the current display unitset_clean(bool)-- start/stop clean cycle (automatically powers on)set_temperature_unit(TemperatureUnit)-- set device display unit:"C"or"F"
OolerBLEState
Dataclass with fields: power, mode, set_temperature, actual_temperature, water_level, clean, temperature_unit.
Types
OolerMode--Literal["Silent", "Regular", "Boost"]TemperatureUnit--Literal["C", "F"]OolerConnectionError-- raised when all retry attempts are exhausted (inherits fromBleakError)
Concurrency & Reconnection
Connection serialization
All connection attempts are serialized through an internal asyncio.Lock. If connect() is called while another connection is already in progress, the second caller waits for the first to complete and then returns immediately if the connection succeeded. This prevents duplicate connections and race conditions.
Two-level retry
GATT write operations use a two-level retry strategy:
- Immediate retry -- if a write fails with a transient BLE error (e.g., ESP32 proxy hiccup), the operation is retried immediately without reconnecting.
- Reconnect + retry -- if the immediate retry also fails, the library forces a full disconnect/reconnect cycle (with a 0.5s backoff) and retries the operation once more.
If all three attempts fail, an OolerConnectionError is raised.
async_poll() uses a similar pattern: if the poll fails, it reconnects and retries once.
Handled exception types
The library catches BleakError, EOFError, BrokenPipeError, and asyncio.TimeoutError during GATT operations. These cover the common failure modes seen with ESP32 BLE proxies.
Disconnect handling
When the BLE connection drops unexpectedly, the internal client reference is cleared immediately so is_connected returns False. Registered callbacks are fired to notify consumers of the state change. The library does not automatically reconnect -- the consumer (e.g., a Home Assistant integration) is responsible for triggering reconnection on the next advertisement or poll cycle.
ESP32 BLE Proxy Considerations
Notification slots
ESP32 BLE proxies (ESPHome) have a global limit of 12 notification registrations across all connected devices. This library subscribes to 4 notification characteristics per device:
- Power, Mode, Set Temperature, Actual Temperature
Water level and clean status are polled (via async_poll()) rather than subscribed to notifications. This means two Ooler devices use 8 of 12 available slots, leaving headroom for other BLE devices.
Connection slots
ESP32 proxies support 3 simultaneous BLE connections by default. Each Ooler device holds one connection slot for as long as it's connected.
Temperature Behavior
The Ooler has a quirk in how it handles temperature units:
- Set temperature (
SETTEMP_CHAR) is always stored and reported in Fahrenheit by the device, regardless of the display unit setting. - Actual temperature (
ACTUALTEMP_CHAR) is reported in whatever unit the device display is set to.
The library handles this automatically:
state.set_temperatureis converted to the current display unit on read.set_temperature(value)accepts a value in the current display unit and converts to Fahrenheit before writing to the device.state.actual_temperatureis passed through as-is from the device.
The display unit is read once on connect and cached. It can be changed via set_temperature_unit().
License
Apache-2.0
Project details
Release history Release notifications | RSS feed
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 ooler_ble_client-0.11.1.tar.gz.
File metadata
- Download URL: ooler_ble_client-0.11.1.tar.gz
- Upload date:
- Size: 23.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/2.3.2 CPython/3.14.3 Darwin/25.3.0
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
5250e6f6f2662826d6408c1b777b77b375b297733e061dc123519b998ec0c26b
|
|
| MD5 |
958bb2118033f7034c0e5aaf2a6b34e2
|
|
| BLAKE2b-256 |
91c61e4a50ec0e77a5320caea3462166a01c98805df258850204b6b546c9a81c
|
File details
Details for the file ooler_ble_client-0.11.1-py3-none-any.whl.
File metadata
- Download URL: ooler_ble_client-0.11.1-py3-none-any.whl
- Upload date:
- Size: 24.0 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/2.3.2 CPython/3.14.3 Darwin/25.3.0
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
73dd56612482e2fd08abf46f01f3b712bddb182c16e2a8c7266e64fa8cee4a67
|
|
| MD5 |
712c066e87f1f9cc31eb4820288ad900
|
|
| BLAKE2b-256 |
02f9a0a1d98dcf860c67749d4f55440209ca671d74661aafd7cddd9e21ef62a3
|