google-health-mcp-server
Retrieves health data from Apple Watch (steps, distance, heart rate, sleep, etc.) unified through Google Health.
Retrieves health data from Fitbit devices (steps, distance, heart rate, sleep, etc.) unified through Google Health.
Provides tools to read daily activity, sleep, heart rate, and body metrics from the Google Health API v4.
Syncs health data into an Obsidian vault as daily markdown notes and a rolling dashboard.
Click on "Install Server".
Wait a few minutes for the server to deploy. Once ready, it will show a "Started" state.
In the chat, type
@followed by the MCP server name and your instructions, e.g., "@google-health-mcp-servershow me my daily activity summary for the last 3 days"
That's it! The server will respond to your query, and you can continue using it as needed.
Here is a step-by-step guide with screenshots.
π©Ί google-health-mcp-server
Hand your whole-body health data to your AI β steps, distance, heart rate, and more from Apple Watch, Fitbit, or Pixel Watch, unified through Google Health.
An MCP server for the Google Health API v4 β read your daily activity, sleep, heart rate, and body metrics from any source connected to Google Health (Apple Watch import, Fitbit Air, Pixel Watch, ...) in Claude or any MCP client, and optionally sync them into an Obsidian vault.
Self-hosted and local-first: it uses your own Google Cloud OAuth app, tokens are stored
on your machine at mode 0600, and no health data leaves your control.
Beta notice. The Google Health API v4 is new (public beta; legacy Fitbit Web API turns down September 2026). Field shapes may still shift. This server normalizes daily values best-effort and always returns the raw rollup points alongside, so nothing is lost if a field mapping in
src/constants.tsneeds a tweak against live data.
Tools
Tool | What it does |
| Per-day table of steps, distance, active calories, active minutes, resting HR, sleep over a date range (or trailing |
| Raw, un-rolled-up points for one dataType β intraday HR, sleep stages, or any type not in the summary. |
| Devices paired to the account β confirm a new device (e.g. Fitbit Air) has connected. |
| Health-check: token works? which scopes granted? which metrics actually return data over the last 7 days? Run this first after setup. |
| Daily markdown notes under |
Related MCP server: Garmin MCP Server
One-time setup
1. Google Cloud project + API
Create or pick a project at https://console.cloud.google.com/.
Enable the Google Health API: https://console.cloud.google.com/apis/library/health.googleapis.com
2. OAuth consent screen
Go to APIs & Services β OAuth consent screen. User type External.
Fill in the app name and your email.
Under Audience β Test users, add your own Google account.
On the Data access page, add these scopes (search "Google Health"):
β¦/auth/googlehealth.profile.readonlyβ¦/auth/googlehealth.settings.readonlyβ¦/auth/googlehealth.activity_and_fitness.readonlyβ¦/auth/googlehealth.health_metrics_and_measurements.readonlyβ¦/auth/googlehealth.sleep.readonlyβ¦/auth/googlehealth.nutrition.readonly
β οΈ 7-day token caveat (and why you should stay in Testing). While the consent screen is in Testing status, Google issues refresh tokens that expire after 7 days. You might expect to fix this by publishing to Production β but the Google Health scopes are Google's Restricted tier, and Restricted scopes in Production require a full security verification (CASA assessment). There is no "click through the unverified warning" bypass for Restricted scopes (that bypass only exists for Sensitive scopes). Verification is impractical for a single-user personal tool, so the recommended setup is: keep the app in Testing, add yourself as a Test user, and accept the 7-day expiry. Re-authorize lazily β only when you actually use the data and a call reports the token has lapsed β by re-running
npm run oauth(a ~10s browser approve). Don't bother re-authing in weeks you don't touch the data.
3. OAuth credentials
APIs & Services β Credentials β Create credentials β OAuth client ID.
Application type: Web application.
Under Authorized redirect URIs, add exactly:
http://localhost:47813/callback(Must match
GOOGLE_HEALTH_OAUTH_PORT. Change both together if 47813 is taken.)Copy the Client ID and Client secret.
4. Configure + authorize
npm install
cp .env.example .env # then paste your client ID + secret into .env
npm run oauth # opens a browser; approve access. Tokens saved to
# ~/.config/google-health-mcp/tokens.json (mode 0600)
npm run buildVerify it works:
node dist/index.js # should print "Google Health MCP server running on stdio"β¦or call connection_status once registered in your MCP client.
Register in Claude
Add to ~/Library/Application Support/Claude/claude_desktop_config.json and restart Claude:
{
"mcpServers": {
"google-health": {
"command": "node",
"args": ["/Users/<you>/projects/google-health-mcp-server/dist/index.js"],
"env": {
"GOOGLE_HEALTH_TOKENS_PATH": "/Users/<you>/.config/google-health-mcp/tokens.json",
"OBSIDIAN_VAULT_PATH": "/Users/<you>/Documents/Obsidian Vault"
}
}
}
}The runtime reads the client ID/secret back from the tokens file, so they don't need to be
in the MCP env block β only the token path and (optionally) the vault path.
Obsidian sync (optional)
Interactive queries via the MCP tools are always live and pull on demand. The Obsidian notes +
dashboard only need a periodic refresh, so set OBSIDIAN_VAULT_PATH and either call
sync_health_to_obsidian from Claude or run the CLI:
npm run sync # incremental sync since last runFor an automated refresh, a launchd template is in launchd/. Copy it to
~/Library/LaunchAgents/, replace __USERNAME__, and
launchctl bootstrap gui/$(id -u) <plist>. It runs every 4 days (well under the 7-day
refresh-token expiry) and is best-effort: if the token has lapsed it logs a "re-auth needed"
note, fires a macOS notification, and exits cleanly (no crash-loop) β re-auth with
npm run oauth (~10s browser approve).
The 4-day cadence narrows staleness, it does not prevent expiry. The cron can only refresh the access token; the refresh token's 7-day Testing-mode cap is fixed by Google regardless of how often it runs, and getting a new refresh token requires the one-time interactive browser approval in
npm run oauthβ that step can't be scripted into a cron. What the cadence change buys you is a couple of chances per week to notice an expiry (via the notification) instead of going silent for up to a month.
Develop
npm run build # tsc --noCheck β dist/
npm run typecheck # strict type check (excludes tests)
npm test # vitest β pure logic (civil-date math, value extraction)
npm run dev # build + run the server on stdioSee CLAUDE.md for architecture notes and known pitfalls.
This server cannot be installed
Maintenance
Resources
Unclaimed servers have limited discoverability.
Looking for Admin?
If you are the server author, to access and configure the admin panel.
Latest Blog Posts
- Your AI Chatbot Just Exposed Your CEO's Salary to an InternBy Om-Shree-0709 on .Agent IdentityMCP SecurityOAuth Delegation
- Why MCP Servers Need Execution Sandboxing (And Why Your Current Stack Isn't Enough)By Om-Shree-0709 on .Agentic AiPrompt InjectionWebAssembly
MCP directory API
We provide all the information about MCP servers via our MCP API.
curl -X GET 'https://glama.ai/api/mcp/v1/servers/madfreakz/google-health-mcp-server'
If you have feedback or need assistance with the MCP directory API, please join our Discord server