Wraps the nb CLI to give Claude proper note-taking capabilities without shell escaping headaches. Exposes operations for creating, editing, searching, and organizing Markdown notes, todos, and bookmarks through JSON parameters instead of raw shell commands. Handles notebook context explicitly so you can maintain separate note collections per project. Commands include nb.add for new notes with tags and folders, nb.search for full-text queries, nb.todo for task management, and nb.move for reorganization. Automatically creates notebooks and manages Git commits in the background. Particularly helpful if you're coordinating multiple LLMs on the same codebase and need structured, searchable notes that survive backticks and code blocks. Requires nb installed locally and supports configuration via environment variables or CLI flags.
MCP server wrapping the nb CLI for LLM-friendly note-taking.
Using nb directly via shell has two problems for LLM assistants:
Backtick escaping: Markdown content with backticks triggers shell command substitution, corrupting notes.
Notebook context: nb assumes a default notebook, making per-project use awkward.
This MCP server solves both by:
Install nb by following the official instructions:
nb installation guide.
From crates.io:
cargo install nb-mcp-server
Or download a prebuilt binary from GitHub Releases.
cargo build --release
With default notebook from environment:
NB_MCP_NOTEBOOK=myproject ./target/release/nb-mcp
Or via CLI argument (takes precedence):
./target/release/nb-mcp --notebook myproject
Disable commit and tag signing in the notebook repository:
./target/release/nb-mcp --notebook myproject --no-commit-signing
Allow new notes at the notebook root instead of requiring a folder:
./target/release/nb-mcp --notebook myproject --allow-top-level-notes
Print the installed version:
./target/release/nb-mcp --version
Show the resolved notebook path and state directory:
./target/release/nb-mcp --show-paths
Add to your MCP client configuration (e.g., .mcp.json):
{
"mcpServers": {
"nb": {
"command": "/path/to/nb-mcp",
"args": ["--notebook", "myproject"]
}
}
}
The canonical access path is the multiplexed nb tool with a command
parameter, which reduces the token footprint of the MCP server.
The args field must be a JSON object. Stringified JSON payloads are rejected.
Unknown args fields are rejected instead of ignored; use the exact command
schema fields or documented aliases.
Returned identifiers such as coordination/mcp/1 or
myproject:coordination/mcp/1 are nb selectors, not filesystem paths in the
current repository. Notebook storage is managed by nb configuration.
The notebook argument must be a bare notebook name. Use folder for folder
paths and id / selector for note selectors. Existing-item commands accept
copied selectors such as myproject:coordination/mcp/1, but reject conflicts
with a separate notebook argument.
All commands are also available as direct first-class tools with typed
schemas: add, show, edit, delete, move, list, search, todo,
do, undo, tasks, bookmark, folders, mkdir, import, status,
notebooks. These bypass the multiplexed command dispatch. The multiplexed
nb tool remains as the compact/backcompat compatibility surface.
| Command | Description | Key Arguments |
|---|---|---|
nb.add | Create a note | title, content, tags[], folder required by default |
nb.show | Read a note | id (alias: selector) |
nb.edit | Update a note | id (alias: selector), content, mode (replace default, append, prepend) |
nb.delete | Delete a note | id (alias: selector) |
nb.move | Move or rename a note | id (alias: selector), destination |
nb.list | List notes | folder, tags[], limit ([ ] / [x] indicate todo status; leading glyphs are item markers) |
nb.search | Full-text search | queries[] (required), mode (any default, all), tags[] |
| Command | Description | Key Arguments |
|---|---|---|
nb.todo | Create a todo | folder required by default, title, optional description (alias: content), optional tasks[], tags[] |
nb.do | Mark complete | id (alias: selector), optional task_number |
nb.undo | Reopen | id (alias: selector), optional task_number |
nb.tasks | List todos | optional status (open or closed), optional recursive (true default) |
| Command | Description | Key Arguments |
|---|---|---|
nb.bookmark | Save a URL | url, folder required by default, title, tags[], comment |
nb.import | Import file/URL | source, folder required by default, filename, convert |
nb.folders | List folders | parent |
nb.mkdir | Create folder | path |
nb.notebooks | List notebooks only | (none) |
nb.status | Notebook info | (none) |
Create a note with code:
{
"command": "nb.add",
"args": {
"title": "API Design Notes",
"content": "# API Design\n\nUse `GET /items` for listing.\n\n```python\nresponse = client.get('/items')\n```",
"tags": ["design", "api"],
"folder": "docs"
}
}
Search for notes:
{
"command": "nb.search",
"args": {
"queries": ["API", "design"],
"mode": "any",
"tags": ["design"]
}
}
For multi-LLM projects, consider using consistent tag prefixes (optional). Example categories and prefixes:
| Category | Pattern | Examples |
|---|---|---|
| Collaborator | llm-<name> | llm-claude, llm-gpt |
| Component | component-<name> | component-api, component-ui |
| Task type | task-<type> | task-bug, task-feature |
| Status | status-<state> | status-review, status-blocked |
Priority order:
notebook argument (highest)--notebook flagNB_MCP_NOTEBOOK environment variableIf no notebook can be resolved, commands fail with a configuration error. The
server does not fall back to nb's default notebook.
If the resolved notebook does not exist, the server creates it automatically.
Use --no-create-notebook to disable automatic creation.
Logs are written to ~/.local/state/nb-mcp/{project}--{worktree}.log (XDG-compliant).
For Git worktrees, logs are named after both the master project and the worktree basename to avoid collisions between multiple MCP server instances.
Use --show-paths to print the resolved notebook path and state directory.
By default, note-creating commands require a folder argument so agents do not
accidentally litter project notebook roots. This applies to nb.add, nb.todo,
nb.bookmark, and nb.import. Use nb.mkdir to create new folders and
nb.folders to list existing folders.
Set NB_MCP_ALLOW_TOP_LEVEL_NOTES=true or pass --allow-top-level-notes to
permit root-level note creation.
Mutating commands warn after successful writes when the notebook argument
targets a notebook other than the project default. Cross-notebook writes remain
allowed for collaboration across teams, but the warning helps catch accidental
notebook/folder confusion.
The notebook argument accepts only bare notebook names, not selector syntax.
For example, use notebook: "other-team" with folder: "todos/mcp", not
notebook: "other-team:todos/mcp".
Control log level with RUST_LOG:
RUST_LOG=debug nb-mcp --notebook myproject
Use --no-commit-signing to disable commit and tag signing in the notebook
repository. The server updates the notebook repository's local Git config so
signing prompts do not block MCP tool calls.
See the contribution guide and code of conduct:
io.github.ericm1018/skillfm-llm-cost-optimizer-openai-anthropic-usage
io.github.mikerawsonnz/llm-orchestration-agent
io.github.mikerawsonnz/authenticated-llm-agent
labforgedev/copilot-memory-mcp
csoai-org/agent-prompt-injection-firewall-mcp
io.github.mikerawsonnz/authenticated-multi-llm-agent