mcp-signal
Local Signal MCP server: read via Signal Desktop, send via signal-cli
README
mcp-signal
A local Model Context Protocol (MCP) server that reads Signal Desktop history from the local encrypted database via
signal-exportand sends outbound messages viasignal-cli.
mcp-signal focuses on the core workflow for personal Signal automation — list chats, read messages, search messages, inspect groups, and send messages to direct or group chats. Everything runs locally; stdio transport with no network listener.
Heads up — mixed backend. Read/search comes from the local Signal Desktop database. Sending uses
signal-cli, which must be installed and linked to a Signal account separately. Ifsignal-cliis unavailable, read/search still works but send tools do not.
Features
- List direct and group chats from Signal Desktop
- Read recent messages from a chat
- Search messages within one chat or across all chats
- List group chats with
signal-cligroup IDs for outbound use - Send a message to:
- a direct recipient by phone number
- a group by group ID
- a chat by exact chat name (with ambiguity checks)
- Runs entirely on your machine; stdio transport with no network listener
Setup
Prerequisites
- Python 3.13+
- uv
- Signal Desktop with an existing local message database
signal-cliinstalled and linked if you want outbound sends
Installation
-
Clone this repository
git clone https://github.com/Sealjay/mcp-signal.git cd mcp-signal -
Install dependencies
uv sync -
Install
signal-cli(optional — only needed for outbound sends)On macOS, the simplest route is Homebrew:
brew install signal-cli
Configure outbound sends
The server auto-loads a local .env.local file from the repo root if present. This file is gitignored and is the recommended place for machine-local config.
cat > .env.local <<'EOF'
SIGNAL_ACCOUNT="+441234567890"
EOF
Optional environment variables:
| Variable | Purpose |
|---|---|
SIGNAL_CLI_PATH |
Override the signal-cli binary path |
SIGNAL_DATA_DIR |
Override the Signal Desktop data directory |
SIGNAL_DB_PASSWORD |
Password for encrypted desktop DBs if needed |
SIGNAL_DB_KEY |
Raw key for encrypted desktop DBs if needed |
Environment variables set in the shell take precedence over .env.local.
Link signal-cli (first run only)
mcp-signal does not manage linking itself. Link the local signal-cli device first:
signal-cli link -n "signal-mcp"
Scan the QR code in the Signal mobile app (Settings → Linked Devices → Link New Device).
Do not pass -a / --account to link on current signal-cli versions — linking a new secondary device does not take a phone number there.
After the QR is accepted, confirm the linked account is visible:
signal-cli listAccounts
That account should match the SIGNAL_ACCOUNT value in .env.local.
signal-cli stores its linked-account state under its own local data directory (typically ~/.local/share/signal-cli/data on macOS/Linux). That state lives outside this repository and is not committed by mcp-signal.
Verify everything is connected:
uv run signal-mcp smoke
MCP client configuration
All clients launch the server the same way over stdio. On macOS, you may need the absolute path to uv — see macOS: uv PATH below.
Claude Code
The quickest route is the CLI:
claude mcp add --transport stdio signal --scope user -- uv run --directory /absolute/path/to/mcp-signal signal-mcp serve
Alternatively, add to .mcp.json at your project root (or ~/.claude.json for a user-scoped server):
{
"mcpServers": {
"signal": {
"type": "stdio",
"command": "uv",
"args": ["run", "--directory", "/absolute/path/to/mcp-signal", "signal-mcp", "serve"]
}
}
}
If you edit the file directly, restart the Claude Code session to pick it up.
Claude Desktop
Add to ~/Library/Application Support/Claude/claude_desktop_config.json (macOS):
{
"mcpServers": {
"signal": {
"command": "uv",
"args": ["run", "--directory", "/absolute/path/to/mcp-signal", "signal-mcp", "serve"]
}
}
}
Restart Claude Desktop. You should see signal listed as an available integration.
Cursor
Add to ~/.cursor/mcp.json:
{
"mcpServers": {
"signal": {
"command": "uv",
"args": ["run", "--directory", "/absolute/path/to/mcp-signal", "signal-mcp", "serve"]
}
}
}
Restart Cursor.
macOS: uv PATH
GUI apps (Claude Desktop, Cursor) don't always inherit the PATH from your interactive terminal, so uv may fail with spawn uv ENOENT. Fix by using the absolute path to uv in command:
- Homebrew —
/opt/homebrew/bin/uv(Apple Silicon) or/usr/local/bin/uv(Intel) - Manual install — run
which uvin your terminal to find it
Example:
{
"mcpServers": {
"signal": {
"command": "/opt/homebrew/bin/uv",
"args": ["run", "--directory", "/absolute/path/to/mcp-signal", "signal-mcp", "serve"]
}
}
}
Architecture
| Component | Description |
|---|---|
| MCP server | Python/FastMCP, stdio transport |
| Read path | signal-export reading the local Signal Desktop database |
| Send path | signal-cli JSON-RPC launched on demand |
| State | No separate cache; reads directly from Signal Desktop data |
Data flow
- The MCP client launches
signal-mcp serveover stdio. - Read/search tools call
signal-exportagainst the local Signal Desktop database. - Group listing and outbound sends call
signal-cli -a ACCOUNT jsonRpc. - Results are returned as structured JSON.
Project structure
mcp-signal/
src/mcp_signal/
config.py
main.py
reader.py
server.py
signal_cli.py
tests/
CLAUDE.md
LICENSE
README.md
SECURITY.md
Tools
| Tool | Purpose |
|---|---|
list_chats |
List direct and group chats from Signal Desktop |
read_messages |
Read messages from a specific chat |
search_messages |
Search messages within one chat or across all chats |
list_groups |
List groups from signal-cli, including group IDs |
send_message |
Send a text message to a direct recipient or group |
get_status |
Show desktop DB / signal-cli / account readiness |
Privacy and security
- No cloud relay. No network listener. All data stays on your machine.
- Read/search uses your local Signal Desktop data only.
- Send operations require a locally configured
signal-cliaccount. .env.localis intended for local secrets such asSIGNAL_ACCOUNTand is not committed.signal-clilinked-device state is stored in its own local app data directory, outside this repo, and is not committed.
See SECURITY.md for how to report vulnerabilities.
Limitations
- Prompt-injection risk: as with many MCP servers, this one is subject to the lethal trifecta. Malicious incoming messages could attempt to instruct an agent to exfiltrate other messages. Treat the tool surface accordingly and review outbound actions before approving them.
- Mixed backend: chat history comes from Signal Desktop, while outbound sends come from
signal-cli. - No attachments: text-only send.
- No real-time notifications: polling/read only.
- Single account per MCP instance.
- Group sends need
signal-cli: local DB reads alone do not provide enough information to send to groups safely.
Development
uv sync
uv run signal-mcp smoke
uv run pytest
uv run ruff check .
Troubleshooting
signal-clinot found — confirmsignal-cliis onPATHor setSIGNAL_CLI_PATHin.env.local. On macOS,brew install signal-cliis the simplest route.- Read/search works but sends fail —
signal-cliis not linked orSIGNAL_ACCOUNTis not set. Runsignal-cli listAccountsto verify, then check.env.local. signal-cli linkhangs or fails — do not pass-a/--accounttolinkon current versions. Runsignal-cli link -n "signal-mcp"and scan the QR from your phone.- MCP client can't launch the server —
argsmust contain an absolute path to the repo, not relative. Ifuvitself fails withspawn uv ENOENT, see macOS:uvPATH. - No messages returned — confirm Signal Desktop is installed and has message history. The read path queries the local Signal Desktop database directly.
Contributing
Contributions welcome via pull request. Please:
- Run
uv run ruff check .before pushing. - Ensure
uv run pytestpasses.
See CLAUDE.md for the full development workflow.
Licence
MIT Licence — see LICENSE.
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.