Turn a Markdown file into a local MCP server, with zero dependencies.
Project description
mdMCP
Turn a Markdown file into a local MCP server, with zero dependencies.
pip install mdmcp
Quickstart
1. Write a tool file:
## Tool: health_check
Check if a URL is reachable.
### Parameters
- url (string, required): URL to check
### Execute
\```bash
STATUS=$(curl -sf -o /dev/null -w '%{http_code}' -m 5 "$MDMCP_URL")
echo "HTTP $STATUS"
\```
2. Serve it:
mdmcp serve tools.md
3. Connect to Claude Code:
{
"mcpServers": {
"my-tools": {
"command": "mdmcp",
"args": ["serve", "tools.md"]
}
}
}
That's it. Your Markdown is now an MCP server.
How it works
## Tool: namedefines a tool### Parametersdefines input schema (envvar-based, no string interpolation)### Executecontains a bash script inside a fenced code block- Parameters are passed as
$MDMCP_<NAME>environment variables
Commands
mdmcp list tools.md # List available tools
mdmcp run tools.md tool_name '{}' # Run a tool once
mdmcp serve tools.md # Start MCP stdio server
Security
- Minimal env: only PATH, HOME, USER, SHELL, LANG, TERM are inherited. API keys and tokens are NOT exposed to tool scripts.
- No string interpolation: parameters are passed as environment variables, not injected into shell code.
- Required param validation: missing required parameters return an error before execution.
- Timeout + process cleanup: 30s timeout with process group kill to prevent zombies.
Tips
- Always quote
"$MDMCP_VAR"in bash to prevent word splitting - Use
${MDMCP_VAR:-default}for optional params with bash-side defaults - You can use
python3 -cinside bash for complex logic (seeexamples/data-tools.md) - Use
set -euat the top of scripts for safer execution
Examples
See the examples/ directory:
- ops-tools.md — disk usage, port check, git log, docker status, health check
- dev-tools.md — find TODOs, run linter, check port (from Codex dogfood)
- data-tools.md — CSV summary, JSON inspect, file diff (from Gemini dogfood)
Known limitations
- Execute blocks only support
```fences (not~~~) - Parameter names must be
\w+(no spaces or special chars) - Nested
```inside heredocs will confuse the parser - All parameter values are strings in bash; cast manually if needed
License
MIT
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file markdown_mcp_server-0.2.0.tar.gz.
File metadata
- Download URL: markdown_mcp_server-0.2.0.tar.gz
- Upload date:
- Size: 9.7 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.13
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
5b84f0063f5d6c503815eb70662d12a96015f3dc5405430e976ce8f69706ef82
|
|
| MD5 |
1bd8ddc56c70785859583850f245d3d7
|
|
| BLAKE2b-256 |
96249e07b00fe0232379532fe852050a692d91b60b8f8d08166b8fe98aff8ec8
|
File details
Details for the file markdown_mcp_server-0.2.0-py3-none-any.whl.
File metadata
- Download URL: markdown_mcp_server-0.2.0-py3-none-any.whl
- Upload date:
- Size: 8.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.13
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
a788ebf59993fc26ab189cdf46fb11f12856c62c734709715e0678802973a098
|
|
| MD5 |
71819612b48b9c0e8082b72f4542bd6a
|
|
| BLAKE2b-256 |
7c0d8f3e411eecafc0e0d52e2f790fab4c6bb3398a18cf536327bb4a38936fcb
|