io.github.sns45/dear-claude
Spawn Claude Code instances from 7 platforms when 'Dear Claude' is mentioned.
Ask AI about io.github.sns45/dear-claude
Powered by Claude Β· Grounded in docs
I know everything about io.github.sns45/dear-claude. Ask me about installation, configuration, usage, or troubleshooting.
0/500
Reviews
Documentation
Dear Claude
MCP server that triggers local Claude Code instances from external platforms.
Say "Dear Claude" in Linear, GitHub, Jira, GitLab, Notion, or Obsidian β and a Claude Code instance spins up to handle it.
Your notes become architecture. Your tasks become pull requests.
What is this?
Dear Claude is an MCP (Model Context Protocol) server that watches your project management tools for the phrase "Dear Claude". When detected, it spawns a local Claude Code instance that:
- Reads the issue/comment/note context
- Executes the requested task (code, review, create tasks, etc.)
- Posts results back to the originating platform
- Persists sessions for 7 days so you can have multi-turn conversations
No Anthropic API keys needed. Works with your existing Claude Code subscription. 100% local and private β your code never leaves your machine.
Supported Platforms
| Platform | Trigger on issue/PR | Trigger on comment | Comment back | Emoji reactions | Sub-tasks | PR/MR review |
|---|---|---|---|---|---|---|
| GitHub | Yes | Yes | Yes | Yes | - | Yes |
| Linear | Yes | Yes | Yes | Yes | Yes | - |
| Jira | Yes | Yes | Yes | - | Yes | - |
| GitLab | Yes | Yes | Yes | Yes | - | Yes |
| Notion | Yes | Yes | Yes | - | - | - |
| Obsidian | Yes | - | Yes | - | - | - |
Cross-Platform Orchestration
Instances from any platform get API access to all configured platforms. This enables workflows like:
- Write a spec in Obsidian β "Dear Claude, create these tasks in Linear"
- Discuss on Linear β "Dear Claude, code this on GitHub"
- Review on GitHub β "Dear Claude, resolve the merge conflicts"
- Parallel coding β Claude spawns multiple instances, one per branch, using git worktrees
Quick Start
Install in one line
claude mcp add dear-claude -- bunx dear-claude start --mcp
That's it. Start Claude Code and Dear Claude is ready.
Prerequisites
- Claude Code CLI installed (
claudecommand available) - Bun runtime (for
bunx) - Tailscale with Funnel enabled (for webhooks from external platforms)
Manual setup (alternative)
If you prefer manual configuration, add to ~/.claude.json under mcpServers:
{
"mcpServers": {
"dear-claude": {
"command": "bunx",
"args": ["dear-claude", "start", "--mcp"],
"env": {
"DEAR_CLAUDE_PORT": "3334",
"GITHUB_CLIENT_ID": "...",
"GITHUB_CLIENT_SECRET": "...",
"GITHUB_WEBHOOK_SECRET": "...",
"LINEAR_CLIENT_ID": "...",
"LINEAR_CLIENT_SECRET": "...",
"LINEAR_WEBHOOK_SECRET": "..."
}
}
}
}
Then start Claude Code:
claude
The MCP server starts automatically, sets up Tailscale Funnel, and prints your public webhook URLs.
Platform Setup
Tailscale Funnel
Dear Claude uses Tailscale Funnel for stable public HTTPS URLs to receive webhooks. Setup is mostly automatic.
-
Install Tailscale:
# macOS brew install tailscale # Linux curl -fsSL https://tailscale.com/install.sh | sh -
Authenticate:
tailscale up -
Enable Funnel in the admin console: https://login.tailscale.com/admin/acls
- Add the
"funnel"capability to your ACL policy
- Add the
-
The server auto-configures Funnel on startup. Your public URL will be:
https://<your-hostname>.ts.net/dc
Tip: Run
tailscale serve status --jsonto verify your config. The health check auto-repairs the Funnel config every 10 seconds.
GitHub
Option A: GitHub App (recommended)
- Go to GitHub Settings β Developer settings β GitHub Apps β New GitHub App
- Set these fields:
- Homepage URL:
https://<your-hostname>.ts.net/dc - Callback URL:
https://<your-hostname>.ts.net/dc/oauth/callback/github - Webhook URL:
https://<your-hostname>.ts.net/dc/webhook/github - Webhook secret: generate a random string
- Homepage URL:
- Under Permissions:
- Repository: Issues (Read & Write), Pull Requests (Read & Write), Contents (Read & Write)
- Under Subscribe to events:
- Issue comments, Pull request review comments
- Copy credentials and set env vars:
GITHUB_CLIENT_ID=Iv1.abc123... GITHUB_CLIENT_SECRET=abc123... GITHUB_WEBHOOK_SECRET=your-webhook-secret - Install the app on your repos
- Complete OAuth: visit
https://<your-hostname>.ts.net/dc/setup/github
Option B: Personal Access Token (simpler, no webhooks)
- Go to GitHub Settings β Developer settings β Personal access tokens β Tokens (classic)
- Create a token with scopes:
repo,write:discussion - Set:
GITHUB_ACCESS_TOKEN=ghp_...
Note: With a PAT alone you won't get webhook-triggered instances. You'd use the
spawn_instanceMCP tool or/api/spawnendpoint instead.
| Environment Variable | Description |
|---|---|
GITHUB_CLIENT_ID | GitHub App client ID |
GITHUB_CLIENT_SECRET | GitHub App client secret |
GITHUB_WEBHOOK_SECRET | Webhook signature verification secret |
GITHUB_ACCESS_TOKEN | Personal access token (alternative to OAuth) |
Linear
- Go to Linear Settings β API β OAuth Applications β New Application
- Set the callback URL:
https://<your-hostname>.ts.net/dc/oauth/callback/linear - Copy Client ID and Client Secret
- Enable the Webhooks toggle on the OAuth app
- Go to Linear Settings β API β Webhooks β New Webhook:
- URL:
https://<your-hostname>.ts.net/dc/webhook/linear - Copy the Signing Secret
- Enable events: Comments (create), Issues (create, update)
- URL:
- Set env vars:
LINEAR_CLIENT_ID=your-client-id LINEAR_CLIENT_SECRET=your-client-secret LINEAR_WEBHOOK_SECRET=your-signing-secret - Complete OAuth: visit
https://<your-hostname>.ts.net/dc/setup/linear
After OAuth, only issues/comments from your authenticated Linear account trigger Claude.
Alternative: Use a Personal API Key (LINEAR_ACCESS_TOKEN=lin_api_...) from Linear Settings β Account β API.
| Environment Variable | Description |
|---|---|
LINEAR_CLIENT_ID | OAuth client ID |
LINEAR_CLIENT_SECRET | OAuth client secret |
LINEAR_WEBHOOK_SECRET | Webhook signing secret |
LINEAR_ACCESS_TOKEN | Personal API key (alternative to OAuth) |
Jira Cloud
- Create an API token at https://id.atlassian.com/manage-profile/security/api-tokens
- Set env vars:
JIRA_DOMAIN=mycompany # Your Jira subdomain (mycompany.atlassian.net) JIRA_USER_EMAIL=you@example.com # Your Atlassian account email JIRA_API_TOKEN=ATATT3x... # The API token you just created JIRA_WEBHOOK_SECRET=optional-secret # Optional shared secret - Create a webhook in Jira:
- Go to Jira Admin β System β Webhooks β Create webhook
- URL:
https://<your-hostname>.ts.net/dc/webhook/jira- If using
JIRA_WEBHOOK_SECRET, append it:?secret=YOUR_SECRET
- If using
- Select events:
issue_created,issue_updated,comment_created
- Save the webhook
Claude can create sub-tasks, transition issue status, and add comments via the Jira REST API v2.
| Environment Variable | Description |
|---|---|
JIRA_DOMAIN | Jira subdomain (e.g. mycompany for mycompany.atlassian.net) |
JIRA_USER_EMAIL | Your Atlassian email |
JIRA_API_TOKEN | API token from Atlassian |
JIRA_WEBHOOK_SECRET | Optional shared secret for webhook verification |
GitLab
- Create a Personal Access Token at GitLab β Settings β Access Tokens
- Scopes:
api,read_repository,write_repository
- Scopes:
- Set env vars:
GITLAB_ACCESS_TOKEN=glpat-... GITLAB_WEBHOOK_SECRET=your-secret - Add a webhook to your project (or group):
- Go to Settings β Webhooks
- URL:
https://<your-hostname>.ts.net/dc/webhook/gitlab - Secret token: same as
GITLAB_WEBHOOK_SECRET - Trigger events: Comments, Issues events, Merge request events
- Save
For self-hosted GitLab, also set GITLAB_URL=https://your-gitlab-instance.com.
| Environment Variable | Description |
|---|---|
GITLAB_ACCESS_TOKEN | Personal access token |
GITLAB_WEBHOOK_SECRET | Webhook secret token |
GITLAB_URL | GitLab instance URL (default: https://gitlab.com) |
Notion
Option A: Internal Integration (simpler)
- Go to https://www.notion.so/my-integrations β New integration
- Give it a name, select your workspace
- Copy the Internal Integration Secret
- Set:
NOTION_ACCESS_TOKEN=ntn_... - Share pages/databases with the integration (click "..." on a page β Connections β Add your integration)
Option B: OAuth (public app)
- Create an OAuth integration at https://www.notion.so/my-integrations
- Set callback URL:
https://<your-hostname>.ts.net/dc/oauth/callback/notion - Set env vars:
NOTION_CLIENT_ID=your-client-id NOTION_CLIENT_SECRET=your-secret - Complete OAuth: visit
https://<your-hostname>.ts.net/dc/setup/notion
Webhook setup
Notion doesn't have native webhooks yet. To trigger Claude from Notion:
- Use Notion's automation rules with a webhook action (if available)
- Or use a third-party service like Zapier/Make to POST to
https://<your-hostname>.ts.net/dc/webhook/notion - Set
NOTION_WEBHOOK_SECRETif you want signature verification
| Environment Variable | Description |
|---|---|
NOTION_ACCESS_TOKEN | Internal integration token |
NOTION_CLIENT_ID | OAuth client ID |
NOTION_CLIENT_SECRET | OAuth client secret |
NOTION_WEBHOOK_SECRET | Webhook verification secret |
Obsidian
Obsidian integration works via filesystem watching β no webhooks needed. Claude watches your vault for files containing "Dear Claude" and responds by appending to the same file.
- Set env var with the absolute path to your vault:
OBSIDIAN_VAULT_PATH=/Users/yourname/Documents/MyVault - That's it! Write "Dear Claude, ..." in any
.mdfile and save.
Claude's response appears as a callout block appended to the same note. The frontmatter gets a claude-status field (processing β done / error).
How it works:
- Watches for
.mdfile changes in the vault - Ignores files in
.obsidian/,.trash/, and dotfile directories - 2-second debounce to avoid triggering on every keystroke
- Supports wikilink references (
[[other-note]]) β Claude resolves and reads them - Supports embedded images β Claude can see and analyze them
| Environment Variable | Description |
|---|---|
OBSIDIAN_VAULT_PATH | Absolute path to your Obsidian vault |
OBSIDIAN_WATCH_DEBOUNCE_MS | Debounce delay in ms (default: 2000) |
Usage
Trigger Format
Write "Dear Claude" (case-insensitive, with a space) anywhere in:
- GitHub issue/PR comments
- Linear issue descriptions or comments
- Jira issue descriptions or comments
- GitLab issue/MR descriptions or comments
- Notion page comments
- Obsidian
.mdfiles
Example
GitHub PR Comment:
Dear Claude, please review this code for bugs and security issues.
Claude responds on GitHub:
Claude Instance Started (Instance:
abc12345) Processing your request...
Task Completed Found 2 issues:
- SQL injection in
user.ts:45- Missing input validation in
api.ts:102Created PR #15 with fixes.
Instance Orchestration
Claude instances can spawn other instances for parallel work:
Dear Claude, code tasks 1-5 in parallel. Each task should be a separate branch.
Claude will:
- Parse the tasks
- Spawn 5 child instances via the
/api/spawnendpoint - Each child works in its own git worktree (same repo, different branch)
- Each child creates a PR when done
- Parent polls child statuses
All Environment Variables
# Server
DEAR_CLAUDE_PORT=3334
TAILSCALE_HOSTNAME= # Optional: auto-detected
# GitHub
GITHUB_CLIENT_ID=
GITHUB_CLIENT_SECRET=
GITHUB_WEBHOOK_SECRET=
GITHUB_ACCESS_TOKEN=
# Linear
LINEAR_CLIENT_ID=
LINEAR_CLIENT_SECRET=
LINEAR_WEBHOOK_SECRET=
LINEAR_ACCESS_TOKEN=
# Jira Cloud
JIRA_DOMAIN=mycompany
JIRA_USER_EMAIL=you@example.com
JIRA_API_TOKEN=
JIRA_WEBHOOK_SECRET=
# GitLab
GITLAB_ACCESS_TOKEN=
GITLAB_WEBHOOK_SECRET=
GITLAB_URL= # Default: https://gitlab.com
# Notion
NOTION_CLIENT_ID=
NOTION_CLIENT_SECRET=
NOTION_WEBHOOK_SECRET=
NOTION_ACCESS_TOKEN=
# Obsidian
OBSIDIAN_VAULT_PATH=
OBSIDIAN_WATCH_DEBOUNCE_MS=2000
# Optional
GIPHY_API_KEY= # For fun GIF reactions in responses
CLI Commands
# Start the server (standalone mode)
bun run src/index.ts start
# Start as MCP server (stdio, for Claude Code)
bun run src/index.ts start --mcp
# Check server and platform status
bun run src/index.ts status
# List instances
bun run src/index.ts instances
# Setup instructions for a platform
bun run src/index.ts setup <platform>
MCP Tools
When running as an MCP server inside Claude Code, these tools are available:
| Tool | Description |
|---|---|
list_platforms | List configured platforms and their status |
list_instances | List all Claude instances (filter by status) |
get_instance_status | Get detailed status of a specific instance |
get_instance_messages | Get conversation history for an instance |
kill_instance | Terminate a running instance |
get_running_instances | List currently running instance IDs |
spawn_instance | Spawn a new Claude instance for a task |
get_project_instances | List all instances in a project group |
HTTP API
The server also exposes REST endpoints on localhost:3334:
| Endpoint | Method | Description |
|---|---|---|
/health | GET | Health check + platform status |
/webhook/:platform | POST | Webhook receiver |
/api/instances | GET | List instances (?project_id= filter) |
/api/instances/:id | GET | Get instance details + children |
/api/instances/:id/kill | POST | Kill a running instance |
/api/spawn | POST | Spawn a new instance programmatically |
/api/platforms | GET | List configured platforms |
/setup/:platform | GET | Start OAuth flow |
/oauth/callback/:platform | GET | OAuth callback |
POST /api/spawn
{
"prompt": "Implement the login page",
"repo_url": "https://github.com/owner/repo",
"branch": "feature/login",
"base_branch": "main",
"parent_instance_id": "optional-parent-id",
"project_id": "optional-project-id"
}
Architecture
Webhooks / File Watcher
ββββββββββ ββββββββββ ββββββββ ββββββββββ ββββββββββββ ββββββββββ
β GitHub β β Linear β β Jira β β GitLab β β Obsidian β β Notion β
βββββ¬βββββ βββββ¬βββββ ββββ¬ββββ βββββ¬βββββ ββββββ¬ββββββ βββββ¬βββββ
β β β β β β
ββββββββββββ΄βββββ¬βββββ΄ββββββββββ΄βββββββββββββ΄βββββββββββββ
β
βΌ
βββββββββββββββββββββββ
β Tailscale Funnel β
β (Public HTTPS URL) β
βββββββββββ¬ββββββββββββ
β
βΌ
βββββββββββββββββββββββ
β dear-claude β
β MCP Server β
β β
β β’ Trigger detection β
β β’ Instance manager β
β β’ Platform adapters β
β β’ Spawn API β
β β’ SQLite DB β
βββββββββββ¬ββββββββββββ
β
βΌ
βββββββββββββββββββββββ
β Claude Code β
β Instances β
β (Agent SDK) β
β β
β β’ Git worktrees β
β β’ Cross-platform β
β API access β
β β’ Child spawning β
βββββββββββββββββββββββ
Instance Lifecycle
PENDING β RUNNING β IDLE β (follow-up) β RUNNING β IDLE β ... β EXPIRED (7 days)
β COMPLETED
β FAILED
- PENDING: Trigger detected, instance queued
- RUNNING: Claude Code actively processing
- IDLE: Waiting for follow-up "Dear Claude" mentions
- COMPLETED: Task finished successfully
- FAILED: Error occurred
- EXPIRED: 7-day TTL exceeded, instance cleaned up
Development
# Install dependencies
bun install
# Run dev mode
bun run dev
# Type check
bunx tsc --noEmit
# Build
bun run build
# Run tests
bun test
Troubleshooting
Tailscale
- "Tailscale not running": Open the Tailscale app (macOS) or
sudo systemctl start tailscaled && tailscale up(Linux) - "Funnel not enabled": Visit https://login.tailscale.com/admin/acls and add Funnel capability
- Funnel disappears: Another
tailscale serve/tailscale funnelcommand may have overwritten it. The health check auto-repairs within 10 seconds. Verify withtailscale serve status --json.
Webhooks
- Not triggering: Check
bun run src/index.ts statusto verify the server is up. Test withcurl https://<your-hostname>.ts.net/dc/health. - "Invalid signature": Verify the webhook secret matches in both the platform config and your env vars.
- GitHub: The app subscribes to
issue_commentevents. To trigger on a new PR, post a comment β PR descriptions alone won't trigger.
OAuth
- Token expired: Re-visit
https://<your-hostname>.ts.net/dc/setup/<platform>to re-authenticate. - 401 errors: The stored OAuth token may have been revoked. Delete the stale token from
data/dear-claude.dband re-authenticate.
Instances
- Stuck in PENDING: Check that Claude Code CLI (
claude) is installed and accessible in your PATH. - Working directory issues: Instances create workspaces under
data/workspaces/. Ensure write permissions.
License
MIT
