fitness-mcp
A personal, remote MCP server for fitness data:
- Garmin Connect activity data via
garminconnect - Hevy workout data via the Hevy API
Designed to run on a Raspberry Pi behind nginx at something like:
https://mcp.home.owenrumney.co.uk/mcp
Tools
Garmin
list_garmin_activitieslist_garmin_activities_by_dateget_garmin_last_activityget_garmin_activityget_garmin_activity_detailsget_garmin_activity_splitsget_garmin_activity_weatherget_garmin_activity_types
Hevy
list_hevy_workoutsget_hevy_workoutlist_hevy_routineslist_hevy_routine_folderslist_hevy_exercise_templates
Local setup
Requires Python 3.12+.
uv sync
cp .env.example .env
Edit .env, then run the Garmin login helper once if your Garmin account needs MFA:
set -a
source .env
set +a
uv run fitness-mcp-garmin-login
Run the server locally over Streamable HTTP:
uv run fitness-mcp --transport streamable-http --host 0.0.0.0 --port 8000
Health check:
curl http://127.0.0.1:8000/health
MCP endpoint:
http://127.0.0.1:8000/mcp
If MCP_AUTH_TOKEN is set, callers must send:
Authorization: Bearer <MCP_AUTH_TOKEN>
Docker on Raspberry Pi
cp .env.example .env
mkdir -p data
# edit .env
docker compose build
docker compose run --rm fitness-mcp uv run fitness-mcp-garmin-login
docker compose up -d
The compose file publishes only to localhost:
127.0.0.1:8000 -> container:8000
Put nginx in front of it and proxy:
https://mcp.home.owenrumney.co.uk/mcp -> http://127.0.0.1:8000/mcp
See deploy/nginx.conf for an example site config.
Environment
MCP_AUTH_TOKEN=replace-with-a-long-random-value
GARMIN_EMAIL=you@example.com
GARMIN_PASSWORD=replace-me
GARMIN_TOKEN_STORE=/data/garmin
HEVY_API_KEY=replace-me
HEVY_BASE_URL=https://api.hevyapp.com/v1
Notes
- Garmin Connect uses unofficial/private APIs through
garminconnect, so MFA friction, rate limits, or upstream breakage are possible. - The server currently exposes read-only tools only.
- If ChatGPT's MCP connector requires OAuth rather than bearer auth, this server will need an OAuth wrapper in front of the same
/mcpbackend.






