Clean up common development artifacts (node_modules, .venv, caches) to reclaim disk space
Project description
devklean
Reclaim disk space by safely cleaning up development artifacts — node_modules, .venv, build caches, and other regenerable directories — with a trash-backed safety net so nothing is ever lost to a typo.
Why
Development trees accumulate gigabytes of regenerable junk. devklean finds it, shows you exactly what it found and how big it is, and removes it to your native system trash — the Recycle Bin on Windows, Trash on macOS, the freedesktop trash on Linux — not rm -rf, so a mistaken delete is recoverable from your trash. It refuses to touch dangerous paths, makes large deletions require an explicit typed confirmation, and keeps a history of what it removed.
Installation
# Recommended: isolated install as a standalone CLI tool
pipx install devklean
# Or into the current environment
pip install devklean
Requires Python 3.8+. Runtime dependencies are send2trash (used for all deletions) and tomli (only on Python < 3.11).
Quick start
# See what could be cleaned under the current directory — no changes made
devklean scan
# Scan and delete (with confirmation), moving items to the trash
devklean clean
# Preview a clean without deleting anything
devklean clean --dry-run
Commands
| Command | What it does |
|---|---|
devklean scan [PATH] |
Find cleanable directories and report sizes. Never deletes. |
devklean clean [PATH] |
Scan, then delete (to trash) after confirmation. |
devklean restore |
Explain how to recover deleted items from your system trash. |
devklean history |
Show previous cleanup operations (timestamp, size, strategy, item count). |
devklean doctor |
Inspect and repair the deletion metadata store. |
devklean --version |
Print the version. |
Default command: running devklean with no subcommand defaults to clean — so devklean is devklean clean, and devklean ~/code is devklean clean ~/code. As a convenience, a bare devklean --dry-run (without -i) is treated as a scan preview.
scan
devklean scan # scan current directory
devklean scan ~/projects # scan a specific path
devklean scan --json # machine-readable output for tooling
clean
devklean clean # scan + delete with a y/N confirmation
devklean clean ~/projects # clean a specific path
devklean clean --dry-run # show what *would* be deleted; delete nothing
devklean clean -i # interactive: pick items in a TUI (Linux/macOS only)
devklean clean --allow-symlinks # permit deleting symlinked targets (blocked by default)
devklean clean -y # skip the y/N prompt (large deletions still require typing DELETE)
| Flag | Meaning |
|---|---|
--dry-run |
Report what would be deleted; make no changes. Output says "would delete". |
-i, --interactive |
Choose items in a terminal UI (SPACE select, A all, D none, ENTER confirm, Q quit). Linux/macOS only — see Platform support. |
--allow-symlinks |
Allow deleting symbolic links. Off by default (symlinks are blocked). |
-y, --yes |
Skip the standard confirmation. Deletions over the size threshold still require typing DELETE. |
restore
devklean moves items to your operating system's own trash, which the OS owns — so recovery is done through your file manager's trash, not through devklean. This command just shows you how:
devklean restore # explains how to recover from the Recycle Bin / Trash
- Windows — open the Recycle Bin and restore the item.
- macOS — open Trash in Finder and "Put Back".
- Linux — open Trash in your file manager and restore.
Run devklean history to see what was removed and when.
history
devklean history # human-readable table of past cleanups
devklean history --json # machine-readable
doctor
devklean doctor # report corrupt metadata; prompt before removing it
devklean doctor --yes # remove corrupt records without prompting
Usage example
$ devklean clean ~/code/my-app
devklean scanning /home/me/code/my-app...
TYPE SIZE PATH
──────────────────────────────────────────────────────────────────────
Node.js 612.0 MB /home/me/code/my-app/node_modules
Python venv 188.0 MB /home/me/code/my-app/.venv
Build output 34.0 MB /home/me/code/my-app/dist
──────────────────────────────────────────────────────────────────────
TOTAL 834.0 MB
Delete 3 directories (~834.0 MB)? (y/N) y
✓ /home/me/code/my-app/node_modules
✓ /home/me/code/my-app/.venv
✓ /home/me/code/my-app/dist
✓ Cleaned 3 directories, freed ~834.0 MB.
Platform support
devklean runs on Linux, macOS, and Windows. All core commands — scan,
clean, history, doctor, and restore — work on every platform.
Known limitation: interactive mode (-i / --interactive) is Linux/macOS
only for now. It is built on Python's curses module, which is not available
on Windows. Running devklean clean -i on Windows prints a clear message and
exits without making changes, rather than crashing — use devklean clean
(non-interactive) instead. Windows support for interactive mode may come in a
future release.
Configuration
devklean reads TOML config with the precedence project > global > built-in defaults:
- A project file
.devklean.toml, discovered by walking up from the current directory. - The global file
~/.config/devklean/config.toml(or$XDG_CONFIG_HOME/devklean/config.toml).
# ~/.config/devklean/config.toml or ./.devklean.toml
# Directory names to never touch (merged into the ignore list)
exclude = ["node_modules", ".git"]
[defaults]
dry_run = false
interactive = false
default_yes = false # skip the y/N prompt (the large-deletion DELETE gate still applies)
theme = "default" # "default" or "mono"
confirm_threshold = 1073741824 # bytes; deletions >= this require typing DELETE (default 1 GiB)
path = "."
[targets]
exclude = ["dist"] # remove built-in target *types*
[targets.custom]
".turbo" = "Turborepo cache" # add custom target directory types
[ignore]
paths = ["/keep/this/node_modules"] # absolute paths to skip
directories = ["vendor"] # directory names to skip
Scalar keys from the project file override the global file; list keys (exclude, ignore.*) are unioned. Unknown keys and malformed TOML produce a warning rather than a crash.
Color follows the theme setting and is automatically disabled when output is piped or NO_COLOR is set.
Logs
devklean writes detailed structured logs (commands, scanned/deleted paths, sizes, errors) to:
~/.cache/devklean/logs/latest.log # or $XDG_CACHE_HOME/devklean/logs/latest.log
Logs rotate (5 backups) and are kept entirely separate from terminal output.
FAQ
Why didn't it delete something?
devklean refuses to delete paths that should never be removed: the filesystem root, your home directory, mounted drive roots, protected system directories (/etc, /usr, …), and symbolic links. Symlinks can be allowed with --allow-symlinks. Each refusal prints the specific reason.
It asked me to type DELETE — why?
Deletions whose total size meets the confirm_threshold (default 1 GiB) require an explicit typed confirmation as an extra guard against large accidental deletes. This gate is intentionally not bypassed by --yes or default_yes; only --dry-run skips it.
Where do deleted files go? Can I get them back?
Items are moved to your operating system's native trash (Recycle Bin on Windows, Trash on macOS, the freedesktop trash on Linux), not permanently deleted. Recover them through your file manager's trash UI — devklean does not own that trash and can't move items back. Run devklean history to see what was removed, or devklean restore for recovery instructions.
How do I exclude a folder?
Add its name to exclude (global or project .devklean.toml), or an absolute path to [ignore].paths. See Configuration.
Where are the logs?
~/.cache/devklean/logs/latest.log (see Logs).
How do I turn off colors?
Set NO_COLOR=1, pipe the output, or set theme = "mono" in config.
Something looks wrong with my history data.
Run devklean doctor to detect and remove corrupt metadata records.
Contributing
See CONTRIBUTING.md. Bug reports and PRs are welcome.
License
MIT — see LICENSE.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file devklean-1.0.2.tar.gz.
File metadata
- Download URL: devklean-1.0.2.tar.gz
- Upload date:
- Size: 50.9 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
fad3759fb46fd7e2b1006a87ee482fb4a3fd8c1531e72f6cd850eb9e46988a01
|
|
| MD5 |
673549e8449c19b97122a836aba04583
|
|
| BLAKE2b-256 |
0c797e7ddd276fc26c29fb507fe374bf0a8b5631edf3669697796e7f98203e0b
|
Provenance
The following attestation bundles were made for devklean-1.0.2.tar.gz:
Publisher:
release.yml on smurftyy/devklean
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
devklean-1.0.2.tar.gz -
Subject digest:
fad3759fb46fd7e2b1006a87ee482fb4a3fd8c1531e72f6cd850eb9e46988a01 - Sigstore transparency entry: 2028007671
- Sigstore integration time:
-
Permalink:
smurftyy/devklean@8c0f4c7cfa6edd45d11a4cae6bccba5c8550cf09 -
Branch / Tag:
refs/tags/v1.0.2 - Owner: https://github.com/smurftyy
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@8c0f4c7cfa6edd45d11a4cae6bccba5c8550cf09 -
Trigger Event:
push
-
Statement type:
File details
Details for the file devklean-1.0.2-py3-none-any.whl.
File metadata
- Download URL: devklean-1.0.2-py3-none-any.whl
- Upload date:
- Size: 45.2 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
b11656fee18b3c580e66750b618224bca154b688fa280a8e9ff468da7e25fb71
|
|
| MD5 |
66350a6c79fd7049c1631391eaf2cbd6
|
|
| BLAKE2b-256 |
cdfdfd7527bdc05feb4d785e2c193ec819411ec0d97f9429a215ece1492b8a2e
|
Provenance
The following attestation bundles were made for devklean-1.0.2-py3-none-any.whl:
Publisher:
release.yml on smurftyy/devklean
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
devklean-1.0.2-py3-none-any.whl -
Subject digest:
b11656fee18b3c580e66750b618224bca154b688fa280a8e9ff468da7e25fb71 - Sigstore transparency entry: 2028007789
- Sigstore integration time:
-
Permalink:
smurftyy/devklean@8c0f4c7cfa6edd45d11a4cae6bccba5c8550cf09 -
Branch / Tag:
refs/tags/v1.0.2 - Owner: https://github.com/smurftyy
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@8c0f4c7cfa6edd45d11a4cae6bccba5c8550cf09 -
Trigger Event:
push
-
Statement type: