Run a high-recall, multi-pass email inbox cleanup. Pattern-based subject queries catch 25x more archivable email than sender scans alone.
assistant skills install inbox-cleanupA playbook for large-scale email inbox cleanup. The core insight: sender-based scans are low-recall. Subject/body pattern queries catch 25x more archivable email. This skill is a multi-pass pipeline built around that insight.
Works with any connected email provider. Adapt query syntax to whatever the provider supports — the strategy (what to search for, how to decide what to archive) is universal.
Do this before touching anything. Ask the user:
1. Aggressiveness level
2. Age threshold Archive everything older than X days? Common choices: 30 / 60 / 90 days. Or no age filter.
3. VIP senders to protect Ask: "Are there any senders that might look like cold outreach but you actually care about? Think: specific individuals at investors, advisors, your lawyer, accountant, recruiters you're actively working with."
Build an explicit keep list. Do not archive anything matching it, ever, regardless of aggressiveness.
4. Categories to confirm before archiving These need a sample + explicit approval before bulk action:
Scan the inbox first for high-stakes items that should be surfaced, not archived. Look for:
| Signal | Why it matters |
|---|---|
| "past due", "overdue", "final notice", "balance due" | Outstanding invoice — financial consequence |
| "will be suspended", "account suspension", "service interruption" | Service shutoff — operational consequence |
| "collections", "case #", "recovery" in sender domain | Collections agency — credit/legal consequence |
| "signature required", "agreement", "DocuSign pending" | Legal action needed |
| Government TLDs (.gov), "IRS", "state of", "department of" | Regulatory — can't be skipped |
Surface these to the user before running the cleanup. They're easy to miss buried in a big inbox.
Run these passes in order. Each pass should paginate to exhaustion (keep fetching while more results exist). After each pass, show the user a count + 5 sample subjects before archiving anything.
Search for all inbox messages older than the user's age threshold (e.g. 30 days). Typically 50–80% of the archivable backlog. Always show a sample before bulk archiving.
Note on result caps: Some providers cap query results (e.g. ~5,000). If a query returns exactly at the cap, archive that batch and re-run the same query — the next batch will surface. Repeat until it returns fewer than the cap.
Ask the user for their first name and company name, then search for subject lines containing patterns like:
[FirstName] -, [FirstName],, for [FirstName], hi [FirstName], hey [FirstName], [FirstName] |
[CompanyName] -, [CompanyName]?, for [CompanyName], re: [CompanyName], [CompanyName] AI
These are the highest-recall patterns for cold outreach and partnership spam. A startup founder's inbox will see the biggest wins here.
Search for subject lines containing:
Search for:
Search for subject lines containing:
Calendar response confirmations are pure noise. Safe to bulk archive without review.
Search for subject lines containing:
Cross-check against urgency triage first — filter out any "past due" or "final notice" items before archiving this batch.
Search for messages from sender domains ending in .shop, .biz, .xyz, .info, .club, .online.
Disproportionately spam. Safe to bulk archive.
After the above passes, run a sender frequency count on what remains. Any sender with 3+ emails not on the keep list is a candidate for bulk archive. Show grouped list to user for approval.
For emails not caught by pattern queries, use LLM-based classification in Standard/Aggressive mode. Flag as cold outreach if 3+ signals are present:
apollo.io, outreach.io, lemlist.com, instantly.ai, salesloft.comRe: prefix, no quoted text from user in body)When the user's inbox management trust stage is 0 (flag-only), or when a batch exceeds 1,000 operations at stage 1, default to --dry-run mode:
--dry-run on all archive callsbun run scripts/gmail-commit.ts commit --run-id "<run-id>"bun run scripts/gmail-commit.ts cancel --run-id "<run-id>"At stage 2 or for small batches at stage 1, archive directly (but still log for audit/reversal).
Archive operations are logged to an operation log for resumability. If a pass fails mid-run (rate limit, daily quota, OAuth expiry, crash):
bun run scripts/gmail-runs.ts list. If a recent run shows status: "interrupted", offer to resume it.bun run scripts/gmail-archive.ts archive --resume "<run-id>". This skips already-committed chunks and retries pending ones.interrupted log entry with a resume hint. Do not retry until after midnight PT — offer to resume the run later.All archive outputs now include a run_id. Pass --run-id to group multiple passes under one run, and --phase to label the pipeline phase (e.g. --phase "noise_archive").
run_id for each passbun run scripts/gmail-reverse.ts --run-id <id> --thread <message-id>"After cleanup, propose Gmail filters so the same categories don't re-accumulate. This bridges cleanup (drain backlog once) and inbox-management (keep inbox clean on schedule).
Note: Filter creation capabilities vary by provider. The
gmail-auto-filters.tsscript handles Gmail. If the provider doesn't support programmatic filter creation, give the user manual instructions instead.
Filters are permanent behavior changes. Unlike a one-time archive, a filter silently skips the inbox for every future matching email. A wrong filter means the user misses emails they were expecting — with no indication anything happened. Always confirm with the user before creating filters.
One-time bulk archiving and permanent auto-archiving are different risk levels. The auto-filter script only derives candidates from patterns marked "Yes" below:
| Pattern | Safe as permanent filter? | Notes |
|---|---|---|
| noreply / no-reply / donotreply senders | Yes | Automated senders, never personal |
| Calendar responses (accepted/declined in subject) | Yes | Pure noise |
| Specific spam domains identified during cleanup | Yes | Domain-level, not pattern-level |
| Sketchy TLDs (.shop, .biz, .xyz, .info) | Yes | High spam signal, low false positive risk |
| Known newsletter senders confirmed during cleanup | Yes | User just explicitly confirmed unwanted |
| Generic phrases ("quick question", "checking in") | Risky | Real colleagues use these — don't filter |
| Name/company subject patterns ("for [Name]", "[Company] -") | No | Too broad — will catch real emails |
| Age-based | No | Not generally supported as a filter condition |
After the cleanup pipeline completes (Phase 5 post-cleanup report), invoke:
# Preview: show what filters would be created (no confirmation prompt)
bun run scripts/gmail-auto-filters.ts preview --run-id "<cleanup-run-id>"
# Generate: show plan, confirm with user, then create
bun run scripts/gmail-auto-filters.ts generate --run-id "<cleanup-run-id>"
If --run-id is omitted, the script finds the most recent completed cleanup run automatically.
The script:
auto/* label (e.g. auto/no-reply, auto/calendar, auto/newsletter, auto/sketchy-tld)Every auto-filter applies an auto/* label instead of silently archiving. This gives the user an audit trail — search label:auto/calendar to see what was caught. Labels are created automatically if they don't exist.
Tell the user:
label:auto/no-reply)bun run scripts/gmail-manage.ts filters --action delete --filter-id "<id>"From a single cleanup session on a startup founder's inbox (April 2026):
| Pass | Approx. catch |
|---|---|
| Older than 30 days | ~7,200 |
| Name-personalized subject patterns | ~35,000 |
| Company-name subject patterns | ~50,000 |
| Sketchy TLDs (.shop/.biz/.xyz) | ~3,741 |
| Newsletters/digests | ~1,014 |
| Calendar responses | ~142 |
| Generic cold outreach phrases | ~23 |
| Completed DocuSigns | ~34 |
Total: ~90,000+ emails in one session. The name/company pattern passes alone accounted for ~85k. This is why patterns dominate sender scans.