A community-driven registry for Claude, Cursor, Windsurf, Cline & more. Not affiliated with Anthropic.
Are you the author? Sign in to claim
Apple Mail MCP server — 47 tools with SQLite-powered millisecond search across 250K+ emails, .emlx parser, batch operati
The most comprehensive Apple Mail MCP server - 48 tools with SQLite-powered millisecond search across 250K+ emails.
| Feature | Other MCPs | che-apple-mail-mcp |
|---|---|---|
| Total Tools | ~20 | 47 |
| Language | Python | Swift (Native) |
| Search Speed | Seconds (AppleScript) | Milliseconds (SQLite) |
| Search Fields | Subject/Sender | Subject/Sender/Recipient/Date |
| Batch Operations | No | Up to 50 emails per call |
| Mailbox Management | Basic | Full CRUD |
| Email Colors | No | 7 flag colors + background |
| VIP Management | No | Yes |
| Rule Management | Partial | Full CRUD |
| Signatures | No | Yes |
| Raw Headers/Source | No | Yes |
# Clone and build
git clone https://github.com/kiki830621/che-apple-mail-mcp.git
cd che-apple-mail-mcp
swift build -c release
# Copy to ~/bin and add to Claude Code
# --scope user : available across all projects (stored in ~/.claude.json)
# --transport stdio: local binary execution via stdin/stdout
# -- : separator between claude options and the command
mkdir -p ~/bin
cp .build/release/CheAppleMailMCP ~/bin/
claude mcp add --scope user --transport stdio che-apple-mail-mcp -- ~/bin/CheAppleMailMCP
💡 Tip: Always install the binary to a local directory like
~/bin/. Avoid placing it in cloud-synced folders (Dropbox, iCloud, OneDrive) as file sync operations can cause MCP connection timeouts.
Then grant Automation permission in System Settings > Privacy & Security > Automation.
For full details see CHANGELOG.md.
attachmentFragment cluster + fallback parityattachmentFragment indent across all 3 callers + removed dead MailController.attachmentScript helper that bypassed v2.7.0's race-mitigation delays (#61, #62)CHE_MAIL_ATTACHMENT_DELAY_BETWEEN / _TRAILING (#63, #64)get_email_metadata SQLite path now falls back to AppleScript on error — last read-tool gap closed; all 8 SQLite-first read tools now have parity fallback (#71).partial.emlx + observabilityData index, causing html_body to begin with "sion: 1.0\n\n<base64>" for some Android Gmail messages — raw base64 leaked into LLM context and triggered AUP false-positives downstream (#72)save_attachment now reads from Attachments/<rowId>/<part_id>/<filename> cache when .partial.emlx body is empty — no more silent 0-byte writes for IMAP messages with stripped binaries (#66)SQLite ... fast path failed for rowId=...; falling through to AppleScript) (#69)forward_email plain mode now embeds RFC 3676 > quoted original (parity with reply_email's #43 fix) (#44)bool / [String] no longer silently coerced (#35)@) (#41)cc_additional deduplicates case-insensitively (#34)~/.ssh, Keychains, TCC db, browser cookies) + symlink-resolved + new MAIL_MCP_ATTACHMENT_ROOTS env allow-list (#38)id as Int at handler boundary — defeats AppleScript predicate injection (#50)reply_email runtime (#37, #45) + smoke matrix templates (#46, #47)format parametercompose_email / create_draft / reply_email / forward_email) gain format: "plain" | "markdown" | "html" param (closes #14, #15)message-composition capability spec| Tool | Description |
|---|---|
list_accounts | List all mail accounts |
get_account_info | Get account details |
| Tool | Description |
|---|---|
list_mailboxes | List all mailboxes (folders) |
create_mailbox | Create a new mailbox |
delete_mailbox | Delete a mailbox |
get_special_mailboxes | Get special mailbox names (inbox, drafts, sent, trash, junk, outbox) |
| Tool | Description |
|---|---|
list_emails | List emails in a mailbox |
get_email | Get full email content |
search_emails | Search by subject/content |
get_unread_count | Get unread count |
get_email_headers | Get all email headers |
get_email_source | Get raw email source |
get_email_metadata | Get metadata (forwarded, replied, size) |
| Tool | Description |
|---|---|
mark_read | Mark as read/unread |
flag_email | Flag/unflag email |
set_flag_color | Set flag color (7 colors) |
set_background_color | Set email background color |
mark_as_junk | Mark as junk/not junk |
move_email | Move to another mailbox |
copy_email | Copy to another mailbox |
delete_email | Delete email (to trash) |
| Tool | Description |
|---|---|
compose_email | Send new email (supports cc/bcc/attachments; format: plain/markdown/html; optional from_address for multi-account sender selection — see #131) |
reply_email | Reply to email. Optional: cc_additional, attachments, save_as_draft, format (since v2.4.0). Plain mode embeds RFC 3676 > quoted original (since v2.5.0 / #43) |
forward_email | Forward email. Optional body + format. Plain mode embeds RFC 3676 > quoted original (since v2.5.0+ / #44) |
redirect_email | Redirect email (keeps original sender) |
open_mailto | Open mailto URL |
Reply to a thread, add extra CC, attach files, and save as a draft for human review before sending:
reply_email(
id="<message id from search_emails>",
mailbox="INBOX",
account_name="iCloud",
body="Reply text",
cc_additional=["x@y.com"],
attachments=["/path/to/file.pdf"],
save_as_draft=true
)
| Tool | Description |
|---|---|
list_drafts | List draft emails |
create_draft | Create a draft (supports attachments; optional from_address for multi-account sender selection — see #131) |
| Tool | Description |
|---|---|
list_attachments | List email attachments |
save_attachment | Save attachment to disk |
| Tool | Description |
|---|---|
list_vip_senders | List VIP senders |
| Tool | Description |
|---|---|
list_rules | List mail rules |
get_rule_details | Get rule details |
create_rule | Create a new rule |
delete_rule | Delete a rule |
enable_rule | Enable/disable a rule |
| Tool | Description |
|---|---|
list_signatures | List email signatures |
get_signature | Get signature content |
| Tool | Description |
|---|---|
list_smtp_servers | List SMTP servers |
| Tool | Description |
|---|---|
check_for_new_mail | Check for new mail |
synchronize_account | Sync IMAP account |
| Tool | Description |
|---|---|
extract_name_from_address | Extract name from email address |
extract_address | Extract email from full address |
get_mail_app_info | Get Mail.app info |
import_mailbox | Import mailbox from file |
search_emails / list_emailsBoth tools return an envelope object { results, returned, limit, truncated } — not a bare array (changed in v2.14.0, #204). Read the matches from .results:
| Field | Meaning |
|---|---|
results | Array of result objects (per-object fields unchanged from the pre-envelope shape). search_emails objects carry id, subject, sender, date_received, account_name, mailbox, to, plus account_id when the account UUID is resolvable. list_emails objects carry id, subject, sender. |
returned | Number of objects in results |
limit | Effective limit applied to the query |
truncated | true when more results are available than were returned — raise limit or narrow the query to retrieve the rest (definitive on the SQLite fast path; a best-effort heuristic on the AppleScript fallback — see below) |
truncated is definitive on the SQLite fast path (it fetches limit + 1 internally); on the AppleScript fallback it is a best-effort returned == limit heuristic. Any "enumerate → batch process" consumer should check truncated before assuming it has the full set.
git clone https://github.com/kiki830621/che-apple-mail-mcp.git
cd che-apple-mail-mcp
swift build -c release
Edit ~/Library/Application Support/Claude/claude_desktop_config.json:
{
"mcpServers": {
"che-apple-mail-mcp": {
"command": "/full/path/to/che-apple-mail-mcp/.build/release/CheAppleMailMCP"
}
}
}
# Copy to ~/bin and register (user scope = available in all projects)
mkdir -p ~/bin
cp .build/release/CheAppleMailMCP ~/bin/
claude mcp add --scope user --transport stdio che-apple-mail-mcp -- ~/bin/CheAppleMailMCP
open "x-apple.systempreferences:com.apple.preference.security?Privacy_Automation"
# For Claude Desktop
osascript -e 'quit app "Claude"' && sleep 2 && open -a "Claude"
# For Claude Code - start a new session
claude
"List all my mail accounts"
"Show unread emails in Gmail inbox"
"Search for emails about 'quarterly report'"
"Send an email to john@example.com about the meeting"
"Flag important emails in red"
"Create a rule to move newsletters to a folder"
"Use list_accounts to show my accounts"
"Use search_emails to find emails containing 'invoice'"
"Use set_flag_color to mark email ID 12345 as blue"
"Use check_for_new_mail to refresh"
set_flag_color)| Index | Color |
|---|---|
| 0 | Red |
| 1 | Orange |
| 2 | Yellow |
| 3 | Green |
| 4 | Blue |
| 5 | Purple |
| 6 | Gray |
| -1 | Clear |
set_background_color)blue, gray, green, none, orange, purple, red, yellow
Most read tools prefer Apple Mail's local Envelope Index (SQLite) and on-disk .emlx message files over AppleScript IPC, with transparent AppleScript fallback when the SQLite path can't satisfy a request:
| Tool | SQLite/.emlx path | AppleScript fallback |
|---|---|---|
get_email | ✓ | ✓ on any error |
get_emails_batch | ✓ (per item) | ✓ (per item) |
get_email_headers | ✓ | ✓ on any error |
get_email_source | ✓ | ✓ on any error |
search_emails | ✓ | ✓ when reader unavailable |
list_attachments | ✓ | ✓ on any error |
save_attachment | ✓ | ✓ on any error |
get_email_metadata | ✓ | ✓ on any error (since #71) |
For save_attachment's read path the fast path is 10–100× faster than AppleScript (per #12 measurements). Other tools' speedup ratios depend on request shape; in general, large bulk reads see the biggest gain.
The fast path requires:
~/Library/Mail/V10/....emlx storageExchange (EWS) accounts in Apple Mail do not materialize .emlx files — message bodies live on the server and are fetched on demand. For these accounts, all 8 read tools (including get_email_metadata since #71) transparently degrade to AppleScript IPC (which is correct but slower). Symptoms:
When the fast path fails for a non-EWS account, the failure is logged to stderr (since #69). Run the binary in a terminal and watch stderr to distinguish:
EnvelopeIndexReader init failed: ... — DB unreachable (commonly: Full Disk Access missing)SQLite get_email fast path failed for rowId=N: ... — per-message failure (e.g., .partial.emlx only, malformed MIME, file not yet synced)Both cases transparently fall through to AppleScript with ... falling through to AppleScript in the log line, so behavior is preserved while observability is restored.
| Problem | Solution |
|---|---|
| Server disconnected | Rebuild with swift build -c release |
| Not allowed to send Apple events | Add permissions in System Settings > Automation |
| Mail.app not responding | Ensure Mail.app is running with configured accounts |
| Commands timing out | Large mailboxes take longer; try specific searches |
| Bulk fetch slower than expected | Watch stderr for ... falling through to AppleScript lines. EWS/Exchange accounts always fall back (see Performance & Storage); other accounts logging fallback indicate a fixable .emlx issue |
save_attachment fails with -1728 "Can't get account" or -1719 "Invalid mailbox index" | Since #173 both errors come back with an actionable hint naming the failing reference (account / mailbox / message). Common causes: two Mail.app accounts share the same display_name, or an email-form account_name maps to several accounts — see Account Disambiguation below. |
Mail.app's AppleScript account "<display_name>" selector is not unique when two accounts share the same display_name — a common pattern when an iCloud catch-all alias forwards a Gmail address back to itself, or when Google Workspace + personal Gmail overlap. Any AppleScript-routed tool (save_attachment fallback, get_email, mark_read, etc.) will then non-deterministically pick the wrong account → -1728 / -1719 errors.
The fix: pass account_id (Mail.app's globally-unique UUID) alongside account_name. When provided, save_attachment uses Mail.app's account id "<UUID>" selector instead, bypassing the ambiguity:
// Tool call: save_attachment with account_id
{
"id": "273214",
"mailbox": "[Gmail]/全部郵件",
"account_name": "alice@example.com",
"account_id": "C38E0583-47F8-4468-BE70-43155C15549D", // ← disambiguates
"attachment_name": "report.pdf",
"save_path": "/tmp/report.pdf"
}
Discovering account_id:
search_emails results — each object in the results array (a SearchResult) carries an account_id field alongside account_name (populated by decoding the account UUID from the SQLite mailboxes.url authority via MailboxURL.decode — Mail.app's storage convention encodes the account UUID in the mailbox URL authority; there is no direct SELECT mailboxes.account_id). Recommended: pass it through directly.~/Library/Mail/V10/MailData/Signatures/AccountsMap.plist. The top-level keys are the UUIDs; the AccountURL value contains the matching email address percent-encoded in the authority.tell application "Mail" to get id of every account returns the UUID list.Backward compatibility: account_id is optional. When omitted (or empty), tools fall back to the legacy account "<display_name>" path — behavior identical to pre-#101 — with one save_attachment exception (#173): when account_name contains @ (email-shaped, the form SQLite-path tools like search_emails emit), save_attachment first reverse-looks-it-up in AccountsMap and silently upgrades to the account id "<UUID>" selector (the upgrade is logged to stderr). Exactly one match → that UUID; several accounts behind one address (iCloud catch-all + Gmail) → an actionable error listing every candidate instead of a raw -1728; no match → the legacy display-name path, unchanged. Edge: a Mail account whose description legitimately contains @ and happens to equal another account's email now resolves in the email namespace first — pass account_id explicitly to pin the selector. Other tools keep the strict pre-#101 fallback (the cross-tool sweep is #176).
Scope: account_id is accepted across the AppleScript-routed tools that reference mail by account. It began with save_attachment (#101); the #104 sweep then added the 13 single-message / movement / relay / mailbox tools below:
save_attachment (#101) — the precursormark_read, flag_email, set_flag_color, set_background_color, mark_as_junkmove_email, copy_email, delete_emailreply_email, forward_email, redirect_emailcreate_mailbox, delete_mailboxThe surface has since expanded beyond the #104 set:
resolveAccountIdForTool chokepoint across all 14 AppleScript-routed write handlers (so an email-form account_name resolves to the UUID selector, not just an accepted account_id).account_id through the read-tool AppleScript fallbacks (list_emails / search_emails / get_email / headers / source / metadata / attachments / get_unread_count) via resolveMailboxRef / resolveMsgRef (the PR-E that was previously deferred is now done).get_special_mailboxes accepts account_id / account_name for per-account special-mailbox real names.check_for_new_mail and synchronize_account gained the account_id escape hatch (synchronize_account accepts account_id alone).Still not covered by account_id (tracked): get_account_info / list_mailboxes (#202).
compose_email / create_draft do not exhibit the display_name-collision defect — they make new outgoing message rather than referencing existing mail by account, so they never emit an account "<display_name>" selector. Multi-account sender selection is now available via the optional from_address parameter (#131) — pass any one of your configured Mail.app email addresses ("alice@example.com" or RFC 5322 form "Alice <alice@example.com>") to set the sender of the outgoing message; omit to use Mail.app's default account. Use list_accounts to discover the addresses configured on the running Mac.
Cross-account move/copy is not supported via account_id (#129 — from #127 verify). move_email and copy_email accept a single account_id, which is threaded through both the source msgRef and the destination mailboxRef. The architectural choice is correct (movement stays within one account, because Mail.app's AppleScript verb move msg to <mailboxRef> requires the destination mailbox to be expressed relative to a single account context). Mail.app's UI permits cross-account move via drag-and-drop, but the AppleScript-routed move_email / copy_email tools cannot replicate that — calling move_email with account_id of one account while expecting the destination to_mailbox to be resolved against a different account silently picks the wrong account's mailbox of that name (if both accounts happen to have one) or raises -1719 "Invalid mailbox index". If you need a copy of the message's contents under a different account, you can manually rebuild it via save_attachment + compose_email — note this is not a true move/copy: original metadata (Message-ID, received-date, flags, labels) and message identity are not preserved.
.emlx file parser, with AppleScript fallback for EWS / unparseable .emlxNSAppleScriptContributions are welcome! Please feel free to submit a Pull Request.
MIT License - see LICENSE for details.
Created by Che Cheng (@kiki830621)
If you find this useful, please consider giving it a star!
Run Claude Code as an MCP server so any agent can delegate coding tasks to it
MCP server integration for DaVinci Resolve Studio
mcp-language-server gives MCP enabled clients access semantic tools like get definition, references, rename, and diagnos
Browser automation using accessibility snapshots instead of screenshots