obsidian

>

Skill file

Preview skill file
---
name: obsidian
description: >
  Activate when the user mentions their Obsidian vault, notes, tags,
  frontmatter, daily notes, backup, or sync. Route operations across MCP,
  Obsidian CLI/app actions, and git sync with safe defaults.
metadata:
  version: "2.0"
  author: bitbonsai
---

# Obsidian Skill

## Routing Policy

Use the backend that best matches user intent:

1. **MCP (default for vault data operations)**
   - Read/write/patch/move/search notes
   - Frontmatter and tag updates
   - Metadata and batch note operations

2. **Obsidian CLI/App context (only when app context is needed)**
   - Open a note in Obsidian from URI
   - Trigger app/plugin workflows that MCP cannot perform

3. **CLI git (sync/backup workflows)**
   - Initialize repo, configure remote, commit, pull, push
   - Periodic or manual vault backup/sync requests

When a request is ambiguous, pick MCP first unless the user explicitly asks for sync/backup/git/app behavior.

## Gotchas

1. **patch_note rejects multi-match by default.** With `replaceAll: false`, if `oldString` appears more than once the call fails and returns `matchCount`. Set `replaceAll: true` only when you mean it, or add surrounding context to make the match unique.

2. **patch_note matches inside frontmatter.** The replacement runs against the full file including the YAML block. A generic string like `title:` will match frontmatter fields. Include enough context to target the right occurrence.

3. **patch_note forbids empty strings.** Both `oldString` and `newString` must be non-empty and non-whitespace. To delete text, use `newString` with a single space or restructure the note with `write_note`.

4. **search_notes returns minified JSON.** Fields are abbreviated: `p` (path), `t` (title), `ex` (excerpt), `mc` (matchCount), `ln` (lineNumber), `uri` (obsidianUri). Hard cap of 20 results regardless of `limit`.

5. **search_notes multi-word queries score terms individually AND as a phrase.** Each term is OR-matched, so a document matching any term appears in results. The full phrase gets an additional scoring boost.

6. **write_note auto-creates directories.** Parent folders are created recursively. In `append`/`prepend` mode, if the note doesn't exist it's created. Frontmatter is merged (new keys override) in append/prepend; replaced entirely in overwrite.

7. **delete_note requires exact path confirmation.** `confirmPath` must be character-identical to `path`. No normalization, no trailing-slash tolerance. Mismatch silently fails with `success: false`.

8. **move_file needs double confirmation.** Both `confirmOldPath` and `confirmNewPath` must exactly match their counterparts. Use `move_note` for markdown renames (text-aware, no confirmation needed); use `move_file` only for binary files or when you need binary-safe moves.

9. **manage_tags reads from two sources but writes to one.** `list` merges frontmatter tags + inline `#hashtags`. `add`/`remove` only modify the frontmatter `tags` array. Inline tags are never touched.

10. **read_multiple_notes never rejects.** Uses `allSettled` internally. Failed files appear in the `err` array; successful ones in `ok`. Always check both. Hard limit of 10 paths per call.

## Error Recovery

| Error | Next step |
|-------|-----------|
| patch_note "Found N occurrences" | Add surrounding lines to `oldString` to make it unique, or set `replaceAll: true` |
| delete_note / move_file confirmation mismatch | Re-read the note path with `read_note` or `list_directory`, then retry with the exact string |
| search_notes returns 0 results | Try single keywords instead of phrases, toggle `searchFrontmatter`, or broaden with partial terms |
| read_multiple_notes partial `err` | Verify failed paths with `list_directory`, fix typos or missing extensions, retry only failed ones |

## Git Sync Mode

When the user asks to "sync", "backup", or "store my vault with git", use CLI git with this behavior:

1. Run a **preflight** before changing anything:
   - `git` available
   - current directory is a git repo (or prompt to initialize)
   - `git config user.name` and `git config user.email` are set
   - at least one remote exists for push/pull sync

2. If preflight is incomplete, ask exactly one targeted question with a recommended default.
   - Use askuserquestion for decisions that materially change behavior.
   - Good examples:
     - "No git repo found. Initialize one in this vault now? (Recommended: Yes)"
     - "No remote configured. Set up GitHub remote now via gh if available, or provide remote URL? (Recommended: Set up via gh)"
     - "Local and remote diverged. Try `git pull --rebase` now? (Recommended: Yes)"

3. Safe sync sequence (never force push by default):
   - `git add -A`
   - `git commit -m "vault sync: YYYY-MM-DD HH:mm"` (skip commit if no changes)
   - `git pull --rebase`
   - `git push`

4. `gh` is optional:
   - Use `gh` only for remote bootstrapping (create repo / set origin) when requested.
   - Do not require `gh` for normal sync once remote is configured.

5. Stop on conflicts and report clear next steps.
   - Do not auto-resolve merge conflicts silently.
   - Explain what failed and what user should run next.

## Obsidian CLI Mode

When the user asks for app-context operations (active file, open in editor, daily notes with templates, backlinks), use the Obsidian CLI directly via shell commands.

1. Run a **preflight** before first CLI use:
   - Resolve the CLI binary using the first match from these candidates:

     | Priority | macOS | Linux | Windows |
     |----------|-------|-------|---------|
     | 1 | `obsidian` (PATH) | `obsidian` (PATH) | `obsidian.exe` or `Obsidian.com` (PATH) |
     | 2 | `/Applications/Obsidian.app/Contents/MacOS/obsidian-cli` | — | — |
     | 3 | `/Applications/Obsidian.app/Contents/MacOS/Obsidian` | — | — |

     > **Obsidian 1.12.7+ installer** bundles a dedicated `obsidian-cli` binary (~10x
     > faster than the legacy Electron-based CLI: ~25ms vs ~250ms per call). On macOS,
     > after installing the 1.12.7+ installer, disable then re-enable the CLI in
     > Settings > General > Advanced to update PATH registration. This replaces the old
     > `~/.zprofile` PATH entry with a `/usr/local/bin/obsidian` symlink pointing to
     > `obsidian-cli`.
     >
     > On Linux, PATH registration creates a symlink at `/usr/local/bin/obsidian`
     > (or `~/.local/bin/obsidian` as fallback). On Windows, the installer places an
     > `Obsidian.com` terminal redirector alongside `Obsidian.exe`.
     >
     > **Note:** The priority table and stale PATH check are verified on macOS only.
     > Linux and Windows may also bundle `obsidian-cli` with the 1.12.7+ installer,
     > but this has not been confirmed. Contributions welcome via issue or PR.

   - **Stale PATH check (macOS):** If priority 1 resolved `obsidian` on PATH, check
     whether it points to the fast binary or the slow Electron launcher:

     | Resolved path | Meaning | Action |
     |---------------|---------|--------|
     | `/usr/local/bin/obsidian` → `obsidian-cli` | 1.12.7 symlink registration | None — fast binary |
     | `/Applications/.../MacOS/obsidian` | Old `~/.zprofile` entry (pre-1.12.7 registration or 1.12.7 installer without re-registering) | Check if `obsidian-cli` exists in the bundle |

     If `obsidian` resolves to the MacOS directory (not `/usr/local/bin`) AND
     `/Applications/Obsidian.app/Contents/MacOS/obsidian-cli` exists, tell the user:
     _"Obsidian 1.12.7+ is installed but PATH still points to the slower Electron
     binary. In Obsidian, go to Settings > General > Advanced and disable then
     re-enable the CLI to update PATH registration."_
     Continue with whichever priority matched — this is advisory, not blocking.

   - Check Obsidian is running: `pgrep -xiq obsidian` (macOS/Linux) or `tasklist /FI "IMAGENAME eq Obsidian.exe" /NH` (Windows)
   - If either fails, tell the user and fall back to MCP tools + `obsidian://` URIs

2. Vault targeting: `obsidian vault="VaultName" <command>`. The vault name is the folder basename unless `OBSIDIAN_VAULT_NAME` is set.

3. Key commands:
   ```bash
   # Read the currently active file
   obsidian read

   # Read a specific file
   obsidian read file="My Note"

   # Open a file in Obsidian
   obsidian open path="Notes/example.md"

   # Open today's daily note
   obsidian daily

   # Append to daily note
   obsidian daily:append content="- [ ] New task"

   # Search (Obsidian's own search, different from MCP's BM25)
   obsidian search query="meeting notes" limit=10

   # List all tags with frequency
   obsidian tags sort=count counts

   # Get backlinks for a note
   obsidian backlinks file="My Note"

   # Find unresolved links
   obsidian unresolved
   ```

4. Run `obsidian help` for the full command reference. The CLI evolves with Obsidian releases.

5. **When to use CLI vs MCP:**
   - MCP for reads/writes/search/tags/frontmatter (sandboxed, validated, works headless)
   - CLI for active file, daily notes with template expansion, backlinks, open in editor, plugin commands
   - If unsure, prefer MCP

## Resources

Load these only when needed, not on every invocation.

- [Tool Patterns](resources/tool-patterns.md) - read when you need a tool's response shape, mode details, or the move_note vs move_file decision
- [Obsidian Conventions](resources/obsidian-conventions.md) - read when creating/writing note content (link syntax, frontmatter fields, daily note format, template variables)
- [Git Sync](resources/git-sync.md) - read when user asks for backup/sync/store-vault workflows with git/gh

Source

Creator's repository · bitbonsai/mcpvault

View on GitHub

Security

Security checks in progress
Results will appear here once audits complete
What this skill can do
Reads your filesConnects to the internetRuns code on your machine
Checked by 3 independent security firms
Does it try to trick the AI?Not yet checkedPending · Gen Agent Trust Hub
Does it sneak in hidden code?Not yet checkedPending · Socket
Does it have known bugs?Not yet checkedPending · Snyk