obsidian-link
obsidian-link
Connect a project's Claude Code configuration (agents, skills) to an Obsidian vault for browsing, searching, and cross-referencing.
Directionality:
- Global agents/skills: Obsidian is source of truth, symlinked into
~/.claude/ - Per-project agents/skills: Project repo is source of truth, symlinked into Obsidian
Mode Dispatch
Parse the user's input to determine mode:
/obsidian-linkor/obsidian-link link→ Link mode (default)/obsidian-link status→ Status mode/obsidian-link unlink <project>→ Unlink mode/obsidian-link init→ Init mode (configure plan frontmatter)
Configuration
Resolving the vault path
Check in order:
OBSIDIAN_VAULTenvironment variable~/.obsidian-vaultfile (single line, absolute path)
If found, confirm with the user. If not found, ask. Once confirmed, offer to save to ~/.obsidian-vault.
Directory structure
<vault>/ClaudeCode/
├── README.md # Auto-generated index (wikilinks)
├── Agents/
│ ├── global/
│ └── <project>/
├── Skills/
│ ├── global/
│ └── <project>/
└── Plans/ # ~/.claude/plans → here
Preflight (runs before every mode)
Before executing any mode, silently check the current setup state. This takes a few seconds and informs how you interact with the user.
Checks
| Check | How | States |
|---|---|---|
| Vault path | ~/.obsidian-vault file or $OBSIDIAN_VAULT |
configured / missing |
| Vault directory | <vault>/ClaudeCode/ exists |
exists / missing |
| Plans symlink | readlink ~/.claude/plans |
linked-to-vault / linked-elsewhere / missing |
| Plan frontmatter | Search ~/.claude/CLAUDE.md for ## Plan Files section |
configured / missing |
| Project linked | <vault>/ClaudeCode/Agents/<project>/ has symlinks |
linked / not-linked |
Behaviour
Summarise the setup state in a compact block before proceeding with the requested mode:
obsidian-link preflight:
Vault: /path/to/vault (OK)
Plans: ~/.claude/plans → vault (OK)
Frontmatter: configured in CLAUDE.md (OK) | not configured
Project: <project> linked (6 agents, 2 skills) | not linked
Then, based on state and requested mode:
- If vault path is missing → Ask the user before continuing with any mode. Cannot proceed without it.
- If frontmatter is not configured and mode is
linkorstatus→ Append a suggestion after the mode completes:Tip: Run
/obsidian-link initto configure plan frontmatter in your CLAUDE.md — plans will then be browsable in Obsidian with Dataview, tags, and graph view. - If frontmatter is not configured and mode is
init→ Expected, proceed normally. - If frontmatter is already configured and mode is
init→ Show current config in preflight, then Step 2 of Init mode handles the replace-or-keep flow. - If project is not linked and mode is
status→ Note it in the status report, suggest running/obsidian-linkto link.
The preflight should be lightweight and informational — never block a mode from running (except when vault path is missing). Its purpose is context, not gatekeeping.
Link Mode (default)
Step 1: Resolve vault + ensure directories
Confirm vault path, then: mkdir -p "<vault>/ClaudeCode/{Agents/global,Skills/global,Plans}"
Step 2: Plans symlink
Check ~/.claude/plans:
- Missing →
ln -sf "<vault>/ClaudeCode/Plans" "$HOME/.claude/plans" - Exists, target matches vault → skip (OK)
- Exists, points elsewhere → warn and skip (do not overwrite)
Step 3: Detect project
git rev-parse --show-toplevel → basename, lowercased. If not in a git repo, ask for project name and root path.
Step 4: Link project agents (idempotent)
If <repo>/.claude/agents/ exists, mkdir -p "<vault>/ClaudeCode/Agents/<project>".
For each .md file, before creating a symlink, check existing state:
target="<vault>/ClaudeCode/Agents/<project>/<file>"
expected="<repo>/.claude/agents/<file>"
current=$(readlink "$target" 2>/dev/null)
if [ "$current" = "$expected" ]; then
# already_linked — skip
elif [ -L "$target" ]; then
# wrong target — overwrite (count as updated)
ln -sf "$expected" "$target"
else
# new — create
ln -sf "$expected" "$target"
fi
Track counts: new, already_linked, updated.
Step 5: Link project skills (idempotent)
Same pattern as Step 4, but for each skill directory in <repo>/.claude/skills/.
Step 6: Sync global skills into ~/.claude
For each skill directory in <vault>/ClaudeCode/Skills/global/ (must contain SKILL.md):
- If
~/.claude/skills/<name>doesn't exist → create symlink - If it exists and points to vault → skip
- If it exists and points elsewhere → skip (never overwrite non-Obsidian skills)
Step 7: Sync global agents into ~/.claude
~/.claude/agentsdoesn't exist →ln -sf "<vault>/ClaudeCode/Agents/global" "$HOME/.claude/agents"- Exists as symlink to vault → skip
- Exists as real directory or wrong symlink → warn, skip
Step 8: Broken symlink detection
After all linking:
find "<vault>/ClaudeCode/Agents" "<vault>/ClaudeCode/Skills" -type l ! -exec test -e {} \; -print
Report broken links with their targets. Suggest cleanup but never auto-delete.
Step 9: Generate index note
Write <vault>/ClaudeCode/README.md (fully regenerated each run):
---
type: index
title: Claude Code Hub
updated: <YYYY-MM-DD>
tags:
- claude-code
- index
---
# Claude Code Hub
## Projects
| Project | Agents | Skills | Status |
|---------|--------|--------|--------|
| <name> | <count> | <count> | OK / N broken |
## Global
| Type | Name | Link |
|------|------|------|
| Agent | <name> | [[<name>]] |
| Skill | <name> | [[<name>]] |
## Plans
<count> plan files in ClaudeCode/Plans/
Use [[wikilinks]] for all agent and skill names so Obsidian's graph view connects them. Scan Agents/ and Skills/ subdirs (excluding global/) for project names.
Step 10: Report
obsidian-link: <project>
Vault: <vault-path>
Project: <project> (<repo-root>)
Agents: <N> new, <N> already linked
Skills: <N> new, <N> already linked
Broken: none | <N> broken (listed above)
Plans: ~/.claude/plans → vault (OK) | WARNING: points elsewhere | created
Global: <N> skills synced, agents symlink OK
Index: ClaudeCode/README.md updated
Status Mode
Scans the vault's ClaudeCode/ directories (not the filesystem) to report health of all linked projects.
Steps
- Resolve vault path (same as Link mode Step 1)
- Scan
<vault>/ClaudeCode/Agents/andSkills/for project subdirs (excludingglobal/) - For each project: count symlinks, count broken symlinks, report health
- Check global sync: agents symlink target, skills presence
- Check Plans symlink
- Regenerate index note (Step 9 from Link mode)
Report format
obsidian-link status
Vault: <vault-path>
Projects:
project-a — 16 agents, 1 skill, 0 broken
project-b — 4 agents, 2 skills, 1 broken
broken: project-b/old-agent.md → /path/that/moved
Global: agents OK, 3 skills synced
Plans: ~/.claude/plans → vault (OK)
Index: ClaudeCode/README.md updated
Unlink Mode
Removes a project's symlinks from the vault. Never deletes repo files.
Steps
- Resolve vault path
- Validate
<project>exists in<vault>/ClaudeCode/Agents/<project>orSkills/<project> - Show what will be removed (counts per directory)
- Ask for explicit user confirmation before proceeding
- Remove the project's directories:
These dirs contain only symlinks — repo source files are untouched.rm -rf "<vault>/ClaudeCode/Agents/<project>" rm -rf "<vault>/ClaudeCode/Skills/<project>" - Regenerate index note
Report format
obsidian-link unlink: <project>
Removed: 16 agent links, 1 skill link
Source files in <repo-root> are untouched.
Index: ClaudeCode/README.md updated
Init Mode
Interactively configures ~/.claude/CLAUDE.md so that Claude always adds Obsidian-friendly frontmatter when creating plan files. This is a one-time setup — safe to re-run (idempotent).
Step 1: Resolve vault path
Same as Link mode. Needed to generate the correct ~/.claude/plans symlink path in instructions.
Step 2: Check for existing frontmatter instructions
Search ~/.claude/CLAUDE.md for a ## Plan Files section (or similar frontmatter block referencing plans/).
- Found → Show the user the current config, ask if they want to replace or keep it.
- Not found → Proceed to Step 3.
Step 3: Ask the user about their frontmatter fields
Present a conversational prompt like:
I'll add instructions to your global CLAUDE.md so plan files always get Obsidian-friendly frontmatter.
Here are some common fields — which would you like?
Default fields (always included unless explicitly removed):
| Field | Example | Purpose |
|---|---|---|
type |
plan |
Obsidian note type for Dataview/queries |
title |
User Auth OAuth Flow |
Human-readable title |
status |
draft | approved | done |
Plan lifecycle tracking |
created |
2026-03-05 |
Creation date |
tags |
[plan, <project>] |
Obsidian tag navigation |
Optional fields (suggest these, let user pick):
| Field | Example | Purpose |
|---|---|---|
id |
<project>-oauth-flow |
Unique slug for cross-referencing |
projects |
[proj-<project>] |
Multi-project attribution |
priority |
high | medium | low |
Prioritisation |
complexity |
1-5 |
Estimation (per user's preference) |
related |
[[other-plan]] |
Wikilink to related plans |
linear |
PROJ-123 |
Linear/issue tracker reference |
updated |
2026-03-05 |
Last-modified date |
Let the user add, remove, or rename fields freely. They may also suggest entirely custom fields — accept any valid YAML key.
Step 4: Ask about filename convention
Present the current convention (or a sensible default):
Filename format:
<project>-<kebab-slug>.md(3–6 words) e.g.myapp-user-auth-oauth-flow.md,backend-stripe-billing-integration.mdWant to adjust this? (e.g. date prefix, different separator, no project prefix)
Accept the user's preference or confirm the default.
Step 5: Generate the CLAUDE.md section
Build a ## Plan Files section containing:
- The filename convention
- A YAML frontmatter template with the chosen fields
- Brief inline comments showing example values
Example output (will vary based on user choices):
## Plan Files
When writing plan files (in `~/.claude/plans/`, which symlinks to `ObsidianVault/Plans/`), always include this frontmatter so they're browsable in Obsidian:
Filename should be kebab-case, prefixed with the project name, 3-6 words total, reflecting the plan content. E.g. `myapp-user-auth-oauth-flow.md`, `backend-stripe-billing-integration.md`.
\```yaml
---
type: plan
id: <project>-<kebab-slug>
title: <human-readable title>
status: draft | approved | done
projects:
- <project ID>
tags:
- plan
- <project tag>
created: "<YYYY-MM-DD>"
---
\```
Step 6: Preview and confirm
Show the user the exact block that will be inserted (or will replace the existing block). Ask for confirmation before writing.
Step 7: Write to CLAUDE.md
- If replacing an existing
## Plan Filessection: replace from## Plan Filesto the next##heading (or end of file). - If inserting new: add before the first
---horizontal rule after the initial sections, or append before# Code Styleif that heading exists. Use best judgement to place it logically. - Never overwrite unrelated sections.
Step 8: Report
obsidian-link init: complete
CLAUDE.md: ~/.claude/CLAUDE.md updated
Section: ## Plan Files (inserted | replaced)
Fields: type, title, status, created, tags, id, projects
Filename: <project>-<kebab-slug>.md
Plans will now include Obsidian frontmatter automatically.
Notes
- Never auto-delete broken symlinks — report them and let the user decide
- Never overwrite non-Obsidian configs — if
~/.claude/agentsis a real directory, warn and skip - Idempotent by design — running Link mode repeatedly produces the same result with accurate counts
- Index note is disposable — fully regenerated each run, safe to delete manually