Convert wearable health data from vendor schemas to Open mHealth schemas
Project description
omh-shim
Convert wearable health data from vendor schemas to Open mHealth schemas.
Status
v1.0 — initial public release. Public API is stable; converter coverage will continue to expand.
Install
pip install git+https://github.com/jupyterhealth/omh-shim.git@v1.0.1
Usage
Timestamp-based data types (one reading at a known instant):
from omh_shim import convert
omh_record = convert(
source="ow_normalized",
data_type="heart_rate",
sample={
"timestamp": "2026-04-09T08:30:00+00:00",
"type": "heart_rate",
"value": 72,
"unit": "bpm",
"source": {"source_name": "Oura Ring", "device_model": "Oura Gen 3"},
},
)
Daily data types (step_count, physical_activity, sleep_duration,
oxygen_saturation)
aggregate over a calendar day, so they REQUIRE an explicit timezone so the day
boundaries reflect the user's local day rather than silently assuming UTC:
from datetime import UTC
from zoneinfo import ZoneInfo
# UTC-anchored upstream data
convert(
source="oura_raw",
data_type="step_count",
sample={"day": "2026-04-09", "steps": 8432},
tz=UTC,
)
# User's local timezone
convert(
source="oura_raw",
data_type="step_count",
sample={"day": "2026-04-09", "steps": 8432},
tz=ZoneInfo("America/Los_Angeles"),
)
Every conversion returns the full IEEE 1752.1 data-point envelope —
{"header": ..., "body": ...} — with UUID, schema_id components, creation
timestamp, modality, and external_datasheets auto-populated from the sample's
source metadata (a nested source dict, or the device implied by source for
raw feeds like oura_raw):
convert(
source="oura_raw",
data_type="heart_rate",
sample={"bpm": 72, "source": "rest", "timestamp": "2026-04-09T03:15:00+00:00"},
)
# Returns:
# {
# "header": {
# "uuid": "...",
# "schema_id": {"namespace": "omh", "name": "heart-rate", "version": "2.0"},
# "source_creation_date_time": "...",
# "modality": "sensed",
# "external_datasheets": [{"datasheet_type": "manufacturer", "datasheet_reference": "Oura Ring"}]
# },
# "body": {
# "heart_rate": {"value": 72.0, "unit": "beats/min"},
# "effective_time_frame": {"date_time": "2026-04-09T03:15:00Z"}
# }
# }
convert raises ConversionError for unknown (source, data_type) pairs,
invalid sample shapes, naive (timezone-less) datetimes, or a missing tz
for daily data types. It raises ValidationError if the converter output
fails schema validation.
Supported sources and data types
source |
data_type values |
|---|---|
oura_raw |
heart_rate, heart_rate_variability, oxygen_saturation, step_count, sleep_duration, sleep_episode, physical_activity |
ow_normalized |
heart_rate, heart_rate_variability, oxygen_saturation, step_count, sleep_duration, sleep_episode, physical_activity |
Note: heart_rate_variability targets the local placeholder schema
local:heart-rate-variability:1.0 (Open mHealth has not published a canonical
HRV schema as of 2026-04). The local: namespace is deliberate — downstream
consumers should not assume OMH-standard interoperability for HRV records.
Served schemas without a converter
omh-shim also vendors clinical OMH body schemas — blood pressure, blood
glucose, body temperature, body weight, respiratory rate, and RR interval —
that have no convert() converter. They exist so consumers can serve and validate OMH
bodies offline from a single pinned source:
from omh_shim import known_ids, load_schema
"omh:blood-pressure:4.0" in known_ids() # True
schema = load_schema("omh:blood-glucose:4.0") # vendored JSON schema, all $refs resolvable offline
These are tracked as SERVED_NO_CONVERTER in tests/test_schema_coverage.py
(the authoritative list) and refreshed alongside the converter schemas by
tools/refresh_schemas.py.
Note: the vendored body-weight-3.0 schema carries an upstream $id typo
(it self-references body-weight-2.0.json). It is kept verbatim to preserve
byte-for-byte fidelity with the pinned source; validation resolves by
registered URI rather than $id, so it is harmless. Tracked upstream at
openmhealth/schemas#38.
Adding a new source
See docs/adding-a-source.md for a step-by-step
guide to adding a new device source (converter functions, test fixtures,
documentation).
Mapping references
docs/mappings/oura_raw.md— Oura Ring v2 API → OMH (body fields)docs/mappings/ow_normalized.md— Open Wearables normalized API → OMH (body fields)docs/mappings/ieee-1752-header.md— IEEE 1752.1 data-point header envelope
Credits
omh_shim/sources/oura_raw.py ports converter mapping logic with permission
from dicristea/oura-clinical-workbench.
See AUTHORS.md.
License
MIT. See LICENSE.
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 omh_shim-1.2.0.tar.gz.
File metadata
- Download URL: omh_shim-1.2.0.tar.gz
- Upload date:
- Size: 36.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 |
9d5ba450603c81845ac9b4ae16dff1fc8c995c0232d9c7eef8f6b30a582f2004
|
|
| MD5 |
6189df275f2a127ee4e16529e48820d9
|
|
| BLAKE2b-256 |
c7ca053beaa8859d27fea1ddf47ccfa154387cec9274064f99e7b81351e065e8
|
Provenance
The following attestation bundles were made for omh_shim-1.2.0.tar.gz:
Publisher:
release.yaml on jupyterhealth/omh-shim
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
omh_shim-1.2.0.tar.gz -
Subject digest:
9d5ba450603c81845ac9b4ae16dff1fc8c995c0232d9c7eef8f6b30a582f2004 - Sigstore transparency entry: 1712451319
- Sigstore integration time:
-
Permalink:
jupyterhealth/omh-shim@dfb0f92c714089003137e09f6c6f36f89b3ce84f -
Branch / Tag:
refs/tags/v1.2.0 - Owner: https://github.com/jupyterhealth
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yaml@dfb0f92c714089003137e09f6c6f36f89b3ce84f -
Trigger Event:
workflow_dispatch
-
Statement type:
File details
Details for the file omh_shim-1.2.0-py3-none-any.whl.
File metadata
- Download URL: omh_shim-1.2.0-py3-none-any.whl
- Upload date:
- Size: 52.4 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 |
ae1baaf6f50f15c775c15b56ccb13c5bead2b876477946c69321dec14876a10f
|
|
| MD5 |
0aa04b96b5193db768d3e70268b759c1
|
|
| BLAKE2b-256 |
1df710bca4d0d6f88f5519f167e44be3c5c5bcd131388b816a890fceb65ebfbc
|
Provenance
The following attestation bundles were made for omh_shim-1.2.0-py3-none-any.whl:
Publisher:
release.yaml on jupyterhealth/omh-shim
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
omh_shim-1.2.0-py3-none-any.whl -
Subject digest:
ae1baaf6f50f15c775c15b56ccb13c5bead2b876477946c69321dec14876a10f - Sigstore transparency entry: 1712451460
- Sigstore integration time:
-
Permalink:
jupyterhealth/omh-shim@dfb0f92c714089003137e09f6c6f36f89b3ce84f -
Branch / Tag:
refs/tags/v1.2.0 - Owner: https://github.com/jupyterhealth
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yaml@dfb0f92c714089003137e09f6c6f36f89b3ce84f -
Trigger Event:
workflow_dispatch
-
Statement type: