Obsidian MCP Server
The Obsidian MCP Server enables AI models to interact with Obsidian vaults via secure API communication, providing:
File Operations: List, retrieve, append, and update files in the vault, including hierarchical directory structures.
Search Capabilities: Perform full-text searches with context, execute complex JsonLogic queries, and filter files based on content or metadata.
Property Management: Retrieve and update YAML frontmatter properties, manage tags, titles, and custom fields.
Advanced Tools: Use JsonLogic for complex filtering, manage timestamps, and intelligently merge or replace properties.
Integration: Works with Obsidian's Local REST API plugin and provides a standardized interface via the Model Context Protocol.
Security: Offers API key authentication, rate limiting, and SSL options for secure operations.
Enables AI models to interact with Obsidian vaults through a standardized interface, providing file operations, search capabilities, and property management for knowledge bases in Obsidian
Click on "Install Server".
Wait a few minutes for the server to deploy. Once ready, it will show a "Started" state.
In the chat, type
@followed by the MCP server name and your instructions, e.g., "@Obsidian MCP Serversearch for notes about machine learning projects"
That's it! The server will respond to your query, and you can continue using it as needed.
Here is a step-by-step guide with screenshots.
Tools
Fourteen tools grouped by shape — readers fetch notes and metadata, writers create or surgically edit content, managers reconcile tags and frontmatter, and a guarded escape hatch dispatches Obsidian command-palette commands.
Tool Name | Description |
| Read a note as raw content, full structured form (content + frontmatter + tags + stat, with optional outgoing links), structural document map, or a single section. |
| List notes and subdirectories under a vault path. Recursive walk (default depth 2, max depth 20; 1000-entry cap) with optional |
| List every tag found across the vault with usage counts, including hierarchical parents. Optional |
| List Obsidian command-palette commands, optionally filtered by |
| Search the vault by text, JSONLogic, or BM25-ranked Omnisearch (when the plugin is reachable). Results paginate via opaque cursors. |
| Create a note, replace a single section in place, or — with |
| Append content to a note. Without |
| Surgical |
| Body-wide search-replace inside a single note. Literal or regex matching with whole-word, whitespace-flexible, and case-sensitivity options; supports capture-group replacement. |
| Atomic |
| Add, remove, or list tags. Defaults to the frontmatter |
| Permanently delete a note. Elicits human confirmation when the client supports it. |
| Open a file in the Obsidian app UI, with |
| Execute an Obsidian command-palette command by ID. Opt-in via |
obsidian_get_note
Read a note in one of four projections, addressed by vault path, the active file, or a periodic note (daily, weekly, monthly, quarterly, yearly).
format: "content"— raw markdown bodyformat: "full"— content, frontmatter, tags, and file metadata; passincludeLinks: trueto also parse outgoing wiki and markdown link references from the body (vault-internal only — external URLs are filtered)format: "document-map"— catalog of headings, block references, and frontmatter fieldsformat: "section"— single heading/block/frontmatter section value (requiressection); heading sections include the full subtree under that heading
Pair the document-map projection with obsidian_patch_note to discover edit targets before patching.
obsidian_search_notes
Up to three search modes selected by mode:
text— substring match with surrounding context windows.contextLengthcontrols characters of context per side of each match (default 100; bump it for more context per hit). OptionalpathPrefixfilter (text mode only — passingpathPrefixin any other mode is rejected withpath_prefix_invalid_mode).jsonlogic— JSONLogic tree evaluated againstpath,content,frontmatter.<key>,tags, andstat.{ctime,mtime,size}; customglobandregexpoperatorsomnisearch— BM25-ranked search via the community Omnisearch plugin. Supports quoted phrases,-exclusion,path:/ext:filters, typo tolerance, and PDF + OCR coverage (via Text Extractor). Only present in the mode enum when the plugin's HTTP server is reachable at startup; the upstream hard-caps results at 50 — narrow the query to surface more (the response carriestruncated: truewhen the cap was likely hit).
Results paginate via opaque cursors per the MCP 2025-11-25 spec: omit cursor for the first page, then pass nextCursor from the prior response. Every result carries totalCount (post-path-policy, pre-pagination); nextCursor is omitted on the last page. Text-mode hits are additionally clipped per file at maxMatchesPerHit (default 10) so a single match-heavy note can't blow the response budget — clipped hits carry truncated: true and totalMatches.
obsidian_write_note
Create or surgically replace, with a protective default against accidental whole-file overwrites.
Without
section— full-filePUT. Refuses to clobber an existing file unlessoverwrite: trueis set. Thefile_exists(Conflict) error suggestsobsidian_patch_note/obsidian_append_to_note/obsidian_replace_in_notefor in-place edits.With
section—PATCH-with-replace against the named heading/block/frontmatter field, leaving the rest of the file untouched. Theoverwriteflag is ignored in section mode.
The output reports created: true when the call brought a new file into existence; false when it replaced an existing one or targeted a section. Every mutating tool also returns previousSizeInBytes and currentSizeInBytes so an agent can spot accidental clobbers, unexpected upstream behavior, or a typo path that landed at the wrong file.
obsidian_append_to_note
A combined upsert + section-append primitive that mirrors the upstream Local REST API behavior:
Without
section—POSTto/vault/{path}. Appends when the file exists, creates the file with your content as the entire body when it doesn't. The output'screated: trueflags the second branch so the agent can notice when a typo path or a not-yet-created daily note silently turned into a brand-new file.With
section—PATCH-with-append against the named heading, block reference, or frontmatter field. The file must exist (PATCH preflight throwsnote_missingotherwise). PasscreateTargetIfMissing: trueto bring the section itself into existence inside an existing file. Block-reference targets concatenate adjacent to the block line without a separator — include a leading newline incontentif you want one.
previousSizeInBytes is 0 on the upsert-create branch and the actual file size otherwise; currentSizeInBytes is the post-write size read from the upstream after the operation. Compare deltas against Buffer.byteLength(content) to detect auto-newline injection or concurrent writers.
obsidian_patch_note
Surgical edits at a single document target.
operation: "append"adds after the sectionoperation: "prepend"adds before the sectionoperation: "replace"swaps it outTargets: heading path, block reference ID, or frontmatter field
Use obsidian_get_note with format: "document-map" to discover what targets exist before patching.
obsidian_replace_in_note
Body-wide search-replace for edits that don't fit obsidian_patch_note's structural targets. The note is fetched, replacements are applied sequentially (each sees the previous output), and the result is written back in a single PUT.
Per-replacement options:
useRegex— treatsearchas an ECMAScript regex. WithuseRegex: true, the replacement honors$1/$&capture-group references.caseSensitive— whenfalse, match case-insensitivelywholeWord— wrap the pattern in\b…\b; works in both literal and regex modesflexibleWhitespace— substitute any run of whitespace insearchwith\s+. Literal mode only — has no effect whenuseRegex: true(express it directly).replaceAll— whenfalse, only the first match is replaced
Literal mode preserves $1 / $& in the replacement verbatim — only useRegex: true expands capture-group references.
obsidian_manage_tags
Add, remove, or list tags on a note. Operates on one of two representations, defaulting to the canonical Obsidian frontmatter location:
location: 'frontmatter'(default) — only the frontmattertags:array; the note body is left untouchedlocation: 'inline'— only inline#tagsyntax in the body;addappends#tagat end-of-filelocation: 'both'— opt-in reconciliation across both representations
add ensures the tag is present in the requested location(s); remove strips it; list ignores the input tags array. Inline #tag occurrences inside fenced code blocks are intentionally left alone.
obsidian_delete_note
Permanently delete a note. When the client supports elicit, the server requests human confirmation before issuing the DELETE and the prompt includes the file's byte size — destructive blast radius visible before the user confirms. Without elicitation, the destructiveHint annotation surfaces the operation in the host's approval flow. The output reports previousSizeInBytes (size at the moment of deletion) and currentSizeInBytes: 0.
obsidian_execute_command
Dispatch an Obsidian command-palette command by ID (discoverable via obsidian_list_commands). Behavior is command-dependent — some commands open UI, others delete files or close the vault.
Off by default. When OBSIDIAN_ENABLE_COMMANDS is unset, both obsidian_execute_command and its discovery partner obsidian_list_commands are wrapped with disabledTool() — absent from tools/list (the LLM can't invoke them) but still visible in the operator-facing manifest with a hint to enable them.
Related MCP server: Obsidian RAG MCP Server
Path policy (folder-scoped permissions)
Three optional env vars gate which vault paths each tool can target. Default unset = full vault for both reads and writes — backwards compatible.
Goal | Config |
Default (current behavior) | all unset |
Read everywhere, write only in |
|
Read only |
|
Read-only deployment — no writes anywhere |
|
Matching is prefix-based with implicit recursion, case-insensitive, with trailing slashes normalized. projects/ matches projects/a.md, projects/sub/b.md, etc.
Write paths are implicitly readable — you can't sanely edit what you can't see. So a read passes when the target matches READ_PATHS or WRITE_PATHS.
OBSIDIAN_READ_ONLY=true short-circuits before the path checks — every write tool and the command-palette pair are wrapped with disabledTool() at startup (absent from tools/list), and any write that still reaches the service is denied at runtime regardless of WRITE_PATHS.
Denies are typed path_forbidden (JSON-RPC code Forbidden) with the active scope echoed back in data.recovery.hint and data.activeScope, so the LLM can self-correct without inspecting server logs. Search results from obsidian_search_notes are filtered against READ_PATHS silently — surfacing a "we hid N hits" indicator would defeat the gate.
Tag listing is vault-wide. obsidian_list_tags and the obsidian://tags resource aggregate tag names across the whole vault and are not narrowed by OBSIDIAN_READ_PATHS — they take no path to gate, so tag names (never note contents) from outside the read scope can surface.
The startup banner logs the active scope so operators can verify their config at boot.
Resources
Type | URI | Description |
Resource |
| A note in the vault — content, frontmatter, tags, and file metadata. |
Resource |
| All tags found across the vault, with usage counts. |
Resource |
| Server reachability, auth status, plugin/Obsidian version info, and the plugin manifest. |
All resource data is also reachable via tools — obsidian_get_note for obsidian://vault/{+path}, obsidian_list_tags for obsidian://tags. Resources exist for clients that prefer attaching a specific note or vault snapshot to a conversation.
Features
Built on @cyanheads/mcp-ts-core:
Declarative tool and resource definitions — single file per primitive, framework handles registration and validation
Unified error handling — handlers throw, framework catches, classifies, and formats. Tools advertise their failure surface via typed
errors[]contracts.Server-level
instructionsoninitialize— surfaces deployment-specific orientation (active path policy, read-only mode, command-palette toggle) to spec-compliant clients alongside the static tool/resource catalogPluggable auth on the HTTP transport:
none,jwt,oauthStructured logging with optional OpenTelemetry tracing
STDIO and Streamable HTTP transports
The server itself is stateless — every tool call hits the Local REST API directly. The framework's storage backends, request-state KV, and progress streams aren't used here; Obsidian is single-vault and there's nothing to persist between calls.
Obsidian-specific:
Wraps the Obsidian Local REST API plugin — typed client, deterministic error mapping
Section-aware editing across headings, block references, and frontmatter fields via
PATCH-with-target operationsTag reconciliation across both representations: frontmatter
tags:array and inline#tagsyntax (skipping fenced code blocks)Search across up to three modes: text, JSONLogic, and (when the plugin is reachable) BM25-ranked Omnisearch — cursor-paginated per the MCP 2025-11-25 spec, with per-file match clipping in text mode
Optional human-in-the-loop confirmation for destructive deletes via
ctx.elicitFolder-scoped read/write permissions via
OBSIDIAN_READ_PATHS/OBSIDIAN_WRITE_PATHSand a globalOBSIDIAN_READ_ONLYkill switch — denies are typedpath_forbiddenwith the active scope echoed back in the error dataOpt-in command-palette pair (
obsidian_list_commands+obsidian_execute_command) — registered only whenOBSIDIAN_ENABLE_COMMANDS=trueForgiving path resolution on
obsidian_get_noteandobsidian_open_in_ui— silently retries case-mismatched paths against the canonical filename, throwsConflicton ambiguous case matches, and enrichesNotFoundwithDid you mean: …?suggestions when only near-matches exist.obsidian_delete_noteis deliberately excluded — a destructive op shouldn't silently rewrite the target path.
Getting started
Add the following to your MCP client configuration file. The Obsidian Local REST API plugin must be installed and enabled in your vault — see Prerequisites.
{
"mcpServers": {
"obsidian-mcp-server": {
"type": "stdio",
"command": "bunx",
"args": ["obsidian-mcp-server@latest"],
"env": {
"MCP_TRANSPORT_TYPE": "stdio",
"MCP_LOG_LEVEL": "info",
"OBSIDIAN_API_KEY": "your-local-rest-api-key"
}
}
}
}Or with npx (no Bun required):
{
"mcpServers": {
"obsidian-mcp-server": {
"type": "stdio",
"command": "npx",
"args": ["-y", "obsidian-mcp-server@latest"],
"env": {
"MCP_TRANSPORT_TYPE": "stdio",
"MCP_LOG_LEVEL": "info",
"OBSIDIAN_API_KEY": "your-local-rest-api-key"
}
}
}
}For Streamable HTTP, set the transport and start the server. Inline env vars work for one-off runs; for repeated use, copy values into .env (see .env.example) and run bun run start:http.
MCP_TRANSPORT_TYPE=http OBSIDIAN_API_KEY=... bun run start:http
# Server listens at http://127.0.0.1:3010/mcp by defaultPrerequisites
Bun v1.3.11 or higher (or Node.js v24+).
The Obsidian Local REST API plugin v4.0.0 or later installed and enabled in your vault. Generate an API key in Settings → Community Plugins → Local REST API and copy it into
OBSIDIAN_API_KEY.This server defaults to
http://127.0.0.1:27123for simplicity. Enable "Non-encrypted (HTTP) Server" in the plugin settings to use it. To use the always-on HTTPS port instead, setOBSIDIAN_BASE_URL=https://127.0.0.1:27124; the plugin's self-signed cert is handled byOBSIDIAN_VERIFY_SSL=false(the default).
Installation
Clone the repository:
git clone https://github.com/cyanheads/obsidian-mcp-server.gitNavigate into the directory:
cd obsidian-mcp-serverInstall dependencies:
bun installConfigure environment:
cp .env.example .env # edit .env and set OBSIDIAN_API_KEY
Configuration
Variable | Description | Default |
| Required. Bearer token for the Obsidian Local REST API plugin. | — |
| Base URL of the Local REST API plugin. Use |
|
| Verify the TLS certificate. Default |
|
| Per-request timeout in milliseconds. |
|
| Opt-in flag for the command-palette pair ( |
|
| Comma-separated vault-relative folder allowlist for read operations. Prefix-based with implicit recursion; case-insensitive; trailing slashes normalized. Unset = full vault. Write paths are implicitly readable. | unset |
| Comma-separated vault-relative folder allowlist for write operations. Same syntax as | unset |
| Global kill switch. When |
|
| Override URL for the Omnisearch plugin's HTTP server. When unset, derives from | derived |
| Transport: |
|
| Host for the HTTP server. |
|
| Port for the HTTP server. |
|
| Endpoint path for the JSON-RPC handler. |
|
| Public origin override for TLS-terminating reverse-proxy deployments (landing page, Server Card, RFC 9728 metadata). | unset |
| Auth mode: |
|
| Required when | — |
| When |
|
| Log level (RFC 5424). |
|
| Directory for log files (Node.js only). |
|
| Enable OpenTelemetry instrumentation (spans, metrics, completion logs). |
|
See .env.example for the full list of optional overrides.
Running the server
Local development
Build and run the production version:
# One-time build bun run rebuild # Run the built server bun run start:stdio # or bun run start:httpRun checks and tests:
bun run devcheck # Lint, format, typecheck, security, changelog sync bun run test # Vitest test suite bun run lint:mcp # Validate MCP definitions against spec
Docker
docker build -t obsidian-mcp-server .
docker run --rm -e OBSIDIAN_API_KEY=your-key -p 3010:3010 obsidian-mcp-serverThe Dockerfile defaults to HTTP transport, stateless session mode, and logs to /var/log/obsidian-mcp-server. OpenTelemetry peer dependencies are installed by default — build with --build-arg OTEL_ENABLED=false to omit them.
The image binds to 0.0.0.0 inside the container (required for Docker port mapping). For any deployment reachable beyond your own machine, set MCP_AUTH_MODE=jwt (with MCP_AUTH_SECRET_KEY) or oauth — otherwise the listener forwards your OBSIDIAN_API_KEY to the vault on behalf of every caller.
Project structure
Directory | Purpose |
|
|
| Server-specific environment variable parsing ( |
| Local REST API client, frontmatter operations, section extractor, domain types. |
| Tool definitions ( |
| Resource definitions ( |
| Prompt definitions (currently empty — CRUD/search shape doesn't benefit from a structured template). |
| Vitest tests mirroring |
| Upstream OpenAPI spec for the Local REST API plugin and the generated |
| Per-version release notes; |
Development guide
See CLAUDE.md for development guidelines and architectural rules. The short version:
Handlers throw, framework catches — no
try/catchin tool logicUse
ctx.logfor request-scoped logging,ctx.statefor tenant-scoped storageRegister new tools and resources via the barrels in
src/mcp-server/*/definitions/index.tsWrap external API calls: validate raw → normalize to domain type → return output schema; never fabricate missing fields
Contributing
Issues and pull requests are welcome. Run checks and tests before submitting:
bun run devcheck
bun run testLicense
Apache-2.0 — see LICENSE for details.
This server cannot be installed
Maintenance
Latest Blog Posts
MCP directory API
We provide all the information about MCP servers via our MCP API.
curl -X GET 'https://glama.ai/api/mcp/v1/servers/cyanheads/obsidian-mcp-server'
If you have feedback or need assistance with the MCP directory API, please join our Discord server