让设备成为 Claude Code CLI 的物理审批按钮 — 电脑作为 TCP 服务端
Project description
claude-code-buddy-bridge
通过网络协议实现物理审批按钮 — 支持手机、嵌入式设备等任何支持 TCP 的设备
Anthropic 的官方 claude-desktop-buddy 固件让设备变成 Claude 的物理审批按钮——但它只支持特定硬件和蓝牙通信。
claude-code-buddy-bridge 提供了一个更通用的解决方案:通过标准 TCP 网络协议通信,让任何支持网络的设备(手机、嵌入式设备、单片机等)都可以作为 Claude Code 的物理审批按钮。
flowchart TD
A["Claude Code CLI\nPermissionRequest 钩子"]
B["ccbb 守护进程\n(TCP 服务端)"]
C["设备\n(TCP 客户端)"]
D["按键决策\n批准或中止"]
E["继续执行"]
F["中止任务"]
A --> B
B --"TCP/IP 网络"--> C
C --> D
D --"决策结果"--> B
B --> E
B --> F
功能
- 零侵入:通过 Claude Code 原生 Hook 接入,不需要修改任何项目文件
- Fail-open:守护进程未运行时,CC 自动回退到自己的权限对话框
- TCP 网络通信:使用标准 TCP/IP 协议,跨平台、跨设备兼容
- 多设备支持:手机、嵌入式设备、单片机等任何支持 TCP 的设备均可连接
- 支持多设备同时连接:多个设备可以同时连接并接收审批请求
- 完整上下文传递:所有 Claude Code Hook 的原始信息都会传递给设备
- 支持多语言:不再有中文等非 ASCII 字符的限制
- 并发串行:多个并发 hook 请求排队,不会同时争抢设备
- EOF 竞争检测:若 CC 提前终止 hook 进程,立即清空设备显示,不会傻等超时
支持的设备
任何支持 TCP 客户端的设备都可以使用:
- 📱 智能手机:通过 App 或脚本连接
- 🔧 嵌入式设备:ESP32、Arduino、Raspberry Pi 等
- 💻 电脑/服务器:通过脚本或程序连接
- 🕹️ 单片机:任何支持网络功能的 MCU
快速开始
1. 启动守护进程(电脑作为服务端)
首先在电脑上运行 ccbb daemon:
cd /workspace
uv sync
uv run ccbb daemon
默认监听 0.0.0.0:9876,可以通过环境变量自定义:
CCBB_TCP_HOST=192.168.1.100 CCBB_TCP_PORT=8888 uv run ccbb daemon
2. 连接设备(作为 TCP 客户端)
任何支持 TCP 的设备都可以连接。以下是几种连接方式:
方式一:使用示例客户端(测试用)
python3 examples/tcp_device_client.py
如果设备在另一台机器上:
CCBB_TCP_HOST=192.168.1.100 CCBB_TCP_PORT=8888 python3 examples/tcp_device_client.py
方式二:使用手机 App
编写一个简单的 TCP 客户端 App,连接到电脑的 IP 和端口。
方式三:使用嵌入式设备
// ESP32 示例代码
#include <WiFi.h>
#include <WiFiClient.h>
const char* ssid = "your_wifi_ssid";
const char* password = "your_wifi_password";
const char* host = "192.168.1.100";
const int port = 9876;
WiFiClient client;
void setup() {
WiFi.begin(ssid, password);
while (WiFi.status() != WL_CONNECTED) { delay(500); }
if (client.connect(host, port)) {
client.println("{\"cmd\": \"permission\", \"id\": \"test\", \"decision\": \"once\"}");
}
}
3. 注入 Claude Code Hook
uv run ccbb install
# 只拦截 Bash 工具(更精准):
# uv run ccbb install --tools Bash
这条命令会自动在 ~/.claude/settings.json 中写入配置。
4. 使用
打开 Claude Code,触发一个需要审批的操作(如执行 Bash 命令):
- 在设备上发送批准指令 → 批准(
allow) - 在设备上发送拒绝指令 → 拒绝(
deny)
设备不在线?ccbb 超时后自动 fail-open,CC 弹出自己的对话框。
常用命令
| 命令 | 说明 |
|---|---|
ccbb install |
注入 hook 到 Claude Code 配置 |
ccbb install --tools Bash Write |
只拦截指定工具 |
ccbb daemon |
启动守护进程(TCP 服务端) |
ccbb daemon -v |
调试模式(显示详细日志) |
ccbb status |
检查守护进程是否在线 |
ccbb uninstall |
移除 hook 配置 |
环境变量
| 变量 | 说明 |
|---|---|
CCBB_TCP_HOST |
TCP 服务端监听地址(默认 0.0.0.0) |
CCBB_TCP_PORT |
TCP 服务端监听端口(默认 9876) |
TCP 协议说明
守护进程作为 TCP 服务端,设备作为客户端连接。双方通过 JSON 行协议通信。
从服务端到设备
时间同步:
{"time": [1234567890, 28800]}
快照(状态更新):
{
"total": 1,
"running": 0,
"waiting": 1,
"msg": "approve: Bash",
"entries": ["10:30 Bash: ls -la"],
"tokens": 0,
"tokens_today": 0,
"prompt": {
"id": "req_12345",
"tool": "Bash",
"hint": "ls -la"
},
"context": {
"tool_use_id": "req_12345",
"tool_name": "Bash",
"tool_input": {
"command": "ls -la"
}
}
}
context 字段(可选):包含 Claude Code Hook 的完整原始信息,设备可以根据需要展示或使用此信息。
从设备到服务端
审批决策:
{
"cmd": "permission",
"id": "req_12345",
"decision": "once"
}
决策值可以是:
once:批准deny:拒绝
确认响应(从服务端到设备):
{"ack": "permission", "ok": true, "n": 0}
设备开发指南
基本流程
- 连接到 TCP 服务端:连接到电脑的 IP 和端口
- 接收快照消息:解析 JSON 格式的状态更新
- 显示审批请求:当
waiting > 0时,显示prompt中的信息 - 发送决策:用户操作后,发送包含
cmd、id和decision的 JSON
最小实现示例
import socket
import json
# 连接到服务端
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.connect(("192.168.1.100", 9876))
# 接收消息
while True:
data = s.recv(1024)
if not data:
break
msg = json.loads(data.decode())
# 检查是否有待审批请求
if msg.get("waiting", 0) > 0:
prompt = msg["prompt"]
print(f"审批请求: {prompt['tool']} - {prompt['hint']}")
# 模拟用户输入
decision = input("批准(a)或拒绝(d)? ").strip().lower()
if decision == "a":
s.sendall(json.dumps({
"cmd": "permission",
"id": prompt["id"],
"decision": "once"
}).encode() + b"\n")
elif decision == "d":
s.sendall(json.dumps({
"cmd": "permission",
"id": prompt["id"],
"decision": "deny"
}).encode() + b"\n")
s.close()
macOS 开机自启(launchd)
# 先确认 ccbb 安装路径
which ccbb
# 编辑 plist,将路径替换为上一步的输出
cp extras/dev.ccbb.daemon.plist ~/Library/LaunchAgents/
# 编辑文件,修改 ProgramArguments 中的路径
launchctl load ~/Library/LaunchAgents/dev.ccbb.daemon.plist
卸载:
launchctl unload ~/Library/LaunchAgents/dev.ccbb.daemon.plist
rm ~/Library/LaunchAgents/dev.ccbb.daemon.plist
开发
git clone https://github.com/oh-myfun/claude-code-buddy-bridge
cd claude-code-buddy-bridge
uv sync --extra dev
# 运行测试
uv run pytest
# 直接运行
uv run ccbb daemon -v
项目结构
ccbb/
├── src/ccbb/
│ ├── __init__.py 版本号
│ ├── bridge.py 守护进程核心(TCP 服务端 + Unix Socket 服务器)
│ ├── hook.py 被 Claude Code 调用的 hook 脚本
│ └── cli.py 命令行入口(daemon / install / status)
├── examples/
│ └── tcp_device_client.py TCP 设备客户端示例
├── extras/
│ └── dev.ccbb.daemon.plist macOS launchd 配置模板
├── pyproject.toml
└── README.md
致谢
协议格式参考了 Anthropic 的 claude-desktop-buddy。
核心架构设计参考了 CharmYue/cc-buddy-bridge 和 cuiqingwei/claude-desktop-buddy-bridge——尤其是 EOF 竞争检测、permission_lock 串行化和 fail-open 设计。
License
Project details
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file claude_code_buddy_bridge-0.1.0.tar.gz.
File metadata
- Download URL: claude_code_buddy_bridge-0.1.0.tar.gz
- Upload date:
- Size: 1.7 MB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
d1987b004753ee7d482e672ae573cda652823689554e0a06fa0dcf32013af3f5
|
|
| MD5 |
cd058fdf8a2773c3966f881f3143409e
|
|
| BLAKE2b-256 |
63799280371ea15b8a035c26ad49901ac20166b3e692eb6ada1b59476acdda1c
|
Provenance
The following attestation bundles were made for claude_code_buddy_bridge-0.1.0.tar.gz:
Publisher:
python-publish.yml on oh-myfun/claude-code-buddy-bridge
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
claude_code_buddy_bridge-0.1.0.tar.gz -
Subject digest:
d1987b004753ee7d482e672ae573cda652823689554e0a06fa0dcf32013af3f5 - Sigstore transparency entry: 1458866575
- Sigstore integration time:
-
Permalink:
oh-myfun/claude-code-buddy-bridge@95ecb17503b39bb90edc2209784419882b61bb9a -
Branch / Tag:
refs/tags/v0.1.0 - Owner: https://github.com/oh-myfun
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
python-publish.yml@95ecb17503b39bb90edc2209784419882b61bb9a -
Trigger Event:
release
-
Statement type:
File details
Details for the file claude_code_buddy_bridge-0.1.0-py3-none-any.whl.
File metadata
- Download URL: claude_code_buddy_bridge-0.1.0-py3-none-any.whl
- Upload date:
- Size: 16.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
6dc2a391a61972c86c7a73ad27930800adfc7bc39c53bdc107c73c40eddf7980
|
|
| MD5 |
f0d67f47be9f4c26bba30efb22b44860
|
|
| BLAKE2b-256 |
0dcf4423ed5723c994ac0b7c8b3a9931e586eff77144fb4b96c49ff95e39644b
|
Provenance
The following attestation bundles were made for claude_code_buddy_bridge-0.1.0-py3-none-any.whl:
Publisher:
python-publish.yml on oh-myfun/claude-code-buddy-bridge
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
claude_code_buddy_bridge-0.1.0-py3-none-any.whl -
Subject digest:
6dc2a391a61972c86c7a73ad27930800adfc7bc39c53bdc107c73c40eddf7980 - Sigstore transparency entry: 1458866666
- Sigstore integration time:
-
Permalink:
oh-myfun/claude-code-buddy-bridge@95ecb17503b39bb90edc2209784419882b61bb9a -
Branch / Tag:
refs/tags/v0.1.0 - Owner: https://github.com/oh-myfun
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
python-publish.yml@95ecb17503b39bb90edc2209784419882b61bb9a -
Trigger Event:
release
-
Statement type: