loading…
Search for a command to run...
loading…
A lightweight Notion MCP server that minimizes token usage by returning Markdown instead of raw JSON, enabling efficient read/write operations on Notion pages.
A lightweight Notion MCP server that minimizes token usage by returning Markdown instead of raw JSON, enabling efficient read/write operations on Notion pages.
A lightweight Notion MCP server that minimises token usage compared to the official Notion MCP.
Instead of returning raw Notion API JSON, every tool returns the smallest useful representation: pages come back as plain Markdown, search results as a single line per hit. In practice this cuts token consumption significantly on read-heavy workflows.
| Tool | Description |
|---|---|
notion_search |
Search pages by title. Returns one line per result in TOON format. |
notion_read |
Read a page as Markdown. |
notion_write |
Append Markdown to an existing page. |
notion_create_page |
Create a sub-page under a parent, optionally with content. |
Search results and page creation responses use the compact TOON format:
Page title | page-id | https://notion.so/... | parent:<parent-id>
This keeps search responses to a single line per result instead of hundreds of tokens of JSON.
notion_read)| Block type | Markdown output |
|---|---|
| Paragraph | Plain text |
| Heading 1 / 2 / 3 | # / ## / ### |
| Bulleted list item | - item |
| Numbered list item | 1. item, 2. item, … |
| Code block | ```lang … ``` |
| Table | Pipe-delimited rows |
| Child page | [Subpage: title](id) |
Inline formatting (bold, italic, bold-italic, strikethrough, underline, inline code, links) is preserved as Markdown.
notion_write)The same block types are supported on write. Markdown is parsed and appended to the target page as native Notion blocks.
The following block types are recognised by Notion but not yet handled. They are silently skipped on read and cannot be written:
---)Empty paragraph blocks between different list types are not round-tripped. When a page has an explicit empty paragraph separating, say, a bulleted list from a numbered list, that separator is lost on a read → write cycle. The "\n\n" that Markdown already places between blocks is visually equivalent but does not produce a Notion empty-paragraph block on write. This is an inherent limitation of a flat Markdown representation and will stay until a v2 format is designed.
Table header flag is not round-tripped. Notion tables can be created without a column-header row (has_column_header: false). The current writer always sets has_column_header: true because standard Markdown tables imply a header. Tables written back will look identical but have the header flag set.
# From source
git clone https://github.com/your-username/tiny-notion-mcp
cd tiny-notion-mcp
pip install .
Or install directly with uv:
uv tool install .
Set your Notion integration token as an environment variable:
export NOTION_TOKEN=secret_...
NOTION_API_KEY is accepted as an alias.
Add the server to your MCP configuration:
{
"mcpServers": {
"tiny-notion-mcp": {
"command": "tiny-notion-mcp",
"env": {
"NOTION_TOKEN": "secret_..."
}
}
}
}
If you installed with uv into an isolated environment, use the full path:
{
"mcpServers": {
"tiny-notion-mcp": {
"command": "uv",
"args": ["run", "--", "tiny-notion-mcp"],
"env": {
"NOTION_TOKEN": "secret_..."
}
}
}
}
# Install with dev dependencies
pip install -e ".[dev]"
# Run tests
pytest
Tests use a stub NotionClient and make no real API calls.
NotionClient interface is injected so the core logic can be tested in isolation.Add this to claude_desktop_config.json and restart Claude Desktop.
{
"mcpServers": {
"tiny-notion-mcp": {
"command": "npx",
"args": []
}
}
}