Deterministic runtime compatibility analyzer
The static analyzer for runtime compatibility.
Predict why a repository will fail on your machine before you run it.
repofail answers one question: Will this repository actually run here?
It inspects both the repo and your machine - then reports deterministic incompatibilities before you install anything.
Why Β· Example Β· Works on Β· Install Β· Usage Β· Rules Β· CI Β· Contracts Β· FAQ
pip install repofail
cd /path/to/any/repo
repofailYou get a compatibility score and a list of deterministic blockers (Node version, Python range, CUDA, lock file, spec drift, etc.). No install of the repoβs dependencies. No cloud. No AI.
Most tools install dependencies.
Few tools tell you:
- Your Node version violates
engines.node. - Docker targets the wrong architecture.
- CUDA is hard-coded with no fallback.
- CI and local Python versions drifted.
repofail inspects both the repository and your machine - then reports deterministic incompatibilities before install or runtime.
- Python - requires-python, PyTorch/CUDA, ABI wheels, spec drift
- Node - engines.node, native modules, lock files, EOL
- Go - go.mod version, CGO dependencies, OS build tags
- Rust - rust-version, system lib crates, target-specific deps
- Docker - base image, platform mismatch, CUDA
- ML - CUDA hard-coding, GPU memory, Apple Silicon wheels
- Monorepos - multi-language, subproject detection
Run it against any local clone.
Deterministic spec violation detected - engines.node requires 22.x, host is 20.x.
| Scenario | Without repofail | With repofail |
|---|---|---|
| Node engine mismatch | Clone β npm install β EBADENGINE β search, fix, retry |
repofail . β "Node 22.x required, host is 20.x" + suggested fix in <1s |
| CUDA on laptop | Clone β pip install β run β RuntimeError: CUDA unavailable |
repofail . β "Hard-coded CUDA path, host has no GPU" before you run |
| Spec drift (Python) | CI passes, local fails; pyproject says 3.11, Docker uses 3.9 | repofail . β "Spec drift - 3 distinct Python targets" + where they differ |
Try the demos: node engine, spec drift, CUDA hardcoded.
From PyPI (recommended)
pip install repofailOne-liner (curl)
curl -sSL https://raw.githubusercontent.com/jayvenn21/repofail/main/install.sh | bashpipx (isolated CLI)
pipx install repofailHomebrew (formula: jayvenn21/homebrew-tap)
brew tap jayvenn21/tap
brew install jayvenn21/tap/repofailFrom source (development)
git clone https://github.com/jayvenn21/repofail.git
cd repofail
pip install -e .# Scan
repofail # Scan current dir
repofail -p /path/to/repo # Scan specific repo
repofail -j # JSON output (machine-readable)
repofail -m # Markdown output
repofail -v # Verbose: rule IDs and low-confidence hints
repofail --ci # CI mode: exit 1 if HIGH rules fire
repofail --fail-on MEDIUM # CI: fail on MEDIUM or higher (default: HIGH)
repofail -r # Save failure report when rules fire (opt-in telemetry)
# AI-powered explanations (requires REPOFAIL_API_KEY or Ollama)
repofail . --ai # Plain English explanation + fix suggestions
repofail . --ai --model ollama/llama3 # Use local model (no data leaves your machine)
repofail . --ai --model claude-sonnet-4-20250514 # Use Anthropic
# Init
repofail init # Interactive config generator
repofail init --yes # Non-interactive (defaults)
# Rules
repofail -e list # List all rules
repofail -e spec_drift # Explain a rule
# Contracts
repofail gen . # Generate env contract to stdout
repofail gen . -o contract.json
repofail check contract.json
# Fleet
repofail a /path # Audit: scan all repos in directory
repofail a /path -j # Audit with JSON output
repofail sim . -H host.json # Simulate: would this work on target host?
repofail s # Stats: local failure counts (from -r reports)
repofail s -j # Stats with JSON output- 0 - No deterministic violations (or scan completed successfully)
- 1 - Violations detected (with
--ci) or target host has issues (withsim) - 2 - Invalid usage / bad input (e.g. not a directory, contract violation)
Option A - Reusable action (comment on PR + fail CI)
name: repofail
on:
pull_request:
branches: [main, master]
jobs:
check:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: jayvenn21/repofail/.github/actions/repofail@main
with:
path: .
fail_on: HIGH
comment: 'true'
upload_artifact: 'true'
pr_number: ${{ github.event.pull_request.number }}The action installs repofail, runs a compatibility check, comments the Markdown report on the PR, uploads the JSON artifact, and fails the job if violations meet the threshold.
Option B - Inline (no comment)
- uses: actions/checkout@v4
- uses: actions/setup-python@v5
with:
python-version: "3.12"
- run: pip install repofail
- run: repofail --ciExits 1 if HIGH rules fire. Use --fail-on MEDIUM to be stricter.
Option C - Lock + verify (enforcement)
Pin the runtime once, then fail CI on drift:
repofail lock # generates repofail.lock.json (python, node, arch, os, cuda, docker_base)
repofail verify # exit 1 if host doesn't match lockCommit repofail.lock.json. In CI, run repofail verify so builds only pass on the locked environment.
Fleet compliance
Scan many repos and get violations, most common drift, and risk clusters:
repofail fleet ~/org --policy org.policy.yamlPolicy YAML (optional): fail_on: HIGH, max_repos: 500, max_depth: 4. With fail_on: HIGH, exit code is 1 if any repo has a HIGH finding.
Option D - GitHub App (zero config)
Install the repofail GitHub App on your repos and every PR gets an automatic compatibility comment - no workflow file needed.
## repofail Β· compatibility report
Compatibility score: π΄ ββββββββββ 32%
### Hard failures
β Hard-coded CUDA path, host has no GPU.
Likely error: RuntimeError: CUDA unavailable
### Runtime risks
β οΈ Spec drift - 3 distinct Python targets across configs.
Self-host with Docker or Railway. See github-app/README.md for setup.
repofail gen . -o contract.json
repofail check contract.jsonVersioned runtime expectations. Teams share contracts. CI checks drift. Generated contracts report the installed repofail version (from PyPI/Homebrew) so tooling stays traceable.
repofail is deterministic by default - fast AST rules, no hallucination. The --ai flag layers plain English explanations on top of the scan results using your own API key or a local model.
pip install repofail[ai] # installs litellm
# Set your key (OpenAI, Anthropic, or skip for Ollama)
export REPOFAIL_API_KEY=sk-...
# Scan with AI explanation
repofail . --aiWhat the AI does:
- Translates each finding into plain English a beginner can understand
- Explains what error you'll actually see if you try to run the repo
- Suggests specific fixes (commands, config changes, code edits)
- Gives an overall "will this work on my machine?" verdict
What the AI does NOT do:
- It never decides whether there's a problem - that's the deterministic scanner
- It never sees your source code - only the structured scan results (a few KB of JSON)
- Zero false positives from detection - the rules are the source of truth
Supported providers:
| Provider | Model example | Cost per scan | Setup |
|---|---|---|---|
| OpenAI | gpt-4o-mini (default) |
~$0.001 | REPOFAIL_API_KEY=sk-... |
| Anthropic | claude-sonnet-4-20250514 |
~$0.001 | REPOFAIL_API_KEY=sk-ant-... |
| Ollama | ollama/llama3 |
$0.00 | No key needed, runs locally |
# Use a specific model
repofail . --ai --model claude-sonnet-4-20250514
# Fully local - no data leaves your machine
repofail . --ai --model ollama/llama3repofail works fully offline by default. The
--aiflag adds AI-powered explanations using your own API key (OpenAI, Anthropic) or a local model via Ollama. No data is sent anywhere unless you opt in.
| Tool | Reads Repo | Inspects Host | Predicts Failure | CI Enforceable |
|---|---|---|---|---|
| pip | β | β | β | β |
| Docker | β | β | β | β |
| repofail | β | β | β | β |
Deterministic rule coverage - repofail includes checks across:
- Spec violations - Python requires-python, Node engines.node, Go go.mod, Rust rust-version
- Architecture mismatches - Apple Silicon vs amd64 Docker, Go/Rust OS build tags
- Hardware constraints - CUDA requirements, GPU memory
- Toolchain gaps - missing compilers, CGO, Rust system crates, node-gyp
- Runtime drift - CI vs Docker vs local inconsistencies
- Environment shape - multi-service RAM pressure, port collisions
See all rules: repofail -e list Β· Explain one: repofail -e <rule_id>
Rule reference
| Rule | Severity | When |
|---|---|---|
| Torch CUDA mismatch | HIGH | Hard-coded CUDA, host has no GPU |
| Python version violation | HIGH | Host outside requires-python range |
| Go version mismatch | HIGH | go.mod go directive > host Go |
| Rust version mismatch | HIGH | Cargo.toml rust-version > host rustc |
| Spec drift | HIGH | pyproject vs Docker vs CI - inconsistent Python |
| Node engine mismatch | HIGH | package.json engines.node vs host |
| Lock file missing | HIGH | package.json has deps, no lock file |
| Go CGO no compiler | MEDIUM | CGO deps but no gcc/clang |
| Go OS build tags | MEDIUM | Build tags exclude current host OS |
| Rust target platform | MEDIUM | Target-specific deps for a different OS |
| Apple Silicon wheel mismatch | MEDIUM/HIGH | arm64 + x86-only packages or Docker amd64 |
| β¦ | repofail -e list |
Scoring model
Compatibility Score = 100 β Ξ£(weight Γ confidence Γ determinism)
| Severity | Weight | Determinism |
|---|---|---|
| HIGH | 45 | 1.0 for spec violations |
| MEDIUM | 20 | 0.8β1.0 |
| LOW | 7 | 0.5β1.0 |
| INFO | 5 | structural only |
Determinism scale: 1.0 = guaranteed failure Β· 0.75 = high likelihood Β· 0.6 = probabilistic (spec drift) Β· 0.5 = structural risk
Score floors at 10%. When score β€15% with HIGH rules: "- fatal deterministic violations present".
repofail/
cli.py # Typer CLI (scan, init, lock, verify, fleet, gen, check, sim)
engine.py # Rule runner
init.py # Interactive config generator
scanner/ # Repo + host inspection (Python, Node, Go, Rust, Docker)
rules/ # Deterministic rule implementations
lock.py # Runtime lock / verify
fleet.py # Audit, simulate, fleet scan
Extensible via .repofail/rules.yaml or .repofail.yaml (generated by repofail init).
Does repofail install or run my project?
No. It only reads configs and (optionally) inspects Python/JS for patterns. No pip install, no npm install, no execution.
Does it need the internet?
No. It runs fully offline. Host inspection uses local subprocesses (e.g. node --version).
Why βdeterministicβ?
Same repo + same host β same result. No ML, no heuristics that change between runs. Rules are based on config and code.
Can I add my own rules?
Yes. Put a .repofail/rules.yaml (or repofail-rules.yaml) in the repo and define conditions on repo.* and host.*. See repofail -e list for built-in rule IDs.
What if my repo is clean?
You get a high score (e.g. 96β100%) and βNo deterministic blockers detected.β repofail does not invent problems.
| Tool | What it answers | Overlap |
|---|---|---|
| CodeRabbit / Greptile | "Is this PR good code?" (LLM review) | None - code quality, not runtime |
| CodeQL / Snyk | "Does this code have vulnerabilities?" | None - security, not compatibility |
| pip / npm / Docker | "Install these deps" | Finds problems after you hit the error |
| go-runtime-compat | "Will this Go code fail in a container?" | Go only |
| repofail | "Will this repo fail on this machine before you run it?" | Cross-language, pre-execution, deterministic |
repofail is the only tool that combines repo scanning + host inspection + failure prediction across Python, Node, Go, Rust, Docker, and CUDA - before you run anything.
- Python / Node / Docker / CUDA scanning
- Go scanner (go.mod, CGO, build tags)
- Rust scanner (rust-version, target platforms, system crates)
-
repofail init- interactive config generator - Runtime lock enforcement (
repofail lock/verify) - Fleet compliance mode (
repofail fleet) - AI-powered explanations (
--aiflag, BYOK, Ollama support) - GitHub App - auto-comment on PRs with environment-specific warnings
- Java scanner (JNI, JVM version, native bindings)
- Web dashboard - paste a GitHub URL, get a report
- Community rule marketplace (
repofail-community-rules)
- You need dependency resolution - use pip, npm, poetry, etc. repofail does not install or resolve.
- You need security scanning - use Dependabot, Snyk, or similar. repofail is compatibility-only.
- You want βAI suggested fixesβ - repofail gives deterministic, rule-based suggestions only.
- You run only in one environment - if every dev and CI use the same OS/runtime, the value is smaller (still useful for drift and contracts).
pip install -e ".[dev]"
pytest tests/ -vQuick checks: bash -n install.sh (syntax). The GitHub Action runs on every PR in this repo (see .github/workflows/repofail.yml); to test the reusable action, use it in another repoβs workflow on a PR.
MIT - see LICENSE.

