plugins/claude-ops/skills/ops-inbox/SKILL.md
Full inbox management across all channels — WhatsApp (wacli), Email (Gmail MCP), Slack (MCP), Telegram (user-auth MCP), Discord (webhook + REST read), Notion (MCP — comments, mentions, assigned tasks). Scans FULL inbox (not just unread), identifies messages needing replies, archives handled conversations.
npx skillsauth add davepoon/buildwithclaude ops-inboxInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Before executing, load available context:
Preferences: Read ${CLAUDE_PLUGIN_DATA_DIR:-$HOME/.claude/plugins/data/ops-ops-marketplace}/preferences.json
default_channels — which channels to scan by defaultsecrets_manager / doppler — how to resolve channel credentials if not in envDaemon health: Read ${CLAUDE_PLUGIN_DATA_DIR}/daemon-health.json
wacli-sync status — if not running or auth needed, skip WhatsApp and surface the issue~/.wacli/.health for live auth status before any wacli commandOps memories: Check ${CLAUDE_PLUGIN_DATA_DIR}/memories/ before drafting any reply:
contact_*.md — load profile for the contact you're about to reply topreferences.md — apply user's communication style and language preferencestopics_active.md — check for active threads or deadlines related to this contactdonts.md — never violate these restrictions in draftsHealth file — check ~/.wacli/.health BEFORE any wacli command:
status=connected → proceed normallystatus=needs_auth → prompt user: "Run wacli auth in terminal, scan QR"status=needs_reauth → prompt user: "WhatsApp session expired. Run wacli auth to re-pair"wacli doctor --json| Command | Usage | Output |
|---------|-------|--------|
| wacli doctor --json | Check auth/connected/lock/FTS | {data: {authenticated, connected, lock_held, fts_enabled}} |
| wacli chats list --json | All chats | {data: [{JID, Name, Kind, LastMessageTS}]} |
| wacli messages list --chat "<JID>" --limit N --json | Messages for chat | {data: {messages: [{FromMe, Text, Timestamp, SenderName, ChatName}]}} |
| wacli messages search --query "<text>" --json | FTS search | Same as above |
| wacli contacts --search "<name>" --json | Contact lookup | Contact objects |
| wacli send --to "<JID>" --message "<msg>" | Send text | Success/error |
| wacli history backfill --chat="<JID>" --count=50 --requests=2 --wait=30s --idle-exit=5s --json | Fetch older messages | Backfill result |
| Command | Usage | Output |
|---------|-------|--------|
| gog gmail search "in:inbox" --max 50 -j --results-only --no-input | Full inbox scan | JSON array of threads |
| gog gmail thread get <threadId> -j | Get full thread with all messages | Full message JSON |
| gog gmail get <messageId> -j | Get single message | Message JSON |
| gog gmail archive <messageId> ... --no-input --force | Archive messages (remove from inbox) | Archive result |
| gog gmail archive --query "<gmail-query>" --max N --force | Archive by query | Archive result |
| gog gmail send --to "<email>" --subject "<subj>" --body "<body>" | Send email | Send result |
| gog gmail send --reply-to-message-id <msgId> --reply-all --body "text" | Reply all | Send result |
| gog gmail mark-read <messageId> ... --no-input | Mark as read | Result |
| gog gmail labels list -j | List all labels | Labels JSON |
If CLAUDE_CODE_EXPERIMENTAL_AGENT_TEAMS=1 is set, use Agent Teams when processing "all channels" mode. This enables:
Team setup (only when flag is enabled, "all channels" mode):
TeamCreate("inbox-channels")
Agent(team_name="inbox-channels", name="whatsapp-scanner", ...)
Agent(team_name="inbox-channels", name="email-scanner", ...)
Agent(team_name="inbox-channels", name="slack-scanner", ...)
Agent(team_name="inbox-channels", name="telegram-scanner", ...)
Agent(team_name="inbox-channels", name="notion-scanner", ...)
Each agent scans its channel and reports back classified results. You then process NEEDS_REPLY items across all channels in priority order.
If the flag is NOT set, process channels sequentially or use fire-and-forget subagents.
${CLAUDE_PLUGIN_ROOT}/../../bin/ops-unread 2>/dev/null || echo '{}'
All channel credentials come from env vars or CLI auth — no hardcoded secrets.
| Variable | Default | Purpose |
| ------------------- | ----------- | ---------------------------------------------------- |
| GMAIL_ACCOUNT | auto-detect | Gmail account for gog CLI |
| SLACK_MCP_ENABLED | false | Set true when Slack MCP server is configured |
| TELEGRAM_ENABLED | false | Set true when Telegram user-auth MCP is configured |
| NOTION_MCP_ENABLED| false | Set true when Notion MCP integration is configured |
| WACLI_STORE | ~/.wacli | wacli store directory |
Do NOT just check unread. Scan the FULL recent inbox for each channel and classify every conversation:
CRITICAL SAFETY RULE — NEVER SEND WITHOUT UNDERSTANDING: Before drafting or sending ANY reply on ANY channel, you MUST have read the FULL conversation history (20+ messages) and PROVEN you understand it by summarizing:
Failure mode this prevents: An agent reads only the last message "je kan het toch uit Klaviyo halen?" and replies "Welke data heb je nodig?" — completely wrong because the contact was telling the user to pull data themselves (they have 2FA), not asking for data. Without the full thread, the reply was nonsensical and confused the contact.
Hard rule: if you cannot summarize the conversation arc in 2 sentences, you have not read enough messages. Go back and read more.
The user does NOT remember every thread. For EVERY message you present, you MUST build full context BEFORE showing it. Never show just a subject line and ask "what do you want to do?" — the user needs to understand what it's about first.
For every NEEDS REPLY item, gather this context automatically:
gog gmail thread get / wacli messages list --limit 20), not just the last message. Summarize the full conversation arc.gog gmail search "from:<contact_email>" --max 10 — recent email historywacli contacts --search "<name>" --json — WhatsApp presencewacli messages search --query "<name>" --json --limit 5 — recent WhatsApp mentionsgog gmail search "subject:<keywords>" --max 5 — related email threadswacli messages search --query "<topic keywords>" --json --limit 5 — related WA messages~/.claude/plugins/data/ops-ops-marketplace/memories/ for any stored context about this contact or topicWhen presenting a NEEDS REPLY item:
━━━ [Contact Name] — [Subject] ━━━
Who: [role, company, relationship — from contact search]
History: [last 3 interactions across channels]
Thread: [2-3 sentence summary of full conversation arc]
Last msg: [full body of their last message]
Context: [related threads/decisions/deadlines found]
Draft reply: "[contextually aware draft based on all above]"
[Send] [Edit] [Read full thread] [Skip]
When drafting replies:
Use the full thread history to maintain conversation continuity
Reference specific points from their message
Match the contact's communication style (formal/casual, language)
If ops-memories has preferences for this contact, apply them
Never generate a generic reply — every draft must show you read the full thread
NEEDS REPLY — other party sent last message, awaiting your response
WAITING — you sent last message, waiting for them (no action needed)
HANDLED — conversation concluded, can be archived
FYI — newsletters, notifications, automated messages (bulk archive)
For each channel, detect availability at runtime:
gog CLI first. If gog unavailable, try mcp__gog__gmail_* MCP tools. If neither, report unavailable.~/.wacli/.health for keepalive daemon status. If status=needs_auth or status=needs_reauth, do NOT attempt wacli commands — instead prompt the user: "WhatsApp needs re-authentication. Run wacli auth in a separate terminal and scan the QR code, then type 'done'." Use AskUserQuestion: [Done — re-paired], [Skip WhatsApp]. On Done, restart the daemon: launchctl kickstart -k gui/$(id -u)/com.claude-ops.wacli-keepalive, wait 5s, re-check health. If no health file exists, fall back to wacli doctor for auth/connection status. If outdated (405 error), advise rebuilding from source.mcp__claude_ai_Slack__*). Check SLACK_MCP_ENABLED env var.TELEGRAM_ENABLED env var. Never use BotFather bots.${CLAUDE_PLUGIN_ROOT}/bin/ops-discord read <CHANNEL_ID> --limit 20 --json. Requires DISCORD_BOT_TOKEN (v1 is channel-scoped — no DM/gateway support yet). Pre-configured read list lives at ${CLAUDE_PLUGIN_DATA_DIR}/preferences.json under discord.inbox_channels (array of channel IDs). If neither a bot token nor a read list is configured, skip Discord with a one-line note ("Discord not configured — run /ops:setup discord") rather than prompting — ops-inbox is not a setup flow. Rule 3 still applies to /ops:setup.mcp__claude_ai_Notion__* or self-hosted Notion MCP). Check NOTION_MCP_ENABLED env var. Searches workspace for recent comments, mentions, and assigned tasks.Parse pre-gathered data for initial counts (unread is just a starting signal).
For each channel, run a FULL scan (not just unread):
in:inbox (not is:unread) via gog gmail search -a $GMAIL_ACCOUNT -j --results-only --no-input --max 30 "in:inbox". For each thread, read the last message to determine who sent it last. Check for DRAFT or SENT labels. Before suggesting to send a draft, verify no reply was already sent in the thread.wacli chats list --json to get all chats. Filter to non-archived chats with LastMessageTS in the last 7 days. For each, fetch the FULL conversation via wacli messages list --chat <JID> --limit 20 --json (20 messages, not 5 — you need the full thread). Parse data.messages[] with fields FromMe, Text, Timestamp, ChatName. Understand which messages are from the user (FromMe: true) vs the contact (FromMe: false). Classify by last message FromMe field.Display the full inbox:
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
OPS ► INBOX MANAGER
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
📱 WhatsApp [N need reply] | [N waiting] | [N archive]
📧 Email [N need reply] | [N waiting] | [N FYI]
💬 Slack [N need reply] | [N waiting]
✈️ Telegram [N need reply] | [N waiting]
──────────────────────────────────────────────────────
Use batched AskUserQuestion calls (max 4 options each). Only show channels that are configured and have messages. If <=4 total options, use a single call.
AskUserQuestion call 1:
[All channels (fastest — one pass)]
[WhatsApp only]
[Email only]
[More...]
AskUserQuestion call 2 (only if "More..."):
[Slack only]
[Telegram only]
[Skip — already done]
If only 3 channels are configured, "All channels" + 3 channel options = 4, fits in one call. Then process the selected channel(s).
Phase 1 — Classify:
wacli chats list --jsonLastMessageTS in the last 7 dayswacli messages list --chat "<JID>" --limit 20 --json — get 20 messages, NOT 5. You need the full conversation thread to understand context.data.messages[] — fields: FromMe, Text, Timestamp, ChatName, SenderNameFromMe: true (user sent) vs FromMe: false (contact sent)FromMe: false (they sent last)FromMe: true (you sent last)Phase 2 — Build context for NEEDS REPLY chats (run in parallel): For each NEEDS REPLY chat:
wacli messages search --query "<contact_name>" --json --limit 10 — mentions in other chatsgog gmail search -j --results-only --no-input --max 5 "from:<name> OR to:<name>" — email history~/.claude/plugins/data/ops-ops-marketplace/memories/contact_*.md for stored profilewacli messages search --query "<topic keywords>" --json --limit 5 — related WA messagesgog gmail search -j --results-only --no-input --max 3 "<topic keywords>" — related emailsFromMe: true messages in this chat, note: language (NL/EN), formality, emoji usage, typical response lengthPhase 3 — Present with full context:
📱 WHATSAPP — NEEDS REPLY (with context)
━━━ 1. [Contact Name] ━━━
Who: [role, company, relationship — from contact search]
History: [last 3 interactions across channels]
Conversation: [2-3 sentence summary of the full chat thread]
Their message: [full text of their last message(s)]
Your last msg: [what you said before they replied]
Context: [related threads/topics found]
Language: [NL/EN — match the user's previous messages in this chat]
Draft reply: "[context-aware draft matching user's style + language]"
[Send] [Edit] [Read full thread] [More...]
If "More...":
[Archive] [Skip]
📱 WHATSAPP — WAITING (no action needed)
N. [Contact] — you said: "[your last message]" — [time ago]
Thread: [1-line summary of what you're waiting for]
Use AskUserQuestion for each NEEDS REPLY chat.
When drafting WhatsApp replies:
Reply via: wacli send --to "<JID>" --message "<msg>"
wacli CLI reference (v0.5.0):
| Command | Usage | Notes |
|---------|-------|-------|
| wacli doctor | wacli doctor --json | Check auth/connected/lock/FTS status |
| wacli auth | wacli auth | QR pairing (interactive — shows QR in terminal) |
| wacli sync | wacli sync --follow --refresh-contacts --refresh-groups | Persistent sync (managed by launchd keepalive) |
| wacli sync --once | wacli sync --once --idle-exit=10s | One-shot sync, exits when idle |
| wacli chats list | wacli chats list --json | All chats with JID, Name, Kind, LastMessageTS |
| wacli messages list | wacli messages list --chat "<JID>" --limit 5 --json | Messages: ChatJID, FromMe, Text, Timestamp, SenderName |
| wacli messages search | wacli messages search --query "<text>" --json | FTS5 search across all messages |
| wacli contacts | wacli contacts --search "<name>" --json | Contact lookup by name |
| wacli send | wacli send --to "<JID>" --message "<msg>" | Send text message |
| wacli history backfill | wacli history backfill --chat="<JID>" --count=50 --requests=2 --wait=30s --idle-exit=5s --json | Fetch older messages from phone (needs store lock) |
Health file contract (~/.wacli/.health):
Before ANY wacli command, read ~/.wacli/.health:
status=connected → proceed normallystatus=needs_auth → prompt user: "Run wacli auth in terminal, scan QR"status=needs_reauth → prompt user: "WhatsApp session expired. Run wacli auth to re-pair"wacli doctor --jsonRequesting backfill for @lid chats with empty messages:
The keepalive daemon holds the store lock, so you can't run backfill directly. Instead:
~/.wacli/.backfill_jids (one per line)launchctl kickstart -k gui/$(id -u)/com.claude-ops.wacli-keepalivewacli troubleshooting:
@lid JIDs (linked device format) may return empty messages → request backfill via the daemon (see above)cd /tmp && git clone https://github.com/Lifecycle-Innovations-Limited/wacli.git && cd wacli && go build -o /usr/local/bin/wacli ./cmd/wacli/launchctl bootout gui/$(id -u)/com.claude-ops.wacli-keepaliveneeds_auth to health file; prompt user for QR scanneeds_reauth; user needs wacli auth re-pairPhase 1 — Classify:
in:inbox (NOT is:unread) via gog gmail search -a $GMAIL_ACCOUNT -j --results-only --no-input --max 30 "in:inbox"gog gmail thread get -a $GMAIL_ACCOUNT <threadId> -j — read ALL messages, not just the last oneFrom header and labelIds (SENT, DRAFT)Phase 2 — Build context for NEEDS REPLY items (run in parallel): For each NEEDS REPLY thread, gather:
gog gmail search -j --results-only --no-input --max 10 "from:<sender_email>" — their recent emails to youwacli contacts --search "<sender_name>" --json — WhatsApp contactwacli messages search --query "<sender_name>" --json --limit 5 — recent WhatsApp mentionsgog gmail search -j --results-only --no-input --max 5 "subject:<keywords>" — related threadsPhase 3 — Present with full context:
📧 EMAIL — NEEDS REPLY (with context)
━━━ 1. [Sender] — [Subject] ━━━
Who: [sender's role, company — from contact search]
History: [last 3 email exchanges with this person]
Thread summary: [2-3 sentences covering the full conversation arc]
Their message: [full body of their last message — NOT truncated]
Related: [any related threads or pending decisions found]
Draft reply: "[context-aware draft using full thread + contact history]"
[Send draft] [Edit draft] [Read full thread] [More...]
If "More...":
[Archive] [Skip]
📧 EMAIL — DRAFTS (unsent)
N. [Recipient] — [Subject] (draft ready to send)
📧 EMAIL — FYI / ARCHIVE
N. [Sender] — [Subject] (newsletter/notification)
For each NEEDS REPLY:
a) Read full thread + draft reply
b) Archive (no reply needed)
c) Skip
For FYI section:
x) Archive all FYI at once
Use AskUserQuestion for each NEEDS REPLY email with options [Read + Reply] / [Archive] / [Skip].
When replying, draft the reply and use AskUserQuestion to confirm:
Reply to [Sender] — [Subject]:
"[drafted reply]"
[Send] [Edit] [Skip]
For FYI bulk archive, use AskUserQuestion:
Archive N FYI/newsletter emails?
[list of subjects]
[Archive all N] [Review each] [Skip]
Draft replies via gog gmail send. Archive via gog gmail archive <messageId> ... --no-input --force.
Use Slack MCP tools with query: "in:*" (NOT is:unread — scan full recent activity, not just unread) for mentions.
For each result, show channel, sender, preview. Read thread for context.
a) Read thread
b) Reply
c) Mark read / skip
Telegram integration must authenticate as the user's personal account (user-auth via tdlib/MTProto), NOT a BotFather bot. The goal is to manage real conversations just like WhatsApp via wacli.
Use the Telegram user-auth MCP server if available.
✈️ TELEGRAM — NEEDS REPLY
1. [Contact] — [preview] — [time ago]
a) Read thread + reply
b) Archive
c) Skip
If no Telegram user-auth tool is available, report: "Telegram not configured — needs user-auth MCP server (tdlib/MTProto)".
Notion serves as a knowledge base and task management channel. Unlike messaging channels, Notion "inbox" items are:
Phase 1 — Discover and scan:
mcp__claude_ai_Notion__notion-search:
query: "" (empty string returns recent pages) or topic-specific termsfilter: {"property": "object", "value": "page"} to limit to pages (not databases)last_edited_time descending to surface recent activitymcp__claude_ai_Notion__notion-fetch with the page URL/IDmcp__claude_ai_Notion__notion-get-comments with the page ID — scan comment authors and timestamps to determine which need repliesPhase 2 — Classify:
For each page with comments or mentions:
Phase 3 — Present with context:
📓 NOTION — NEEDS REPLY
━━━ 1. [Page Title] — [Database Name] ━━━
Page: [page URL]
Comment by: [commenter name] — [time ago]
Comment: "[full comment text]"
Page context: [2-3 sentence summary of the page content]
Draft reply: "[context-aware reply to the comment]"
[Reply] [View page] [Skip] [More...]
If "More...":
[Mark resolved] [Archive]
📓 NOTION — ASSIGNED TASKS
N. [Task title] — [database] — Status: [status] — Due: [date]
Context: [1-line summary]
📓 NOTION — RECENTLY UPDATED (FYI)
N. [Page title] — updated by [person] — [time ago]
Use AskUserQuestion for each NEEDS REPLY item.
When replying to Notion comments:
mcp__claude_ai_Notion__notion-create-comment with the page ID and reply textWhen updating tasks:
mcp__claude_ai_Notion__notion-update-page to change status, add notesAPI fallback (when MCP is down):
If Notion MCP tools fail or are unavailable but NOTION_API_KEY is set, fall back to direct API:
curl -s -H "Authorization: Bearer $NOTION_API_KEY" -H "Notion-Version: 2022-06-28" \
-H "Content-Type: application/json" \
-X POST https://api.notion.com/v1/search \
-d '{"sort":{"direction":"descending","timestamp":"last_edited_time"},"page_size":10}'
If NOTION_MCP_ENABLED is not set or Notion MCP tools are unavailable, report: "Notion not configured — set NOTION_MCP_ENABLED=true and add Notion integration via claude.ai or self-hosted MCP".
Discord v1 support is channel-scoped (webhook send + REST read). DM + gateway are deferred to a v2 issue.
${CLAUDE_PLUGIN_DATA_DIR}/preferences.json → discord.inbox_channels[]. If empty and DISCORD_GUILD_ID is set, fall back to bin/ops-discord channels --json (list the guild's text channels and let the user pick via AskUserQuestion, ≤4 per Rule 1 — paginate with [More...]).${CLAUDE_PLUGIN_ROOT}/bin/ops-discord read "<CHANNEL_ID>" --limit 20 --json
<@user-id>) or is a direct question.send path documented in skills/ops-comms/SKILL.md → Discord send.If bin/ops-discord exits 1 with {"error":"no discord credential configured — run /ops:setup discord"}, print a single-line note and continue to the next channel — do not prompt inside the inbox flow.
💬 DISCORD — activity (last 7d)
#channel-name [N messages] | [M need reply]
After all selected channels are processed, print:
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
INBOX ZERO ✓ — [timestamp]
Processed: [N] messages | Replied: [N] | Archived: [N]
━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━
If $ARGUMENTS specifies a channel (e.g. whatsapp), skip the menu and go directly to that channel.
Use TaskCreate for each channel being processed. Update with TaskUpdate as messages are replied/archived/skipped. Gives the user a live inbox-zero progress bar.
After processing, offer to schedule recurring inbox checks via AskUserQuestion:
[Schedule inbox check every 2 hours] [Schedule morning + evening] [No schedule]
Use CronCreate if selected. Show existing schedules with CronList.
tools
Assesses the current state of the startup project and recommends what to focus on next. Use when there is a need or a question from the user to understand what the next steps are or what to focus on next.
data-ai
Use at the start of any conversation about a startup idea, product validation, founder strategy, or work inside a `startup/` workspace. Establishes file conventions, voice-input handling, subagent dispatch rules, and how to update each artifact safely. Activate before invoking any other startup-superpowers skill.
tools
Manages the founder's survey-based validation — crafting the right questions, deploying a survey to the internet, and analyzing results against hypotheses. Use when the founder wants to run a survey, create survey questions, validate hypotheses at scale, check how a survey is going, understand whether a survey is the right tool right now, or deploy a question set to get quantitative signal. Also bring this up if you believe that creating a survey to collect quantitative evidence may be useful at this point.
development
Guides the founder through designing and optionally building the simplest MVP or prototype that validates their current hypotheses. Use when the founder wants to build something to test assumptions, discusses what to build next, wants to interpret results from a live MVP, or is deciding whether the current approach is still right. Also use when a founder proposes something to build — the skill will check whether the proposed form is the simplest thing that generates honest signal.