SSH MCP Server
Provides LLM clients with safe, persistent SSH access to remote machines through the Model Context Protocol. Maintains shell sessions that preserve environment state between commands, enabling multi-step workflows and interactive diagnostics on remote systems.
README
SSH MCP Server
A Model Context Protocol (MCP) server that gives LLM clients safe, persistent SSH access to remote machines.
Table of Contents
- Overview
- Key Features
- Architecture
- Installation
- Running the Server
- Client Integrations
- Host Configuration
- Session Management
- Authentication Modes
- Timeouts & Inactivity Handling
- Directory Structure
- Using the MCP Tools
- Testing
- Troubleshooting
- Security Considerations
- Contributing
- License
Overview
ssh-mcp lets MCP-compatible clients (such as Claude Code, Cursor, or custom MCP inspectors) control remote machines through SSH. Once hosts are registered, the server maintains persistent shell sessions that retain environment state between commands—ideal for multi-step workflows, long-running processes, or interactive diagnostics.
The server is implemented in TypeScript on top of the official @modelcontextprotocol/sdk.
Key Features
- MCP-compliant: exposes functionality through standard MCP tool definitions.
- Persistent sessions: keep shell sessions alive, preserving working directory, environment variables, and process state across multiple commands.
- Stored host profiles: manage SSH targets through durable JSON configuration (
~/.ssh-mcp/hosts.json). - Flexible authentication: supports passwords, private keys, and SSH agent forwarding (fallback).
- Timeout & cleanup safeguards: sessions auto-close after prolonged inactivity; commands are marked and monitored for completion.
- Structured listings: query active sessions and saved hosts directly from the MCP client.
Architecture
MCP Client ─┬─> add-host / edit-host / remove-host
│
├─> list-hosts
│
├─> start-session ─┬─> PersistentSession (ssh2 shell)
│ ├─> exec (reuses shell, captures stdout/stderr)
│ └─> close-session / auto-timeout
│
└─> list-sessions
Persistent configuration → ~/.ssh-mcp/hosts.json
Each active session maintains:
- SSH connection via
ssh2 - Interactive shell (
conn.shell) to support multi-command pipelines - Buffered output with unique UUID markers to detect command completion
- Inactivity timer (defaults to 2 hours)
Installation
Install the published package from npm so the ssh-mcp-sessions executable is available in your environment.
Global install (preferred)
npm install -g ssh-mcp-sessions
Launch the server anywhere by running ssh-mcp-sessions.
Project-local install
Inside your project:
npm install ssh-mcp-sessions
Run the server with npx ssh-mcp-sessions (or reference ./node_modules/.bin/ssh-mcp-sessions).
Building from source is only required for contributing. See the Contributing section if you plan to work on the codebase itself.
Claude Desktop
Add an entry to ~/Library/Application Support/Claude/claude_desktop_config.json (macOS) or the appropriate config path on Windows/Linux:
{
"mcpServers": {
"ssh-mcp": {
"command": "npx",
"args": [
"ssh-mcp-sessions"
]
}
}
}
Restart Claude Desktop after saving the file. You can now use the MCP Inspector or the command palette (Cmd/Ctrl+Shift+O) to call tools like add-host and start-session.
Claude Code (VS Code extension)
Update the Claude Code workspace settings (.vscode/settings.json or global settings) with:
{
"claude.mcpServers": {
"ssh-mcp": {
"command": "npx",
"args": [
"ssh-mcp-sessions"
]
}
}
}
Reload the window. The MCP panel will list ssh-mcp, and commands are available via the command palette (Ctrl/Cmd+Shift+P → “Claude: Run MCP Tool”).
Codex (OpenAI GPT-4o/5 with MCP)
Create or edit ~/.config/openai-codex/mcp.toml (the path may differ per platform—use the location documented by the client). Add:
[mcpServers."ssh-mcp"]
command = "npx"
args = ["ssh-mcp-sessions"]
Restart Codex or re-open the MCP inspector. The ssh-mcp tools will appear under the configured servers list.
Cursor IDE
Open Cursor settings → “Model Context Protocol” (or edit ~/Library/Application Support/Cursor/mcp.json directly) and include:
{
"mcpServers": {
"ssh-mcp": {
"command": "npx",
"args": [
"ssh-mcp-sessions"
]
}
}
}
After saving, reload Cursor. The MCP sidebar exposes the server; you can invoke tools via chat or the command palette (Cmd/Ctrl+Shift+L).
Tip: If you prefer an explicit path instead of relying on
npx, replace the command/args with the absolute path to the executable (node_modules/.bin/ssh-mcp-sessionsfor local installs or/usr/local/lib/node_modules/ssh-mcp-sessions/build/index.jsfor global installs).
Running the Server
ssh-mcp-sessions
The server is purely stdio-based. Once running it prints:
SSH MCP Server running on stdio
You can register it with Claude Code or any other MCP client by pointing to the executable installed from npm:
{
"mcpServers": {
"ssh-mcp": {
"command": "npx",
"args": [
"ssh-mcp-sessions"
]
}
}
}
If you prefer to rely on PATH resolution (e.g., after a global install), you can simplify the entry to { "command": "ssh-mcp-sessions" }.
Note: The server no longer accepts CLI arguments for host/user/password. Everything is configured dynamically via MCP tools.
Host Configuration
Host Storage
- Hosts are persisted in
~/.ssh-mcp/hosts.json. - The directory is created automatically if it does not exist.
- File format:
{
"hosts": [
{
"id": "host",
"host": "host.local",
"port": 22,
"username": "user",
"password": "...", // optional
"keyPath": "~/.ssh/id_rsa" // optional
}
]
}
Fields:
id(string) — unique identifier used by all session commands.host(string) — hostname or IP.port(number, default 22) — SSH port.username(string) — SSH user.password(optional string) — password auth.keyPath(optional string) — private key path; tilde expansion supported.- If neither
passwordnorkeyPathis provided, the server attempts to use the local SSH agent viaSSH_AUTH_SOCK(with agent forwarding enabled).
- If neither
The MCP tools ensure this file remains well-formed; never edit it manually unless you know what you’re doing.
Adding Hosts
Tool: add-host
{
"host_id": "user@host.local",
"host": "host.local",
"port": 22,
"username": "user",
"password": "optional",
"keyPath": "optional"
}
host_id: new identifier. Must be unique.host: hostname or IP.port: optional (defaults to 22); provide integer > 0.username: SSH user.passwordorkeyPath: optional; configure one or rely on agent.
Example (Claude Code command palette or inspector):
/mcp mcp-remote-ssh add-host {"host_id":"host","host":"host.local","username":"user"}
Listing Hosts
Tool: list-hosts
Returns text with one host per line:
id=host host=host.local:22 user=user auth=agent
auth values:
password— password field presentkey— keyPath presentagent— neither password nor keyPath; agent fallback active
Editing Hosts
Tool: edit-host
{
"host_id": "user@host.local",
"port": 2222,
"password": "new-pass"
}
Only supply the properties you want to change. Omitted fields remain unchanged; providing null to a field is not supported—set an empty string or remove the host instead.
Removing Hosts
Tool: remove-host
{
"host_id": "user@host.local"
}
Deletes the entry from hosts.json. Active sessions using that host must be closed manually.
Session Management
Starting a Session
Tool: start-session
{
"host_id": "user@host.local"
}
Optionally you can supply sessionId; otherwise, a UUID is returned.
Example response:
Listing Sessions
Tool: list-sessions
Shows all active sessions with metadata:
session=fff4… host=host.local:22 user=user uptime=3m12s lastCommand=ls -la
Executing Commands
Tool: exec
{
"session_id": "fff4b34b-56dd-4711-9555-c04e8b64249b",
"command": "pwd"
}
- Commands are sanitized (trimmed, length-limited).
- Output is captured from the persistent shell and returned as plain text.
- Non-zero exit codes raise
McpErrorwith stderr in the message.
Example output:
/home/user
Closing Sessions
Tool: close-session
{
"sessionId": "fff4b34b-56dd-4711-9555-c04e8b64249b"
}
Note: session IDs for
close-sessionusesessionId(camelCase) to remain backwards compatible with the underlying tool definition.
Legacy Helper
Function execSshCommand(hostId, command, sessionId?) remains exported for programmatic use and simply delegates through the session machinery described above.
Authentication Modes
- Password — stored in
hosts.json; transmitted tossh2during connection. - Private key —
keyPathread at runtime; supports encrypted keys (prompt user to setSSH_MCP_KEY_PASSPHRASEbefore launch if needed). - SSH agent (fallback) — if neither password nor key is set and
SSH_AUTH_SOCKis present, the agent is passed tossh2(agentForward: true).
Timeouts & Inactivity Handling
- Each session has a global inactivity timeout (default 2 hours). Timer resets whenever a command executes successfully.
- If the timer elapses, the session cleans up the SSH connection, shell, and resolver buffer, and removes itself from
activeSessions. - Command completion uses a UUID marker:
printf '__MCP_DONE__{uuid}%d\n' $?. Output before the marker is returned; numeric code after the marker becomes the exit status.
Directory Structure
ssh-mcp/
├── build/ # Compiled JS output (npm run build)
├── src/index.ts # Primary MCP server implementation
├── test/ # Vitest tests (CLI-only; integration tests skipped)
├── package.json
├── README.md # This document
└── ~/.ssh-mcp/hosts.json # Created at runtime (per user)
Using the MCP Tools
Below is a typical workflow using Claude Code (commands start with /mcp), but the same JSON payloads apply to any MCP inspector.
-
Add host
/mcp mcp-remote-ssh add-host {"host_id":"host","host":"host.local","username":"user"} -
Start session
/mcp mcp-remote-ssh start-session {"host_id":"host"}→ returns
session_id -
Run commands
/mcp mcp-remote-ssh exec {"session_id":"<id>","command":"pwd"} /mcp mcp-remote-ssh exec {"session_id":"<id>","command":"ls -la"} -
Inspect
/mcp mcp-remote-ssh list-sessions /mcp mcp-remote-ssh list-hosts -
Close session
/mcp mcp-remote-ssh close-session {"sessionId":"<id>"}
Testing
Unit tests (Vitest):
npm run test
Integration smoke tests for SSH are not included by default because they require external infrastructure. You can manually validate with the workflow above.
Troubleshooting
| Symptom | Possible Cause | Suggested Action |
|---|---|---|
Host 'xyz' already exists |
Duplicate host_id |
Use edit-host or pick a new ID. |
Host 'xyz' not found |
Missing entry | Run list-hosts to confirm; add host again if needed. |
Error (code X): … |
Remote command returned non-zero | Inspect the command output. The session remains open. |
Session disappears from list-sessions |
Inactivity timeout reached | Start a new session or reduce idle periods. |
| Permission denied (publickey) | Missing credentials | Ensure keyPath or agent has the right key. |
Invalid key path |
keyPath resolved to undefined or missing file |
Provide an absolute/tilde path that exists. |
Security Considerations
- Treat
~/.ssh-mcp/hosts.jsonas sensitive; it may contain passwords or key paths. - Prefer key-based or agent authentication where possible.
- Limit
hosts.jsonpermissions:chmod 600 ~/.ssh-mcp/hosts.json. - Sessions inherit all privileges of the configured SSH user.
- Long-running sessions can be closed manually or rely on the inactivity timeout.
Contributing
- Fork the repo and create a branch.
- Make your changes with tests and documentation updates.
- Run
npm run buildandnpm run testbefore submitting a PR. - Follow the Code of Conduct.
Issues and feature requests are welcome via GitHub.
License
Happy automating! If this project improves your workflow, please star the repository or share feedback. Your contributions help make remote development safer and simpler for everyone.
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.