Inbox Cleanup

Email

Run a high-recall, multi-pass email inbox cleanup. Pattern-based subject queries catch 25x more archivable email than sender scans alone.

Install
assistant skills install inbox-cleanup

compatibility:Designed for Vellum personal assistants

Inbox Cleanup Skill

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


Phase 1: Preference Capture

Do this before touching anything. Ask the user:

1. Aggressiveness level

  • Conservative — newsletters with unsubscribe headers + obvious spam only
  • Standard — above + cold outreach heuristics (subject patterns, unknown senders)
  • Aggressive — above + anything from senders with no prior thread history

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:

  • Financial/billing alerts
  • Legal/contracts
  • Account suspension notices
  • Government/regulatory

Phase 2: Urgency Triage (do this before any archiving)

Scan the inbox first for high-stakes items that should be surfaced, not archived. Look for:

SignalWhy 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 domainCollections 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.


Phase 3: The Query Pipeline

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.

Pass 1: Age-based bulk (biggest catch)

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.

Pass 2: Personalized cold outreach (subject patterns)

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.

Pass 3: Generic cold outreach phrases

Search for subject lines containing:

  • "quick question", "quick note", "checking in"
  • "following up", "just following up", "circling back"
  • "would love to connect", "15 minutes", "quick call"
  • "intro call", "reaching out", "came across your"
  • "are you the right person", "happy to hop on"

Pass 4: No-reply & newsletters

Search for:

  • Messages from noreply/no-reply/donotreply sender addresses
  • Subject lines containing "unsubscribe", "newsletter", "weekly digest", "monthly digest"

Pass 5: Calendar noise

Search for subject lines containing:

  • "accepted:", "declined:", "tentative:"
  • "has accepted", "has declined", "invitation:"

Calendar response confirmations are pure noise. Safe to bulk archive without review.

Pass 6: Transactional/receipts

Search for subject lines containing:

  • "your order", "order confirmation", "your receipt"
  • "shipment", "has shipped", "delivered"

Cross-check against urgency triage first — filter out any "past due" or "final notice" items before archiving this batch.

Pass 7: Sketchy TLDs

Search for messages from sender domains ending in .shop, .biz, .xyz, .info, .club, .online.

Disproportionately spam. Safe to bulk archive.

Pass 8: High-volume repeat senders

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.


Phase 4: Cold Outreach Classification

For emails not caught by pattern queries, use LLM-based classification in Standard/Aggressive mode. Flag as cold outreach if 3+ signals are present:

  • Sender domain not in user's contact/thread history
  • No prior reply from user to this sender
  • Subject contains user's name + company together (personalization ≠ trust)
  • Body contains: "came across your company", "I help companies like", "reaching out because", "15 minutes", "quick call", "are you the right person"
  • Sender domain is a known outreach tool: apollo.io, outreach.io, lemlist.com, instantly.ai, salesloft.com
  • Email is not a reply (no Re: prefix, no quoted text from user in body)

Dry-Run Defaults

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:

  1. Run the pipeline with --dry-run on all archive calls
  2. At the end, show a summary: counts by phase with example subjects
  3. Ask the user to confirm before committing: "This would archive X,XXX emails across Y passes. Commit?"
  4. If confirmed, commit via bun run scripts/gmail-commit.ts commit --run-id "<run-id>"
  5. If rejected, cancel via 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).


Error Recovery & Resume

Archive operations are logged to an operation log for resumability. If a pass fails mid-run (rate limit, daily quota, OAuth expiry, crash):

  1. Check for interrupted runs before starting a new cleanup: bun run scripts/gmail-runs.ts list. If a recent run shows status: "interrupted", offer to resume it.
  2. Resume: bun run scripts/gmail-archive.ts archive --resume "<run-id>". This skips already-committed chunks and retries pending ones.
  3. Daily quota (403): The archive script detects daily quota exhaustion and writes an interrupted log entry with a resume hint. Do not retry until after midnight PT — offer to resume the run later.
  4. Rate limit (429): Handled automatically with exponential backoff (up to 5 retries for batch operations). No user intervention needed.

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").


Phase 5: Post-Cleanup

  1. Report totals — how many archived per pass, which categories, and the run_id for each pass
  2. Update blocklist — remember which senders/domains were archived; use for faster future passes
  3. Surface any urgents found — if financial/legal/suspension items surfaced during the pass, present them now with recommended actions
  4. Mention reversal — remind the user: "If any of these archives were wrong, I can reverse specific threads: bun run scripts/gmail-reverse.ts --run-id <id> --thread <message-id>"

Phase 6: Permanent Filter Setup

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.ts script 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.

Which patterns are safe as permanent 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:

PatternSafe as permanent filter?Notes
noreply / no-reply / donotreply sendersYesAutomated senders, never personal
Calendar responses (accepted/declined in subject)YesPure noise
Specific spam domains identified during cleanupYesDomain-level, not pattern-level
Sketchy TLDs (.shop, .biz, .xyz, .info)YesHigh spam signal, low false positive risk
Known newsletter senders confirmed during cleanupYesUser just explicitly confirmed unwanted
Generic phrases ("quick question", "checking in")RiskyReal colleagues use these — don't filter
Name/company subject patterns ("for [Name]", "[Company] -")NoToo broad — will catch real emails
Age-basedNoNot generally supported as a filter condition

Running auto-filter generation

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:

  1. Reads the cleanup run's op-log to extract archived patterns
  2. Derives filter candidates from safe categories only
  3. Fetches existing Gmail filters and skips duplicates
  4. Shows the user a confirmation dialog listing every filter that will be created, its criteria, and its label — the user must explicitly approve before any filter is created
  5. Creates one filter per logical category with an auto/* label (e.g. auto/no-reply, auto/calendar, auto/newsletter, auto/sketchy-tld)
  6. Logs all filter creations to the op-log for audit and reversal

Label strategy

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.

After filter creation

Tell the user:

  • How many filters were created and what each covers
  • How to find auto-archived emails (search by label, e.g. label:auto/no-reply)
  • How to remove a filter: bun run scripts/gmail-manage.ts filters --action delete --filter-id "<id>"

Reference: Proven Catch Rates

From a single cleanup session on a startup founder's inbox (April 2026):

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

CreatorVellum
LicenseMIT
Updated1 month ago
SecurityVerified
View on GitHub

The Personal AI you were promised

GET STARTED