Python library for communicating with Specialized Turbo e-bikes over Bluetooth Low Energy
Project description
specialized-turbo
Python library for talking to Specialized Turbo e-bikes (Vado, Levo, Creo) over Bluetooth Low Energy. Reads speed, power, cadence, battery, motor temp, odometer, assist level, range. Can also write settings like assist level and acceleration.
Async, built on bleak. Includes a CLI. Protocol docs in docs/protocol.md.
Installation
pip install specialized-turbo
Quick start
Stream telemetry
import asyncio
from specialized_turbo import SpecializedConnection, TelemetryMonitor
async def main():
async with SpecializedConnection("DC:DD:BB:4A:D6:55", pin="946166") as conn:
monitor = TelemetryMonitor(conn)
await monitor.start()
async for msg in monitor.stream():
print(f"{msg.field_name} = {msg.converted_value} {msg.unit}")
asyncio.run(main())
Read the snapshot
async with SpecializedConnection("DC:DD:BB:4A:D6:55", pin="946166") as conn:
monitor = TelemetryMonitor(conn)
await monitor.start()
await asyncio.sleep(5)
snap = monitor.snapshot
print(f"Speed: {snap.motor.speed_kmh} km/h")
print(f"Battery: {snap.battery.charge_pct}%")
print(f"Power: {snap.motor.rider_power_w} W (rider) + {snap.motor.motor_power_w} W (motor)")
print(f"Cadence: {snap.motor.cadence_rpm} RPM")
print(f"Assist: {snap.motor.assist_level}")
Query a single value
from specialized_turbo import SpecializedConnection, Sender, BatteryChannel
async with SpecializedConnection("DC:DD:BB:4A:D6:55", pin="946166") as conn:
msg = await conn.request_value(Sender.BATTERY, BatteryChannel.CHARGE_PERCENT)
print(f"Battery: {msg.converted_value}%")
Write commands
async with SpecializedConnection("DC:DD:BB:4A:D6:55", pin="946166") as conn:
await conn.set_assist_level(2) # TRAIL
await conn.set_acceleration(50.0) # 50%
await conn.set_shuttle(25)
await conn.set_assist_percentage(0, 35) # ECO = 35%
CLI
Scan for bikes:
specialized-turbo scan
specialized-turbo scan --timeout 15
Stream telemetry:
specialized-turbo telemetry DC:DD:BB:4A:D6:55 --pin 946166
specialized-turbo telemetry DC:DD:BB:4A:D6:55 --pin 946166 --format json
specialized-turbo telemetry DC:DD:BB:4A:D6:55 --pin 946166 --duration 30
Read a single value:
specialized-turbo read list # show available fields
specialized-turbo read battery_charge_percent DC:DD:BB:4A:D6:55 --pin 946166
specialized-turbo read speed DC:DD:BB:4A:D6:55 --pin 946166 --format json
Write a value:
specialized-turbo write list # show writable fields
specialized-turbo write assist_level 2 DC:DD:BB:4A:D6:55 --pin 946166 # set to TRAIL
specialized-turbo write acceleration 50 DC:DD:BB:4A:D6:55 --pin 946166 # 50% sensitivity
Dump GATT services (debugging):
specialized-turbo services DC:DD:BB:4A:D6:55 --pin 946166
Available fields
| Field | Unit | Writable | Description |
|---|---|---|---|
battery_capacity_wh |
Wh | Total battery capacity | |
battery_remaining_wh |
Wh | Remaining energy | |
battery_health |
% | Battery health | |
battery_temp |
°C | Battery temperature | |
battery_charge_cycles |
cycles | Number of charge cycles | |
battery_voltage |
V | Battery voltage | |
battery_current |
A | Battery current draw | |
battery_charge_percent |
% | State of charge | |
rider_power |
W | Rider pedal power | |
cadence |
RPM | Pedaling cadence | |
speed |
km/h | Current speed | |
odometer |
km | Total distance | |
assist_level |
-- | yes | OFF / ECO / TRAIL / TURBO |
motor_temp |
°C | Motor temperature | |
motor_power |
W | Electric motor power | |
peak_assist |
% | ECO / TRAIL / TURBO percentages | |
shuttle |
-- | yes | Shuttle mode value (0-100) |
wheel_circumference |
mm | yes | Wheel circumference setting |
assist_lev1_pct |
% | yes | ECO assist percentage |
assist_lev2_pct |
% | yes | TRAIL assist percentage |
assist_lev3_pct |
% | yes | TURBO assist percentage |
fake_channel |
-- | Bit-coded internal channel | |
acceleration |
% | yes | Acceleration sensitivity |
TCX2+ bikes have additional fields (range, altitude, gradient, calories, system temperature, and more). See docs/protocol.md for the full list.
Protocol support
Four protocol generations exist:
| Protocol | Message format | Encryption |
|---|---|---|
| TCU1 | [sender][channel][data] |
None |
| TCX2 | 2-byte parameter ID + CRC-16 | Optional AES-128-CTR |
| TCX3 | Same as TCX2 | Optional AES-128-CTR |
| TCX4 | Same as TCX2 | Optional AES-128-CTR |
TCX2/3/4 share one wire format and differ only in which parameters the bike supports. The BLEProfile enum (TCU1 / TCX) controls which GATT UUIDs to use.
See docs/protocol.md for the full spec.
Pairing
The bike needs a 6-digit PIN for BLE pairing, shown on its TCU screen. Pass it via --pin (CLI) or pin= (Python).
On Windows, bleak's WinRT backend can handle passkey pairing programmatically. If that doesn't work, pair through Windows Bluetooth Settings first, then connect without --pin.
Some newer bikes use numeric comparison instead of passkey entry. On those, pair through your OS Bluetooth settings first.
Development
uv sync --extra dev
uv run pytest
License
MIT
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 specialized_turbo-0.5.0.tar.gz.
File metadata
- Download URL: specialized_turbo-0.5.0.tar.gz
- Upload date:
- Size: 52.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.13
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
3d3515b776cf3e03f251b50dfffbaacdc8152561fe337bb52712136899424b41
|
|
| MD5 |
ae54bcc17b143526e7649b74d6a30d8e
|
|
| BLAKE2b-256 |
5aead3db9e6b0b66b3ac99b5798fd0f039349685e51a12401a742e9c675db809
|
Provenance
The following attestation bundles were made for specialized_turbo-0.5.0.tar.gz:
Publisher:
publish.yml on JamieMagee/specialized-turbo
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
specialized_turbo-0.5.0.tar.gz -
Subject digest:
3d3515b776cf3e03f251b50dfffbaacdc8152561fe337bb52712136899424b41 - Sigstore transparency entry: 1638659245
- Sigstore integration time:
-
Permalink:
JamieMagee/specialized-turbo@41181424dc70b7b9dac8435f030c922473fd067a -
Branch / Tag:
refs/tags/v0.5.0 - Owner: https://github.com/JamieMagee
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@41181424dc70b7b9dac8435f030c922473fd067a -
Trigger Event:
release
-
Statement type:
File details
Details for the file specialized_turbo-0.5.0-py3-none-any.whl.
File metadata
- Download URL: specialized_turbo-0.5.0-py3-none-any.whl
- Upload date:
- Size: 38.6 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.13
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
5e86dbb2ab0d4123efd57e2129591e29a80d0f604014a61594970003e6d25ef6
|
|
| MD5 |
34dc884ff947736fb36b7bcaa6521c94
|
|
| BLAKE2b-256 |
9181d2491f2ad7f37b82fedd3d9635cceee3d4533fcc5cfbf4d815f10e8ef60f
|
Provenance
The following attestation bundles were made for specialized_turbo-0.5.0-py3-none-any.whl:
Publisher:
publish.yml on JamieMagee/specialized-turbo
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
specialized_turbo-0.5.0-py3-none-any.whl -
Subject digest:
5e86dbb2ab0d4123efd57e2129591e29a80d0f604014a61594970003e6d25ef6 - Sigstore transparency entry: 1638659350
- Sigstore integration time:
-
Permalink:
JamieMagee/specialized-turbo@41181424dc70b7b9dac8435f030c922473fd067a -
Branch / Tag:
refs/tags/v0.5.0 - Owner: https://github.com/JamieMagee
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@41181424dc70b7b9dac8435f030c922473fd067a -
Trigger Event:
release
-
Statement type: