Skip to main content

Python 3 API wrapper for Garmin Connect

Project description

GitHub Release GitHub Activity License Project Maintenance

Donate via PayPal Sponsor on GitHub

Python: Garmin Connect

The Garmin Connect API library comes with two examples:

  • example.py - Simple getting-started example showing authentication, token storage, and basic API calls
  • demo.py - Comprehensive demo providing access to 130+ API methods organized into 13 categories for easy navigation
$ ./demo.py
🏃‍♂️ Full-blown Garmin Connect API Demo - Main Menu
==================================================
Select a category:

  [1] 👤 User & Profile
  [2] 📊 Daily Health & Activity
  [3] 🔬 Advanced Health Metrics
  [4] 📈 Historical Data & Trends
  [5] 🏃 Activities & Workouts
  [6] ⚖️ Body Composition & Weight
  [7] 🏆 Goals & Achievements
  [8]  Device & Technical
  [9] 🎽 Gear & Equipment
  [0] 💧 Hydration & Wellness
  [a] 🔧 System & Export
  [b] 📅 Training plans
  [c]  Golf

  [q] Exit program

Make your selection:

API Coverage Statistics

  • Total API Methods: 134+ unique endpoints (snapshot)
  • Categories: 13 organized sections
  • User & Profile: 4 methods (basic user info, settings)
  • Daily Health & Activity: 9 methods (today's health data)
  • Advanced Health Metrics: 12 methods (fitness metrics, HRV, VO2, training readiness, running tolerance)
  • Historical Data & Trends: 9 methods (date range queries, weekly aggregates)
  • Activities & Workouts: 38 methods (comprehensive activity, workout management, typed workout uploads, scheduling, import, edit description / exercise sets)
  • Body Composition & Weight: 8 methods (weight tracking, body composition)
  • Goals & Achievements: 15 methods (challenges, badges, goals)
  • Device & Technical: 7 methods (device info, settings)
  • Gear & Equipment: 7 methods (gear management, tracking)
  • Hydration & Wellness: 12 methods (hydration, nutrition, blood pressure, menstrual)
  • System & Export: 4 methods (reporting, logout, GraphQL)
  • Training Plans: 3 methods (plans, plan by ID, adaptive plan by ID)
  • Golf: 3 methods (scorecard summary, scorecard detail, shot data)

Interactive Features

  • Enhanced User Experience: Categorized navigation with emoji indicators
  • Smart Data Management: Interactive weigh-in deletion with search capabilities
  • Comprehensive Coverage: All major Garmin Connect features are accessible
  • Error Handling: Robust error handling with user-friendly prompts
  • Data Export: JSON export functionality for all data types

Donate via PayPal Sponsor on GitHub

A comprehensive Python3 API wrapper for Garmin Connect, providing access to health, fitness, and device data.

📖 About

This library enables developers to programmatically access Garmin Connect data including:

  • Health Metrics: Heart rate, sleep, stress, body composition, SpO2, HRV
  • Activity Data: Workouts, typed workout uploads (running, cycling, swimming, walking, hiking), workout scheduling, exercises, training status, performance metrics, import-style uploads (no Strava re-export)
  • Nutrition: Daily food logs, meals, and nutrition settings
  • Golf: Scorecard summaries, scorecard details, shot-by-shot data
  • Device Information: Connected devices, settings, alarms, solar data
  • Goals & Achievements: Personal records, badges, challenges, race predictions
  • Historical Data: Trends, progress tracking, date range queries

Compatible with all Garmin Connect accounts. See https://connect.garmin.com/

📦 Installation

Install from PyPI:

pip install --upgrade garminconnect curl_cffi

Run demo software (recommended)

Clone the repo, then:

python3 -m venv .venv --copies
source .venv/bin/activate  # On Windows: .venv\Scripts\activate
pip install -e ".[example]"

python3 ./example.py   # simple getting-started example
python3 ./demo.py      # comprehensive demo (130+ API methods)

🛠️ Development

This project uses PDM for dependency management and task automation.

⚠️ Important: Create a virtual environment first on externally-managed Python installs (Debian/Ubuntu) to avoid system package conflicts.

python3 -m venv .venv --copies
source .venv/bin/activate  # On Windows: .venv\Scripts\activate
pip install pdm
python -m pdm install --group :all
pre-commit install --install-hooks  # optional but recommended

Note: Using python -m pdm instead of pdm avoids PATH issues on some Windows setups where pip install pdm places the pdm executable outside the directories on PATH. Once pdm install has run, subsequent pdm run ... commands work normally because the venv's Scripts/ directory is on PATH while the venv is active.

Development commands:

pdm run format      # Auto-format code (isort, black, ruff --fix)
pdm run lint        # Check code quality (isort, ruff, black, mypy)
pdm run codespell   # Check spelling
pdm run test        # Run test suite
pdm run testcov     # Run tests with coverage report
pdm run all         # Run all checks (lint + codespell + pre-commit + test)
pdm run clean       # Clean build artifacts and cache files
pdm run build       # Build package for distribution
pdm run publish     # Build and publish to PyPI
pdm run --list      # Show all available commands

Run pdm run format && pdm run lint && pdm run test before submitting PRs.

🔐 Authentication

Authentication uses the same mobile SSO flow as the official Garmin Connect Android app. No browser is needed.

How it works:

  1. First login: Authenticates via sso.garmin.com/mobile/api/login using the Android app's client ID. If MFA is required, a callback (prompt_mfa) prompts for the one-time code.
  2. Token exchange: The service ticket is exchanged for DI OAuth Bearer tokens (access_token + refresh_token) via diauth.garmin.com. Tokens are stored at ~/.garminconnect/garmin_tokens.json.
  3. Auto-refresh: Before each API request the library checks whether the DI token is about to expire and refreshes it automatically — no user interaction required.

Session lifetime:

  • DI tokens auto-refresh indefinitely as long as the refresh token remains valid.
  • A full re-login with credentials (and possibly MFA) is only needed if the refresh token itself expires or is revoked.

Token storage:

~/.garminconnect/garmin_tokens.json   # saved automatically, mode 0600

Resilient login (multi-strategy + token validation):

login() tries several authentication strategies in order (mobile, SSO widget, web portal — each with and without TLS impersonation) and only declares success when the resulting token is actually accepted by the API. If a strategy obtains a token the API later rejects (a region/account-specific condition — see #369), the library transparently falls through to the next strategy. Set Garmin(..., verify_login=False) to restore the legacy "first token wins" behavior.

Cached-token gotcha & self-healing: when a tokenstore is supplied, login() loads those tokens before the strategy chain and short-circuits if they load — so stale/poisoned cached tokens used to fail every run. The library now detects this: if cached tokens are rejected by the API, it discards them and performs a fresh credential login automatically. To force a clean slate yourself (e.g. between a failed resume and a retry), call:

g.logout()            # clears in-memory auth + cached tokens (uses GARMINTOKENS)
g.logout(tokenstore)  # or pass an explicit path

🧪 Testing

Run example.py once first to create saved tokens in ~/.garminconnect, then:

pdm run test        # Run all tests
pdm run testcov     # Run tests with coverage report

Optional: keep test tokens isolated

export GARMINTOKENS="$(mktemp -d)"
python3 ./example.py   # create a fresh token file for tests
pdm run test

Note: Tests use VCR cassettes to record/replay API responses. If tests fail with authentication errors, ensure valid tokens exist in ~/.garminconnect (run example.py first).

📦 Publishing

For package maintainers:

Setup PyPI credentials:

pip install twine
# Edit with your preferred editor, or create via here-doc:
# cat > ~/.pypirc <<'EOF'
# [pypi]
# username = __token__
# password = <PyPI_API_TOKEN>
# EOF
[pypi]
username = __token__
password = <PyPI_API_TOKEN>

Recommended: use environment variables and restrict file perms

chmod 600 ~/.pypirc
export TWINE_USERNAME="__token__"
export TWINE_PASSWORD="<PyPI_API_TOKEN>"

Publish new version:

pdm run publish    # Build and publish to PyPI

Alternative publishing steps:

pdm run build      # Build package only
pdm publish        # Publish pre-built package

🤝 Contributing

We welcome contributions! Here's how you can help:

  • Report Issues: Bug reports and feature requests via GitHub issues
  • Submit PRs: Code improvements, new features, documentation updates
  • Testing: Help test new features and report compatibility issues
  • Documentation: Improve examples, add use cases, fix typos

Before contributing:

  1. Set up your dev environment (see Development above)
  2. Format and lint: pdm run format && pdm run lint
  3. Run tests: pdm run test
  4. Follow existing code style and patterns

Jupyter Notebook

Explore the API interactively with our reference notebook.

Python Code Examples

import os
from datetime import date
from garminconnect import Garmin

# First run: logs in and saves tokens to ~/.garminconnect
# Subsequent runs: loads saved tokens and auto-refreshes
client = Garmin(
    os.getenv("EMAIL"),
    os.getenv("PASSWORD"),
    prompt_mfa=lambda: input("MFA code: "),
)
client.login("~/.garminconnect")

# Get today's stats
today = date.today().isoformat()
stats = client.get_stats(today)

# Get heart rate data
hr_data = client.get_heart_rates(today)
print(f"Resting HR: {hr_data.get('restingHeartRate', 'n/a')}")

Typed Workouts (Pydantic Models)

The library includes optional typed workout models for creating type-safe workout definitions:

pip install garminconnect[workout]
from garminconnect.workout import (
    RunningWorkout, WorkoutSegment,
    create_warmup_step, create_interval_step, create_cooldown_step,
    create_repeat_group,
)

# Create a structured running workout
workout = RunningWorkout(
    workoutName="Easy Run",
    estimatedDurationInSecs=1800,
    workoutSegments=[
        WorkoutSegment(
            segmentOrder=1,
            sportType={"sportTypeId": 1, "sportTypeKey": "running"},
            workoutSteps=[create_warmup_step(300.0)]
        )
    ]
)

# Upload and optionally schedule it
result = client.upload_running_workout(workout)
client.schedule_workout(result["workoutId"], "2026-03-20")

# Delete a workout or remove it from the calendar
client.delete_workout(workout_id)
client.unschedule_workout(scheduled_workout_id)

Available workout classes: RunningWorkout, CyclingWorkout, SwimmingWorkout, WalkingWorkout, HikingWorkout, MultiSportWorkout, FitnessEquipmentWorkout

Helper functions: create_warmup_step, create_interval_step, create_recovery_step, create_cooldown_step, create_repeat_group

Additional Resources

  • Simple Example: example.py - Getting started guide
  • Comprehensive Demo: demo.py - All 130+ API methods
  • API Documentation: Comprehensive method documentation in source code
  • Test Cases: Real-world usage examples in tests/ directory

🙏 Acknowledgments

Special thanks to all contributors who have helped improve this project:

  • Community Contributors: Bug reports, feature requests, and code improvements
  • Issue Reporters: Helping identify and resolve compatibility issues
  • Feature Developers: Adding new API endpoints and functionality
  • Documentation Authors: Improving examples and user guides

This project thrives thanks to community involvement and feedback.

💖 Support This Project

If you find this library useful for your projects, please consider supporting its continued development and maintenance:

🌟 Ways to Support

  • ⭐ Star this repository - Help others discover the project
  • 💰 Financial Support - Contribute to development and hosting costs
  • 🐛 Report Issues - Help improve stability and compatibility
  • 📖 Spread the Word - Share with other developers

💳 Financial Support Options

Donate via PayPal Sponsor on GitHub

Why Support?

  • Keeps the project actively maintained
  • Enables faster bug fixes and new features
  • Supports infrastructure costs (testing, AI, CI/CD)
  • Shows appreciation for hundreds of hours of development

Every contribution, no matter the size, makes a difference and is greatly appreciated! 🙏

Project details


Release history Release notifications | RSS feed

This version

0.3.5

Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

garminconnect-0.3.5.tar.gz (61.7 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

garminconnect-0.3.5-py3-none-any.whl (57.3 kB view details)

Uploaded Python 3

File details

Details for the file garminconnect-0.3.5.tar.gz.

File metadata

  • Download URL: garminconnect-0.3.5.tar.gz
  • Upload date:
  • Size: 61.7 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: pdm/2.26.1 CPython/3.13.5 Linux/6.12.88+deb13-amd64

File hashes

Hashes for garminconnect-0.3.5.tar.gz
Algorithm Hash digest
SHA256 b4de13fa5e6c581348cbd8110afb9095dc68273d6cfb1284ed13cae7df0f6b02
MD5 d20428e680eff8db02dfb2f270aafd49
BLAKE2b-256 64d8df118bbbf9a4634bf335cba7baf66f0618babbbf6982c7c43560f9d96d32

See more details on using hashes here.

File details

Details for the file garminconnect-0.3.5-py3-none-any.whl.

File metadata

  • Download URL: garminconnect-0.3.5-py3-none-any.whl
  • Upload date:
  • Size: 57.3 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: pdm/2.26.1 CPython/3.13.5 Linux/6.12.88+deb13-amd64

File hashes

Hashes for garminconnect-0.3.5-py3-none-any.whl
Algorithm Hash digest
SHA256 7a34c408ab4b94a69f67baeda96e8c8aef674447389bbb39beea0e5e49c2a29d
MD5 0e8796422bc4ad78db5a8f83732970fd
BLAKE2b-256 19bbc35ed445156311b9408aa2a10b6c9199392ea1fa26144de947d544c5601a

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page