iMessage Max
Click on "Install Server".
Wait a few minutes for the server to deploy. Once ready, it will show a "Started" state.
In the chat, type
@followed by the MCP server name and your instructions, e.g., "@iMessage MaxWhat did Nick say about the dinner plans yesterday?"
That's it! The server will respond to your query, and you can continue using it as needed.
Here is a step-by-step guide with screenshots.
iMessage Max
A high-performance MCP (Model Context Protocol) server for iMessage that lets AI agents read, search, and send your messages with proper contact resolution.
Built in Swift for native macOS integration - single binary, no runtime dependencies.
Distribution Status
The project now ships a single Swift implementation:
GitHub releases
Homebrew
source builds
Codex plugin metadata
Claude Desktop MCPB metadata
The old Python package has been retired and removed from the repository.
Everything current lives under swift/.
Features
11 Intent-Aligned Tools - Work the way you naturally ask questions, not raw database queries
Contact Resolution - See names instead of phone numbers via macOS Contacts
Smart Image Handling - Efficient image variants (vision/thumb/full) to avoid token bloat
Session Grouping - Messages grouped into conversation sessions with gap detection
Attachment Tracking - Know which images are available locally vs offloaded to iCloud
Native Performance - Swift with raw SQLite3, Core Image GPU acceleration
Read-Only Safe - Only reads from chat.db, send requires explicit permission
Why This Exists
Most iMessage tools expose raw database structures, requiring 3-5 tool calls per user intent. This MCP provides intent-aligned tools:
"What did Contact A and I talk about yesterday?"
→ find_chat(participants=["Contact A"]) + get_messages(since="yesterday")
"Show me the exact details for this thread before I reply"
→ get_chat_details(chat_id="chat123")
"Show me photos from the group chat"
→ list_attachments(chat_id="chat123", type="image")
"Find where we discussed the launch timeline"
→ search(query="launch timeline")Common Agent Workflows
The tools work best when an agent uses them as short workflows instead of isolated one-off calls.
Agents should treat chat_id values like chat123 as internal handles for tool calls and exact sends. When explaining results to a person, use the returned chat name, group name, or participant-derived label instead of saying "Chat 123."
Find the right conversation, then read it
find_chat(participants=["Contact A"])
get_chat_details(chat_id="chat123")
get_messages(chat_id="chat123", since="yesterday", limit=50)Use this when the person matters more than the exact thread id.
Search first, then zoom in
search(query="launch timeline", limit=10)
get_context(message_id="msg_456", before=5, after=10)Use this when you know the topic but not where it was discussed.
Check what needs attention
get_unread()
get_active_conversations(hours=24, min_exchanges=2)Use this to surface unread threads and active conversations after a broad chat-list sweep.
Work with attachments safely
list_attachments(chat_id="chat123", type="image", since="30d")
get_attachment(attachment_id="att123", variant="vision")Use list_attachments to discover the message where files were shared first. It still returns exact attachment ids and local-availability state before you fetch a file.
Send with exact targeting when it matters
find_chat(participants=["Contact A", "Contact B"])
send(chat_id="chat456", text="Please use the latest draft")For sensitive sends, prefer resolving the exact chat first and then using chat_id so the message lands in the intended thread.
Installation
Homebrew (Recommended)
brew tap cyberpapiii/tap
brew install imessage-maxFrom Source
git clone https://github.com/cyberpapiii/imessage-max.git
cd imessage-max/swift
swift build -c release
# Binary is at .build/release/imessage-maxFor local development, advanced setup, and the signed install workflow, see:
Client Icon Metadata
iMessage Max ships icons for the main MCP protocol surface and the client packaging surfaces that use their own metadata:
MCP
2025-11-25initialize responses include PNGserverInfo.icons.Each tool advertises a compact PNG tool icon.
Codex plugin metadata lives in
.codex-plugin/plugin.jsonand usesassets/codex/icon.pngplusassets/codex/logo.png.Claude Desktop / MCPB metadata lives in
mcpb/manifest.jsonand uses PNG assets undermcpb/assets/.
The committed PNG source set is under assets/icons/ at 16x16, 32x32,
64x64, 128x128, 256x256, and 512x512.
Setup
1. Grant Full Disk Access
Required to read ~/Library/Messages/chat.db:
Open System Settings → Privacy & Security → Full Disk Access
Click + to add the binary
For Homebrew installs: The binary is at /opt/homebrew/Cellar/imessage-max/VERSION/bin/imessage-max (not the symlink at /opt/homebrew/bin/). Find it with:
# Open the folder containing the actual binary
open $(dirname $(readlink -f $(which imessage-max)))For source builds: Add .build/release/imessage-max from your clone directory.
Tip: In the file picker, press ⌘+Shift+G and paste the path to navigate directly.
2. Grant Contacts Access
Required for resolving phone numbers to names. The app will request access on first run, or add manually:
System Settings → Privacy & Security → Contacts → add imessage-max
3. Configure Your MCP Client
Add imessage-max to your MCP client's server configuration.
Many MCP clients use a JSON structure like this:
For Homebrew:
{
"mcpServers": {
"imessage": {
"command": "/opt/homebrew/Cellar/imessage-max/VERSION/bin/imessage-max"
}
}
}For source builds:
{
"mcpServers": {
"imessage": {
"command": "/path/to/imessage-max/swift/.build/release/imessage-max"
}
}
}If your client uses a different config format, point it at the same binary path.
4. Reconnect Your MCP Client
After saving the config, reconnect or restart your MCP client. The server should appear in the available tools, and you can verify the connection with diagnose.
Tools
find_chat
Find chats by participants, name, or recent content.
find_chat(participants=["Contact A"]) # Find a direct chat
find_chat(participants=["Contact A", "Contact B"]) # Find a group with both
find_chat(name="Project Group") # Find by chat name
find_chat(contains_recent="latest draft") # Find by recent contentget_chat_details
Inspect a known thread without opening the full conversation.
get_chat_details(chat_id="chat123") # Participants, handles, state, last message
get_chat_details(chat_id="chat123", include_shared_summary=false) # Skip recent shared summaryget_messages
Retrieve messages with flexible filtering. Returns metadata for media.
get_messages(chat_id="chat123", limit=50) # Recent messages
get_messages(chat_id="chat123", since="24h") # Last 24 hours
get_messages(chat_id="chat123", from_person="Contact A") # From specific personget_attachment
Retrieve image content by attachment ID with resolution variants.
get_attachment(attachment_id="att123") # Default: vision (1568px)
get_attachment(attachment_id="att123", variant="thumb") # Quick preview (400px)
get_attachment(attachment_id="att123", variant="full") # Original resolutionVariant | Resolution | Use Case | Token Cost |
| 1568px | AI analysis, OCR | ~1,600 tokens |
| 400px | Quick preview | ~200 tokens |
| Original | Maximum detail | Varies |
list_chats
Browse recent chats with previews.
list_chats(limit=20) # Recent chats
list_chats(is_group=True) # Only group chats
list_chats(since="7d") # Active in last weeksearch
Full-text search across messages.
search(query="draft") # Search all messages
search(query="budget", from_person="Contact A") # From specific person
search(query="launch", is_group=True) # Only in group chatsget_context
Get messages surrounding a specific message.
get_context(message_id="msg_123", before=5, after=10)get_active_conversations
Find chats with recent back-and-forth activity.
get_active_conversations(hours=24)
get_active_conversations(is_group=True, min_exchanges=3)list_attachments
Browse shared items grouped by message. Each row includes exact attachment ids for follow-up fetches.
list_attachments(type="image", since="7d")
list_attachments(chat_id="chat123", type="any")get_unread
Get unread threads or unread messages. Default is summary by chat.
get_unread() # Summary by chat for last 7 days
get_unread(since="24h") # Summary by chat for last 24 hours
get_unread(format="messages") # Row-level unread messagessend
Send a message or file attachment (requires Automation permission for Messages.app).
send(to="Contact A", text="Checking in")
send(chat_id="chat123", text="Please use the latest draft")
send(chat_id="chat123", file_paths=["/path/save-the-date.jpg"])
send(to="Contact A", file_paths=["/path/reference.png"], text="Sharing the file here")Rules:
Exactly one of
toorchat_idAt least one of
textorfile_pathsIf both are provided, files are sent first and text is sent last
Release Checks
For a lightweight pre-release routine, use:
Additional send note:
reply_tois currently unsupportedRisky sends, including group/file/long-message sends, require user confirmation. Clients with MCP elicitation support may be prompted inline; other clients should call
sendagain withconfirm: trueafter reviewing the destination and content.
Send result semantics:
status: "sent"means the message or attachment was confirmed successfullystatus: "pending_confirmation"means Messages accepted an attachment send, but it was not confirmed as finished within the polling windowstatus: "cancelled"means a confirmation prompt was declined/cancelled before Messages.app was invokedstatus: "failed"means the send failedstatus: "ambiguous"means the target could not be resolved safely
Notes:
pending_confirmationis a normal non-fatal attachment state, not the same as a hard failureexact chat sends target the existing conversation identified by
chat_idJSON-shaped tools return MCP
structuredContentas well as legacy text content for older clients
Examples:
{"status":"sent","success":true,...}means delivery was confirmed within the polling window{"status":"pending_confirmation","success":false,...}means Messages accepted the attachment, but the MCP could not yet confirm final completion
diagnose
Troubleshoot configuration and permission issues.
diagnose() # Returns: database status, contacts count, permissions, capabilitiesTroubleshooting
Contacts showing as phone numbers
Run diagnose to check status. If contacts_authorized is false:
Add the
imessage-maxbinary to System Settings → Privacy & Security → Contacts
"Database not found" error
Add the imessage-max binary to System Settings → Privacy & Security → Full Disk Access
Images show "attachment_offloaded" error
Some attachments are stored in iCloud, not on disk. list_attachments includes nested attachment summaries with available: true/false for each file. To download offloaded attachments, open the conversation in Messages.app.
MCP client not loading the server
Check config file syntax is valid JSON
Verify the binary path is correct
Reconnect or fully restart your MCP client
Architecture
┌─────────────────┐ ┌─────────────────┐ ┌─────────────────┐
│ MCP Client / │◄───►│ iMessage Max │◄───►│ chat.db │
│ Agent │ │ (Swift MCP) │ │ (SQLite) │
└─────────────────┘ └────────┬────────┘ └─────────────────┘
│
▼
┌─────────────────┐
│ Contacts.app │
│ (CNContactStore)│
└─────────────────┘Requirements
macOS 13+ (Ventura or later)
Full Disk Access permission
Contacts permission (for name resolution)
Automation permission for Messages.app (send only)
Advanced Setup
For HTTP mode, local background service setup, development commands, and contributor-focused workflow details, see:
License
MIT
This server cannot be installed
Maintenance
Resources
Unclaimed servers have limited discoverability.
Looking for Admin?
If you are the server author, to access and configure the admin panel.
Latest Blog Posts
MCP directory API
We provide all the information about MCP servers via our MCP API.
curl -X GET 'https://glama.ai/api/mcp/v1/servers/cyberpapiii/imessage-max'
If you have feedback or need assistance with the MCP directory API, please join our Discord server