Adaptive AI training coach MCP server backed by Garmin Connect
Project description
coach-mcp
An opinionated AI training coach as an MCP server. It pulls your real data from Garmin Connect and prescribes with authority — science-based load management (ACWR), code-enforced injury gates (the server rejects plans that violate an active injury restriction, no matter what the LLM says), and persistent coaching memory so decisions, rationale, and your adaptation patterns survive between conversations. It will tell you "no" when your enthusiasm exceeds your capacity.
All health data and credentials stay on your machine — see Security & Privacy.
Quickstart
You need Python 3.12+, a free Garmin Connect account, and an MCP client (Claude Code, Claude Desktop, or Cursor).
Option A: uvx (after PyPI release)
No install step — your MCP client runs the server on demand:
uvx garmin-coach-mcp
Jump to Connect your MCP client and use uvx as
the command.
Option B: from source
git clone https://github.com/snoozelieb/coach-mcp.git
cd coach-mcp
python -m venv .venv
# Linux/macOS:
source .venv/bin/activate
# Windows:
.venv\Scripts\activate
pip install -r requirements.txt
cp .env.example .env # then edit: GARMIN_EMAIL, GARMIN_PASSWORD
python server.py
Connect your MCP client
The server needs two environment variables: GARMIN_EMAIL and
GARMIN_PASSWORD. Optional: COACH_DATA_DIR (where your coaching data lives)
and ANTHROPIC_API_KEY (only for the standalone daily_loop.py --llm script).
From a source checkout, a .env file works too.
Claude Code
claude mcp add coach-mcp \
--env GARMIN_EMAIL=you@example.com \
--env GARMIN_PASSWORD=your_garmin_password \
-- uvx garmin-coach-mcp
Or in .mcp.json:
{
"mcpServers": {
"coach-mcp": {
"command": "uvx",
"args": ["garmin-coach-mcp"],
"env": {
"GARMIN_EMAIL": "you@example.com",
"GARMIN_PASSWORD": "your_garmin_password",
"COACH_DATA_DIR": "/path/to/your/coach-data"
}
}
}
}
Running from source instead: claude mcp add coach-mcp -- python /full/path/to/coach-mcp/server.py
Claude Desktop
In claude_desktop_config.json (Settings → Developer → Edit Config):
{
"mcpServers": {
"coach-mcp": {
"command": "uvx",
"args": ["garmin-coach-mcp"],
"env": {
"GARMIN_EMAIL": "you@example.com",
"GARMIN_PASSWORD": "your_garmin_password",
"COACH_DATA_DIR": "/path/to/your/coach-data"
}
}
}
}
Cursor
In .cursor/mcp.json (project) or ~/.cursor/mcp.json (global):
{
"mcpServers": {
"coach-mcp": {
"command": "uvx",
"args": ["garmin-coach-mcp"],
"env": {
"GARMIN_EMAIL": "you@example.com",
"GARMIN_PASSWORD": "your_garmin_password",
"COACH_DATA_DIR": "/path/to/your/coach-data"
}
}
}
}
If you installed with pip install garmin-coach-mcp instead of uvx, use
"command": "garmin-coach-mcp" with no args in any of the blocks above.
First run
-
Create your profile. From a source checkout, run the interactive wizard:
python scripts/setup_wizard.pyIt creates your athlete profile, training config, and empty plan/memory files in the data directory. Alternatively, create the two required files by hand and let the coach fill in the rest via conversation:
echo '{"personal":{"name":null},"injury_history":[],"life_constraints":{}}' > data/athlete.json echo '{"events":[],"current_block":{"phase":"base"}}' > data/training_config.json
-
Pull your Garmin baseline. In your MCP client, say:
"Run refresh_athlete_baseline and set up my training."
The coach pulls your name, weight, age, HR data, and training capacity from Garmin, then starts the onboarding conversation — goals, constraints, injury history, race calendar.
-
Garmin MFA / expired session. Garmin logins are token-cached. If tools start returning
AUTH_REQUIRED, recover with:python scripts/garmin_login.pyIt does a fresh credential login, prompts for the MFA code if Garmin asks, and saves new tokens. Restart the MCP server afterwards.
How it works
- Snapshot first — every coaching conversation starts from
get_coaching_snapshot(): current time context, 7-day week grid (rest days explicit), fitness metrics, plan adherence, open anomalies, injuries, sleep gate. - Load hierarchy before prescribing — overall ACWR (injury gate, 0.8–1.3 sweet spot), then sport-specific ACWR (spike detection), then sport-specific CTL (race readiness).
- Hard gates are code, not vibes —
update_weekly_planandpush_plan_to_garminreject sessions that violate an active injury's restricted activities, and every non-rest session must carry apurposeor the save is refused. - Curiosity with memory — planned-vs-actual anomalies (missed session, type mismatch, activity on a rest day) register once with a lifecycle (open → asked → resolved); the coach asks you what happened instead of silently assuming.
- Everything persists — decisions, approvals, adaptation patterns, and season lifecycle (race debriefs, phase transitions) live in local JSON and carry across sessions.
MCP surface
48 tools — you don't call them directly; the coach uses them during conversation:
| Category | Tools |
|---|---|
| Coaching core | get_coaching_snapshot (canonical, sectioned), get_compliance_report, get_coaching_score |
| Planning | get_weekly_plan, update_weekly_plan, push_plan_to_garmin, get_week_constraints, get_weekly_prescription, get_periodization_status, update_phase |
| Garmin data | query_metrics (kind=fitness/intensity/daily/readiness/personal_records), get_activities_range |
| Athlete | get_athlete, update_athlete, set_ftp, set_threshold_pace, analyze_ftp_test, refresh_athlete_baseline, refresh_fitness_history, get_onboarding_guide |
| Methodology | get_methodology, update_methodology |
| Races | races (action=list/add/update/research), remove_race |
| Strength | sync_strength_session, get_strength_baseline, approve_progression, set_exercise_preference, generate_strength_workout, add_exercise |
| Injuries | diagnose_injury, research_injury, update_injury_status |
| Research | research_exercise, list_exercises, research_sport |
| Memory | log_coaching_decision, get_active_decisions, update_decision_status, record_athlete_response, get_response_patterns, resolve_anomaly |
| Approvals | propose_coaching_action, list_pending_approvals, approve_proposal, reject_proposal |
| Interactive | generate_smart_brief, interactive_check_in |
Every tool carries MCP annotations (read-only / destructive / idempotent / open-world), enforced by tests.
5 prompts: weekly_planning, morning_brief, injury_assessment,
week_review, onboarding.
6 resources: coach://athlete/profile, coach://plan/current,
coach://config/training, coach://coaching/decisions, coach://context/now,
coach://coaching/doctrine (the long-form coaching doctrine).
Security & Privacy
Everything stays on your machine:
- Credentials:
GARMIN_EMAIL/GARMIN_PASSWORDlive in your MCP client config or a local.env. Garmin OAuth tokens are cached in a local token store (.garth/garmin_tokens.json). - Health data: all coaching data (profile, plans, fitness history, sleep, coaching memory) is local JSON in your data directory. There is no backend, no telemetry, no analytics.
- What leaves your machine: requests to Garmin's own API (your
credentials/tokens, sent only to Garmin); whatever your MCP client sends to
its LLM as part of the conversation; optional public web-page fetches when
the coach researches a race, injury, or exercise; and, only if you run
daily_loop.py --llm, one request to the Anthropic API. - Single athlete per data directory by design. For multiple athletes, run
separate server instances with separate
COACH_DATA_DIRs.
See SECURITY.md for details and how to report issues.
Data directory
Resolution order: COACH_DATA_DIR env var → data/ in a source checkout → a
per-user data directory (created on first run for installed packages). The
only file shipped with the package is methodology.json (safety rules, race
templates, personas); everything personal is created locally and never
committed.
Advanced
# HTTP / SSE transport instead of stdio
COACH_TRANSPORT=streamable-http FASTMCP_PORT=8000 garmin-coach-mcp
# Code Mode (search/execute meta-tools instead of 48 individual tools)
pip install fastmcp[code-mode]
COACH_CODE_MODE=1 garmin-coach-mcp
# Standalone morning audit
python scripts/daily_loop.py # template-based brief
python scripts/daily_loop.py --llm # LLM brief (needs ANTHROPIC_API_KEY)
# Tests (1,260 tests; clean checkouts use committed sanitized fixtures)
pip install -r requirements-dev.txt
python -m pytest -q
Architecture
server.py registers tools from the coach/ package (11 tool modules, pure
parsers, a typed pydantic storage layer, CTL/ATL/ACWR fitness math, a Garmin
client with token-first auth, and a workout builder that pushes structured
workouts to your watch). The project went through a five-phase modernization —
auth rebuild, schema layer, hard gates, sectioned snapshot, packaging — whose
full history and rationale live in
docs/UPGRADE_ROADMAP.md. Development conventions
are in CLAUDE.md.
License
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 garmin_coach_mcp-1.0.0.tar.gz.
File metadata
- Download URL: garmin_coach_mcp-1.0.0.tar.gz
- Upload date:
- Size: 400.4 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
501915b4335f76a897dfbbb791df2eccb74945697a3b40885c9275d6a027e529
|
|
| MD5 |
5bc9082f788cdb49031839d4b5291ff7
|
|
| BLAKE2b-256 |
a7dc158c5d3c569685315cc7bfe26b181770924e53b1a438717a4a4a50303a18
|
Provenance
The following attestation bundles were made for garmin_coach_mcp-1.0.0.tar.gz:
Publisher:
release.yml on snoozelieb/coach-mcp
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
garmin_coach_mcp-1.0.0.tar.gz -
Subject digest:
501915b4335f76a897dfbbb791df2eccb74945697a3b40885c9275d6a027e529 - Sigstore transparency entry: 1779699700
- Sigstore integration time:
-
Permalink:
snoozelieb/coach-mcp@a7a7e722c5dfd6ba83208e7b8a608a29d0e2825d -
Branch / Tag:
refs/tags/v1.0.0 - Owner: https://github.com/snoozelieb
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@a7a7e722c5dfd6ba83208e7b8a608a29d0e2825d -
Trigger Event:
push
-
Statement type:
File details
Details for the file garmin_coach_mcp-1.0.0-py3-none-any.whl.
File metadata
- Download URL: garmin_coach_mcp-1.0.0-py3-none-any.whl
- Upload date:
- Size: 204.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
3f630724ebced5bc8748da29fd1645a6fa75a3dda69e689e14f325c5dab1c16b
|
|
| MD5 |
1bbf7901adef5085d8a43d9a5dc789bc
|
|
| BLAKE2b-256 |
d20ef05d600d17d4e9c88140c3ef43351e576b2d5ac97bb0f741be30730066fc
|
Provenance
The following attestation bundles were made for garmin_coach_mcp-1.0.0-py3-none-any.whl:
Publisher:
release.yml on snoozelieb/coach-mcp
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
garmin_coach_mcp-1.0.0-py3-none-any.whl -
Subject digest:
3f630724ebced5bc8748da29fd1645a6fa75a3dda69e689e14f325c5dab1c16b - Sigstore transparency entry: 1779699815
- Sigstore integration time:
-
Permalink:
snoozelieb/coach-mcp@a7a7e722c5dfd6ba83208e7b8a608a29d0e2825d -
Branch / Tag:
refs/tags/v1.0.0 - Owner: https://github.com/snoozelieb
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@a7a7e722c5dfd6ba83208e7b8a608a29d0e2825d -
Trigger Event:
push
-
Statement type: