npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2026 – Pkg Stats / Ryan Hefner

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

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 --fix to remove them
  • Vue & Nuxt - Scan .vue single-file components (@vue/compiler-sfc); choose Vue during qleaner init for pages / layouts exclusions and .vue globs (typical for Nuxt apps)
  • Prune Unused Code - Remove unused variables, functions, and classes (with dry-run); -r / --report prints a file, line, name, and kind table
  • Prune Console Logs - Strip console debug 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 includes foundByRisk and you pass --list-risk-categories (see controllers/list.jsrunSurgicalPass). -i / --list-risk-categories-info prints the risk category index (paths to utils/consoleLogHighRiskPatterns.js and utils/consoleLogMediumRiskPatterns.js) and returns without scanning. --list-risk-categories alone does not skip the scan—use -d for a dry run or -i for docs only.
  • Duplicate Detection - Find and clean duplicate functions and classes; -r / --report prints 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 / --report prints per-step dry-run tables for all four steps (console log tiers, unused code, duplicates, exports). --list-risk-categories limits extra console-log tier tables to that step only; -i prints 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, including parentGraph.fixes fingerprints for surgical passes when present)
  • Dry-Run Mode - Where supported, -d / --dry-run matches 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 use scan --auto-fix to move all reported unused files to .trash without 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 qleaner

Use with npx (No Installation)

npx qleaner <command>

Local Development Dependency

yarn add qleaner --dev
# or
npm install qleaner --save-dev

GitHub 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

  1. Sets up Node.js (default 20).
  2. Runs npm install -g qleaner@<version> (default 1.5.1).
  3. Restores unused-check-cache.json from the Actions cache (key uses qleaner.config.json and package-lock.json).
  4. Runs read-only captures in the job workspace:
    • qleaner tidy <scan-path> …tidy_report.txt
    • qleaner image <assets> <code> …image_report.txt (skipped if image paths are empty)
    • qleaner summaryhealth_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.json at the repo root (run qleaner init locally once; do not run init in CI — it is interactive).
  • Add unused-check-cache.json and .trash/ to .gitignore ( init can 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.1uses: 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.1 must match an existing tag on trevismurithi/react-cleaner (e.g. v1.5.1 or 1.5.1 — the ref is exact).
  • No image scan: set image-assets-path and image-code-path to "" (or omit and rely on defaults).
  • Report only on PRs: workflow_dispatch runs Qleaner but skips the comment steps (if: github.event_name == 'pull_request').
  • You do not copy .github/scripts/pr-report.js into your repo; it ships in the published qleaner npm 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 init

This interactive command will ask you:

  1. Path to scan - Default directory for unused-file scans and for commands that use qleaner.config.json (e.g. exports, prune, tidy)
  2. Package manager (npm/yarn/pnpm)
  3. Image path resolution - Relative or alias paths vs absolute paths from public/ (e.g. /images/logo.png)
  4. Framework (React, Next.js, or Vue) - Sets entry-point patterns and, for Vue, default excludeDirPrint (layouts, pages)
  5. 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 src

3. 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-run

Commands

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:

  1. Path to scan - Directory Qleaner uses by default for scans and code-analysis commands
  2. Package manager (npm, yarn, or pnpm)
  3. Image path resolution - Relative/alias vs absolute-from-public/ paths
  4. Framework (React, Next.js, or Vue) - Entry files and Vue-specific directory exclusions
  5. Configuration file for path aliases - Most important question! Select which config file Qleaner should use:
    • tsconfig.json - Standard TypeScript configuration
    • jsconfig.json - Standard JavaScript configuration
    • tsconfig.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.json or tsconfig.json to resolve imports like @/components/Button or ~/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 paths field of qleaner.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-run

Note: 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 config path and exclusions as implemented in command.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 as dep is not shared: on scan this is auto-fix, on dep -u is --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 --table

qleaner 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 .trash without 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.png where / is the root)
  • -a, --alias - Images use alias/import paths (e.g., @/assets/images/logo.png or ~/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-prune

Supported Image Reference Patterns:

Qleaner automatically detects images referenced through various patterns:

  • ES6 Imports: import logo from './logo.png'
  • CommonJS Require: require('./logo.png') or require('./logo.png')
  • Dynamic Imports: import('./logo.png') or import('./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') or url('/images/logo.png')
  • Styled-Components: css`background-image: url('./logo.png')` or styled.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 axios

Note: 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 containing package.json to 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 (on dep only; scan uses -u for --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 --uninstall

Note: 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 (via scan with clearCache as implemented in unusedExports)
  • -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 (-r on this command is --fresh-scan)

Examples:

qleaner exports . --dry-run
qleaner exports . --dry-run --report
qleaner exports . --fix --report

Note: 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 --help may describe flags differently; runtime behavior matches controllers/list.js (pruneConsoleLogs, runSurgicalPass) as summarized below.

Arguments:

  • [pathToScan] - Directory to scan (default .); globs come from qleaner.config.json like other prune commands

Options:

  • -d, --dry-run - Report counts only (no edits). Prints totals as N (high-risk sensitive: H, medium: M, low: L). With --list-risk-categories also set, prints per-tier hit tables (file, line, preview) when the dry-run result includes foundByRisk (capped per tier in controllers/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 -d to dry-run and optionally show hit tables as above, or use -i for 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 src

qleaner 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> - code or images depending on which kind of deletion you want to revert

Examples:

qleaner undo code
qleaner undo images

qleaner 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-run scan (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-categories for that step).
  • --list-risk-categories - Console-log step only: per-tier hit tables when foundByRisk is present (also enabled when -r is set).
  • -i / --list-risk-categories-info - Print the console-log risk category index; tidy continues with remaining steps.

Sequence (see controllers/list.js):

  1. Console logs — dry-run (counts + tier tables with -r or --list-risk-categories) → apply if --auto-fix
  2. Unused internal codeprune dry-run (detail table with -r) → apply if needed
  3. Duplicates — dry-run (detail table with -r) → apply if needed
  4. Unused exports — dry-run (detail table with -r) → apply with fix if needed
  5. Unused files scanscan with dryRun: true only when --auto-fix ran (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-fix

qleaner 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 --dependencies

Summary 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:

  1. codeAlias and paths:

    • If codeAlias is set (e.g., "tsconfig.json"), Qleaner will read path aliases from that config file. The paths field will be ignored.
    • If codeAlias is null, Qleaner will use the paths field for manual path alias configuration.
    • Getting this right is crucial for accurate file resolution in large codebases!
  2. isRootFolderReferenced and alias: These should not both be true or both be false. Set one to true and the other to false.

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, out
  • coverage, .turbo, .vite, .cache
  • .vercel, .netlify, storybook-static
  • generated, prisma, graphql, supabase, drizzle, __generated__

Framework-Specific Entry Points

React:

  • index.js, index.jsx, index.ts, index.tsx
  • main.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.jsx
  • layout.tsx, layout.jsx
  • middleware.ts, middleware.js
  • error.tsx, error.jsx, loading.tsx, loading.jsx
  • not-found.tsx, not-found.jsx
  • global-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.json in 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-cache

File Deletion

When you run qleaner scan without --dry-run:

  • Default — Qleaner prompts you to select unused files and whether to move them to .trash or delete permanently (askDeleteFiles).
  • --auto-fix / -u (or autoFix: true in config, merged like other scan options) — Skips the prompt and moves all reported unused files to .trash in one step (moveToTrash).

Recommended Workflow:

  1. Run with --dry-run first to see what would be deleted
  2. Review the results carefully
  3. Run without --dry-run and either use the interactive flow or --auto-fix if you want everything moved to .trash without prompts
  4. Prefer Move to .trash when using the interactive flow
  5. Test your application
  6. Empty .trash when confident

How It Works

Code Analysis

  1. File Discovery - Uses fast-glob to find all code files matching patterns with intelligent exclusion handling
  2. Path Alias Resolution - Reads path aliases from your jsconfig.json or tsconfig.json (or manual paths config) to understand how imports are resolved. This enables accurate file finding even in large codebases with complex import structures.
  3. AST Parsing - Parses JavaScript/TypeScript with Babel; .ts / .mts / .cts are parsed without the JSX plugin so TypeScript angle-bracket assertions (e.g. <Type>expr) are valid, while .tsx / .js / .jsx keep JSX parsing. Vue & Nuxt: .vue SFCs are compiled with @vue/compiler-sfc; script blocks use lang for the TS/JSX split. Image and dependency scanning understand .vue files (utils/astParser.js, utils/fileProcessing.js, controllers/image.js). For Nuxt, run qleaner init, choose Vue, and set path to your app root (e.g. src/ or project root).
  4. Module Resolution - Uses enhanced-resolve with 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 ./components to ./components/index.ts)
  5. Dependency Graph - Builds a comprehensive graph of file dependencies, tracking both imports and exports
  6. Unused Detection - Identifies files with no incoming imports (except entry points in excludeFilePrint and paths matching excludeDirPrint). Files that are only used via re-exports (barrel files) are treated as used when downstream files import the exported symbols or *.

Image Analysis

  1. Image Discovery - Finds all image files (png, jpg, jpeg, svg, gif, webp) using glob patterns with exclusion support
  2. 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') or url('/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
  3. Path Normalization - Normalizes paths based on alias/root configuration, handling both relative and absolute paths
  4. 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-fix

Use 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.

  1. Fork the repository
  2. Create your feature branch (git checkout -b feature/amazing-feature)
  3. Commit your changes (git commit -m 'Add some amazing feature')
  4. Push to the branch (git push origin feature/amazing-feature)
  5. Open a Pull Request

📄 License

MIT

Links

📚 Additional Resources

Tips for Best Results

  1. Get configuration right first - Most important! Ensure codeAlias points to the correct jsconfig.json or tsconfig.json file, or manually configure paths if you don't use these config files. Incorrect configuration leads to false positives and missed dependencies.
  2. Verify your config file - After running qleaner init, check that codeAlias matches your project's actual config file. For monorepos, you may need tsconfig.app.json or tsconfig.base.json instead of tsconfig.json.
  3. Run scans regularly - Set up a cron job or CI check to catch unused files early
  4. Use dry-run first - Always preview with --dry-run before deleting anything
  5. Review entry points - Make sure excludeFilePrint in config includes all entry point files (index.tsx, main.js, etc.)
  6. Clear cache when needed - After major refactoring, dependency changes, or config file updates, use --clear-cache for accurate results
  7. Test after deletion - Always test your application thoroughly after removing files
  8. Use table format for large outputs - The -t flag makes results easier to read
  9. Check summary regularly - Use qleaner summary to monitor project health
  10. 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:

  1. Check your configuration - Most commonly, this happens when codeAlias is incorrect or paths is not configured properly. Verify that Qleaner can resolve your import aliases:
    • Ensure codeAlias points 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-cache after fixing config
  2. These might be entry points - Add them to excludeFilePrint in your config or use -F flag
  3. Verify the file is actually imported - Check with qleaner list <file-path> to see if it's referenced
  4. Check for dynamic imports - Some dynamic imports may not be detected

Q: Images aren't being detected. A:

  1. Verify your -a (alias) or -r (root-referenced) flag matches how images are referenced in code
  2. Check if the image path patterns match what's in your code
  3. Try --clear-cache to rebuild the image graph
  4. 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 -e flag to speed up scans
  • Consider scanning smaller subdirectories separately
  • Ensure your codeAlias config is correct - incorrect config can cause unnecessary resolution attempts

Q: How do I know if my configuration is correct? A:

  • After qleaner init, verify codeAlias in qleaner.config.json matches your project's config file
  • Run a test scan: qleaner scan src --dry-run and check if files you know are used are incorrectly reported as unused
  • If you see false positives, check:
    1. Does your jsconfig.json/tsconfig.json exist and have correct paths configuration?
    2. Is codeAlias pointing to the right file?
    3. If using manual paths, do the patterns match your import statements?
  • For monorepos, you may need tsconfig.app.json or tsconfig.base.json instead of tsconfig.json

Q: How do I undo deletions? A:

  • If you used "Move to .trash", files are safely stored in .trash directory 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 -p or path in config) first to generate the cache
  • The dependency path might need to match exactly - try partial paths (e.g., react instead of react/dist/react.production.min.js)
  • Check if the dependency is in node_modules or 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