Skip to content

Commit

Permalink
docs: biggg readme update
Browse files Browse the repository at this point in the history
  • Loading branch information
tahnok committed Oct 4, 2024
1 parent d150d0c commit 9cff691
Show file tree
Hide file tree
Showing 4 changed files with 239 additions and 31 deletions.
101 changes: 85 additions & 16 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,32 +1,48 @@
# Colmi R02 Client

A python client to connect to the Colmi R02 family of Smart Rings.
Open source python client to read your data from the Colmi R02 family of Smart Rings. 100% open source, 100% offline.

Can be used either as a command line app (via `colmi_r02_client` and `colmi_r02_util`)
The colmi R02 ring includes the following sensors:

Inspired by https://github.com/atc1441/ATC_RF03_Ring/
- Accelerometer
- step tracking
- sleep tracking
- gestures (maybe...?)
- Heart Rate (HR)
- Blood Oxygen (SPO2)

I found out about this from atc1441 and his work on [ATC_RF03](https://github.com/atc1441/ATC_RF03_Ring/) and this
[Hackaday coverage](https://hackaday.com/2024/06/16/new-part-day-a-hackable-smart-ring/)

# Status
Got questions or ideas? [Send me an email](mailto:[email protected]) or [open an issue](https://github.com/tahnok/colmi_r02_client/issues/new)

- [x] real time heart rate and SPO2
Are you hiring? [Send me an email](mailto:[email protected])

## How to buy

You can get it on [here on AliExpress](https://www.aliexpress.com/item/1005006631448993.html). If that link is dead try searching for "COLMI R02", I got mine from "Colmi official store". It cost me $CAD 22 shipped.

## Reverse engineering status

- [x] Real time heart rate and SPO2
- [x] Step logs (still don't quite understand how the day is split up)
- [x] Heart rate logs (periodic measurement)
- [x] Set ring time
- [x] Set HR log frequency
- [ ] SPO2 logs
- [ ] "Stress" measurement
- [ ] Sleep tracking
- [ ] Set HR log frequency
- [x] Set ring time
- [ ] "Stress" measurement

# TODO
## Planned Feature

- add more CLI functionlity
- pretty print HR and steps
- sync all data to a file or sqlite db?
- add more CLI functionlity
- pretty print HR and steps
- sync all data to a file or sqlite db
- simple web interface

# Getting started
## Getting started

## With the command line
### Using the command line

If you don't know python that well, I **highly** recommend you install [pipx](https://pipx.pypa.io/stable/installation/). It's puprpose built for managing python packages intended to be used as standalone programs and it will keep your computer safe from the pitfalls of python packaging. Once install you can do

Expand Down Expand Up @@ -58,6 +74,59 @@ Starting reading, please wait.
[81, 81, 79, 79, 79, 79]
```

## With the library / sdk
The most up to date and comprehensive help for the command line can be found running

```sh
colmi_r02_client --help
```

```
Usage: colmi_r02_client [OPTIONS] COMMAND [ARGS]...
Options:
--debug / --no-debug
--record / --no-record Write all received packets to a file
--address TEXT Bluetooth address
--name TEXT Bluetooth name of the device, slower but will work
on macOS
--help Show this message and exit.
Commands:
get-heart-rate-log Get heart rate for given date
get-heart-rate-log-settings Get heart rate log settings
get-real-time-heart-rate Get real time heart rate.
info Get device info and battery level
set-heart-rate-log-settings Get heart rate log settings
set-time Set the time on the ring, required if you...
```

### With the library / sdk

You can use the `colmi_r02_client.client` class as a library to do your own stuff in python. I've tried to write a lot of docstrings, which are visable on [the docs site](https://tahnok.github.io/colmi_r02_client/)

## Communication Protocol Details

I've kept a lab notebook style stream of consciousness notes on https://notes.tahnok.ca/, starting with [2024-07-07 Smart Ring Hacking](https://notes.tahnok.ca/blog/2024-07-07+Smart+Ring+Hacking) and eventually getting put under one folder. That's the best source for all the raw stuff.

At a high level though, you can talk to and read from the ring using BLE. There's no binding or security keys required to get started. (that's kind of bad, but the range on the ring is really tiny and I'm not too worried about someone getting my steps or heart rate information. Up to you).

The ring has a ble GATT service with the UUID `6E40FFF0-B5A3-F393-E0A9-E50E24DCCA9E`. It has two important characteristics:

1. RX: `6E400002-B5A3-F393-E0A9-E50E24DCCA9E`, which you write to
2. TX: `6E400003-B5A3-F393-E0A9-E50E24DCCA9E`, which you can "subscribe" to and is where the ring responds to packets you have sent.

This closely ressembles the [Nordic UART Service](https://docs.nordicsemi.com/bundle/ncs-latest/page/nrf/libraries/bluetooth_services/services/nus.html) and UART/Serial communications in general.

### Packet structure

The ring communicates in 16 byte packets for both sending and receiving. The first byte of the packet is always a command/tag/type. For example, the packet you send to ask for the battery level starts with `0x03` and the response packet also starts with `0x03`.

The last byte of the packet is always a checksum/crc. This value is calculated by summing up the other 15 bytes in the packet and taking the result modulo 255. See `colmi_r02_client.packet.checksum`

The middle 14 bytes are the "subdata" or payload data. Some requests (like `colmi_r02_client.set_time.set_time_packet`) include additional data. Almost all responses use the subdata to return the data you asked for.

Some requests result in multiple responses that you have to consider together to get the data. `colmi_r02_client.steps.SportDetailParser` is an example of this behaviour.

If you want to know the actual packet structure for a given feature's request or response, take a look at the source code for that feature. I've tried to make it pretty easy to follow even if you don't know python very well. There are also some tests that you can refer to for validated request/response pairs and human readable interpretations of that data.

You can use the `colmi_r02_client.client` class as a library to do your own stuff in python. See `docs/` for the documentation on that.
Got questions or ideas? [Send me an email](mailto:[email protected]) or [open an issue](https://github.com/tahnok/colmi_r02_client/issues/new)
7 changes: 2 additions & 5 deletions colmi_r02_client/__init__.py
Original file line number Diff line number Diff line change
@@ -1,7 +1,4 @@
"""
This is a python client for the Colmi R02 Smart ring.
You can find it for like ~25USD$ on Aliexpress.
[Hackaday coverage](https://hackaday.com/2024/06/16/new-part-day-a-hackable-smart-ring/)
.. include:: ../README.md
:start-line: 2
"""
160 changes: 151 additions & 9 deletions docs/colmi_r02_client.html
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,15 @@
<input type="search" placeholder="Search..." role="searchbox" aria-label="search"
pattern=".+" required>

<h2>Contents</h2>
<ul>
<li><a href="#how-to-buy">How to buy</a></li>
<li><a href="#reverse-engineering-status">Reverse engineering status</a></li>
<li><a href="#planned-feature">Planned Feature</a></li>
<li><a href="#getting-started">Getting started</a></li>
<li><a href="#communication-protocol-details">Communication Protocol Details</a></li>
</ul>


<h2>Submodules</h2>
<ul>
Expand Down Expand Up @@ -50,24 +59,157 @@ <h2>Submodules</h2>
<h1 class="modulename">
colmi_r02_client </h1>

<div class="docstring"><p>This is a python client for the Colmi R02 Smart ring.</p>
<div class="docstring"><p>Open source python client to read your data from the Colmi R02 family of Smart Rings. 100% open source, 100% offline.</p>

<p>The colmi R02 ring includes the following sensors:</p>

<ul>
<li>Accelerometer
<ul>
<li>step tracking</li>
<li>sleep tracking</li>
<li>gestures (maybe...?)</li>
</ul></li>
<li>Heart Rate (HR)</li>
<li>Blood Oxygen (SPO2)</li>
</ul>

<p>I found out about this from atc1441 and his work on <a href="https://github.com/atc1441/ATC_RF03_Ring/">ATC_RF03</a> and this
<a href="https://hackaday.com/2024/06/16/new-part-day-a-hackable-smart-ring/">Hackaday coverage</a></p>

<p>Got questions or ideas? <a href="mailto:[email protected]">Send me an email</a> or <a href="https://github.com/tahnok/colmi_r02_client/issues/new">open an issue</a></p>

<p>Are you hiring? <a href="mailto:[email protected]">Send me an email</a></p>

<h2 id="how-to-buy">How to buy</h2>

<p>You can get it on <a href="https://www.aliexpress.com/item/1005006631448993.html">here on AliExpress</a>. If that link is dead try searching for "COLMI R02", I got mine from "Colmi official store". It cost me $CAD 22 shipped.</p>

<h2 id="reverse-engineering-status">Reverse engineering status</h2>

<ul>
<li><input type="checkbox" class="task-list-item-checkbox" checked disabled> Real time heart rate and SPO2</li>
<li><input type="checkbox" class="task-list-item-checkbox" checked disabled> Step logs (still don't quite understand how the day is split up)</li>
<li><input type="checkbox" class="task-list-item-checkbox" checked disabled> Heart rate logs (periodic measurement)</li>
<li><input type="checkbox" class="task-list-item-checkbox" checked disabled> Set ring time</li>
<li><input type="checkbox" class="task-list-item-checkbox" checked disabled> Set HR log frequency</li>
<li><input type="checkbox" class="task-list-item-checkbox" disabled> SPO2 logs</li>
<li><input type="checkbox" class="task-list-item-checkbox" disabled> Sleep tracking</li>
<li><input type="checkbox" class="task-list-item-checkbox" disabled> "Stress" measurement</li>
</ul>

<h2 id="planned-feature">Planned Feature</h2>

<ul>
<li>add more CLI functionlity</li>
<li>pretty print HR and steps</li>
<li>sync all data to a file or sqlite db</li>
<li>simple web interface</li>
</ul>

<h2 id="getting-started">Getting started</h2>

<h3 id="using-the-command-line">Using the command line</h3>

<p>If you don't know python that well, I <strong>highly</strong> recommend you install <a href="https://pipx.pypa.io/stable/installation/">pipx</a>. It's puprpose built for managing python packages intended to be used as standalone programs and it will keep your computer safe from the pitfalls of python packaging. Once install you can do</p>

<div class="pdoc-code codehilite">
<pre><span></span><code>pipx<span class="w"> </span>install<span class="w"> </span>git+https://github.com/tahnok/colmi_r02_client
</code></pre>
</div>

<p>Once that is done you can look for nearby rings using</p>

<div class="pdoc-code codehilite">
<pre><span></span><code>colmi_r02_util<span class="w"> </span>scan
</code></pre>
</div>

<pre><code>Found device(s)
Name | Address
--------------------------------------------
R02_341C | 70:CB:0D:D0:34:1C
</code></pre>

<p>Once you have your address you can use it to do things like get real time heart rate</p>

<div class="pdoc-code codehilite">
<pre><span></span><code>colmi_r02_client<span class="w"> </span>--address<span class="o">=</span><span class="m">70</span>:CB:0D:D0:34:1C<span class="w"> </span>get-real-time-heart-rate
</code></pre>
</div>

<pre><code>Starting reading, please wait.
[81, 81, 79, 79, 79, 79]
</code></pre>

<p>The most up to date and comprehensive help for the command line can be found running</p>

<div class="pdoc-code codehilite">
<pre><span></span><code>colmi_r02_client<span class="w"> </span>--help
</code></pre>
</div>

<pre><code>Usage: colmi_r02_client [OPTIONS] COMMAND [ARGS]...

Options:
--debug / --no-debug
--record / --no-record Write all received packets to a file
--address TEXT Bluetooth address
--name TEXT Bluetooth name of the device, slower but will work
on macOS
--help Show this message and exit.

Commands:
get-heart-rate-log Get heart rate for given date
get-heart-rate-log-settings Get heart rate log settings
get-real-time-heart-rate Get real time heart rate.
info Get device info and battery level
set-heart-rate-log-settings Get heart rate log settings
set-time Set the time on the ring, required if you...
</code></pre>

<h3 id="with-the-library-sdk">With the library / sdk</h3>

<p>You can use the <code><a href="colmi_r02_client/client.html">colmi_r02_client.client</a></code> class as a library to do your own stuff in python. I've tried to write a lot of docstrings, which are visable on <a href="https://tahnok.github.io/colmi_r02_client/">the docs site</a></p>

<h2 id="communication-protocol-details">Communication Protocol Details</h2>

<p>I've kept a lab notebook style stream of consciousness notes on <a href="https://notes.tahnok.ca/">https://notes.tahnok.ca/</a>, starting with <a href="https://notes.tahnok.ca/blog/2024-07-07+Smart+Ring+Hacking">2024-07-07 Smart Ring Hacking</a> and eventually getting put under one folder. That's the best source for all the raw stuff.</p>

<p>At a high level though, you can talk to and read from the ring using BLE. There's no binding or security keys required to get started. (that's kind of bad, but the range on the ring is really tiny and I'm not too worried about someone getting my steps or heart rate information. Up to you).</p>

<p>The ring has a ble GATT service with the UUID <code>6E40FFF0-B5A3-F393-E0A9-E50E24DCCA9E</code>. It has two important characteristics:</p>

<ol>
<li>RX: <code>6E400002-B5A3-F393-E0A9-E50E24DCCA9E</code>, which you write to</li>
<li>TX: <code>6E400003-B5A3-F393-E0A9-E50E24DCCA9E</code>, which you can "subscribe" to and is where the ring responds to packets you have sent.</li>
</ol>

<p>This closely ressembles the <a href="https://docs.nordicsemi.com/bundle/ncs-latest/page/nrf/libraries/bluetooth_services/services/nus.html">Nordic UART Service</a> and UART/Serial communications in general.</p>

<h3 id="packet-structure">Packet structure</h3>

<p>The ring communicates in 16 byte packets for both sending and receiving. The first byte of the packet is always a command/tag/type. For example, the packet you send to ask for the battery level starts with <code>0x03</code> and the response packet also starts with <code>0x03</code>.</p>

<p>The last byte of the packet is always a checksum/crc. This value is calculated by summing up the other 15 bytes in the packet and taking the result modulo 255. See <code><a href="colmi_r02_client/packet.html#checksum">colmi_r02_client.packet.checksum</a></code></p>

<p>The middle 14 bytes are the "subdata" or payload data. Some requests (like <code><a href="colmi_r02_client/set_time.html#set_time_packet">colmi_r02_client.set_time.set_time_packet</a></code>) include additional data. Almost all responses use the subdata to return the data you asked for.</p>

<p>Some requests result in multiple responses that you have to consider together to get the data. <code><a href="colmi_r02_client/steps.html#SportDetailParser">colmi_r02_client.steps.SportDetailParser</a></code> is an example of this behaviour.</p>

<p>You can find it for like ~25USD$ on Aliexpress.</p>
<p>If you want to know the actual packet structure for a given feature's request or response, take a look at the source code for that feature. I've tried to make it pretty easy to follow even if you don't know python very well. There are also some tests that you can refer to for validated request/response pairs and human readable interpretations of that data.</p>

<p><a href="https://hackaday.com/2024/06/16/new-part-day-a-hackable-smart-ring/">Hackaday coverage</a></p>
<p>Got questions or ideas? <a href="mailto:[email protected]">Send me an email</a> or <a href="https://github.com/tahnok/colmi_r02_client/issues/new">open an issue</a></p>
</div>

<input id="mod-colmi_r02_client-view-source" class="view-source-toggle-state" type="checkbox" aria-hidden="true" tabindex="-1">

<label class="view-source-button" for="mod-colmi_r02_client-view-source"><span>View Source</span></label>

<div class="pdoc-code codehilite"><pre><span></span><span id="L-1"><a href="#L-1"><span class="linenos">1</span></a><span class="sd">&quot;&quot;&quot;</span>
</span><span id="L-2"><a href="#L-2"><span class="linenos">2</span></a><span class="sd">This is a python client for the Colmi R02 Smart ring.</span>
</span><span id="L-3"><a href="#L-3"><span class="linenos">3</span></a>
</span><span id="L-4"><a href="#L-4"><span class="linenos">4</span></a><span class="sd">You can find it for like ~25USD$ on Aliexpress.</span>
</span><span id="L-5"><a href="#L-5"><span class="linenos">5</span></a>
</span><span id="L-6"><a href="#L-6"><span class="linenos">6</span></a><span class="sd">[Hackaday coverage](https://hackaday.com/2024/06/16/new-part-day-a-hackable-smart-ring/)</span>
</span><span id="L-7"><a href="#L-7"><span class="linenos">7</span></a><span class="sd">&quot;&quot;&quot;</span>
</span><span id="L-2"><a href="#L-2"><span class="linenos">2</span></a><span class="sd">.. include:: ../README.md</span>
</span><span id="L-3"><a href="#L-3"><span class="linenos">3</span></a><span class="sd"> :start-line: 2</span>
</span><span id="L-4"><a href="#L-4"><span class="linenos">4</span></a><span class="sd">&quot;&quot;&quot;</span>
</span></pre></div>


Expand Down
Loading

0 comments on commit 9cff691

Please sign in to comment.