FALLOW(1)

NAME

fallowCodebase intelligence for TypeScript and JavaScript. Free static layer: unused code, duplication, circular deps,…

SYNOPSIS

$npm install -g fallow

INFO

483 stars
13 forks
0 views

DESCRIPTION

Codebase intelligence for TypeScript and JavaScript. Free static layer: unused code, duplication, circular deps, complexity hotspots, architecture boundaries. Optional paid runtime layer: hot-path review and cold-path deletion evidence from real production traffic. Rust-native, sub-second, 90 framework plugins.

README

fallow

Codebase intelligence for TypeScript & JavaScript.
Free static analysis for unused code, duplication, complexity, and architecture drift.
Optional runtime intelligence for hot paths, cold paths, and runtime-backed code decisions.
Built for AI-assisted development. No AI inside.
Rust-native. Zero config. Sub-second.

CI Coverage crates.io npm MIT License Documentation


npx fallow
 Dead code   3 unused files, 12 unused exports, 2 unused deps       18ms
 Duplication 4 clone groups (2.1% of codebase)                      31ms
 Complexity  7 functions exceed thresholds                           4ms
 Total       26 issues across 847 files                             53ms

Static analysis is free and open source. Runtime intelligence is optional.

90 framework plugins. No Node.js runtime required for static analysis. No config needed for the first run.

Fallow builds a project-wide understanding of your TS/JS codebase instead of checking one file at a time. Use it to review AI-generated changes faster, clean up dead code, reduce duplication, find risky complexity, and enforce architecture boundaries. Add the runtime layer when you want to know what actually executed in production.

Fallow is the codebase truth layer your coding agent can call. It is not an AI assistant.

Install

npx fallow                  # Run without installing
npm install -g fallow       # Or install globally (macOS, Linux, Windows)
cargo install fallow-cli    # Or via Cargo

Start here

fallow                      # Dead code + duplication + health
fallow dead-code            # Cleanup candidates
fallow dupes                # Repeated logic
fallow health               # Complexity + refactor targets
fallow fix --dry-run        # Preview automatic cleanup

What it finds

  • Dead code: unused files, exports, dependencies, types, cycles, boundaries, stale suppressions
  • Duplication: repeated blocks from exact to semantic clones
  • Complexity: high-risk functions, file scores, hotspots, and refactor targets
  • Architecture drift: boundary violations across layers and modules

Why Fallow exists

Linters check files. TypeScript checks types. Fallow checks the codebase.

It builds a module graph across the whole project so it can find problems that file-local tools cannot:

WhatLinterFallow
Unused variable in a functionyesno
Unused export that nothing importsnoyes
File that nothing importsnoyes
Circular dependency across modulesnoyes
Duplicate code blocks across filesnoyes
Dependency in package.json never importednoyes

Full comparison: fallow vs ESLint, Biome, knip, ts-prune

Why teams using AI need Fallow

AI accelerates code creation. It does not eliminate review, cleanup, or architecture drift.

When Claude Code, Codex, Cursor, or other tools generate changes, teams still need to know:

  • did this introduce dead code?
  • did it duplicate logic that already existed?
  • did complexity get worse?
  • did the change cross a boundary it should not cross?
  • is this code on a hot path or a cold one?
  • what should the reviewer read closely first?

Fallow answers those questions with deterministic, graph-based analysis and structured output, so both humans and agents can act on facts instead of guesses.

How agents use Fallow

Agents do not need to guess from limited context. They can call Fallow directly via the CLI or MCP.

Common agent workflow:

  1. generate or edit code
  2. run fallow --format json
  3. inspect dead code, duplication, health findings, and per-issue actions
  4. apply safe fixes or adjust the patch before opening a PR
  5. hand the result to a human reviewer with better evidence
npx fallow --format json
npx fallow audit --format json
npx fallow fix --dry-run --format json

See Agent integration for MCP setup and the full list of structured tools.

More static commands

fallow audit                # Audit changed files (verdict: pass/warn/fail)
fallow watch                # Re-analyze on file changes
fallow fix                  # Apply automatic cleanup after previewing

Dead code

Finds unused files, exports, dependencies, types, enum members, class members, unresolved imports, unlisted dependencies, duplicate exports, circular dependencies (including cross-package cycles in monorepos), boundary violations, type-only dependencies, test-only production dependencies, and stale suppression comments. Entry points are auto-detected from package.json fields, framework conventions, and plugin patterns. Arrow-wrapped dynamic imports (React.lazy, loadable, defineAsyncComponent) are tracked as references. Script multiplexers (concurrently, npm-run-all) are analyzed to discover transitive script dependencies. JSDoc tags (@public, @internal, @beta, @alpha, @expected-unused) control export visibility.

fallow dead-code                          # All dead code issues
fallow dead-code --unused-exports         # Only unused exports
fallow dead-code --circular-deps          # Only circular dependencies
fallow dead-code --boundary-violations    # Only boundary violations
fallow dead-code --stale-suppressions     # Only stale suppression comments
fallow dead-code --production             # Exclude test/dev files
fallow dead-code --changed-since main     # Only changed files (for PRs)
fallow dead-code --file src/utils.ts       # Single file (lint-staged integration)
fallow dead-code --include-entry-exports  # Also check exports from entry files
fallow dead-code --group-by owner         # Group by CODEOWNERS for team triage
fallow dead-code --group-by directory     # Group by first directory component
fallow dead-code --group-by package       # Group by workspace package (monorepo)
fallow dead-code --group-by section       # Group by GitLab CODEOWNERS section

Duplication

Finds copy-pasted code blocks across your codebase. Suffix-array algorithm -- no quadratic pairwise comparison.

fallow dupes                              # Default (mild mode)
fallow dupes --mode semantic              # Catch clones with renamed variables
fallow dupes --skip-local                 # Only cross-directory duplicates
fallow dupes --trace src/utils.ts:42      # Show all clones of code at this location

Four detection modes: strict (exact tokens), mild (default, AST-based), weak (different string literals), semantic (renamed variables and literals).

Complexity

Surfaces the most complex functions in your codebase and identifies where to spend refactoring effort.

fallow health                             # Functions exceeding thresholds
fallow health --score                     # Project health score (0-100) with letter grade
fallow health --min-score 70              # CI gate: fail if score drops below 70
fallow health --top 20                    # 20 most complex functions
fallow health --file-scores               # Per-file maintainability index (0-100)
fallow health --hotspots                  # Riskiest files (git churn x complexity)
fallow health --hotspots --ownership      # Add bus factor, owner, drift signals
fallow health --targets                   # Ranked refactoring recommendations
fallow health --targets --effort low      # Only quick-win refactoring targets
fallow health --coverage-gaps             # Static test coverage gaps
fallow health --coverage coverage/coverage-final.json
fallow health --coverage artifacts/coverage.json --coverage-root /home/runner/work/myapp
fallow health --production-coverage ./coverage
fallow health --production-coverage ./coverage --min-invocations-hot 250
fallow health --trend                     # Compare against saved snapshot
fallow health --changed-since main        # Only changed files

Runtime intelligence (optional)

Static analysis answers: what is connected to what?

Runtime intelligence answers: what actually ran?

Fallow Runtime is the optional paid team layer. It uses production coverage as the collection engine (V8 dumps via NODE_V8_COVERAGE=... and Istanbul coverage-final.json files), then merges that evidence into fallow health so teams and coding agents can:

  • review changes on hot production paths more carefully
  • delete cold code with stronger evidence
  • prioritize refactors by runtime importance
  • spot stale feature-flag branches and stale runtime code
  • give agents factual usage data instead of assumptions
fallow license activate --trial --email you@company.com
fallow coverage setup
fallow health --production-coverage ./coverage

Static coverage_gaps and runtime production_coverage are separate layers in the same health surface:

SurfaceFlagInputAnswersLicense
Static test reachability--coverage-gapsnonewhich runtime files/exports have no test dependency pathno
Exact CRAP scoring--coverageIstanbul JSON file or coverage-final.json directoryhow covered each function is for CRAP computationno
Runtime production coverage--production-coverageV8 directory, V8 JSON file, or Istanbul JSON filewhich functions actually executed, which stayed cold, which are hotyes

Setup details:

  • fallow license activate --trial --email ... starts a trial and stores the signed license locally
  • fallow license refresh refreshes the stored license before the hard-fail window
  • fallow coverage setup detects your framework and package manager, installs the sidecar if needed, writes a collection recipe, and resumes from the current setup state on re-run
  • The sidecar can be installed globally or as a project devDependency; fallow resolves FALLOW_COV_BIN, project-local shims, package-manager bin lookups, ~/.fallow/bin/fallow-cov, and PATH
  • fallow health --production-coverage <path> accepts a V8 directory, a single V8 JSON file, or a single Istanbul coverage map JSON file (commonly coverage-final.json)
  • fallow health --coverage <path> accepts a single Istanbul coverage map JSON file or a directory containing coverage-final.json
  • --coverage-root <path> rebases Istanbul file paths before CRAP matching. Use it when coverage was generated in CI or Docker with a different checkout root, for example fallow health --coverage artifacts/coverage-final.json --coverage-root /home/runner/work/myapp
  • V8 dumps that include Node's source-map-cache are remapped through supported source-map paths before analysis, including file paths, relative paths, webpack://..., and vite://...; unsupported virtual schemes safely fall back to raw V8 handling
  • fallow health --changed-since <ref> --production-coverage <path> promotes touched hot paths to a hot-path-changes-needed verdict during change review

Production coverage is merged into the same human, JSON, SARIF, compact, markdown, and CodeClimate outputs as the rest of the health report.

Read more: Static vs runtime intelligence | Production coverage

Audit

Quality gate for AI-generated code and PRs. Combines dead code + complexity + duplication scoped to changed files.

fallow audit                              # Auto-detects base branch
fallow audit --base main                  # Explicit base ref
fallow audit --base HEAD~3               # Audit last 3 commits
fallow audit --format json                # Structured output with verdict

Returns a verdict: pass (exit 0), warn (exit 0, warn-severity only), or fail (exit 1). JSON output includes a verdict field for CI and agent integration.

CI integration

# GitHub Action
- uses: fallow-rs/fallow@v2

GitLab CI -- include the template and extend

include:

Or run directly on any CI

  • run: npx fallow --ci

--ci enables SARIF output, quiet mode, and non-zero exit on issues. Also supports:

  • --group-by owner|directory|package|section -- group output by CODEOWNERS ownership, directory, workspace package, or GitLab CODEOWNERS [Section] headers for team-level triage
  • --summary -- show only category counts (no individual issues)
  • --changed-since main -- analyze only files touched in a PR
  • --changed-workspaces origin/main -- scope monorepo analysis to workspaces containing any changed file (CI primitive; fails hard on git errors so CI never silently widens back to the full repo)
  • --baseline / --save-baseline -- fail only on new issues
  • --fail-on-regression / --tolerance 2% -- fail only if issues grew beyond tolerance
  • --format sarif -- upload to GitHub Code Scanning
  • --format codeclimate -- GitLab Code Quality inline MR annotations
  • --format annotations -- GitHub Actions inline PR annotations (no Action required)
  • --format json / --format markdown -- for custom workflows (JSON includes machine-actionable actions per issue)
  • --format badge -- shields.io-compatible SVG health badge (fallow health --format badge > badge.svg)

Both the GitHub Action and GitLab CI template auto-detect your package manager (npm/pnpm/yarn) from lock files, so install/uninstall commands in review comments match your project.

Adopt incrementally -- surface issues without blocking CI, then promote when ready:

{ "rules": { "unused-files": "error", "unused-exports": "warn", "circular-dependencies": "off" } }

GitLab CI rich MR comments

The GitLab CI template can post rich comments directly on merge requests -- summary comments with collapsible sections and inline review discussions with suggestion blocks.

VariableDefaultDescription
FALLOW_COMMENT"false"Post a summary comment on the MR with collapsible sections per analysis
FALLOW_REVIEW"false"Post inline MR discussions at the relevant lines, with suggestion blocks for unused exports
FALLOW_MAX_COMMENTS"50"Maximum number of inline review comments

In MR pipelines, --changed-since is set automatically to scope analysis to changed files. Previous fallow comments are cleaned up on re-runs.

The comment merging pipeline groups unused exports per file and deduplicates clone reports, keeping MR threads readable.

A GITLAB_TOKEN (PAT with api scope) is recommended for full features (suggestion blocks, cleanup of previous comments). CI_JOB_TOKEN works for posting but cannot delete comments from prior runs.

# .gitlab-ci.yml -- full example with rich MR comments
include:
  - remote: 'https://raw.githubusercontent.com/fallow-rs/fallow/main/ci/gitlab-ci.yml'

fallow: extends: .fallow variables: FALLOW_COMMENT: "true" # Summary comment with collapsible sections FALLOW_REVIEW: "true" # Inline discussions with suggestion blocks FALLOW_MAX_COMMENTS: "30" # Cap inline comments (default: 50) FALLOW_FAIL_ON_ISSUES: "true"

Configuration

Works out of the box. When you need to customize, create .fallowrc.json or run fallow init:

// .fallowrc.json
{
  "$schema": "https://raw.githubusercontent.com/fallow-rs/fallow/main/schema.json",
  "entry": ["src/workers/*.ts", "scripts/*.ts"],
  "ignorePatterns": ["**/*.generated.ts"],
  "ignoreDependencies": ["autoprefixer"],
  "rules": {
    "unused-files": "error",
    "unused-exports": "warn",
    "unused-types": "off"
  },
  "health": {
    "maxCyclomatic": 20,
    "maxCognitive": 15
  }
}

Architecture boundary presets enforce import rules between layers with zero manual config:

{ "boundaries": { "preset": "bulletproof" } } // or: layered, hexagonal, feature-sliced

Run fallow list --boundaries to inspect the expanded rules. TOML also supported (fallow init --toml). The init command auto-detects your project structure (monorepo layout, frameworks, existing config) and generates a tailored config. It also adds .fallow/ to your .gitignore (cache and local data). Scaffold a pre-commit hook with fallow init --hooks. Migrating from knip or jscpd? Run fallow migrate.

See the full configuration reference for all options.

Framework plugins

90 built-in plugins detect entry points, convention exports, config-defined aliases, and template-visible usage for your framework automatically.

CategoryPlugins
FrameworksNext.js, Nuxt, Remix, Qwik, SvelteKit, Gatsby, Astro, Angular, NestJS, Expo, Expo Router, Electron, and more
BundlersVite, Webpack, Rspack, Rsbuild, Rollup, Rolldown, Tsup, Tsdown, Parcel
TestingVitest, Jest, Playwright, Cypress, Storybook, Mocha, Ava
CSSTailwind, PostCSS, UnoCSS
Databases & BackendPrisma, Drizzle, Knex, TypeORM, Kysely, Convex
BlockchainHardhat
MonoreposTurborepo, Nx, Changesets, Syncpack, pnpm

Full plugin list -- missing one? Add a custom plugin or open an issue.

Editor & AI support

Fallow is not an AI assistant. It is the codebase truth layer your assistant can call.

  • VS Code extension -- tree views, status bar, one-click fixes, auto-download LSP binary (Marketplace)
  • LSP server -- real-time diagnostics, hover info, code actions, Code Lens with reference counts
  • MCP server -- AI agent integration for Claude Code, Cursor, Windsurf (fallow-skills)
  • JSON actions array -- every issue in --format json output includes fix suggestions with auto_fixable flag, so agents can self-correct

Performance

Benchmarked on real open-source projects (median of 5 runs, Apple M5).

Dead code: fallow vs knip

ProjectFilesfallowknip v5knip v6vs v5vs v6
zod17417ms577ms300ms34x18x
fastify28619ms791ms232ms41x12x
preact24420ms767ms2.02s39x103x
TanStack/query901170ms2.50s1.28s15x8x
svelte3,337359ms1.73s749ms5x2x
next.js20,4161.66s--------

knip errors out on next.js. fallow completes in under 2 seconds.

Duplication: fallow vs jscpd

ProjectFilesfallowjscpdSpeedup
fastify28676ms1.96s26x
vue/core522124ms3.11s25x
next.js20,4162.89s24.37s8x

No TypeScript compiler, no Node.js runtime needed to analyze your code. Fallow vs linters | Reproduce benchmarks

Suppressing findings

// fallow-ignore-next-line unused-export
export const keepThis = 1;

// fallow-ignore-file // Suppress all issues in this file

Also supports JSDoc visibility tags (/** @public */, /** @internal */, /** @beta */, /** @alpha */) to suppress unused export reports for library APIs consumed externally.

Limitations

fallow uses syntactic analysis -- no type information. This is what makes it fast, but type-level dead code is out of scope. Use inline suppression comments or ignoreExports for edge cases.

Documentation

Contributing

Missing a framework plugin? Found a false positive? Open an issue.

cargo build --workspace && cargo test --workspace

License

MIT

SEE ALSO

clihub4/21/2026FALLOW(1)