Show HN: Pulse – Claude Code 仪表盘,支持通过手机批准工具调用
Show HN: Pulse – Dashboard for Claude Code, approve tool calls from your phone

原始链接: https://github.com/nikitadoudikov/claude-pulse

**Pulse** 是一款为 Claude Code 设计的私有化、本地优先的仪表盘,旨在监控活动、跟踪支出并提供远程控制功能,且无需遥测或外部依赖。 **核心功能:** * **实时仪表盘:** 按小时、天或周可视化令牌(token)使用量和费用。通过“环境办公”视图实时查看 Claude 的状态。 * **远程审批:** 无需端口转发或 Wi-Fi 配置,即可直接通过手机的推送通知批准或拒绝工具调用,即使在移动网络下也能使用。 * **会话管理:** 自动为活动会话创建快照,并支持对整个历史记录进行全文搜索。通过单条命令即可立即恢复丢失的会话。 * **隐私至上:** Pulse 在您的机器上本地运行,以只读模式读取您的 `~/.claude` 文件,且零依赖。没有任何数据会离开您的计算机。 * **简易设置:** 通过 `npx pulse-for-claude-code` 运行或克隆仓库。使用 `claude-pulse install-hooks` 即可与 Claude Code 无缝集成。 Pulse 专为追求透明、持久且便携界面的高级用户设计,旨在安全高效地管理其 Claude Code 工作流。

Pulse 是一款旨在提升 Claude Code 体验的全新开源工具。该本地应用程序由一名学生开发者创建,提供了一个精简的界面,用于监控在终端中运行的 Claude 代理。 Pulse 提供多项关键功能,包括: * **实时监控:** 通过简洁、舒适的设计追踪代理活动。 * **成本追踪:** 直接在应用内查看 Token 使用量和会话成本。 * **远程批准:** 通过手机远程批准工具调用。 * **注重隐私:** 应用程序完全在本地运行,无需账号。 Pulse 现已在 GitHub 上提供下载,为您管理和监督 AI 编码代理会话提供了更便捷的方式。
相关文章

原文

A local dashboard for Claude Code that shows what Claude is doing, what it is spending, and lets you approve its tool calls from your phone. Zero dependencies, nothing leaves your machine.

Overview

Claude Code already writes every session to disk. Pulse reads those files (read only) and turns them into a live dashboard: token spend by hour, day and week, the context fill of your active session, an ambient view of Claude at work, full-text search across everything you have ever run, and a notification with Allow / Allow all / Deny buttons when Claude needs you, on your desktop or your phone. No account, no telemetry, no network calls.

  • Approve from your phone. A push with working Allow / Allow all / Deny buttons. No Wi-Fi setup, no IP, no open port: it works from anywhere, even on cellular.
  • Never lose a session. One command recovers your last session as a readable transcript, and Pulse auto-snapshots active ones, so a crash or a frozen laptop never costs you context.
  • See the spend. Live tokens and API-equivalent cost by hour, day, week, model and project, against budgets you set, with a phone alert when you cross one.
  • Ambient office. A full-screen view of a little mascot working, resting, or waiting on you, with a rough ETA. Quietly addictive on a second monitor.
  • Search everything. Full-text search across every session on disk, one click to the transcript.
  • Local and private. Reads ~/.claude read only, serves on 127.0.0.1, zero dependencies, no telemetry.

Requires Node 18+. Run it with no install:

npx pulse-for-claude-code

Or clone it:

git clone https://github.com/nikitadoudikov/claude-pulse.git
cd claude-pulse
node bin/cli.js

Either way it opens http://127.0.0.1:4317. To get desktop and phone notifications and to approve tool calls, wire the hooks (one command, safe to re-run):

claude-pulse install-hooks   # adds the hooks to ~/.claude/settings.json

Then restart Claude Code, and you are set. Other options:

claude-pulse --port 4317   # change the port
claude-pulse --no-open     # do not open the browser

Run in the foreground and Pulse dies when you close that terminal. To keep it alive independently, run it in the background:

claude-pulse start     # run detached, survives closing the terminal
claude-pulse status    # is it running?
claude-pulse stop      # stop it
claude-pulse restart   # stop and start again

If your terminal crashes, claude-pulse start brings it back in one command, and a background instance is not affected by the crash in the first place.

On macOS you can hand Pulse to the system so it starts at login and respawns itself if it ever dies:

claude-pulse install-service     # start at login, auto-restart
claude-pulse uninstall-service   # remove it

Terminal crashed, laptop froze, hit a session limit? Nothing is lost: Claude Code writes every session to disk as it happens. One command brings the last one back, prints a recap and saves a readable transcript:

claude-pulse recover        # the most recent session
claude-pulse recover 2      # the one before that
claude-pulse recover <id>   # a specific session

It saves a light markdown file under ~/.claude-pulse/exports/ (a 15 MB log becomes a ~180 KB file) and prints a link to read the full transcript in the browser or on your phone. You can also open any session in the dashboard and use open transcript / download .md.

While Pulse runs it also auto-snapshots every recently active session to ~/.claude-pulse/exports/snapshots/ (one file per session, rewritten only when it changes). So the latest state is always on disk even if you never run recover. Set snapshotMinutes to 0 in ~/.claude-pulse.json to turn it off.

To back up everything at once, claude-pulse export-all writes every session into a single small gzipped markdown file, or use download all history on the Sessions screen.

Lost where you did something? The Sessions screen has a search box that scans every session on disk for a word or phrase and jumps you straight to the transcript. It works from your phone too.

The simplest phone control is the ntfy notification itself: it carries working Allow / Allow all / Deny buttons (see above), no network setup at all.

For a richer view, open http://<your-machine>:4317/phone on the same Wi-Fi (needs bindLan: true) to see what Claude is doing right now plus a Pause / Resume button. Pausing stops Claude from running further tools until you resume. Both need the PreToolUse hook wired.

  ┌──────────────┐   writes .jsonl    ┌──────────────────────┐   SSE    ┌──────────────┐
  │  Claude Code │ ─────────────────▶ │  Pulse  (read only)  │ ───────▶ │  dashboard   │
  │  (terminal)  │                    │   127.0.0.1:4317     │          │  + phone     │
  └──────┬───────┘                    └──────────────────────┘          └──────────────┘
         │
         │  hooks:  Notification  ·  Stop  ·  PreToolUse
         ▼
  ┌─────────────────────────────┐
  │  ~/.claude-pulse/           │   pending approvals · decisions · events
  └─────────────────────────────┘

Claude Code logs every session as JSONL under ~/.claude/projects/. Each assistant message carries a usage block (input, output and cache tokens) with a timestamp. Pulse reads those files (read only), caches each file by modification time so unchanged sessions are never re-parsed, and aggregates the numbers. The browser gets live updates over Server-Sent Events. Three small hooks let Claude Code tell Pulse when it needs you, when a turn ends, and when it wants to run a tool.

Notifications when Claude needs you

Claude Code can run a hook when it needs your attention. Point its Notification event at the bundled script and Pulse will show a banner and fire a desktop notification, even if the tab is in the background.

The easy way is one command:

claude-pulse install-hooks     # wires the hooks into ~/.claude/settings.json (safe to re-run)
claude-pulse uninstall-hooks   # removes them

It backs up your settings once, merges next to any hooks you already have, and never adds a duplicate. Restart Claude Code afterwards. To do it by hand instead, add this to ~/.claude/settings.json (use the absolute path to your clone):

{
  "hooks": {
    "Notification": [
      {
        "matcher": "",
        "hooks": [
          { "type": "command", "command": "node /absolute/path/to/claude-pulse/hooks/notify-hook.js" }
        ]
      }
    ],
    "Stop": [
      {
        "matcher": "",
        "hooks": [
          { "type": "command", "command": "node /absolute/path/to/claude-pulse/hooks/stop-hook.js" }
        ]
      }
    ],
    "PreToolUse": [
      {
        "matcher": "",
        "hooks": [
          { "type": "command", "command": "node /absolute/path/to/claude-pulse/hooks/permission-hook.js" }
        ]
      }
    ]
  }
}

Keep claude-pulse running and you are set.

Approve tools from the dashboard (and your phone)

With the PreToolUse hook wired, when Claude wants to run something that needs permission, an approval card appears in Pulse with Allow, Allow all and Deny. Allow all stops asking for the rest of the run.

This is built to never hang Claude. Read only tools pass straight through, and if Pulse is not running, has not heard from you within the approval timeout (60s by default, set approvalTimeoutMs), or hits any error, it falls back to the normal terminal prompt. Nothing breaks if you ignore it. The phone push carries Allow, Allow all and Deny buttons.

To approve from your phone, you only need an ntfyTopic (below) and the ntfy app. The push notification carries Allow, Allow all and Deny buttons, and tapping one sends the answer back through ntfy to a private reply topic that Pulse listens on. No same Wi-Fi, no IP, no open port: it works from anywhere, even on cellular. Pulse only acts on a reply while it is actually waiting for that request, so a stale notification can do nothing.

  Claude wants to run a tool
          │
          ▼
   PreToolUse hook ──▶ Pulse ──push──▶ phone notification
          │                                   │
          │                              tap "Allow"
          │                                   │
          │                    answer returns over ntfy
          │                                   │
          ▼                                   ▼
   hook is still waiting ◀── decision ◀── Pulse (subscribed to the reply topic)
          │
          ▼
   hook returns "allow" ──▶ Claude runs the tool

To get a push on your phone when Claude needs you or finishes, pick a hard to guess topic name, install the free ntfy app and subscribe to that topic, then set it in ~/.claude-pulse.json:

{ "ntfyTopic": "claude-pulse-9f3a7c" }

With the hooks above wired, the Notification hook pushes when Claude is waiting for you, and the Stop hook pushes when a turn finishes (debounced to 30s so a back and forth does not spam you). Anyone who knows the topic can read it, so use a random name.

If you set budgets (below), Pulse also pushes when a rolling window crosses 80% then 100% of its budget, so you find out from your pocket, not by checking.

Copy config.example.json to ~/.claude-pulse.json and edit. Every field is optional.

{
  "plan": "max20",
  "contextLimit": 200000,
  "idleMinutes": 10,
  "approvalTimeoutMs": 60000,
  "budgets": { "fiveHour": 140, "day": 360, "week": 1100 }
}

Anthropic does not publish exact subscription limits, and they are usage based rather than a fixed token count. Pulse cannot read your real plan ceiling, so the budgets above are rough API-equivalent estimates you adjust to match what you observe. The pro, max5 and max20 presets are starting points, not official numbers. Token cost is estimated from public API list prices purely as a usage proxy; subscription users do not pay per token.

Pulse is local-first and opt-in. Out of the box it binds to 127.0.0.1 only, makes no outbound calls, has zero dependencies (no supply chain), and reads ~/.claude read only. Nothing leaves your machine and there is no analytics. Two optional features change that, and both are off until you turn them on:

  • Phone push (ntfyTopic) routes through the public ntfy.sh relay. Approval prompts (with a short command summary) and your taps pass through a topic you name, so anyone who learns the topic can read those prompts and answer them. Use a long random topic, and self-host ntfy or use ntfy access tokens if you want stronger guarantees. Pulse only acts on a reply while it is genuinely waiting for that exact request, so a stale or guessed message cannot approve anything by itself.
  • LAN access (bindLan) binds the server to your whole network so a phone on the same Wi-Fi can open the live /phone page. While it is on, other devices on that network can also read the dashboard and your transcripts, so only enable it on a network you trust. You do not need it for phone approvals (those go through ntfy), so most people should leave it off.

Runtime state, the device token and your config live in your home directory under ~/.claude-pulse/, and are never committed or sent anywhere.

MIT

联系我们 contact @ memedata.com