Read-only MCP server for Apple Mail on macOS.
It gives an LLM or AI agent fast local access to Apple Mail metadata, message bodies, and attachment text without AppleScript and without IMAP/POP/EWS network calls.
Apple Mail already contains the data an agent needs, but it is buried in a local SQLite index and scattered .emlx files.
This project exposes that storage through a small, intent-driven MCP interface so an agent can:
- locate messages by subject, date range, sender, participant, mailbox, or account;
- fetch a full message including metadata, recipients, body, and attachment summary;
- extract readable text from supported attachments;
- operate completely locally and read-only.
In practice, it empowers AI workflows to search and read your mail archive safely and quickly, without relying on Mail.app automation like AppleScript (which damn slow and throws timeouts regularly) or network protocols.
The current tool set is intentionally compact:
| Tool | What it does |
|---|---|
search_messages |
Search by subject, dates, sender, participant, account, or mailbox |
list_accounts |
Discover account identifiers; set include_mailboxes=true for combined account+mailbox discovery |
get_message |
Read one message in full; recipients omitted by default (include_recipients=true to include) |
get_attachment_content |
Extract readable attachment content |
list_mailboxes |
List mailboxes/folders with message counts |
- macOS
- Apple Mail installed and synced at least once
- Rust toolchain (
rustup,cargo)
cargo build --releasecargo install --path .This installs the binary under the name rusty_apple_mail_mcp.
The server supports two operating modes:
In MCP mode, the server communicates via stdin/stdout using the Model Context Protocol. This is the primary mode for integration with AI agents, Claude Code, VS Code, and other MCP-compatible clients.
To start from source:
cargo run --releaseOr with the installed binary:
rusty_apple_mail_mcpFor interactive experimentation, use the MCP Inspector:
npx -y @modelcontextprotocol/inspector ./target/release/rusty_apple_mail_mcpIn CLI mode, you can run individual commands directly from the terminal. This is useful for:
- Scripting — automation of mail search tasks in shell scripts
- Debugging — quick testing without setting up an MCP client
- Integration — piping results to other command-line tools
- One-off queries — when you need a quick answer without starting a persistent server
# List all accounts
rusty_apple_mail_mcp list-accounts
rusty_apple_mail_mcp list-accounts --include-mailboxes
# List all mailboxes
rusty_apple_mail_mcp list-mailboxes
# Search messages
rusty_apple_mail_mcp search --subject-query "invoice"
rusty_apple_mail_mcp search --sender "john@example.com" --limit 10
rusty_apple_mail_mcp search --date-from "2024-01-01" --date-to "2024-12-31"
rusty_apple_mail_mcp search --mailbox "INBOX" --include-body-preview
# Get a specific message
rusty_apple_mail_mcp get-message --message-id "12345"
rusty_apple_mail_mcp get-message --message-id "12345" --body-format html
# Get attachment content
rusty_apple_mail_mcp get-attachment --message-id "12345" --attachment-id "12345:0"CLI mode supports the same configuration options as MCP mode:
| Option | Env Variable | Description |
|---|---|---|
--mail-directory |
APPLE_MAIL_DIR |
Mail data directory (default: ~/Library/Mail) |
--mail-version |
APPLE_MAIL_VERSION |
Envelope Index version (default: V10) |
--account |
APPLE_MAIL_ACCOUNT |
Account selector(s) |
Example:
rusty_apple_mail_mcp --mail-directory ~/Library/Mail --mail-version V10 search --subject-query "meeting"Or with environment variables:
export APPLE_MAIL_DIR="$HOME/Library/Mail"
export APPLE_MAIL_VERSION="V10"
rusty_apple_mail_mcp list-accounts| Feature | MCP Mode | CLI Mode |
|---|---|---|
| Protocol | stdin/stdout (MCP) | Direct command execution |
| Use case | AI agents, IDE integration | Scripting, debugging, one-off queries |
| Persistent process | Yes | No (per-command spawn) |
| Output format | JSON-RPC messages | JSON (human-readable) |
| Real-time streaming | Yes | No (batch output) |
| Error handling | MCP error codes | Exit codes + stderr |
Use MCP mode when:
- Integrating with Claude Code, VS Code, or other MCP clients
- Building AI-powered workflows that need to make multiple queries
- You need a persistent server process
- Your client already speaks MCP
Use CLI mode when:
- Writing shell scripts or automation
- Quick debugging and testing
- Piping results to other tools (
jq,grep, etc.) - Making single queries without overhead of starting a server
- Running from cron jobs or CI/CD pipelines
Example CLI pipeline:
# Find all messages from sender, extract subjects, save to file
rusty_apple_mail_mcp search --sender "boss@company.com" | \
jq -r '.messages[].subject' > ~/meeting-subjects.txt
# Count messages from last month
rusty_apple_mail_mcp search --date-from "2024-12-01" --date-to "2024-12-31" | \
jq '.messages | length'The server is configured only through environment variables:
| Variable | Required | Default | Description |
|---|---|---|---|
APPLE_MAIL_DIR |
no | ~/Library/Mail |
Root folder of the Mail data |
APPLE_MAIL_VERSION |
no | V10 |
Envelope Index version subdirectory |
APPLE_MAIL_ACCOUNT |
no | unset | Comma-separated account selectors such as Work Email or user@example.com; when set, the whole server is restricted to the resolved account(s) |
RUST_LOG |
no | unset | Standard Rust tracing filter used by tracing_subscriber; controls server logs written to stderr |
Example setup:
export APPLE_MAIL_DIR="$HOME/Library/Mail"
export APPLE_MAIL_VERSION="V10"
export APPLE_MAIL_ACCOUNT="Work Email"
export RUST_LOG="warn"The server reads RUST_LOG through tracing_subscriber::EnvFilter, so it accepts the usual Rust tracing filter syntax.
Common values:
error— only errorswarn— warnings and errorsinfo— startup and high-level operational logsdebug— includes per-request debug logstrace— very verbose tracingoff— disables logging
You can also scope logs per module/crate:
rusty_apple_mail_mcp=debugrusty_apple_mail_mcp=trace,rusqlite=warninfo,rmcp=warn
When RUST_LOG enables debug for this crate, search_messages logs timing breakdowns to stderr, including:
- total matched rows
- SQL query time
- metadata hydration time from SQLite
- body preview fallback time
- total request time
If APPLE_MAIL_ACCOUNT is set, the server resolves each selector through macOS ~/Library/Accounts/Accounts4.sqlite and then restricts all tools to the matched Mail account IDs.
- Matching is case-insensitive and trims whitespace.
- Supported selectors include human-friendly account names and email addresses.
- Startup fails fast if a selector matches zero accounts or multiple accounts.
list_accountsreturnsaccount_nameandemailwhen available, so valid selector values are discoverable from the MCP interface itself.
Example minimum .vscode/mcp.json configuration:
{
"servers": {
"mail_mcp": {
"command": "rusty_apple_mail_mcp",
"args": [],
"env": {
"APPLE_MAIL_DIR": "/Users/your-user/Library/Mail",
"APPLE_MAIL_VERSION": "V10",
"APPLE_MAIL_ACCOUNT": "Work Email",
"RUST_LOG": "warn"
}
}
}
}Typical usage pattern:
- Call
list_accounts(optionally withinclude_mailboxes=true) to discover accounts and mailboxes. - Use
search_messagesto build a shortlist of candidates. - Call
get_messageto fetch the full message you care about. - Use
get_attachment_contentwhen you need the text of a particular attachment.
The server is optimized to minimize token consumption:
- Compact tool descriptions — routing hints live in
ServerInfo.instructions(loaded once), not repeated per-tool on every request. - HTML → plain text — HTML email bodies are converted to clean text via DOM parsing (using
scraper), typically 10–20× smaller than raw HTML. status: "success"omitted — the status field only appears on error/not_found/partial responses.- Recipients omitted by default —
get_messageskips To/CC lists unlessinclude_recipients=true. - Compact dates — ISO 8601 without seconds (
2024-09-15T00:00Z). has_bodyremoved — always true for indexed messages; no longer wasting tokens.
{
"subject_query": "invoice",
"account": "imap://ACCOUNT-ID",
"limit": 10
}{
"message_id": "12345",
"include_body": true,
"include_attachments_summary": true,
"body_format": "text",
"include_recipients": false
}include_recipients defaults to false — set it to true when you need the To/CC lists (saves tokens on corporate mail with 50+ recipients).
Instead of calling list_accounts then list_mailboxes separately, use include_mailboxes=true:
{
"include_mailboxes": true
}This returns accounts with their mailboxes grouped, saving one round-trip.
The server draws from two local sources:
- Envelope Index – the SQLite database Apple Mail uses as a metadata index and relationship store
- .emlx files – the canonical bodies and attachments for individual messages
Search queries hit the lightweight index, keeping them fast; bodies and attachments are loaded on-demand from the .emlx files.
Handy commands while working on the codebase:
cargo test
cargo clippy --all-targets --all-features -- -D warnings
cargo doc --no-deps- macOS only
- read-only access only
- stdio transport only
- requires Apple Mail storage to be present on disk
- some binary attachment formats may yield metadata instead of extracted text
Need an AI agent to safely search and read Apple Mail on your Mac? This project provides a clean, read-only MCP layer over Apple Mail’s native storage — fast index-based searches, on‑demand body hydration from .emlx, and zero write access.