Skip to main content

Git repository commit statistics collector and visualizer

Project description

oh-my-gitstats

中文 | English

Collect commit data from multiple git repositories and visualize it as a single interactive HTML report.

Line Chart Heatmap

Features

  • Batch Collection — Recursively scan a directory for all git repositories
  • Incremental Sync — Only fetch new commits since the last collection
  • Automated Workflowgitstats auto runs collect → sync → visualize and opens the browser
  • Line Charts — Switch metric (lines changed / commit count) and granularity (day / week / month); editable date range bar (two date inputs + 30D / 90D / 6M / 1Y / ALL presets), with the legend filtered to the selected range
  • Calendar Heatmaps — View commit activity filtered by year
  • Aggregate and Per-Repo Views — A 2-column grid showing combined or per-repo statistics
  • Repo Detail Modal — Click any repo card to open its line chart, heatmap, and meta info (commits / lines / first / last / local / remote / action)
  • Local / Remote Indicators — Each sync status is rendered as two independent signal lamps for working-tree state and remote-tracking state
  • VS Code Integration — Open repo folders directly from the HTML report

Installation

pip install oh-my-gitstats

Or install from source:

git clone https://github.com/amomorning/oh-my-gitstats.git
cd oh-my-gitstats
pip install -e .

Quick Start

First time, collect from each project directory (paths are recorded to the config automatically):

cd ~/projects && gitstats collect .
cd ~/work && gitstats collect .

After that, a single command runs collect → sync → visualize and opens the browser:

gitstats auto

You can also edit the config file manually to add directories:

# Auto-created at:
~/.gitstats/settings.json
{
  "collect_paths": [
    "/home/user/projects",
    "/home/user/work"
  ]
}

Options:

Option Description
-q, --quiet Suppress output messages
--check Check GitHub archive status (requires network; set GITHUB_TOKEN for private repos)
--no-open Do not open the HTML file in the browser after generation

Commands

collect — Collect commit data

Scan a directory for git repositories and export them to JSON:

gitstats collect /path/to/repos
Option Description
-o, --output Directory to save JSON files (default: ~/.gitstats/data)
-q, --quiet Suppress output messages
--skip Skip repos that already have a JSON file
--check Check GitHub archive status

sync — Incremental sync

Update existing JSON files with only new commits — faster than re-collecting:

gitstats sync
Option Description
-q, --quiet Suppress output messages
--check Check GitHub archive status

Typical workflow with multiple directories:

# One-time: collect from different locations
gitstats collect /path/to/work-projects
gitstats collect /path/to/personal-projects --skip

# Later: update all at once
gitstats sync

visualize — Generate visualization

Create an interactive HTML file from the collected data:

gitstats visualize
Option Description
-o, --output HTML file path (default: ~/.gitstats/stats.html)

Granularity and metric can be switched dynamically in the generated HTML — no need to regenerate.

Configuration

Config file at ~/.gitstats/settings.json (auto-created on first run):

{
  "data_dir": "~/.gitstats/data",
  "output_html": "~/.gitstats/stats.html",
  "collect_paths": []
}
Field Description
data_dir Where JSON files are stored
output_html Where the HTML visualization is generated
collect_paths Directories gitstats auto scans (auto-populated when you run collect)

GitHub Token (Optional)

--check queries the GitHub API to check archive status. Without authentication, only public repositories can be checked (rate limit: 60 requests/hour).

If GITHUB_TOKEN is not set, a warning is printed when using --check.

To check private repositories, set the GITHUB_TOKEN environment variable:

Linux / macOS

export GITHUB_TOKEN=ghp_your_token_here
gitstats sync --check

Windows (PowerShell)

Set for the current session:

$env:GITHUB_TOKEN="ghp_your_token_here"
gitstats sync --check

Set permanently: SettingsSystemAboutAdvanced system settingsEnvironment Variables → User variables → New

Getting a token

  1. Go to GitHubSettingsDeveloper settingsPersonal access tokensTokens (classic)
  2. Click Generate new token (classic)
  3. Give it a name (e.g. oh-my-gitstats)
  4. Under Select scopes, no additional scopes are needed for public repos
  5. To access private repositories, check the repo scope
  6. Click Generate token and copy the value (starts with ghp_)

Use Tokens (classic), not Fine-grained tokens. With a token, the rate limit increases to 5,000 requests/hour.

Output

The generated HTML contains:

  1. Line Chart (01 / Trend) — Metric selector (lines changed / commit count) + granularity selector (day / week / month) + editable date range bar with two <input type="date"> (or <input type="month"> when granularity is month) and preset buttons (30D / 90D / 6M / 1Y / ALL). The range drives and is driven by the ECharts dataZoom; the legend is filtered to show only repositories with commits inside the selected range.

  2. Aggregate Heatmap (02 / Aggregate) — Combined activity across all repos with a year selector (all years / specific year); height auto-adjusts when switching between single-year and multi-year ranges.

  3. Individual Heatmaps (03 / Repositories) — A 2-column grid of per-repo cards. Each card shows the repo name, monospace path, Local + Remote signal lamps (small colored circles labeled L / R, in green / yellow / red / gray), and a Continue / Archived button with an MDI icon (vscode://file/ URI). Click any card to open the detail modal with a per-repo line chart (default granularity day), heatmap, and a 7-cell meta grid (commits / lines / first / last / local / remote / action). Closeable via the × button, a backdrop click, or the Escape key.

Individual Heatmaps

JSON Format

Each repository generates a JSON file (~/.gitstats/data/{repo_name}.json):

{
  "repo_name": "my-project",
  "repo_path": "/absolute/path/to/my-project",
  "last_commit_hash": "a1b2c3d4...",
  "sync_status": "synced",
  "is_archived": false,
  "commits": [
    {
      "timestamp": "2024-01-15T10:30:00",
      "additions": 45,
      "deletions": 12
    }
  ]
}

last_commit_hash — HEAD hash at collection time. During sync, repos with a matching hash are skipped.

sync_status — Sync state, decomposed at render time into two independent Local + Remote signal lamps (the stored JSON value is a single enum):

sync_status Local Remote
synced 🟢 Clean 🟢 Synced
local_changes 🟡 Dirty 🟢 Synced
remote_ahead 🟢 Clean 🟡 Ahead
diverged 🟡 Dirty 🟡 Ahead
local_only_clean 🟢 Clean ⚪ None
local_only_dirty 🟡 Dirty ⚪ None
network_error_clean 🟢 Clean 🔴 Error
network_error_dirty 🟡 Dirty 🔴 Error

is_archived — Whether the repo is archived on GitHub (set by --check). Values: true, false, or null (not checked or check failed). Archived repos show a grayed-out Archived button.

Requirements

  • Python 3.9+
  • click, gitpython, pyecharts, jinja2, requests

Project details


Download files

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

Source Distribution

oh_my_gitstats-0.3.3.tar.gz (34.0 kB view details)

Uploaded Source

Built Distribution

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

oh_my_gitstats-0.3.3-py3-none-any.whl (33.0 kB view details)

Uploaded Python 3

File details

Details for the file oh_my_gitstats-0.3.3.tar.gz.

File metadata

  • Download URL: oh_my_gitstats-0.3.3.tar.gz
  • Upload date:
  • Size: 34.0 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.13.9

File hashes

Hashes for oh_my_gitstats-0.3.3.tar.gz
Algorithm Hash digest
SHA256 59d1b577d16854383fe4189e916bbaec9f5f365490cddd53e10c13e2c4ec9245
MD5 e9af5bee21fb84b05cbb36d4c032a148
BLAKE2b-256 54706cdac54175c9e7f537f2ef93be08b271a3362252483de9f179779215d5d9

See more details on using hashes here.

File details

Details for the file oh_my_gitstats-0.3.3-py3-none-any.whl.

File metadata

  • Download URL: oh_my_gitstats-0.3.3-py3-none-any.whl
  • Upload date:
  • Size: 33.0 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.13.9

File hashes

Hashes for oh_my_gitstats-0.3.3-py3-none-any.whl
Algorithm Hash digest
SHA256 4cc67c4428fb3110ffcfe40f2f36df70be3dbee22a35c572ab693d4c0d241828
MD5 657e281d971ab1917b6d173324af179a
BLAKE2b-256 a0682b25ccd7283ff738c3f188eb7a2d78a02457092712f8b5d4127610e7cea0

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