miyo-kado
Security-first MCP server plugin for Obsidian that gives AI assistants granular, permissioned access to vault notes and files through tools like read, write, delete, rename, search, and open-notes.
README
<p align="center"> <img src="assets/kado_banner_final.svg" alt="MiYo Kado banner" width="480" /> </p>
MiYo Kado -- Obsidian MCP Gateway
Security-first Model Context Protocol server plugin for Obsidian. Gives AI assistants controlled, granular access to your vault through six tools: kado-read, kado-write, kado-delete, kado-rename, kado-search, and kado-open-notes.
Part of the MiYo family. The plugin is referred to as MiYo Kado in the Obsidian community-plugin index and in the settings UI; "Kado" alone is used as a short form throughout this README and the source.
<p align="center"> <img src="assets/MiYo-Kado.png" alt="MiYo Kado logo" width="160" /> </p>
Why MiYo Kado?
Letting an AI assistant talk to your vault sounds great until you realize most integrations give the assistant everything -- every note, every file, full read/write. Kado is built around the opposite default:
- Nothing is exposed by default. You opt every path and every data type in, explicitly.
- Per-key scopes. Different assistants get different keys with different permissions. Revoke any key independently.
- Audit trail. Every allowed and denied request is logged so you can see exactly what an assistant touched.
- Local-first. The MCP server runs inside Obsidian on
127.0.0.1by default. No cloud, no telemetry, no third party.
If you've ever wanted to say "this assistant can read my project notes but not my journal, and can delete drafts but never touch archived material", Kado is for you. For a deeper look at why AI permissioning matters for PKM workflows -- and why guardrails are not the same as real permissions -- see Permissioning your AI.
Features
- Default-deny security -- nothing is accessible until explicitly whitelisted
- Two-tier access control -- global security scope defines what is eligible; per-key scope defines what is permitted
- Five permission gates -- authenticate, global-scope, key-scope, datatype-permission, path-access
- Four data types -- notes (markdown), frontmatter (YAML as JSON), files (binary as base64), Dataview inline fields
- Partial note read/write -- read a slice (
firstXChars,sectionby heading,rangeby line/char) and write in place (append/prepend,insertUnderHeading,replaceSection/replaceRange) without round-tripping the whole body; omitting the mode is byte-for-byte backward compatible - Rename & move --
kado-renamerenames or moves notes and files with backlinks updated automatically; one folder ⇒ rename (needsupdate), across folders ⇒ move (needsdeleteon the source +createon the target). Works best with Obsidian's "Automatically update internal links" on (silent, links updated); with it off the tool is hidden unless you opt in, and then each rename prompts Obsidian's link-update dialog (the file still moves, but inbound links update only when you answer) - Seven search operations -- byName, byTag, byContent, byFrontmatter, listDir, listTags, listNotes
- Optimistic concurrency -- timestamp-based conflict detection on writes
- Rate limiting -- 200 requests/minute per IP
- Audit logging -- NDJSON log with rotation (metadata only, no content)
- Status bar indicator -- the 門 gate glyph shows server state at a glance (listening / off / bind error) and throbs on each tool call, colour-coded for reads vs. writes, with the acting key's name in the tooltip; a rejected call lingers red for a few seconds then self-clears. Click it to open Kado's settings
Architecture & the MiYo ecosystem
MiYo Kado is one component of the MiYo multi-repo system. Kado is the Obsidian-side
MCP gateway: it owns the vault access-control model (two-layer path eligibility + per-key
CRUD scopes) and exposes the MCP tool surface that companion tools such as MiYo Tomo
consume. The authoritative record for cross-repo contracts, system-level architecture, and
governance decisions lives in MiYo Kokoro; this repo's local design docs (below, and
docs/XDD/specs/) defer to Kokoro for project-wide principles. Cross-component contract
changes (e.g. new MCP tools) are handed off to Kokoro via _outbox/for-kokoro/.
Internally Kado follows a four-layer clean architecture: MCP boundary (src/mcp/) →
permission gates + policy (src/core/) → Obsidian adapters (src/obsidian/) → canonical
types (src/types/). See How It Works for the enforcement flow.
Documentation
| Document | Audience | Content |
|---|---|---|
| Installation | Everyone | Community Plugins, BRAT, manual install |
| Configuration Guide | Vault owners | Settings UI, security setup, API key management |
| Client Setup | Vault owners | Claude Code, Claude Desktop, Cursor, Windsurf |
| How It Works | Vault owners | Architecture, security model, enforcement logic, audit log |
| Example Configurations | Vault owners | Common setups with permission matrices |
| API Reference | MCP client developers | Tool schemas, parameters, examples, error codes |
| Development Guide | Contributors | Build, test, lint, architecture, live testing |
| Permissioning your AI | PKM practitioners | Why AI permissioning matters, guardrails vs enforcement |
Roadmap
Tracked as GitHub issues:
- Tag permissions beyond read-only (#81) -- deny permission so tags can exclude matching items from otherwise-allowed paths.
- Granular whitelist / blacklist toggle (#82) -- per-section toggle for mixed strategies.
- Real-time permission testing (#83) -- dry-run in the settings UI.
- Settings import / export (#84) -- backup/restore for the whole config.
- Opt-in content truncation on
kado-read(#85) -- first ~1000 words with a hint that more exists.
Shipped: sub-path key scopes (narrower sub-paths inside an allowed parent) landed in v0.15.0.
Known edge cases
Editing a note the AI wants access to
- Edit-while-reading gap. If you're actively typing in a note and your AI assistant reads it via Kado at the same moment, the assistant may see the version from up to 2 seconds ago -- Obsidian saves your edits to disk on a ~2 second pause. Pause briefly (or press Cmd/Ctrl+S) before asking the AI about your most recent sentence.
- Write while the same note is open and dirty. If an AI tries to write to a note you are currently editing with unsaved keystrokes, Kado refuses the write with a
CONFLICTerror and shows a Notice ("Kado wanted to modify <note> ..."). Your typing always wins. The AI client sees the same conflict signal used for any concurrent change and is expected to re-read and retry, so once you pause typing (~2 s Obsidian autosave) and it retries, its write is applied on top of your latest edits.
Quick Start
- Install the plugin
- Open Settings > MiYo Kado
- Add paths to the global security whitelist (e.g.
notes/,projects/, or**for full vault) - Create an API key and assign it paths and per-data-type permissions
- Enable the server
- Connect your AI client
{
"mcpServers": {
"kado": {
"type": "http",
"url": "http://127.0.0.1:23026/mcp",
"headers": {
"Authorization": "Bearer YOUR_API_KEY"
}
}
}
}
Screenshots
General tab -- server status, port, audit logging.
<p align="center"> <img src="assets/settings-general.png" alt="MiYo Kado General settings tab" width="720" /> </p>
Global Security tab -- whitelist of paths and tags that any key may reference. Permissions are set per data type (Notes, Frontmatter, Dataview, Files) for each path.
<p align="center"> <img src="assets/settings-global-security.png" alt="MiYo Kado Global Security settings tab with two whitelisted paths and one tag" width="720" /> </p>
API Key tab -- per-key access. Each key has its own access mode, paths, tags, and permission matrix, all constrained by the global security scope.
<p align="center"> <img src="assets/settings-api-key.png" alt="MiYo Kado API Key settings tab showing key management, permissions, paths and tags" width="720" /> </p>
Security Model
Every request passes through five gates in order. The first denial stops the chain. For the full enforcement logic, see How It Works.
| # | Gate | Purpose |
|---|---|---|
| 0 | authenticate | Bearer token must match an enabled API key |
| 1 | global-scope | Path must be inside the global whitelist (or outside the blacklist) |
| 2 | key-scope | Path must be inside the key's own scope |
| 3 | datatype-permission | Key must have the required CRUD flag for the data type |
| 4 | path-access | Final path-traversal and validation check |
Architecture
MCP Client -> [MCP API Handler] -> [Kado Core] -> [Obsidian Interface] -> Vault
- MCP API Handler -- Express + Streamable HTTP transport, auth, rate limiting
- Kado Core -- Permission gates, routing, concurrency guard. No MCP or Obsidian imports.
- Obsidian Interface -- Vault adapters for notes, frontmatter, files, inline fields, search
Part of MiYo
Kado is part of MiYo, a small family of Obsidian-adjacent tools focused on giving you control over what your assistants can see and do. MiYo Kado is the gateway component -- the piece that turns your vault into a properly-scoped MCP server. More tools are in the works.
Open tracking
Live issues and upstream references live in GitHub Issues and docs/ai/memory/troubleshooting.md. No open issues at the time of this release.
Support
If MiYo Kado is useful to you and you want to help me keep building, you can support development via:
Issues and pull requests are also very welcome.
Contributing
Contributions are welcome. The short version:
- Open an issue first for anything non-trivial (bugs, features, refactors) so we can align on scope before you invest time.
- Fork & branch from
master. Use a descriptive branch name (e.g.fix/search-tag-case,feat/granular-scopes). - Keep changes focused -- one feature or one fix per PR. See Development Guide for build, test, and lint commands.
- Tests & lint must pass -- run
npm run build,npm test, andnpm run lintbefore pushing. - Conventional commits -- e.g.
feat:,fix:,docs:,refactor:. Release notes are generated from commit history. - Open a PR against
masterand reference the issue. Small, reviewable diffs get merged fastest.
For security issues, please do not open a public issue -- email marcus@mmomm.org instead.
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.