CovLoupe
An MCP server, command line utility, and library for Ruby SimpleCov test coverage analysis.
Documentation
Full documentation is available at https://keithrbennett.github.io/cov-loupe/.
What is cov-loupe?
cov-loupe makes SimpleCov coverage data queryable and actionable through three interfaces:
- MCP server - Lets AI assistants analyze your coverage
- CLI - Fast command-line coverage reports and queries
- Ruby library - Programmatic API for custom tooling
Works with any SimpleCov-generated .resultset.json file—no runtime dependency on your test suite.
Key Features
- ✅ Multiple interfaces - MCP server, CLI, and Ruby API
-
Annotated source code -
-s full|uncovered/--source full|uncoveredwith-c N/--context-lines Nfor context lines - ✅ Staleness detection - Identify outdated coverage (missing files, timestamp mismatches, line count changes)
- ✅ Multi-suite support - Automatic merging of multiple test suites (RSpec + Cucumber, etc.)
- ✅ Flexible path resolution - Works with absolute or relative paths
- ✅ Comprehensive error handling - Context-aware messages for each mode
- ⚠️ Branch coverage limitation - Branch-level metrics are collapsed to per-line totals. Use native SimpleCov reports for branch-by-branch analysis.
Practical Use Cases
- Query coverage data from AI assistants, e.g.:
- "Using cov-loupe, analyze test coverage data and write a report to a markdown file containing a free text analysis of each issue and then two tables, one sorted in descending order of urgency, the other in ascending order of level of effort."
- "Using cov-loupe, generate a table of directories and their average coverage rates, in ascending order of coverage."
- Find files with the lowest coverage
- Investigate specific files or directories
- Generate CI/CD coverage reports
- Create custom pass/fail predicates for scripts and CI - use the library API or CLI JSON output to implement arbitrarily complex coverage rules beyond simple thresholds (e.g., require higher coverage for critical paths, exempt test utilities, track coverage trends)
Quick Start
Installation
gem install cov-loupeUpgrading
If you are upgrading from a previous version, please refer to the Migration Guides.
Generate Coverage Data
# Run your tests with SimpleCov enabled
bundle exec rspec # or your test command
# Verify coverage was generated
ls -l coverage/.resultset.jsonBasic Usage
CLI - View Coverage Table:
cov-loupeCLI - Check Specific File:
cov-loupe summary lib/cov_loupe/model.rb
cov-loupe uncovered lib/cov_loupe/cli.rbCLI - Find the Project Homepage Fast:
Run cov-loupe -h and the banner's second line shows the repository URL. Some terminal applications (e.g. iTerm2) will enable direct clicking the link using modifier keys such as Cmd or Alt.
Usage: cov-loupe [options] [subcommand] [args]
Repository: https://github.com/keithrbennett/cov-loupe # <--- Project URL ---
Ruby Library:
require "cov_loupe"
model = CovLoupe::CoverageModel.new
files = model.list
# => [{ "file" => "lib/cov_loupe/model.rb", "covered" => 114, "total" => 118, "percentage" => 96.61, "stale" => false }, ...]
summary = model.summary_for("lib/cov_loupe/model.rb")
# => { "file" => "lib/cov_loupe/model.rb", "summary" => { "covered" => 114, "total" => 118, "percentage" => 96.61 }, "stale" => false }MCP Server: See MCP Integration Guide for AI assistant setup.
Multi-Suite Coverage Merging
How It Works
When a .resultset.json file contains multiple test suites (e.g., RSpec + Cucumber), cov-loupe automatically merges them using SimpleCov's combine logic. All covered files from every suite become available to the CLI, library, and MCP tools.
Performance: Single-suite projects avoid loading SimpleCov at runtime. Multi-suite resultsets trigger a lazy SimpleCov load only when needed, keeping the tool fast for the simpler coverage configurations.
Current Limitations
Staleness checks: When suites are merged, we keep a single "latest suite" timestamp. This matches prior behavior but may under-report stale files if only some suites were re-run after a change. Use --raise-on-stale (or -S) on the CLI, raise_on_stale: true via the Ruby API, or the MCP tool parameter to turn these warnings into hard failures. A per-file timestamp refinement is planned; until then, treat multi-suite staleness flags as advisory rather than definitive.
Multiple resultset files: Only suites stored inside a single .resultset.json are merged automatically. If your project produces separate resultset files (e.g., different CI jobs writing coverage/job1/.resultset.json, coverage/job2/.resultset.json), you must merge them yourself before pointing cov-loupe at the combined file.
Documentation
Full documentation is available at https://keithrbennett.github.io/cov-loupe/.
User Guides:
- Quick Start - Get up and running in 3 steps
- User Docs Overview - Map of all end-user guides
- Installation - Setup for different environments
- CLI Usage - Command-line reference
- Examples - Common use cases
- Advanced Usage - Staleness detection, error modes, path resolution
- Library API - Ruby API documentation
- Error Handling - Error modes and exceptions
- MCP Integration - AI assistant configuration
- Troubleshooting - Common issues
Special Topics & Prompts:
- CLI Fallback for LLMs - When MCP isn't available
- Codex Env Var Workaround - Passing GEM paths through Codex MCP configs
- Sample MCP Prompts - Ready-to-use ChatGPT/Claude/Gemini prompts
- Migration Guides
Developer Docs:
- Developer Docs Overview - Entry point for contributors
- Architecture - Design and internals
- Branch Coverage - Branch coverage limitations
- Development Guide - Local dev workflow
- Releasing - Release checklist
- Architecture Decision Records - Design history
Requirements
-
Ruby >= 3.2 (required by
mcpgem dependency) - SimpleCov-generated
.resultset.jsonfile -
simplecovgem >= 0.21
Configuring the Resultset
cov-loupe automatically searches for .resultset.json in standard locations (coverage/.resultset.json, .resultset.json, tmp/.resultset.json). For non-standard locations:
# Command-line option (highest priority) - use -r or --resultset
cov-loupe -r /path/to/your/coverage
# Environment variable (project-wide default)
export COV_LOUPE_OPTS="-r /path/to/your/coverage"
# MCP server configuration
# Add to your MCP client config (used as defaults for MCP tools):
# "args": ["-r", "/path/to/your/coverage"]MCP precedence: For MCP tool calls, per-request JSON parameters win over the CLI args used to start the server (including COV_LOUPE_OPTS). If neither is provided, built-in defaults are used (root: '.', raise_on_stale: false, etc.).
See CLI Usage Guide for complete details.
Common Workflows
Find Coverage Gaps
# Files with worst coverage
cov-loupe -o d list # -o = --sort-order, d = descending (worst at end)
cov-loupe list | less # display table in pager, worst files first
cov-loupe list | head -10 # truncate the table
# Specific directory
cov-loupe -g "lib/cov_loupe/tools/**/*.rb" list # -g = --tracked-globs
# Export for analysis
cov-loupe -fJ list > coverage-report.jsonWorking with JSON Output
The -fJ flag enables programmatic processing of coverage data using command-line JSON tools.
Using jq:
# Filter files below 80% coverage
cov-loupe -fJ list | jq '.files[] | select(.percentage < 80)'Using Ruby one-liners:
# Count files below threshold
cov-loupe -fJ list | ruby -r json -e '
puts JSON.parse($stdin.read)["files"].count { |f| f["percentage"] < 80 }
'Using rexe:
rexe is a Ruby gem that enables shorter Ruby command lines by providing command-line options for input and output formats, plus other conveniences. It eliminates the need for explicit JSON parsing and formatting code.
Install: gem install rexe
# Filter files below 80% coverage with pretty-printed JSON output
cov-loupe -fJ list | rexe -ij -mb -oJ 'self["files"].select { |f| f["percentage"] < 80 }'
# Count files below threshold
cov-loupe -fJ list | rexe -ij -mb -op 'self["files"].count { |f| f["percentage"] < 80 }'
# Human-readable output with AmazingPrint
cov-loupe -fJ list | rexe -ij -mb -oa 'self["files"].first(3)'With rexe's -ij -mb options, self automatically becomes the parsed JSON object. The same holds true for JSON output -- using -oJ produces pretty-printed JSON without explicit formatting calls. Rexe also supports YAML input/output (-iy, -oy) and AmazingPrint output (-oa) for human consumption.
When Coverage Rows Are Skipped
If a file in the resultset is missing on disk or has corrupt data, the CLI now logs and warns after rendering the report so operators immediately see that totals may be incomplete. Example table output:
$ cov-loupe list
┌─────────────────────────────┬─────────┬─────────┬────────────┬───────┐
│ File │ Covered │ Total │ % Covered │ Stale │
├─────────────────────────────┼─────────┼─────────┼────────────┼───────┤
│ lib/foo.rb │ 2 │ 3 │ 66.67% │ │
│ lib/bar.rb │ 1 │ 3 │ 33.33% │ │
└─────────────────────────────┴─────────┴─────────┴────────────┴───────┘
WARNING: 1 coverage row skipped due to errors:
- lib/deleted.rb: No coverage data found for file: lib/deleted.rb
Run again with --raise-on-stale to exit when rows are skipped.
Pretty JSON (-fJ) reports still emit valid JSON to stdout; the warning continues to be printed on stderr:
$ cov-loupe -fJ list
{
"files": [
{ "file": "lib/foo.rb", "covered": 2, "total": 3, "percentage": 66.67, "stale": null },
{ "file": "lib/bar.rb", "covered": 1, "total": 3, "percentage": 33.33, "stale": null }
],
"counts": { "total": 2, "ok": 2, "stale": 0 }
}
WARNING: 1 coverage row skipped due to errors:
- lib/deleted.rb: No coverage data found for file: lib/deleted.rb
Run again with --raise-on-stale to exit when rows are skipped.
Use --raise-on-stale true (or -S true) to turn these warnings into hard failures for CI pipelines.
Run rexe -h to see all available options, or visit the rexe project page for more examples.
For comprehensive JSON processing examples, see user/EXAMPLES.md.
CI/CD Integration
# Fail build if coverage is stale (--raise-on-stale or -S)
cov-loupe --raise-on-stale true list || exit 1
# Generate coverage report artifact
cov-loupe -fJ list > artifacts/coverage.jsonInvestigate Specific Files
# Quick summary
cov-loupe summary lib/cov_loupe/model.rb
# See uncovered lines
cov-loupe uncovered lib/cov_loupe/cli.rb
# View in context
cov-loupe -s u -c 3 uncovered lib/cov_loupe/cli.rb # -s = --source (u = uncovered), -c = --context-lines
# Detailed hit counts
cov-loupe detailed lib/cov_loupe/util.rb
# Project totals
cov-loupe totals
cov-loupe -fJ totalsBoolean CLI Options
Boolean flags such as --color (short: -C) and --raise-on-stale (short: -S) require explicit boolean arguments. Recognized literals:
yes |
no |
y |
n |
true |
false |
t |
f |
on |
off |
+ |
- |
1 |
0 |
Each row lists the equivalent true token (left) and false token (right).
cov-loupe --color false # disable ANSI colors explicitly
cov-loupe -C false # short form
cov-loupe --raise-on-stale yes # enforce stale coverage failuresCommands and Tools
CLI Subcommands: list, summary, uncovered, detailed, raw, totals, validate, version
MCP Tools: coverage_summary_tool, coverage_detailed_tool, coverage_raw_tool, uncovered_lines_tool, list_tool, coverage_totals_tool, coverage_table_tool, validate_tool, help_tool, version_tool
📖 See also:
- CLI Usage Guide - Complete command-line reference
- MCP Integration Guide - MCP tools documentation
Troubleshooting
- "command not found" - See Installation Guide
-
"cannot load such file -- mcp" - Requires Ruby >= 3.2. Verify:
ruby -v - "Could not find .resultset.json" - Ensure SimpleCov is configured in your test suite, then run tests to generate coverage. See the Configuring the Resultset section for more details.
- MCP server won't connect - Check PATH and Ruby version in MCP Troubleshooting
-
RVM in sandboxed environments (macOS) - RVM requires
/bin/pswhich may be blocked by sandbox restrictions. Use rbenv or chruby instead.
For more detailed help, see the full Troubleshooting Guide.
Development
# Clone and setup
git clone https://github.com/keithrbennett/cov-loupe.git
cd cov-loupe
bundle install
# Run tests
bundle exec rspec
# Test locally
bundle exec exe/cov-loupe
# Build and install
gem build cov-loupe.gemspec
gem install cov-loupe-*.gemSee dev/DEVELOPMENT.md for more.
SimpleCov Dependency
cov-loupe declares a runtime dependency on simplecov (>= 0.21) to support multi-suite merging using SimpleCov's combine helpers. The dependency is lazy-loaded only when needed, ensuring fast startup for single-suite projects.
Contributing
Contributions are welcome! Please:
- Fork the repository
- Create a feature branch
- Add tests for new functionality
- Ensure all tests pass (
bundle exec rspec) - Submit a pull request
License
MIT License - see LICENSE file for details.
Links
- GitHub: https://github.com/keithrbennett/cov-loupe
- RubyGems: https://rubygems.org/gems/cov-loupe
- Issues: https://github.com/keithrbennett/cov-loupe/issues
- Changelog: RELEASE_NOTES.md
Next Steps
📦 Install: gem install cov-loupe
📖 Read: CLI Usage Guide | MCP Integration
🐛 Report issues: GitHub Issues
⭐ Star the repo if you find it useful!
