Apple Mail MCP Server
A read-only MCP server that lets Claude Desktop interact with Apple Mail on macOS via AppleScript. It enables listing mailboxes, searching emails, and reading email content without making network calls.
README
Apple Mail MCP Server (READ ONLY)
A minimal, read-only MCP (Model Context Protocol) server that lets Claude Desktop interact with Apple Mail on macOS. Uses AppleScript via subprocess — no third-party email libraries, no network calls.
Version
Current: 1.2.1
Versioning follows Semantic Versioning:
- MAJOR — breaking changes to the tool API or behaviour
- MINOR — new tools or non-breaking feature additions
- PATCH — bug fixes and security hardening
What it can do
| Tool | Description |
|---|---|
mail_list_mailboxes |
List every account and mailbox configured in Apple Mail |
mail_search_emails |
Search emails by keyword across all mailboxes (uses Mail's native search index); optional account and mailbox_name filters for scoped searches |
mail_read_email |
Read the full content of a specific email by its opaque ID |
What it will never do
- Delete, trash, move, or archive any email
- Send, reply, forward, or compose any message
- Write any file to disk or export data
- Make network requests or external connections
- Access or decode email attachments
- Provide analytics or aggregate statistics
Prerequisites
- macOS (Apple Mail is macOS-only)
- Python 3.10 or later
- Apple Mail configured with at least one account
- Claude Desktop (or any MCP-compatible client)
Setup
1. Clone the repository
git clone https://github.com/androidua/apple-mail-mcp.git
cd apple-mail-mcp
2. Create a virtual environment and install dependencies
python3 -m venv venv
venv/bin/pip install -r requirements.txt
3. Verify the server starts cleanly
venv/bin/python apple_mail_mcp.py
Press Ctrl-C to stop. If no errors appear, the server is ready.
4. Grant macOS Automation permission
The first time the server runs, macOS will ask whether this process may control Apple Mail. Click OK. You can manage this later in:
System Settings → Privacy & Security → Automation
5. Configure Claude Desktop
Open (or create) ~/Library/Application Support/Claude/claude_desktop_config.json and add the block shown below under "mcpServers".
{
"mcpServers": {
"apple_mail": {
"command": "/path/to/apple-mail-mcp/venv/bin/python",
"args": [
"/path/to/apple-mail-mcp/apple_mail_mcp.py"
]
}
}
}
Replace /path/to/apple-mail-mcp with the absolute path to the directory where you cloned the repository (e.g. /Users/yourname/projects/apple-mail-mcp).
Restart Claude Desktop after saving the file.
Usage examples
Once connected, you can ask Claude things like:
- "List all my email mailboxes"
- "Search my emails for messages from Alice"
- "Find emails with 'invoice' in the subject, show me the top 5"
- "Read the email about the project kickoff" (after a search returns an ID)
Security notes
- No destructive operations. Every AppleScript is read-only.
- Input sanitisation. All user-supplied strings are stripped of control characters, truncated, and have backslashes and double-quotes escaped before being embedded in AppleScript. This prevents script-injection attacks.
- Local only. The server uses stdio transport and never opens a network socket.
- No credentials stored. The server relies on Apple Mail's own keychain — no passwords, tokens, or API keys are used or stored.
Performance
mail_search_emails uses AppleScript's whose clause — a declarative predicate evaluated by Mail's Objective-C runtime — to filter messages by subject and sender. This is fast even on accounts with hundreds of thousands of messages, and works correctly on macOS 26 / Mail 16 (which removed the older search <mailbox> for <keyword> AppleScript command). System mailboxes (Trash, Junk, Spam) are skipped by default.
To further scope a search, pass the optional account and/or mailbox_name parameters — e.g. restrict to account="Yahoo" and mailbox_name="INBOX" to avoid scanning all accounts.
Troubleshooting
| Symptom | Fix |
|---|---|
AppleScript failed … not allowed to send Apple events |
Go to System Settings → Privacy & Security → Automation and enable Mail for your Python process. |
No mailboxes found |
Open Apple Mail and ensure at least one account is signed in. |
| Tool times out | Use account and/or mailbox_name to scope the search, or reduce limit. |
Invalid email_id |
Always pass the email_id back exactly as returned by mail_search_emails. |
Project structure
apple-mail-mcp/
├── apple_mail_mcp.py # MCP server — single file, all tools
├── requirements.txt # Pinned dependencies
├── README.md # This file
└── venv/ # Local virtual environment (not committed)
Changelog
1.2.1 — 2026-03-25
- Fix (critical): remove
proc.stdout.close()/proc.stderr.close()—asyncio.StreamReaderhas no.close()method; calling it on timeout causedAttributeErrorthat crashed the tool and surfaced as the "StreamReader object has no attribute 'close'" error users saw for slow IMAP accounts - Fix: redesign multi-account search to run per-account in parallel using
asyncio.gather(return_exceptions=True)— a slow or offline account (Yahoo!, Hotmail, etc.) can no longer block or crash results from other accounts; each account gets an independent 45-second timeout - Fix: when a specific
accountis provided the original single-script path is preserved (60 s timeout); parallel path is used only for cross-account searches - UX: results now include a warning listing which accounts timed out, rather than crashing silently
1.2.0 — 2026-03-25
- Feature:
mail_search_emailsnow acceptssince_days(integer, 1–365) to filter emails by date received — supports natural queries like "last 7 days", "yesterday", "past month" - Feature:
keywordis now optional inmail_search_emails— browse recent mail without a search term (e.g.since_days=1returns today's mail) - Both filters are combinable:
keyword="invoice" + since_days=30returns invoice emails from the past month - Result headers and "no results" messages now reflect which filters were active
- Note: body content search is intentionally not supported — AppleScript's
whose content containsforces a full body download for every message, making it impractically slow on real mailboxes
1.1.4 — 2026-03-25
- Fix (regression): revert
_script_read_emailto proven account/mailbox iteration — direct AppleScript addressing (mailbox X of account Y) was unreliable for non-standard account types (Gmail, Exchange, shared accounts) - Fix (regression): revert
mail_search_emailsJSON output to flat array[...]— the{"results": [...]}wrapper introduced in v1.1.3 broke Claude AI's ability to extractemail_idvalues from results - Fix: improve AppleScript error categorisation — errors now return actionable messages (Mail not running, Automation permission denied, item not found) instead of a generic fallback; raw AppleScript error text is still logged internally
1.1.3 — 2026-03-25
- Fix (reliability): close asyncio pipe transports before
await proc.wait()on timeout — prevents file descriptor accumulation under repeated Mail.app timeouts - Fix (reliability): anchor
---BODY_START---split to a leading newline — prevents a subject line containing that exact string from corrupting header parsing inmail_read_email - Fix (security): parse search output fields from both ends of the delimiter-split record — a
\x1fbyte in a subject no longer shifts sender/date/is_read columns - Fix (security): extend
_CTRL_STRIP_REto cover C1 controls U+0080–U+009F (including U+0085 NEL which Python'ssplitlines()treats as a line terminator) - Fix (security): log raw
osascriptstderr internally; return a generic error string to callers instead of forwarding script fragments - Perf: replace nested account/mailbox iteration in
_script_read_emailwith direct AppleScript object addressing (mailbox X of account Y) — O(1) lookup instead of O(accounts × mailboxes) name scan - Docs: correct
whosedocstring — it is O(n) per mailbox, not indexed; lowlimitdoes not reduce scan cost - UX: search results now report a warning when rows were silently skipped due to parse errors
1.1.2 — 2026-03-09
- Fix: replaced backslash line-continuation characters (
\) in the_script_read_emailAppleScript template with sequential assignments — AppleScript uses¬for continuation, not\; the invalid characters caused allmail_read_emailcalls to fail with AppleScript syntax error -2741
1.1.1 — 2026-03-09
- Fix: replaced
search <mailbox> for <keyword>AppleScript command with awhoseclause filter — thesearchcommand was removed in Mail 16 (macOS 26) and caused allmail_search_emailscalls to fail with an AppleScript syntax error
1.1.0 — 2026-03-09
- Performance:
mail_search_emailsnow uses Apple Mail's native indexed search (search <mailbox> for <keyword>) instead of brute-force message iteration — dramatically faster on large mailboxes (e.g. Yahoo with 20+ years of email) - Feature: added optional
accountandmailbox_nameparameters tomail_search_emailsfor scoped searches (e.g. search only Yahoo / INBOX) - Default exclusion: Trash, Deleted Messages, Junk, Spam, Bulk Mail are
skipped automatically unless explicitly targeted via
mailbox_name
1.0.0 — 2026-03-09
- Initial release
- Tools:
mail_list_mailboxes,mail_search_emails,mail_read_email - Read-only, AppleScript-based, no network calls
- Input sanitisation against AppleScript injection
Recommended Servers
playwright-mcp
A Model Context Protocol server that enables LLMs to interact with web pages through structured accessibility snapshots without requiring vision models or screenshots.
Magic Component Platform (MCP)
An AI-powered tool that generates modern UI components from natural language descriptions, integrating with popular IDEs to streamline UI development workflow.
Audiense Insights MCP Server
Enables interaction with Audiense Insights accounts via the Model Context Protocol, facilitating the extraction and analysis of marketing insights and audience data including demographics, behavior, and influencer engagement.
VeyraX MCP
Single MCP tool to connect all your favorite tools: Gmail, Calendar and 40 more.
graphlit-mcp-server
The Model Context Protocol (MCP) Server enables integration between MCP clients and the Graphlit service. Ingest anything from Slack to Gmail to podcast feeds, in addition to web crawling, into a Graphlit project - and then retrieve relevant contents from the MCP client.
Kagi MCP Server
An MCP server that integrates Kagi search capabilities with Claude AI, enabling Claude to perform real-time web searches when answering questions that require up-to-date information.
E2B
Using MCP to run code via e2b.
Neon Database
MCP server for interacting with Neon Management API and databases
Exa Search
A Model Context Protocol (MCP) server lets AI assistants like Claude use the Exa AI Search API for web searches. This setup allows AI models to get real-time web information in a safe and controlled way.
Qdrant Server
This repository is an example of how to create a MCP server for Qdrant, a vector search engine.