qleaner
v1.5.1
Published
A CLI tool for analyzing and identifying unused files, images, imports, and dead code in React, Vue, Nuxt, TypeScript, and JavaScript projects to help optimize bundle size and maintain clean codebases
Maintainers
Keywords
Readme
Qleaner
Qleaner (v1.5.1) is a powerful CLI tool for analyzing and cleaning up React, Vue, Nuxt, TypeScript, and JavaScript projects. It helps you identify unused files, images, dependencies, and dead code to optimize your bundle size and maintain a clean codebase.
Features
- Find Unused Code Files - Identify files that aren't imported or used anywhere (including re-export–aware checks so barrel files are not false positives when their exports are used)
- Find Unused Images - Detect image assets that are never referenced in your code
- Unused Dependencies - Discover npm/yarn/pnpm packages that are installed but not used; optionally uninstall them in one step
- Dead Image Links - Find image references in code that point to non-existent files
- Unused Exports - Find exported symbols nothing imports; optional
--fixto remove them - Vue & Nuxt - Scan
.vuesingle-file components (@vue/compiler-sfc); choose Vue duringqleaner initforpages/layoutsexclusions and.vueglobs (typical for Nuxt apps) - Prune Unused Code - Remove unused variables, functions, and classes (with dry-run);
-r/--reportprints a file, line, name, and kind table - Prune Console Logs - Strip
consoledebug calls (log,dir,dirxml,table,debug,info,trace). With-d/--dry-run, reports high / medium / low counts; per-tier hit tables (file, line, preview) print when the dry-run payload includesfoundByRiskand you pass--list-risk-categories(seecontrollers/list.js→runSurgicalPass).-i/--list-risk-categories-infoprints the risk category index (paths toutils/consoleLogHighRiskPatterns.jsandutils/consoleLogMediumRiskPatterns.js) and returns without scanning.--list-risk-categoriesalone does not skip the scan—use-dfor a dry run or-ifor docs only. - Duplicate Detection - Find and clean duplicate functions and classes;
-r/--reportprints a detail table - Tidy - Run the main cleanup pipeline (logs, unused code, duplicates, exports) in one command; optional
[pathToScan](default.) and--auto-fix/-u.-r/--reportprints per-step dry-run tables for all four steps (console log tiers, unused code, duplicates, exports).--list-risk-categorieslimits extra console-log tier tables to that step only;-iprints the console-log risk index (pipeline continues). - Undo - Restore files moved during cleanup from
.trash(code or images) - Project Summary - Get comprehensive statistics about your codebase
- File Size Analysis - Identify the largest files and potential optimization targets
- Dependency Analysis - See which files have heavy dependencies and hotspots
- Smart Caching - Fast incremental scans with intelligent cache invalidation (
unused-check-cache.json, includingparentGraph.fixesfingerprints for surgical passes when present) - Dry-Run Mode - Where supported,
-d/--dry-runmatches the CLI: show what would be deleted without actually deleting (skips prompt) - Interactive Deletion - Choose to delete files permanently or move them to
.trash, or usescan --auto-fixto move all reported unused files to.trashwithout a prompt - Enhanced File Finding - Advanced path resolution using jsconfig/tsconfig for accurate dependency tracking in large codebases with complex import structures
- Advanced Image Detection - Sophisticated AST parsing extracts image references from imports, requires, JSX, CSS, styled-components, template literals, arrays, and more
Installation
Global Installation
npm install -g qleanerUse with npx (No Installation)
npx qleaner <command>Local Development Dependency
yarn add qleaner --dev
# or
npm install qleaner --save-devGitHub Actions
Qleaner ships a composite GitHub Action at the repo root: action.yml (“Qleaner Health Guardian”). It installs the published qleaner package from npm and runs the global qleaner CLI — not yarn start from a cloned dev checkout.
Publish to the GitHub Marketplace by tagging a release (e.g. v1.5.1) that includes action.yml. Pin the same version on npm (qleaner-version input).
What the action does
- Sets up Node.js (default 20).
- Runs
npm install -g qleaner@<version>(default 1.5.1). - Restores
unused-check-cache.jsonfrom the Actions cache (key usesqleaner.config.jsonandpackage-lock.json). - Runs read-only captures in the job workspace:
qleaner tidy <scan-path> …→tidy_report.txtqleaner image <assets> <code> …→image_report.txt(skipped if image paths are empty)qleaner summary→health_summary.txt
No --auto-fix / -u: CI is dry-run only. The job succeeds when every command exits 0.
Use in your repository
Prerequisites
- Commit
qleaner.config.jsonat the repo root (runqleaner initlocally once; do not runinitin CI — it is interactive). - Add
unused-check-cache.jsonand.trash/to.gitignore(initcan do this). - Set
with:paths to match your app layout (not this repo’s Infisical sample tree). - Push a git tag on this repo that matches the Action ref (e.g. tag
v1.5.1→uses: trevismurithi/[email protected]). npm publish alone does not create the tag.
Full workflow (status check + PR health report comment):
Create .github/workflows/qleaner-health.yml in your project and adjust paths. The Action runs Qleaner; the following steps build pr_report.md from the npm-bundled script and post (or update) one sticky PR comment.
name: Qleaner Health Guardian
on:
pull_request:
types: [opened, synchronize, reopened, ready_for_review]
workflow_dispatch:
jobs:
hygiene-check:
runs-on: ubuntu-latest
permissions:
contents: read
pull-requests: write
steps:
- name: Checkout
uses: actions/checkout@v4
- name: Run Qleaner
uses: trevismurithi/[email protected]
with:
qleaner-version: "1.5.1"
scan-path: src
image-assets-path: public
image-code-path: src
tidy-extra-args: "-r"
image-extra-args: "-d -T 0.1"
- name: Build PR health report
if: github.event_name == 'pull_request'
env:
NO_COLOR: "1"
run: |
REPORT_SCRIPT="$(npm root -g)/qleaner/.github/scripts/pr-report.js"
node "$REPORT_SCRIPT" \
qleaner.stats.json health_summary.txt tidy_report.txt image_report.txt \
> pr_report.md
- name: Post PR Health Report
if: always() && github.event_name == 'pull_request'
uses: actions/github-script@v7
with:
script: |
const fs = require('fs');
if (!fs.existsSync('pr_report.md')) return;
const body = fs.readFileSync('pr_report.md', 'utf8');
const marker = '<!-- qleaner-report -->';
const { data: comments } = await github.rest.issues.listComments({
issue_number: context.issue.number,
owner: context.repo.owner,
repo: context.repo.repo,
per_page: 100,
});
const existing = comments.find(c => c.body?.startsWith(marker));
if (existing) {
await github.rest.issues.updateComment({
comment_id: existing.id,
owner: context.repo.owner,
repo: context.repo.repo,
body,
});
} else {
await github.rest.issues.createComment({
issue_number: context.issue.number,
owner: context.repo.owner,
repo: context.repo.repo,
body,
});
}uses: …@v1.5.1must match an existing tag ontrevismurithi/react-cleaner(e.g.v1.5.1or1.5.1— the ref is exact).- No image scan: set
image-assets-pathandimage-code-pathto""(or omit and rely on defaults). - Report only on PRs:
workflow_dispatchruns Qleaner but skips the comment steps (if: github.event_name == 'pull_request'). - You do not copy
.github/scripts/pr-report.jsinto your repo; it ships in the publishedqleanernpm package. The Action step installs that package globally first.
Check-only workflow (no PR comment — omit the last two steps and pull-requests: write).
Action inputs (all optional except you should set paths for your project):
| Input | Default | Purpose |
| --- | --- | --- |
| qleaner-version | 1.5.1 | npm version of qleaner to install |
| node-version | 20 | Node.js for setup-node |
| scan-path | . | First argument to qleaner tidy |
| image-assets-path | (empty) | First argument to qleaner image |
| image-code-path | (empty) | Second argument to qleaner image |
| tidy-extra-args | -r | Extra tidy flags (default -r = per-step dry-run detail tables) |
| image-extra-args | -d -T 0.1 | Extra image flags (dry-run + similarity threshold) |
| skip-image | false | Set true to skip image scan |
| skip-summary | false | Set true to skip summary |
| restore-cache | true | Restore unused-check-cache.json before the run |
Outputs: tidy-report-file, image-report-file, summary-file (filenames in the workspace).
The composite Action does not post to GitHub by itself — use the full workflow above for the PR comment.
This repository (dogfooding)
.github/workflows/qleaner-health.yml uses uses: ./ to test the local action.yml while still installing [email protected] from npm. It uses node .github/scripts/pr-report.js from the checked-out repo instead of $(npm root -g)/qleaner/... (same script, easier for dogfooding). Paths under with: target the bundled Infisical sample tree (src/infisical-main/frontend, etc.) — copy the full workflow above for other repos, not those paths.
Branch protection: After the workflow runs on a PR, you can require status check hygiene-check (job id). If you see “Expected — Waiting for status to be reported”, the workflow did not run (workflow missing on the base branch, Actions disabled, or fork PR awaiting approval).
Quick Start
1. Initialize Configuration
IMPORTANT: Proper configuration is critical for Qleaner to accurately find and resolve files in your project, especially for large codebases. Take time to get this right!
First, set up Qleaner for your project:
qleaner initThis interactive command will ask you:
- Path to scan - Default directory for unused-file scans and for commands that use
qleaner.config.json(e.g.exports,prune,tidy) - Package manager (npm/yarn/pnpm)
- Image path resolution - Relative or alias paths vs absolute paths from
public/(e.g./images/logo.png) - Framework (React, Next.js, or Vue) - Sets entry-point patterns and, for Vue, default
excludeDirPrint(layouts,pages) - Configuration file for path aliases - This is crucial! Select which config file Qleaner should use to resolve import aliases:
tsconfig.json(most common for TypeScript projects)jsconfig.json(for JavaScript projects)tsconfig.app.json(for monorepos or specific app configs)tsconfig.base.json(for monorepo base configs)none(if you don't use path aliases or want to configure manually)
Why this matters: Qleaner uses your project's jsconfig.json or tsconfig.json file to understand how import paths are resolved. This enables accurate file finding even in large codebases with complex path aliases (e.g., @/components, ~/utils, etc.). Selecting the correct config file ensures Qleaner can properly trace dependencies and find all file references.
If you don't use jsconfig/tsconfig: Select "none" during initialization, then manually add your path aliases to the paths field in qleaner.config.json (see Configuration section below).
This creates a qleaner.config.json file in your project root with sensible defaults. It also updates .gitignore: a # Qleaner section is appended with unused-check-cache.json and .trash/ when those patterns are not already ignored (so the dependency cache and soft-delete folder stay out of version control). If .gitignore did not exist, it is created with that section.
2. Scan for Unused Code Files
Scan for unused files. Pass a directory as the first argument ([pathToScan], default .), or rely on the path field in qleaner.config.json when you omit it.
# Dry run (recommended first time) — explicit directory
qleaner scan src --dry-run
# Current directory (same as qleaner scan . --dry-run)
qleaner scan --dry-run
# Actual scan (will prompt for deletion unless --auto-fix)
qleaner scan src3. Scan for Unused Images
Find images that aren't referenced in your code:
# Scan images in public/images against code in src
qleaner image public/images src --dry-run
# With alias paths (e.g., @/assets/images/logo.png)
qleaner image public/images src -a --dry-run
# With root-referenced paths (e.g., /images/logo.png)
qleaner image public/images src -r --dry-runCommands
The CLI is implemented in bin/cli.js. Program name qleaner, tagline A tool to clean up your React code, and qleaner --version match that file (version is read from package.json). Run qleaner --help for the same command and option list Commander registers there.
qleaner init
Initialize Qleaner with an interactive configuration wizard. This command asks you a series of questions to set up Qleaner for your project and creates a qleaner.config.json file in your project root with sensible defaults.
After writing the config, init updates .gitignore: it adds unused-check-cache.json and .trash/ under a # Qleaner heading only when those entries are not already present (missing lines are appended; existing rules are left unchanged).
Configuration is Critical: Getting the configuration right, especially the jsconfig/tsconfig selection, is essential for Qleaner to accurately find files in large codebases. Incorrect configuration may result in false positives (files reported as unused when they're actually used) or missed dependencies.
Questions asked:
- Path to scan - Directory Qleaner uses by default for scans and code-analysis commands
- Package manager (npm, yarn, or pnpm)
- Image path resolution - Relative/alias vs absolute-from-
public/paths - Framework (React, Next.js, or Vue) - Entry files and Vue-specific directory exclusions
- Configuration file for path aliases - Most important question! Select which config file Qleaner should use:
tsconfig.json- Standard TypeScript configurationjsconfig.json- Standard JavaScript configurationtsconfig.app.json- App-specific config (common in monorepos)tsconfig.base.json- Base config (common in monorepos)none- No config file (use manual path configuration)
Vue & Nuxt: Choosing Vue during init sets excludeDirPrint to layouts and pages by default (route files under those directories are not reported as unused). Globs include **/*.{ts,js,jsx,vue}. Nuxt 3 apps typically use the same layout—select Vue, point path at your source tree, and run qleaner scan before tidy or exports.
Why the config file matters:
- Qleaner reads path aliases from your
jsconfig.jsonortsconfig.jsonto resolve imports like@/components/Buttonor~/utils/helpers - This enables accurate dependency tracking even in large projects with complex import structures
- Without the correct config file, Qleaner may not properly resolve aliased imports, leading to incorrect results
If you don't have a jsconfig/tsconfig file:
- Select "none" during initialization
- Manually configure path aliases in the
pathsfield ofqleaner.config.json(see Configuration section) - Example:
"paths": { "@/*": ["./src/*"], "~/*": ["./src/*"] }
Examples:
# Initialize with interactive prompts
qleaner init
# After initialization, verify your config
cat qleaner.config.json
# Then run your first scan
qleaner scan src --dry-runNote: If you already have a qleaner.config.json file, this command will prompt you before overwriting it. You can manually edit the config file instead if needed.
qleaner scan
Scan a directory for unused code files. Merge order: values from qleaner.config.json first, then CLI flags (CLI wins).
Arguments:
[pathToScan]- Directory to scan (default.; combined with configpathand exclusions as implemented incommand.js)
Options:
-e, --exclude-dir <dir...>- Exclude directories from scan (e.g.,-e node_modules dist)-f, --exclude-file <file...>- Exclude specific files by name (e.g.,-f config.js)-F, --exclude-file-print <files...>- Scan but don't report as unused (for entry points)-x, --exclude-extensions <extensions...>- Exclude file extensions (e.g.,-x test.tsx test.ts)-t, --table- Display results in a formatted table-d, --dry-run- Show what would be deleted without actually deleting (skips prompt)-u, --auto-fix- Skip the interactive prompt and move all reported unused files to.trash(only when not using--dry-run; same flag name asdepis not shared: onscanthis is auto-fix, ondep-uis--uninstall)-C, --clear-cache- Clear cache before scanning (useful after major changes)
Examples:
# Basic scan (will prompt for deletion if unused files found)
qleaner scan src
# Dry run with table output (recommended first time)
qleaner scan src --dry-run --table
# Move every reported unused file to .trash without prompting (use after a trusted dry-run)
qleaner scan src --auto-fix
# Exclude test files and multiple directories
qleaner scan src -x test.tsx test.ts test.js test.jsx -e node_modules dist build
# Exclude specific files from being reported as unused (entry points)
qleaner scan src -F index.tsx main.tsx app.tsx
# Clear cache and perform fresh scan
qleaner scan src --clear-cache
# Comprehensive scan with multiple exclusions
qleaner scan src --exclude-dir node_modules dist build --exclude-extensions test.tsx test.ts --tableqleaner image <directory> <rootPath>
Scan for unused images by comparing image files against references in your code. Detects images (png, jpg, jpeg, svg, gif, webp) that exist in your assets directory but are never referenced in your source code.
Arguments:
<directory>- Directory containing image files to scan (e.g.,public/images,src/assets)<rootPath>- Root directory of your source code that references images (e.g.,src,app)
Options:
-u, --auto-prune- Move all reported unused images to.trashwithout a prompt (only when not using--dry-run)-T, --size-threshold-mb <megabytes>- Highlight unused images larger than this size in the report (table column or list suffix; display only)-e, --exclude-dir-assets <dir...>- Exclude directories from image file scan (can specify multiple)-F, --exclude-file-assets <file...>- Exclude specific image files by name pattern-E, --exclude-dir-code <dir...>- Exclude directories when scanning code for image references (e.g.,-E node_modules dist)-S, --exclude-file-code <file...>- Exclude specific files when scanning code for image references-r, --is-root-folder-referenced- Images use root-referenced paths (e.g.,/images/logo.pngwhere/is the root)-a, --alias- Images use alias/import paths (e.g.,@/assets/images/logo.pngor~/assets/logo.png)-t, --table- Display results in a formatted table with columns: Unused Images, In Code, Exists, Size-C, --clear-cache- Clear cache before scanning (recommended after major code changes)-H, --hide-not-found-images- Hide images referenced in code but not found on disk (dead links)-d, --dry-run- Show what would be deleted without actually deleting; does not move files and does not run--auto-prune
Important: Either -r (root-referenced) or -a (alias) must be set, but not both. They cannot have the same value.
Examples:
# Basic image scan with root-referenced paths (e.g., /images/logo.png)
qleaner image public/images src -r
# With alias paths (e.g., @/assets/images/logo.png)
qleaner image src/assets/images src -a
# Dry run with table format
qleaner image public/images src -r --dry-run --table
# Hide dead image links and use table format
qleaner image public/images src -r -H -t
# Exclude directories from code scan
qleaner image public/images src -r -E node_modules dist test
# Exclude specific directories from asset scan
qleaner image src/assets src -a -e icons fonts
# Clear cache and rescan
qleaner image public/images src -r --clear-cache
# Comprehensive scan with all options
qleaner image src/assets src -a --exclude-dir-code node_modules dist --hide-not-found-images --table --dry-run
# Highlight unused images larger than 2 MB (report only)
qleaner image public/images src -r --dry-run -T 2 --table
# Move all unused images to .trash without prompting (after a trusted dry-run)
qleaner image public/images src -r --auto-pruneSupported Image Reference Patterns:
Qleaner automatically detects images referenced through various patterns:
- ES6 Imports:
import logo from './logo.png' - CommonJS Require:
require('./logo.png')orrequire('./logo.png') - Dynamic Imports:
import('./logo.png')orimport('./assets/${name}.png') - JSX Attributes:
<img src="./logo.png" />or<img src={imagePath} /> - React Style Props:
style={{ backgroundImage: "url('./logo.png')" }} - CSS Files:
url('./logo.png')orurl('/images/logo.png') - Styled-Components:
css`background-image: url('./logo.png')`orstyled.div`background: url('./bg.png')` - Arrays:
const images = ['./logo.png', './icon.png'] - Template Literals:
`./logo-${name}.png`or`/images/${type}.png` - String Literals: Any string containing an image path pattern
qleaner list <dependency>
List files that import modules whose resolved paths match your query. The matcher splits <dependency> on / and finds graph nodes whose file path contains every segment; it then prints the union of those nodes’ importedBy files (direct importers). Works well for package names (lodash), nested paths (react-router/dom), or path fragments that appear in resolved file paths.
Arguments:
<dependency>- Dependency or path pattern (e.g.,react-router,lodash,utils/helpers)
Options:
-t, --table- Display results in a table format
Examples:
# List files using react-router
qleaner list react-router
# List files touching a path segment
qleaner list utils/helpers
# List files using lodash with table format
qleaner list lodash --table
# Find files using a specific package
qleaner list axiosNote: Requires unused-check-cache.json. Run qleaner scan (with -p or config path) first.
qleaner dep
List unused production dependencies from package.json by comparing package names to the dependency graph (packages with no importers in the graph are listed).
Options:
-d, --directory <directory>- Directory containingpackage.jsonto read (defaults to current directory)-t, --table- Display results in a table format-u, --uninstall- After listing, uninstall each unused dependency using your configured package manager (ondeponly;scanuses-ufor--auto-fix, not uninstall.)
Examples:
# Find unused dependencies in current directory
qleaner dep
# Check dependencies in a specific package/monorepo directory
qleaner dep -d ./packages/my-package
# Display results in table format
qleaner dep --table
# Uninstall unused production dependencies (use after dry review)
qleaner dep --uninstallNote: Requires unused-check-cache.json. Run qleaner scan first. Only analyzes dependencies, not devDependencies.
qleaner exports
List unused exports (symbols exported from a file that nothing imports, following the same graph as the scan). Use --fix to apply edits that remove those exports.
Arguments:
[pathToScan]- Project directory to analyze (default.)
Options:
-r, --fresh-scan- Clear graph-related state before analyzing (viascanwithclearCacheas implemented inunusedExports)-f, --fix- Fix (remove) the unused exports-d, --dry-run- Only print how many unused exports were found (no table unless--report)--report- Print a table of unreferenced exports (export name and file); works with or without--dry-run(-ron this command is--fresh-scan)
Examples:
qleaner exports . --dry-run
qleaner exports . --dry-run --report
qleaner exports . --fix --reportNote: Requires unused-check-cache.json from qleaner scan over the same project tree you care about. With --fix, files are edited in place—use version control.
qleaner prune
Remove unused internal declarations (unused variables, functions, classes, etc.) via static analysis. Always try --dry-run first.
Arguments:
[pathToScan]- Directory whose files are globbed from config (default.)
Options:
-d, --dry-run- Show what would be deleted without actually deleting (skips prompt)-r, --report- Print a table of unused symbols (file, line, name, kind); works with or without--dry-run
Examples:
qleaner prune . --dry-run -r
qleaner prune . -r
qleaner prune .Note: Operates on files discovered from qleaner.config.json (path and exclusions). Large refactors may update qleaner.stats.json with change statistics.
qleaner prune-logs
Remove console debug calls: log, dir, dirxml, table, debug, info, trace (standalone expression statements only).
Note:
qleaner prune-logs --helpmay describe flags differently; runtime behavior matchescontrollers/list.js(pruneConsoleLogs,runSurgicalPass) as summarized below.
Arguments:
[pathToScan]- Directory to scan (default.); globs come fromqleaner.config.jsonlike other prune commands
Options:
-d, --dry-run- Report counts only (no edits). Prints totals asN (high-risk sensitive: H, medium: M, low: L). With--list-risk-categoriesalso set, prints per-tier hit tables (file, line, preview) when the dry-run result includesfoundByRisk(capped per tier incontrollers/list.js).-i/--list-risk-categories-info- Print the risk-tier category index (pattern file paths and section titles), then return without scanning or editing.--list-risk-categories- Does not skip the scan by itself. Use with-dto dry-run and optionally show hit tables as above, or use-ifor the index-only path.
Heuristics (not a security scanner): Patterns live in utils/consoleLogHighRiskPatterns.js and utils/consoleLogMediumRiskPatterns.js; classification order is high → medium → low.
Examples:
# Risk index only (no scan, no edits) — preferred for CI / docs
qleaner prune-logs -i
# Dry-run counts (hit tables only if you also pass --list-risk-categories)
qleaner prune-logs src --dry-run
qleaner prune-logs src --dry-run --list-risk-categories
# Apply removals (not a dry run)
qleaner prune-logs srcqleaner duplicates
Detect and clean duplicate code patterns across files under the configured path.
Arguments:
[pathToScan]- Directory to scan (default.)
Options:
-d, --dry-run- Show what would be deleted without actually deleting (skips prompt)-r, --report- Print a table of duplicate functions and classes (file, line, name, kind); works with or without--dry-run
Examples:
qleaner duplicates . --dry-run -r
qleaner duplicates . -r
qleaner duplicates .qleaner undo <type>
Restore files from .trash after a cleanup that moved files instead of deleting them permanently.
Arguments:
<type>-codeorimagesdepending on which kind of deletion you want to revert
Examples:
qleaner undo code
qleaner undo imagesqleaner tidy
Tidy up the project — runs a pipeline of checks; each step dry-runs first, then applies edits only if --auto-fix / -u is set and the dry-run reported work.
Arguments:
[pathToScan]- Project root or source directory for globs (default.)
Options:
-u, --auto-fix- After a successful dry-run count for a step, run the real pass (console logs, internal unused code, duplicates, unused exports with--fix). Finishes with a dry-runscan(unused files listed, not moved).-r/--report- During each step’s dry-run, print detail tables: console log tiers, unused code, duplicates, and exports (also enables console-log tier tables; same as passing--list-risk-categoriesfor that step).--list-risk-categories- Console-log step only: per-tier hit tables whenfoundByRiskis present (also enabled when-ris set).-i/--list-risk-categories-info- Print the console-log risk category index; tidy continues with remaining steps.
Sequence (see controllers/list.js):
- Console logs — dry-run (counts + tier tables with
-ror--list-risk-categories) → apply if--auto-fix - Unused internal code —
prunedry-run (detail table with-r) → apply if needed - Duplicates — dry-run (detail table with
-r) → apply if needed - Unused exports — dry-run (detail table with
-r) → apply with fix if needed - Unused files scan —
scanwithdryRun: trueonly when--auto-fixran (reports unused files; does not move them)
Prefer exercising prune, prune-logs, duplicates, exports, and scan manually with --dry-run and -r until you trust the behavior.
Examples:
# Dry-run each step; no edits
qleaner tidy
# Per-step detail tables (CI default: tidy-extra-args "-r")
qleaner tidy -r
# Risk index for console logs (pipeline continues)
qleaner tidy -i
# Apply safe code cleanups when issues are found
qleaner tidy . --auto-fixqleaner summary
Get comprehensive project statistics and insights. (qleaner summary --help uses the short description List all the imports in the project; the sections below describe the full behavior.)
Options:
-l, --largest-files- Show top 10 largest code and image files-d, --dependencies- Show dependency analysis and hotspots
Note: When no options are provided, shows a general project summary with totals.
Examples:
# Full project summary
qleaner summary
# Show largest files
qleaner summary --largest-files
# Show dependency analysis
qleaner summary --dependenciesSummary Output Breakdown:
General Summary (no flags):
- Total code files count
- Total image files count
- Total unused files count
- Total unused images count
- Total dead image links count
- Total files count
Largest Files (-l or --largest-files):
- Top 10 largest code files (sorted by size in KB)
- Top 10 largest image files (sorted by size in MB)
- Total code size and total image size
- Code files exceeding 100 KB (warning list)
Dependencies (-d or --dependencies):
- Top 10 files with heavy dependencies (most imports)
- Top 10 files with light dependencies (fewest imports)
- Top 10 file hotspots (most imported files - files that many other files import)
- Top 10 dependency hotspots (most used dependencies/packages)
- Top 10 dead image hotspots (dead image links referenced by many files)
- Top 10 alive image hotspots (images referenced by many files)
Note: Requires a cache file. Run qleaner scan to generate code cache and/or qleaner image <dir> <root> to generate image cache first.
Configuration
Qleaner can be configured via qleaner.config.json or CLI flags. For qleaner scan, CLI flags override values merged from the config file.
Configuration File Structure
{
"path": "src",
"framework": "react",
"codeAlias": "tsconfig.json",
"paths": {},
"packageManager": "yarn",
"excludeDir": ["node_modules", "dist", "build", ".next"],
"excludeFile": ["payload-types.ts"],
"excludeExtensions": [".test.tsx", ".test.ts"],
"excludeDirPrint": [],
"excludeFilePrint": ["index.js", "index.tsx", "main.js"],
"excludeDirAssets": [],
"excludeFileAssets": [],
"excludeDirCode": ["node_modules", "dist"],
"excludeFileCode": [],
"isRootFolderReferenced": true,
"alias": false,
"autoFix": false,
"autoPrune": false,
"sizeThresholdMb": null
}Configuration Options
| Option | Type | Description |
|--------|------|-------------|
| path | string | Default directory for qleaner scan (when -p is omitted) and for commands that glob from config (exports, prune, prune-logs, duplicates, tidy). Often your app root or src. |
| autoFix | boolean | When true (and not overridden), qleaner scan without --dry-run moves all reported unused files to .trash via moveToTrash instead of opening the interactive delete prompt (same effect as --auto-fix / -u on the scan command). |
| autoPrune | boolean | When true, qleaner image without --dry-run moves all reported unused images to .trash without a prompt (same as --auto-prune / -u on the image command). |
| sizeThresholdMb | number | null | Optional default for highlighting large unused images in the image report (same semantics as --size-threshold-mb / -T). |
| framework | string | One of react, nextjs, or vue (use vue for Vue and Nuxt apps). Sets default excludeFilePrint and, for Vue, default excludeDirPrint (layouts, pages). |
| codeAlias | string | null | Critical! Config file to use for path alias resolution. Options: "tsconfig.json", "jsconfig.json", "tsconfig.app.json", "tsconfig.base.json", or null for manual configuration. Qleaner reads path aliases from this file to resolve imports (e.g., @/components, ~/utils). |
| paths | object | Manual path aliases (only used if codeAlias is null). Object mapping alias patterns to file paths. Example: { "@/*": ["./src/*"], "~/*": ["./src/*"] }. If codeAlias is set, this field is ignored and paths are read from the config file instead. |
| packageManager | string | Package manager used: npm, yarn, or pnpm |
| excludeDir | string[] | Directories to exclude from code scans |
| excludeFile | string[] | Files to exclude by name from code scans |
| excludeExtensions | string[] | File extensions to exclude (e.g., ["test.tsx"]) |
| excludeDirPrint | string[] | Path segments matched against file paths; matching files are excluded from the unused-files result (e.g. Vue defaults layouts, pages) |
| excludeFilePrint | string[] | Entry point files to scan but not report as unused |
| excludeDirAssets | string[] | Directories to exclude from image scans |
| excludeFileAssets | string[] | Image files to exclude by name |
| excludeDirCode | string[] | Directories to exclude when scanning code for image references |
| excludeFileCode | string[] | Files to exclude when scanning code for image references |
| isRootFolderReferenced | boolean | true if image paths are root-referenced (e.g., /images/logo.png) |
| alias | boolean | true if image paths use aliases (e.g., @/assets/images/logo.png) |
Important Configuration Notes:
codeAliasandpaths:- If
codeAliasis set (e.g.,"tsconfig.json"), Qleaner will read path aliases from that config file. Thepathsfield will be ignored. - If
codeAliasisnull, Qleaner will use thepathsfield for manual path alias configuration. - Getting this right is crucial for accurate file resolution in large codebases!
- If
isRootFolderReferencedandalias: These should not both betrueor both befalse. Set one totrueand the other tofalse.
Manual Path Configuration
If you don't use jsconfig.json or tsconfig.json, or need custom path resolution, you can manually configure paths in qleaner.config.json:
{
"codeAlias": null,
"paths": {
"@/*": ["./src/*"],
"~/*": ["./src/*"],
"components/*": ["./src/components/*"],
"utils/*": ["./src/utils/*"]
}
}The paths object follows the same format as TypeScript's path mapping:
- Key: The alias pattern (e.g.,
"@/*"matches@/components,@/utils, etc.) - Value: Array of actual file paths to resolve to (relative to project root)
Example: If you have import Button from '@/components/Button' in your code, and your config has "@/*": ["./src/*"], Qleaner will resolve it to ./src/components/Button.tsx (or .ts, .jsx, .js).
Default Exclusions
Qleaner automatically excludes common build and generated directories:
node_modules,dist,build,.next,outcoverage,.turbo,.vite,.cache.vercel,.netlify,storybook-staticgenerated,prisma,graphql,supabase,drizzle,__generated__
Framework-Specific Entry Points
React:
index.js,index.jsx,index.ts,index.tsxmain.js,main.jsx,main.ts,main.tsx
Vue: Default excludeFilePrint may be empty after init; typical unused-file exclusions use excludeDirPrint (layouts, pages) so framework routes are not flagged.
Next.js:
page.tsx,page.jsx,route.ts,route.jsxlayout.tsx,layout.jsxmiddleware.ts,middleware.jserror.tsx,error.jsx,loading.tsx,loading.jsxnot-found.tsx,not-found.jsxglobal-error.tsx,global-error.jsx_app.tsx,_app.jsx,_document.tsx,_document.jsx_error.tsx,_error.jsx
Caching
Qleaner uses intelligent caching to speed up subsequent scans:
- File Hashing - Files are only re-analyzed if their content changed
- Incremental Updates - Only changed files are re-processed
- Cache Location -
unused-check-cache.jsonin your project root
Some edit-heavy commands (prune, prune-logs, duplicates, exports --fix) may append summary metrics to qleaner.stats.json in the project root.
When to Clear Cache:
- After major refactoring
- When dependencies change significantly
- If you get unexpected results
# Clear cache and rescan
qleaner scan src --clear-cacheFile Deletion
When you run qleaner scan without --dry-run:
- Default — Qleaner prompts you to select unused files and whether to move them to
.trashor delete permanently (askDeleteFiles). --auto-fix/-u(orautoFix: truein config, merged like other scan options) — Skips the prompt and moves all reported unused files to.trashin one step (moveToTrash).
Recommended Workflow:
- Run with
--dry-runfirst to see what would be deleted - Review the results carefully
- Run without
--dry-runand either use the interactive flow or--auto-fixif you want everything moved to.trashwithout prompts - Prefer Move to
.trashwhen using the interactive flow - Test your application
- Empty
.trashwhen confident
How It Works
Code Analysis
- File Discovery - Uses
fast-globto find all code files matching patterns with intelligent exclusion handling - Path Alias Resolution - Reads path aliases from your
jsconfig.jsonortsconfig.json(or manualpathsconfig) to understand how imports are resolved. This enables accurate file finding even in large codebases with complex import structures. - AST Parsing - Parses JavaScript/TypeScript with Babel;
.ts/.mts/.ctsare parsed without the JSX plugin so TypeScript angle-bracket assertions (e.g.<Type>expr) are valid, while.tsx/.js/.jsxkeep JSX parsing. Vue & Nuxt:.vueSFCs are compiled with@vue/compiler-sfc; script blocks uselangfor the TS/JSX split. Image and dependency scanning understand.vuefiles (utils/astParser.js,utils/fileProcessing.js,controllers/image.js). For Nuxt, runqleaner init, choose Vue, and setpathto your app root (e.g.src/or project root). - Module Resolution - Uses
enhanced-resolvewith your project's path aliases to resolve import paths accurately. Supports:- Path aliases (e.g.,
@/components,~/utils) - Relative imports (
./component,../utils) - Node modules (
react,lodash) - Directory index files (automatically resolves
./componentsto./components/index.ts)
- Path aliases (e.g.,
- Dependency Graph - Builds a comprehensive graph of file dependencies, tracking both imports and exports
- Unused Detection - Identifies files with no incoming imports (except entry points in
excludeFilePrintand paths matchingexcludeDirPrint). Files that are only used via re-exports (barrel files) are treated as used when downstream files import the exported symbols or*.
Image Analysis
- Image Discovery - Finds all image files (png, jpg, jpeg, svg, gif, webp) using glob patterns with exclusion support
- Advanced Code Scanning - Uses sophisticated AST traversal to extract image references from:
- ES6 import statements (
import logo from './logo.png') - CommonJS require (
require('./logo.png')) - Dynamic imports (
import('./logo.png')) - JSX attributes (
<img src="./logo.png" />) - React style props (
style={{ backgroundImage: "url('./logo.png')" }}) - CSS files (
url('./logo.png')orurl('/images/logo.png')) - Styled-components (
css`background-image: url('./logo.png')`) - Arrays (
const images = ['./logo.png', './icon.png']) - Template literals (
`./logo-${name}.png`) - String literals (any string containing an image path pattern)
- Spread elements in arrays
- ES6 import statements (
- Path Normalization - Normalizes paths based on alias/root configuration, handling both relative and absolute paths
- Unused Detection - Compares image files against all discovered references to identify truly unused images
📊 Example Workflow
# 1. Initialize configuration (CRITICAL STEP - get this right!)
qleaner init
# Make sure to select the correct jsconfig.json or tsconfig.json file
# Verify the generated qleaner.config.json has the correct codeAlias
# 2. Verify your configuration
cat qleaner.config.json
# Check that codeAlias matches your project's config file
# 3. Scan for unused code files (dry run first!)
qleaner scan src --dry-run --table
# 4. Review results - if you see false positives, check your config
# Fix codeAlias or paths if needed, then clear cache and rescan:
qleaner scan src --clear-cache --dry-run --table
# 5. Scan for unused images (dry run)
qleaner image public/images src -r --dry-run --table
# 6. Check for unused dependencies
qleaner dep --table
# 7. Optional: unused exports, prune, logs, duplicates (dry-run first)
qleaner exports . --dry-run
qleaner prune . --dry-run
qleaner prune-logs -i
qleaner prune-logs src/infisical-main/frontend --dry-run
qleaner duplicates . --dry-run
# 8. Get project summary
qleaner summary
# 9. Get largest files report
qleaner summary --largest-files
# 10. Get dependency analysis
qleaner summary --dependencies
# 11. Once confident with results, run actual scans and delete files
qleaner scan src
qleaner image public/images src -r
# 12. Optional one-shot cleanup (dry-run each step; add --auto-fix to apply)
# qleaner tidy
# qleaner tidy . --auto-fixUse Cases
- Before Deployment - Clean up unused files to reduce bundle size
- Code Review - Verify no unused files are being added
- Refactoring - Find files that can be safely removed
- Optimization - Identify large files and dependency hotspots
- Maintenance - Regular cleanup to keep codebase healthy
Requirements
- Node.js 14+ (LTS recommended)
- npm or yarn
Contributing
Contributions are welcome! Please feel free to submit a Pull Request.
- Fork the repository
- Create your feature branch (
git checkout -b feature/amazing-feature) - Commit your changes (
git commit -m 'Add some amazing feature') - Push to the branch (
git push origin feature/amazing-feature) - Open a Pull Request
📄 License
MIT
Links
- Repository: https://github.com/trevismurithi/react-cleaner
- Issues: https://github.com/trevismurithi/react-cleaner/issues
- NPM Package: https://www.npmjs.com/package/qleaner
📚 Additional Resources
Tips for Best Results
- Get configuration right first - Most important! Ensure
codeAliaspoints to the correctjsconfig.jsonortsconfig.jsonfile, or manually configurepathsif you don't use these config files. Incorrect configuration leads to false positives and missed dependencies. - Verify your config file - After running
qleaner init, check thatcodeAliasmatches your project's actual config file. For monorepos, you may needtsconfig.app.jsonortsconfig.base.jsoninstead oftsconfig.json. - Run scans regularly - Set up a cron job or CI check to catch unused files early
- Use dry-run first - Always preview with
--dry-runbefore deleting anything - Review entry points - Make sure
excludeFilePrintin config includes all entry point files (index.tsx, main.js, etc.) - Clear cache when needed - After major refactoring, dependency changes, or config file updates, use
--clear-cachefor accurate results - Test after deletion - Always test your application thoroughly after removing files
- Use table format for large outputs - The
-tflag makes results easier to read - Check summary regularly - Use
qleaner summaryto monitor project health - For large codebases - The enhanced file finding mechanism with proper config ensures accurate results even in projects with thousands of files and complex import structures
Command Dependencies
Best practice: Run qleaner scan <your-src> (or qleaner scan with path in qleaner.config.json) before commands that need unused-check-cache.json.
| Command | Requires |
|---------|----------|
| qleaner list <dependency> | unused-check-cache.json from qleaner scan |
| qleaner dep | unused-check-cache.json from qleaner scan |
| qleaner exports | unused-check-cache.json from qleaner scan |
| qleaner summary | Code and/or image cache from qleaner scan / qleaner image |
| qleaner prune, prune-logs, duplicates | qleaner.config.json with path; optional unused-check-cache.json for incremental behavior / parentGraph.fixes |
| qleaner tidy | Config, scan cache for the exports step, and the same prerequisites as the sub-steps |
Common Issues
Q: Qleaner reports files as unused that are actually used. A:
- Check your configuration - Most commonly, this happens when
codeAliasis incorrect orpathsis not configured properly. Verify that Qleaner can resolve your import aliases:- Ensure
codeAliaspoints to the correct config file (check if it exists and has the right path mappings) - If using manual
paths, verify the alias patterns match your imports - Try running with
--clear-cacheafter fixing config
- Ensure
- These might be entry points - Add them to
excludeFilePrintin your config or use-Fflag - Verify the file is actually imported - Check with
qleaner list <file-path>to see if it's referenced - Check for dynamic imports - Some dynamic imports may not be detected
Q: Images aren't being detected. A:
- Verify your
-a(alias) or-r(root-referenced) flag matches how images are referenced in code - Check if the image path patterns match what's in your code
- Try
--clear-cacheto rebuild the image graph - Ensure the
<rootPath>argument includes all directories that reference images
Q: Scan is slow on large projects. A:
- First scan is always slower - it builds the complete dependency graph
- Subsequent scans use caching and are much faster (only changed files are re-analyzed)
- The enhanced file finding mechanism with proper config ensures accurate and efficient resolution even in large codebases
- Exclude more directories with
-eflag to speed up scans - Consider scanning smaller subdirectories separately
- Ensure your
codeAliasconfig is correct - incorrect config can cause unnecessary resolution attempts
Q: How do I know if my configuration is correct? A:
- After
qleaner init, verifycodeAliasinqleaner.config.jsonmatches your project's config file - Run a test scan:
qleaner scan src --dry-runand check if files you know are used are incorrectly reported as unused - If you see false positives, check:
- Does your
jsconfig.json/tsconfig.jsonexist and have correctpathsconfiguration? - Is
codeAliaspointing to the right file? - If using manual
paths, do the patterns match your import statements?
- Does your
- For monorepos, you may need
tsconfig.app.jsonortsconfig.base.jsoninstead oftsconfig.json
Q: How do I undo deletions? A:
- If you used "Move to
.trash", files are safely stored in.trashdirectory in your project root - Permanently deleted files can be restored from git if you're using version control
- Always test after deletions before emptying
.trash
Q: list command shows no results but I know the dependency is used.
A:
- Make sure you ran
qleaner scan(with-porpathin config) first to generate the cache - The dependency path might need to match exactly - try partial paths (e.g.,
reactinstead ofreact/dist/react.production.min.js) - Check if the dependency is in
node_modulesor a local file path
Q: dep command shows dependencies I know are used.
A:
- Some dependencies might be used only in config files (webpack.config.js, etc.) which aren't scanned
- Dynamic requires/imports might not be detected
- Check if they're actually used with
qleaner list <dependency-name>
Made with love for clean codebases
