1
0

Initial commit.

This commit is contained in:
Markus Birth 2022-01-24 11:32:40 +01:00
commit 90913631a6
Signed by: mbirth
GPG Key ID: A9928D7A098C3A9A
8 changed files with 235 additions and 0 deletions

2
.gitignore vendored Normal file
View File

@ -0,0 +1,2 @@
/auth.py
/spo2.db3

12
Pipfile Normal file
View File

@ -0,0 +1,12 @@
[[source]]
url = "https://pypi.org/simple"
verify_ssl = true
name = "pypi"
[packages]
garminconnect = "*"
[dev-packages]
[requires]
python_version = "3.9"

89
Pipfile.lock generated Normal file
View File

@ -0,0 +1,89 @@
{
"_meta": {
"hash": {
"sha256": "9a50d04f5b70c0f051d0e240e3cc06c311ecb455e788a312b32cbf4f85fd4e79"
},
"pipfile-spec": 6,
"requires": {
"python_version": "3.9"
},
"sources": [
{
"name": "pypi",
"url": "https://pypi.org/simple",
"verify_ssl": true
}
]
},
"default": {
"certifi": {
"hashes": [
"sha256:78884e7c1d4b00ce3cea67b44566851c4343c120abd683433ce934a68ea58872",
"sha256:d62a0163eb4c2344ac042ab2bdf75399a71a2d8c7d47eac2e2ee91b9d6339569"
],
"version": "==2021.10.8"
},
"charset-normalizer": {
"hashes": [
"sha256:876d180e9d7432c5d1dfd4c5d26b72f099d503e8fcc0feb7532c9289be60fcbd",
"sha256:cb957888737fc0bbcd78e3df769addb41fd1ff8cf950dc9e7ad7793f1bf44455"
],
"markers": "python_version >= '3'",
"version": "==2.0.10"
},
"cloudscraper": {
"hashes": [
"sha256:674fd739f9412188aae8d6614e3e6316939fc0670ef5646abd3d316f1a59d3c2",
"sha256:dda29028c5628b5ba3e4dc43816ed38fd46bd945ef938c420f185586a6d8dff2"
],
"version": "==1.2.58"
},
"garminconnect": {
"hashes": [
"sha256:cddd59eca3009ae6b0c2fd216b8049759680b6f88390cea49afce1ed68ce1bed"
],
"index": "pypi",
"version": "==0.1.44"
},
"idna": {
"hashes": [
"sha256:84d9dd047ffa80596e0f246e2eab0b391788b0503584e8945f2368256d2735ff",
"sha256:9d643ff0a55b762d5cdb124b8eaa99c66322e2157b69160bc32796e824360e6d"
],
"markers": "python_version >= '3'",
"version": "==3.3"
},
"pyparsing": {
"hashes": [
"sha256:18ee9022775d270c55187733956460083db60b37d0d0fb357445f3094eed3eea",
"sha256:a6c06a88f252e6c322f65faf8f418b16213b51bdfaece0524c1c1bc30c63c484"
],
"markers": "python_version >= '3.6'",
"version": "==3.0.7"
},
"requests": {
"hashes": [
"sha256:68d7c56fd5a8999887728ef304a6d12edc7be74f1cfa47714fc8b414525c9a61",
"sha256:f22fa1e554c9ddfd16e6e41ac79759e17be9e492b3587efa038054674760e72d"
],
"markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4, 3.5'",
"version": "==2.27.1"
},
"requests-toolbelt": {
"hashes": [
"sha256:380606e1d10dc85c3bd47bf5a6095f815ec007be7a8b69c878507068df059e6f",
"sha256:968089d4584ad4ad7c171454f0a5c6dac23971e9472521ea3b6d49d610aa6fc0"
],
"version": "==0.9.1"
},
"urllib3": {
"hashes": [
"sha256:000ca7f471a233c2251c6c7023ee85305721bfdf18621ebff4fd17a8653427ed",
"sha256:0e7c33d9a63e7ddfcb86780aac87befc2fbddf46c58dbb487e0855f7ceec283c"
],
"markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4' and python_version < '4'",
"version": "==1.26.8"
}
},
"develop": {}
}

30
README.md Normal file
View File

@ -0,0 +1,30 @@
Garmin SpO2 to Apple Health
===========================
For some reason, Garmin Connect on iOS doesn't sync Blood Oxygen (SpO2) data into Apple Health
despite syncing various other metrics perfectly fine already.
There's a [feature request](https://forums.garmin.com/apps-software/mobile-apps-web/f/garmin-connect-mobile-ios/254977/request-for-spo2-vo2-max-and-respiration-data-to-be-shared-to-apple-health-app)
in their forums from February 2021, but apart from "We've added this to our feature request list"
nothing came of it in the past 11 months. And I found similar requests as old as [over 4 years (June 2018)](https://forums.garmin.com/outdoor-recreation/outdoor-recreation/f/fenix-5-plus-series/147239/spo2-measurements).
Until Garmin finds the 5 minutes to implement this feature, I've resorted to use the wonderful
[garminconnect](https://github.com/cyberjunky/python-garminconnect) Python-library to download
my SpO2 data, a PHP script to serve the data from the last 2 days as JSON and an iOS Shortcut to
sync that to Apple Health.
While it's not the most elegant solution, it works.
Usage
-----
1. Copy `auth.py.example` to `auth.py` and edit it to specify your Garmin Connect credentials
2. Copy `spo2_empty.db3` to `spo2.db3`
3. Run `pipenv install` to install dependencies
4. Make a cronjob running `pipenv run ./fetch.py` regularly
5. Copy the script from the `php` folder to a webserver and edit the file to point to the SQLite
database file.
6. Install the [SpO2 Inserter Shortcut](https://www.icloud.com/shortcuts/7f6f94eb536e4fb1857993bfbc181ccb)
on your iPhone and point it to the PHP script.
7. Optional: Create an Automation in Shortcuts to run this automatically.

6
auth.py.example Normal file
View File

@ -0,0 +1,6 @@
# Your Garmin Connect credentials
USERNAME = "changeme@changeme.com"
PASSWORD = "changeme"
# Number of days to fetch (default: last 3 days)
DAYS_TO_FETCH = 3

61
fetch.py Executable file
View File

@ -0,0 +1,61 @@
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
import datetime
import sqlite3
from auth import USERNAME, PASSWORD, DAYS_TO_FETCH
from pprint import pprint
from time import sleep
from garminconnect import (
Garmin,
GarminConnectConnectionError
)
today = datetime.date.today()
startfrom = today - datetime.timedelta(days=DAYS_TO_FETCH)
print(f"Logging into Garmin Connect using {USERNAME}.")
api = Garmin(USERNAME, PASSWORD)
tries = 0
while True:
try:
api.login()
break
except GarminConnectConnectionError as e:
tries += 1
wait = tries
if wait > 2:
wait = 2
print(f"Error during login! Retry #{tries} in {wait} sec.", flush=True)
sleep(wait)
db = sqlite3.connect("spo2.db3")
reqday = startfrom
while reqday <= today:
print("Querying data for: {}".format(reqday.isoformat()))
sleep = api.get_sleep_data(reqday.isoformat())
#pprint(sleep)
if not "wellnessEpochSPO2DataDTOList" in sleep:
print("No SpO2 data for {}. Skipping day.".format(reqday.isoformat()))
reqday += datetime.timedelta(days=1)
continue
spo2 = sleep["wellnessEpochSPO2DataDTOList"]
#pprint(spo2)
print("Got {} records.".format(len(spo2)))
for rec in spo2:
ts = datetime.datetime.fromisoformat(rec["epochTimestamp"][:-2]) # strip hundreds of seconds
sql = "INSERT OR IGNORE INTO spo2 VALUES (?, ?, ?)"
db.execute(sql, [ts.timestamp(), rec["spo2Reading"], rec["readingConfidence"]])
reqday += datetime.timedelta(days=1)
db.commit()
db.close()
api.logout()

35
php/gcserve.php Normal file
View File

@ -0,0 +1,35 @@
<?php
ini_set('display_errors', '1');
error_reporting(E_ALL);
$DB_FILE = '/home/mbirth/opt/garmin-fetch/spo2.db3'; // path to SQLite database
$DAYS_BACK = 1; // Today minus $DAYS_BACK days
$pdo = new \PDO('sqlite:' . $DB_FILE);
$now = gmmktime(0, 0, 0); // get today at 00:00:00 according to UTC/GMT
$servefrom = $now - $DAYS_BACK * 24 * 60 * 60;
#$servefrom = 1643001299;
$pq = $pdo->prepare('SELECT * FROM spo2 WHERE timestamp_utc >= ?');
$pq->execute(array($servefrom));
$res = $pq->fetchAll();
#header('Content-Type: text/plain');
header('Content-Type: application/json');
$result = array();
foreach ($res as $row) {
$dto = new DateTime('@' . $row['timestamp_utc'], new DateTimeZone('UTC'));
$result[] = array(
'unixtime' => intval($row['timestamp_utc']),
#'datestr' => $dto->format('Y.m.d \A\D \a\t H:i:s \U\T\C'),
'datestr' => $dto->format('c'),
'spo2_value' => intval($row['spo2_percent']),
'spo2_confidence' => intval($row['spo2_confidence']),
);
}
#var_dump($result);
echo json_encode($result);

BIN
spo2_empty.db3 Normal file

Binary file not shown.