A Model Context Protocol (MCP) server that fetches sleep, HRV, and training data from the unofficial Coros API and exposes them to AI assistants like Claude.
No API key required. This server authenticates directly with your Coros Training Hub credentials. Your token is stored securely in your system keyring (or an encrypted local file as fallback), never transmitted anywhere except to Coros.
Ask your AI assistant questions like:
- "How much deep sleep and REM did I get last week?"
- "What was my HRV trend over the last 4 weeks?"
- "Show me my resting heart rate and training load for last week"
- "How many steps did I average per day this month?"
- "List my rides from last month"
- "Show me the details of my last long ride"
- "Create a 90-minute sweet spot workout for me"
- "What's on my training calendar next week?"
- "Schedule my VO2 workout for Thursday"
- "Create a 20-minute strength circuit with squats, lunges, and planks"
| Tool | Description |
|---|---|
authenticate_coros |
Log in with email and password — token stored securely in keyring |
authenticate_coros_mobile |
Log in to the mobile API only (useful for sleep data troubleshooting) |
check_coros_auth |
Check whether a valid auth token is present |
get_daily_metrics |
Fetch daily metrics (HRV, resting HR, training load, VO2max, stamina, and more) for n weeks (default: 4) |
get_sleep_data |
Fetch nightly sleep stages (deep, light, REM, awake) and sleep HR for n weeks (default: 4) |
list_activities |
List activities for a date range with summary metrics |
get_activity_detail |
Fetch full detail for a single activity (laps, HR zones, power zones) |
list_workouts |
List all saved structured workout programs |
create_workout |
Create a new structured workout with named steps and power targets |
delete_workout |
Delete a workout program from the library |
list_planned_activities |
List planned workouts from the Coros training calendar |
schedule_workout |
Schedule an existing workout on a calendar day |
remove_scheduled_workout |
Remove a scheduled workout from the calendar |
create_strength_workout |
Create a structured strength workout with sets, reps, or timed exercises |
list_exercises |
Browse the Coros exercise catalogue, especially for strength workouts |
If you have Claude Code, paste this prompt:
Set up the Coros MCP server from https://github.com/cygnusb/coros-mcp — clone it, create a venv, install it with pip install -e ., add it to my MCP config, then tell me to run 'coros-mcp auth' in my terminal to authenticate.
Claude will handle the installation and guide you through configuration.
git clone https://github.com/cygnusb/coros-mcp.git
cd coros-mcp
python3 -m venv .venv
source .venv/bin/activate # Windows: .venv\Scripts\activate
pip install -e .Or with uv:
uv pip install -e .claude mcp add coros -- python /path/to/coros-mcp/server.pyOr add to Claude Desktop (~/Library/Application Support/Claude/claude_desktop_config.json):
{
"mcpServers": {
"coros": {
"command": "/path/to/coros-mcp/.venv/bin/python",
"args": ["/path/to/coros-mcp/server.py"]
}
}
}Run the following command in your terminal — outside of any Claude session:
coros-mcp authYou will be prompted for your email, password, and region (eu, us, or asia). This stores both the Training Hub web token and the mobile API token (used for sleep data). Your credentials are sent directly to Coros and the tokens are stored securely in your system keyring (or an encrypted local file as fallback). You only need to do this once — the tokens persist across restarts.
Note: The mobile login (
apieu.coros.com) will log you out of the Coros mobile app on your phone. If you want to avoid this, usecoros-mcp auth-webinstead — it stores only the web token, and the mobile token will be obtained automatically when you first request sleep data.
Other auth commands:
coros-mcp auth-web # Web API only — skips mobile login (sleep data obtained lazily)
coros-mcp auth-mobile # Mobile API only (sleep data)
coros-mcp auth-status # Check if authenticated
coros-mcp auth-clear # Remove stored tokensLog in with your Coros credentials. The auth token is stored securely in your system keyring (or an encrypted file as fallback).
{ "email": "you@example.com", "password": "yourpassword", "region": "eu" }Returns: authenticated, user_id, region, message
Authenticate with the Coros mobile API only. This is mainly useful if you need to restore sleep-data access without redoing full web authentication.
{ "email": "you@example.com", "password": "yourpassword", "region": "eu" }Returns: authenticated, user_id, region, message
Check whether valid web and mobile tokens are stored and how long the web token remains valid.
{}Returns: authenticated, user_id, region, expires_in_hours, mobile_authenticated, mobile_token_status
Fetch daily metrics for a configurable number of weeks (default: 4).
{ "weeks": 4 }Returns: records (list), count, date_range
Each record includes:
| Field | Source | Description |
|---|---|---|
date |
— | Date (YYYYMMDD) |
avg_sleep_hrv |
dayDetail | Nightly HRV (RMSSD ms) |
baseline |
dayDetail | HRV rolling baseline |
rhr |
dayDetail | Resting heart rate (bpm) |
training_load |
dayDetail | Daily training load |
training_load_ratio |
dayDetail | Acute/chronic training load ratio |
tired_rate |
dayDetail | Fatigue rate |
ati / cti |
dayDetail | Acute / chronic training index |
distance / duration |
dayDetail | Distance (m) / duration (s) |
vo2max |
analyse (merge) | VO2 Max (last ~28 days) |
lthr |
analyse (merge) | Lactate threshold heart rate (bpm) |
ltsp |
analyse (merge) | Lactate threshold pace (s/km) |
stamina_level |
analyse (merge) | Base fitness level |
stamina_level_7d |
analyse (merge) | 7-day fitness trend |
Fetch nightly sleep stage data for a configurable number of weeks (default: 4).
{ "weeks": 4 }Returns: records (list), count, date_range
Each record includes:
| Field | Description |
|---|---|
date |
Date (YYYYMMDD) — the morning after the sleep |
total_duration_minutes |
Total sleep in minutes |
phases.deep_minutes |
Deep sleep |
phases.light_minutes |
Light sleep |
phases.rem_minutes |
REM sleep |
phases.awake_minutes |
Time awake during the night |
phases.nap_minutes |
Daytime nap time (null if none) |
avg_hr |
Average heart rate during sleep |
min_hr |
Minimum heart rate during sleep |
max_hr |
Maximum heart rate during sleep |
quality_score |
Sleep quality score (null if not computed) |
Note: Sleep data is fetched from the Coros mobile API (
apieu.coros.com), which uses a separate token from the Training Hub web API.coros-mcp authobtains both tokens, but doing so logs you out of the Coros mobile app. Usecoros-mcp auth-webto skip mobile login — the mobile token is then obtained automatically on the first sleep data request. The token expires after ~1 hour but refreshes automatically on subsequent requests.
List activities for a date range.
{ "start_day": "20260101", "end_day": "20260305", "page": 1, "size": 30 }Returns: activities (list), total_count, page
Each activity includes: activity_id, name, sport_type, sport_name, start_time, end_time, duration_seconds, distance_meters, avg_hr, max_hr, calories, training_load, avg_power, normalized_power, elevation_gain
Fetch full detail for a single activity. Requires the sport_type from list_activities.
{ "activity_id": "469901014965714948", "sport_type": 200 }Returns full activity data including laps, HR zones, power zones, and all sport-specific metrics.
Note: Large time-series arrays (
graphList,frequencyList,gpsLightDuration) are stripped from the response to keep it manageable.
List all saved structured workout programs.
{}Returns: workouts (list), count
Each workout includes: id, name, sport_type, sport_name, estimated_time_seconds, exercise_count, exercises (list of steps with name, duration_seconds, power_low_w, power_high_w)
Create a new structured workout. Workouts appear in the Coros app and can be synced to the watch. Steps can be plain steps or repeat groups for intervals.
Plain steps:
{
"name": "Sweet Spot 90min",
"sport_type": 2,
"steps": [
{"name": "15:00 Warmup", "duration_minutes": 15, "power_low_w": 148, "power_high_w": 192},
{"name": "20:00 Sweet Spot", "duration_minutes": 20, "power_low_w": 260, "power_high_w": 275},
{"name": "5:00 Rest", "duration_minutes": 5, "power_low_w": 100, "power_high_w": 150},
{"name": "20:00 Sweet Spot", "duration_minutes": 20, "power_low_w": 260, "power_high_w": 275},
{"name": "30:00 Cooldown", "duration_minutes": 30, "power_low_w": 100, "power_high_w": 192}
]
}With repeat groups (intervals):
{
"name": "3×10min Sweet Spot",
"sport_type": 2,
"steps": [
{"name": "Warmup", "duration_minutes": 10, "power_low_w": 150, "power_high_w": 200},
{"repeat": 3, "steps": [
{"name": "Sweet Spot", "duration_minutes": 10, "power_low_w": 265, "power_high_w": 285},
{"name": "Recovery", "duration_minutes": 3, "power_low_w": 150, "power_high_w": 175}
]},
{"name": "Cooldown", "duration_minutes": 11, "power_low_w": 150, "power_high_w": 200}
]
}sport_type: 2 = Indoor Cycling (default), 200 = Road Bike
Returns: workout_id, name, total_minutes, steps_count, message
Delete a workout program from the Coros account.
{ "workout_id": "476023839273435149" }The workout_id comes from list_workouts.
Returns: deleted, workout_id, message
List planned activities from the Coros training calendar.
{ "start_day": "20260309", "end_day": "20260316" }Returns: activities (list), count, date_range
Add an existing workout from your library to the Coros training calendar.
{ "workout_id": "1234567890", "happen_day": "20260312", "sort_no": 1 }Returns: scheduled, workout_id, happen_day
Remove a scheduled workout from the Coros training calendar.
{
"plan_id": "987654321",
"id_in_plan": "1234567890",
"plan_program_id": "1234567890"
}plan_id, id_in_plan, and plan_program_id come from list_planned_activities. If plan_program_id is missing, you can usually reuse id_in_plan.
Returns: removed, plan_id, id_in_plan
Create a structured strength workout with repeated sets. Exercises must come from the Coros exercise catalogue.
{
"name": "Leg Circuit",
"sets": 3,
"exercises": [
{
"origin_id": "54",
"name": "T1061",
"overview": "sid_strength_squats",
"target_type": 3,
"target_value": 12,
"rest_seconds": 45
},
{
"origin_id": "130",
"name": "T1176",
"overview": "sid_strength_plank",
"target_type": 2,
"target_value": 60,
"rest_seconds": 30
}
]
}target_type: 2 = time in seconds, 3 = reps
Returns: workout_id, name, sets, exercise_count
List the Coros exercise catalogue for a sport type. Default sport_type=4 returns strength exercises.
{ "sport_type": 4 }Returns: exercises (list), count, sport_type
- Python ≥ 3.11
- A Coros account (Training Hub)
coros-mcp/
├── server.py # MCP server with tool definitions
├── coros_api.py # Coros API client (auth, requests, parsers)
├── models.py # Pydantic data models
├── cli.py # CLI commands (auth, auth-mobile, auth-status, auth-clear)
├── auth/ # Token storage (keyring + encrypted file fallback)
├── pyproject.toml # Project metadata & dependencies
└── docs/
└── mobile-token.md # Mobile API token background (legacy reference)
- fastmcp — MCP framework
- httpx — Async HTTP client
- pydantic — Data validation
- pycryptodome — AES encryption for mobile API auth
- keyring — Secure token storage
- python-dotenv —
.envsupport
This project uses the unofficial Coros Training Hub API. The API may change at any time without notice. Use at your own risk.