Count lines of code from GitHub pull requests
Project description
trueloc - Lines of Code Counter
A CLI tool to analyze your coding activity via GitHub pull requests and direct commits:
- Total lines written since any date
- Per-PR and per-commit breakdown showing when each contribution was made
- File type analysis revealing which languages you've worked with most
Counts ALL lines touched across every commit in a PRโeven for squash merges where GitHub only shows the final diff. For example, a PR where you add 1000 lines, delete them, then add 1 line shows +1001 / -1000 (not just +1).
Installation
Recommended (installs in isolated environment):
uv tool install trueloc
Or run directly without installing:
uvx trueloc count USERNAME --since 1m
Alternatively, with pip:
pip install trueloc
Requires the GitHub CLI (gh) to be installed and authenticated:
gh auth login
Usage
# Count lines from PRs and direct commits since a date
trueloc count USERNAME --since 2023-01-01
# Use relative dates
trueloc count USERNAME --since 5d # 5 days ago
trueloc count USERNAME --since 2w # 2 weeks ago
trueloc count USERNAME --since 3m # 3 months ago
trueloc count USERNAME --since 1y # 1 year ago
trueloc count USERNAME --since "last month"
# Specify a date range
trueloc count USERNAME --since 2024-01-01 --until 2024-06-30
# Count only net diff (not per-commit)
trueloc count USERNAME --since 2023-01-01 --net
# Exclude direct commits (PRs only)
trueloc count USERNAME --since 2023-01-01 --no-direct-commits
# Hide file extension breakdown
trueloc count USERNAME --since 2023-01-01 --no-extensions
# Disable caching (fresh API calls)
trueloc count USERNAME --since 2023-01-01 --no-cache
# Output as JSON for scripting/postprocessing
trueloc count USERNAME --since 2023-01-01 --json
# Clear the cache
trueloc clear-cache
CLI Reference
Main Commands
echo '```'
trueloc --help
echo '```'
[1m [0m
[1m [0m[1;33mUsage: [0m[1mtrueloc [OPTIONS] COMMAND [ARGS]...[0m[1m [0m[1m [0m
[1m [0m
Count lines of code from GitHub pull requests.
[2mโญโ[0m[2m Options [0m[2mโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโ[0m[2mโโฎ[0m
[2mโ[0m [1;36m-[0m[1;36m-install[0m[1;36m-completion[0m Install completion for the current shell. [2mโ[0m
[2mโ[0m [1;36m-[0m[1;36m-show[0m[1;36m-completion[0m Show completion for the current shell, to [2mโ[0m
[2mโ[0m copy it or customize the installation. [2mโ[0m
[2mโ[0m [1;36m-[0m[1;36m-help[0m [1;32m-h[0m Show this message and exit. [2mโ[0m
[2mโฐโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโฏ[0m
[2mโญโ[0m[2m Commands [0m[2mโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโ[0m[2mโโฎ[0m
[2mโ[0m [1;36mcount [0m[1;36m [0m Count lines of code from merged PRs and direct commits. [2mโ[0m
[2mโ[0m [1;36mclear-cache [0m[1;36m [0m Clear the disk cache. [2mโ[0m
[2mโ[0m [1;36mcount-local [0m[1;36m [0m Count lines of code from a local git repository. [2mโ[0m
[2mโฐโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโฏ[0m
Count Command Options
echo '```'
trueloc count --help
echo '```'
[1m [0m
[1m [0m[1;33mUsage: [0m[1mtrueloc count [OPTIONS] USERNAME[0m[1m [0m[1m [0m
[1m [0m
Count lines of code from merged PRs and direct commits.
[2mBy default, counts all lines touched across all commits in each PR,[0m
[2mplus direct commits to the main branch (not from PRs).[0m
[2mExample: a single PR where you add 1000 lines, delete them, then add[0m
[2m1 line = +1001 / [0m[1;2;32m-1000[0m[2m (even though the PR's net diff shows only +1).[0m
[2mAdditions and deletions are summed separately across all commits.[0m
[2mUse [0m[1;2;36m-[0m[1;2;36m-net[0m[2m to count only the final diff (net additions/deletions).[0m
[2mUse [0m[1;2;36m-[0m[1;2;36m-no[0m[1;2;36m-direct-commits[0m[2m to exclude direct commits to main branch.[0m
[2mโญโ[0m[2m Arguments [0m[2mโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโ[0m[2mโโฎ[0m
[2mโ[0m [31m*[0m username [1;33mTEXT[0m GitHub username [2;31m[required][0m [2mโ[0m
[2mโฐโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโฏ[0m
[2mโญโ[0m[2m Options [0m[2mโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโ[0m[2mโโฎ[0m
[2mโ[0m [31m*[0m [1;36m-[0m[1;36m-since[0m [1;32m-s[0m [1;33mTEXT[0m Start date (e.g., 5d, [2mโ[0m
[2mโ[0m 2w, 3m, 1y, 'last [2mโ[0m
[2mโ[0m month', 2024-01-01) [2mโ[0m
[2mโ[0m [2;31m[required] [0m [2mโ[0m
[2mโ[0m [1;36m-[0m[1;36m-until[0m [1;32m-u[0m [1;33mTEXT[0m End date (e.g., 1d, [2mโ[0m
[2mโ[0m 'yesterday', [2mโ[0m
[2mโ[0m 2024-12-31) [2mโ[0m
[2mโ[0m [1;36m-[0m[1;36m-no[0m[1;36m-cache[0m [1;33m [0m Disable disk cache [2mโ[0m
[2mโ[0m [1;36m-[0m[1;36m-extensions[0m [1;35m-[0m[1;35m-no[0m[1;35m-extensions[0m [1;33m [0m Show breakdown by file [2mโ[0m
[2mโ[0m extension [2mโ[0m
[2mโ[0m [2m[default: extensions] [0m [2mโ[0m
[2mโ[0m [1;36m-[0m[1;36m-per[0m[1;36m-commit[0m [1;35m-[0m[1;35m-net[0m [1;33m [0m Count all lines [2mโ[0m
[2mโ[0m touched per commit [2mโ[0m
[2mโ[0m (default) vs net diff [2mโ[0m
[2mโ[0m only [2mโ[0m
[2mโ[0m [2m[default: per-commit] [0m [2mโ[0m
[2mโ[0m [1;36m-[0m[1;36m-direct[0m[1;36m-commits[0m [1;35m-[0m[1;35m-no[0m[1;35m-direct-commits[0m [1;33m [0m Include direct commits [2mโ[0m
[2mโ[0m to main branch (not [2mโ[0m
[2mโ[0m from PRs) [2mโ[0m
[2mโ[0m [2m[default: [0m [2mโ[0m
[2mโ[0m [2mdirect-commits] [0m [2mโ[0m
[2mโ[0m [1;36m-[0m[1;36m-json[0m [1;33m [0m Output results as JSON [2mโ[0m
[2mโ[0m for scripting [2mโ[0m
[2mโ[0m [1;36m-[0m[1;36m-repo[0m [1;32m-r[0m [1;33mTEXT[0m Only process a single [2mโ[0m
[2mโ[0m repository (e.g., [2mโ[0m
[2mโ[0m 'owner/repo' or [2mโ[0m
[2mโ[0m 'repo') [2mโ[0m
[2mโ[0m [1;36m-[0m[1;36m-help[0m [1;32m-h[0m [1;33m [0m Show this message and [2mโ[0m
[2mโ[0m exit. [2mโ[0m
[2mโฐโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโฏ[0m
Features
- Per-commit counting (default): Counts every line touched in every commit
- Net diff mode: Alternative mode that only counts final diff (
--net) - Direct commits: Includes commits pushed directly to main (not via PR)
- File extension breakdown: Shows which languages you've worked with
- JSON output: Machine-readable output for scripting (
--json) - Disk caching: Uses diskcache to avoid hammering the GitHub API
- Rate limit handling: Automatically waits when rate limited with progress bar
- Flexible dates: Supports relative (
5d,2w,3m,1y) and natural language (last month)
Caching Strategy
- Immutable data (cached forever): commit stats, PR commits, PR files
- Mutable data (1 day TTL): user repos, merged PR lists
Cache is stored in ~/.cache/trueloc/.
Tech Stack
- Python 3.12+
- Typer - CLI framework
- httpx - HTTP client
- Rich - Terminal output
- diskcache - Persistent caching
- dateparser - Flexible date parsing
- Hatch - Build system
- Ruff + mypy - Linting/typing
Development
# Install dev dependencies
uv sync --group dev
# Run tests
pytest
# Run linting
ruff check .
ruff format .
mypy .
License
MIT
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 trueloc-0.3.0.tar.gz.
File metadata
- Download URL: trueloc-0.3.0.tar.gz
- Upload date:
- Size: 75.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
a35666529467721f0557768dd28c427e00fc3a1d0cdbee1ce465647981822dfa
|
|
| MD5 |
68ca3e2c2c62f99d0a928b6c51a2c5fd
|
|
| BLAKE2b-256 |
44439f66c650f0a8b7e58b530160f804cd82e4b407b754b19dd33c4e3d5942b8
|
Provenance
The following attestation bundles were made for trueloc-0.3.0.tar.gz:
Publisher:
release.yml on basnijholt/trueloc
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
trueloc-0.3.0.tar.gz -
Subject digest:
a35666529467721f0557768dd28c427e00fc3a1d0cdbee1ce465647981822dfa - Sigstore transparency entry: 832850057
- Sigstore integration time:
-
Permalink:
basnijholt/trueloc@953812a48ac7355738f37540239d353c6d1337d8 -
Branch / Tag:
refs/tags/v0.3.0 - Owner: https://github.com/basnijholt
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@953812a48ac7355738f37540239d353c6d1337d8 -
Trigger Event:
release
-
Statement type:
File details
Details for the file trueloc-0.3.0-py3-none-any.whl.
File metadata
- Download URL: trueloc-0.3.0-py3-none-any.whl
- Upload date:
- Size: 22.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
29ed1ea9ce4c5a622efd2a5e6d4e28d118376f056860090d4ee4467152674a96
|
|
| MD5 |
eb9f5c8823ad490080b61c42553abcd7
|
|
| BLAKE2b-256 |
36bf0685ee3a6cc160cf78eb7b55e75bfd4a1bfe4adfa1cffd47914d9075242a
|
Provenance
The following attestation bundles were made for trueloc-0.3.0-py3-none-any.whl:
Publisher:
release.yml on basnijholt/trueloc
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
trueloc-0.3.0-py3-none-any.whl -
Subject digest:
29ed1ea9ce4c5a622efd2a5e6d4e28d118376f056860090d4ee4467152674a96 - Sigstore transparency entry: 832850058
- Sigstore integration time:
-
Permalink:
basnijholt/trueloc@953812a48ac7355738f37540239d353c6d1337d8 -
Branch / Tag:
refs/tags/v0.3.0 - Owner: https://github.com/basnijholt
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@953812a48ac7355738f37540239d353c6d1337d8 -
Trigger Event:
release
-
Statement type: