{/* 此页面由 website/scripts/generate-skill-docs.py 从技能的 SKILL.md 自动生成。请编辑源文件 SKILL.md,而非此页面。 */}

Pinggy 隧道

通过 Pinggy 实现零安装的 SSH 本地主机隧道。

技能元数据

来源可选 —— 使用 hermes skills install official/devops/pinggy-tunnel 安装
路径optional-skills/devops/pinggy-tunnel
版本0.1.0
作者Teknium(teknium1),Hermes Agent
许可证MIT
平台linux, macos, windows
标签Pinggy, Tunnel, Networking, SSH, Webhook, Localhost
相关技能cloudflared-quick-tunnel, webhook-subscriptions

参考:完整 SKILL.md

:::info 以下是 Hermes 在触发此技能时加载的完整技能定义。这是技能激活时代理所看到的指令。 :::

Pinggy 隧道技能

使用 Pinggy SSH 反向隧道将本地服务(开发服务器、webhook 接收器、MCP 端点、演示)暴露到公共互联网。无需安装守护进程 —— 用户自带的 SSH 客户端连接到 a.pinggy.io:443,Pinggy 返回一个公共 HTTP/HTTPS URL。

免费层:60 分钟隧道,随机子域名,无需注册。Pro 层($3/月)为可选,需使用令牌。

使用时机

  • 用户要求”暴露这个本地服务”、“分享我的开发服务器”、“让这个 URL 可公开访问”、“隧道端口 N”、“为 webhook 获取公共 URL”
  • 需要在本地任务期间接收 webhook 回调(Stripe、GitHub、Discord、AgentMail)
  • 与远程方共享一次性 HTTP 演示(MCP 服务器、Ollama/vLLM 端点、仪表盘)
  • 主机有 SSH 但没有 cloudflared / ngrok 二进制文件,安装它们过于繁琐

如果主机已配置 cloudflared,优先使用 cloudflared-quick-tunnel 技能 —— Cloudflare 快速隧道不会在 60 分钟后过期。

前提条件

  • ssh 在 PATH 中(ssh -V)。Linux、macOS 和 Windows 10+ 默认自带。无需额外安装。
  • 在隧道启动前,本地服务正在 127.0.0.1:<端口> 上监听。Pinggy 会返回 URL,但直到本地源启动前,这些 URL 返回 502。

可选:

  • PINGGY_TOKEN 环境变量用于付费 Pro 功能(持久性子域名、自定义域名、多隧道、无 60 分钟限制)。免费层无需凭证。

快速参考

# 端口 8000 的纯 HTTP/HTTPS 隧道(免费层)
ssh -p 443 -o StrictHostKeyChecking=no -o ServerAliveInterval=30 \
    -R0:localhost:8000 free@a.pinggy.io
 
# TCP 隧道(数据库、原始 SSH 等)
ssh -p 443 -o StrictHostKeyChecking=no -R0:localhost:5432 tcp@a.pinggy.io
 
# TLS 隧道(Pinggy 无法解密 —— 在源端使用您自己的证书)
ssh -p 443 -o StrictHostKeyChecking=no -R0:localhost:443 tls@a.pinggy.io
 
# 基本认证网关(b:user:pass)
ssh -p 443 -o StrictHostKeyChecking=no -R0:localhost:8000 \
    "b:admin:secret+free@a.pinggy.io"
 
# Bearer 令牌网关(k:token)
ssh -p 443 -o StrictHostKeyChecking=no -R0:localhost:8000 \
    "k:mysecrettoken+free@a.pinggy.io"
 
# IP 白名单(w:CIDR)
ssh -p 443 -o StrictHostKeyChecking=no -R0:localhost:8000 \
    "w:203.0.113.0/24+free@a.pinggy.io"
 
# 启用 CORS + 强制 HTTPS 重定向
ssh -p 443 -o StrictHostKeyChecking=no -R0:localhost:8000 \
    "co+x:https+free@a.pinggy.io"
 
# Pro 层(持久 URL,无 60 分钟限制)
ssh -p 443 -o StrictHostKeyChecking=no -R0:localhost:8000 "$PINGGY_TOKEN+a.pinggy.io"

操作步骤 —— 启动隧道并获取 URL

模型应使用 terminal 工具。隧道必须在共享期间保持存活,因此作为后台进程运行并解析 stdout 中的公共 URL。

1. 确认本地源已启动

curl -sI http://127.0.0.1:8000/ | head -1
# 期望 HTTP/1.x 200(或任何非连接被拒的响应)

如果尚未有服务在监听,请先启动(例如 python3 -m http.server 8000 --bind 127.0.0.1)。Pinggy 会愉快地返回指向空服务的 URL —— 用户将看到 502 直到源启动。

2. 以后台进程启动隧道

使用 terminal(background=True) 并将输出捕获到日志文件(Pinggy 将 URL 打印到 stdout,然后保持连接打开):

LOG=/tmp/pinggy-8000.log
nohup ssh -p 443 \
    -o StrictHostKeyChecking=no \
    -o UserKnownHostsFile=/dev/null \
    -o ServerAliveInterval=30 \
    -o ServerAliveCountMax=3 \
    -R0:localhost:8000 free@a.pinggy.io \
    > "$LOG" 2>&1 &
echo $! > /tmp/pinggy-8000.pid

StrictHostKeyChecking=no + UserKnownHostsFile=/dev/null 跳过首次运行的主机密钥提示。ServerAliveInterval=30 防止空闲 NAT 断开 SSH 会话。

3. 从日志中解析 URL

sleep 4
grep -oE 'https://[a-z0-9-]+\.[a-z]+\.pinggy\.link' /tmp/pinggy-8000.log | head -1

预期输出类似:

You are not authenticated.
Your tunnel will expire in 60 minutes.
http://yqycl-98-162-69-48.a.free.pinggy.link
https://yqycl-98-162-69-48.a.free.pinggy.link

https://...pinggy.link URL 交给用户。

4. 验证

curl -sI https://<the-url>/ | head -3
# 期望 200/302/本地源实际返回的任何状态

如果收到 502 Bad Gateway,SSH 会话已建立但本地源未在监听 —— 先修复步骤 1。

5. 拆除

kill "$(cat /tmp/pinggy-8000.pid)"
# 或者,如果 pid 文件丢失:
pkill -f 'ssh -p 443 .* free@a\.pinggy\.io'

如果有 terminal(background=True) 返回的 session_id,优先使用 process(action='kill', session_id=...)

通过用户名关键词的访问控制

Pinggy 通过 SSH 用户名以 + 分隔叠加控制标志。当参数包含 + 时,始终引用整个 user@host 参数:

关键词效果
b:user:passHTTP Basic 认证网关
k:tokenBearer 令牌头部网关(Authorization: Bearer ***
w:CIDRIP 白名单(单个 IP 或 CIDR,可重复)
co添加 Access-Control-Allow-Origin: *(CORS)
x:https强制 HTTPS —— 自动将 HTTP 重定向到 HTTPS
a:Name:Value添加请求头部
u:Name:Value更新请求头部
r:Name移除请求头部
qr将 URL 的 QR 码打印到 stdout(方便移动端分享)

自由组合:"b:admin:secret+co+x:https+free@a.pinggy.io"

Web 调试器(可选)

Pinggy 可以将入站流量镜像到 localhost:4300 供检查。在 SSH 命令中添加本地转发:

ssh -p 443 -L4300:localhost:4300 -R0:localhost:8000 free@a.pinggy.io

然后在浏览器中打开 http://localhost:4300 查看实时请求/响应对。

注意事项

  • 免费层有 60 分钟硬性限制。 SSH 会话在 60 分钟时终止;URL 失效。对于更长的共享,使用 PINGGY_TOKEN(Pro)或使用 shell 循环自动重启(注意免费层每次重启 URL 会变化)。
  • 免费层 URL 随机且在重启时变化。 不要收藏,不要粘贴到配置文件中。每次从日志中重新解析。
  • 同时的免费隧道限制为每个源 IP 一个。 从同一台机器启动第二个隧道通常会杀死第一个。Pro 层解除此限制。
  • 用户名中的 + 必须加引号。ssh ... b:admin:secret+free@a.pinggy.io 在 bash 中有效,但在将 + 特殊处理或以编程方式组装时可能出问题。始终用双引号包裹。
  • 没有访问控制标志,不要隧道传输敏感内容。 裸 HTTP 隧道可被任何拥有 URL 的人访问。对于非公共服务,使用 b:k:w:
  • process(action='log') 可能错过 SSH 横幅输出。 Pinggy 打印 URL 后 SSH 会话进入交互模式。始终重定向到日志文件并直接 grep 文件 —— 与 cloudflared-quick-tunnel 相同模式。
  • 首次运行的主机密钥提示。 默认 OpenSSH 配置要求用户接受 Pinggy 的主机密钥。对于无人值守运行,始终传递 -o StrictHostKeyChecking=no -o UserKnownHostsFile=/dev/null
  • TCP 和 TLS 隧道返回 <子域名>.a.pinggy.online:<端口> 对,而非 HTTPS URL。 使用不同的正则表达式解析(tcp:// 和端口)。不要假定每个 Pinggy 隧道都是 HTTP。
  • Pro 模式需要将令牌作为用户名而非标志使用。 使用 "$PINGGY_TOKEN+a.pinggy.io"(没有 free@)。使用令牌还可以添加 :persistent 以获得稳定的子域名 —— 参见 pinggy.io/docs/

配方

将本地源与 Pinggy 隧道结合的组合模式。每个配方自包含 —— 启动源、启动隧道、解析 URL、交给用户。

配方 1 — 接收 webhook 回调

当外部服务(Stripe、GitHub、Discord、AgentMail 等)需要在本地任务期间向可公开访问的 URL 发送 POST 请求时使用。

# 1. 小型捕获服务器:每个请求追加到 /tmp/webhook-hits.log
cat >/tmp/webhook-server.py <<'PY'
import http.server, json, datetime, pathlib
LOG = pathlib.Path("/tmp/webhook-hits.log")
class H(http.server.BaseHTTPRequestHandler):
    def _capture(self):
        n = int(self.headers.get("content-length") or 0)
        body = self.rfile.read(n).decode("utf-8", "replace") if n else ""
        rec = {"t": datetime.datetime.utcnow().isoformat(), "path": self.path,
               "method": self.command, "headers": dict(self.headers), "body": body}
        with LOG.open("a") as f: f.write(json.dumps(rec) + "\n")
        self.send_response(200); self.send_header("content-type","application/json")
        self.end_headers(); self.wfile.write(b'{"ok":true}\n')
    def do_GET(self): self._capture()
    def do_POST(self): self._capture()
    def log_message(self,*a,**k): pass
http.server.HTTPServer(("127.0.0.1", 18080), H).serve_forever()
PY
nohup python3 /tmp/webhook-server.py >/tmp/webhook-server.log 2>&1 &
echo $! >/tmp/webhook-server.pid
 
# 2. 隧道 —— bearer 令牌网关,防止随机人员污染捕获日志
nohup ssh -p 443 -o StrictHostKeyChecking=no -o UserKnownHostsFile=/dev/null \
    -o ServerAliveInterval=30 \
    -R0:localhost:18080 "k:$(openssl rand -hex 12)+free@a.pinggy.io" \
    >/tmp/webhook-pinggy.log 2>&1 &
echo $! >/tmp/webhook-pinggy.pid
sleep 5
URL=$(grep -oE 'https://[a-z0-9-]+\.[a-z]+\.pinggy\.link' /tmp/webhook-pinggy.log | head -1)
echo "Webhook URL: $URL"
 
# 3. 代理工作时,观察命中的记录
tail -f /tmp/webhook-hits.log

$URL 交给需要回调您的服务。拆除:kill $(cat /tmp/webhook-server.pid) $(cat /tmp/webhook-pinggy.pid)

配方 2 — 通过 HTTP/SSE 暴露 MCP 服务器

当远程 MCP 客户端(另一台机器上的 Claude Desktop、队友的编辑器等)需要访问本地机器上运行的 MCP 服务器时使用。仅适用于支持 HTTP 传输的 MCP 服务器 —— stdio 模式的服务器无法被隧道传输。

# 1. 以 HTTP 模式启动 MCP 服务器(示例:FastMCP 服务器在端口 8765)
nohup python3 my_mcp_server.py --transport http --port 8765 \
    >/tmp/mcp-server.log 2>&1 &
echo $! >/tmp/mcp-server.pid
 
# 2. 使用 bearer 令牌隧道 —— MCP 流量不应向互联网开放
TOKEN=$(openssl rand -hex 16)
nohup ssh -p 443 -o StrictHostKeyChecking=no -o UserKnownHostsFile=/dev/null \
    -o ServerAliveInterval=30 \
    -R0:localhost:8765 "k:$TOKEN+free@a.pinggy.io" \
    >/tmp/mcp-pinggy.log 2>&1 &
echo $! >/tmp/mcp-pinggy.pid
sleep 5
URL=$(grep -oE 'https://[a-z0-9-]+\.[a-z]+\.pinggy\.link' /tmp/mcp-pinggy.log | head -1)
echo "MCP URL: $URL"
echo "Bearer token: $TOKEN"

远程客户端连接到 $URL,使用 Authorization: Bearer <TOKEN>。Hermes 自身的原生 MCP 客户端配置:{"transport": "http", "url": "<URL>", "headers": {"Authorization": "Bearer <TOKEN>"}}

配方 3 — 暴露本地 LLM 端点(Ollama / vLLM / llama.cpp)

与远程调用者(另一个代理、手机、队友)共享本地模型。Ollama 监听 :11434,vLLM 和 llama.cpp 通常在 :8000

# 前置条件:模型服务器已在 127.0.0.1:11434 运行(Ollama 默认)
TOKEN=$(openssl rand -hex 16)
nohup ssh -p 443 -o StrictHostKeyChecking=no -o UserKnownHostsFile=/dev/null \
    -o ServerAliveInterval=30 \
    -R0:localhost:11434 "k:$TOKEN+co+free@a.pinggy.io" \
    >/tmp/llm-pinggy.log 2>&1 &
echo $! >/tmp/llm-pinggy.pid
sleep 5
URL=$(grep -oE 'https://[a-z0-9-]+\.[a-z]+\.pinggy\.link' /tmp/llm-pinggy.log | head -1)
echo "Endpoint: $URL"
echo "Token:    $TOKEN"
 
# 验证
curl -s "$URL/api/tags" -H "Authorization: Bearer <TOKEN>" | head

co 启用 CORS,使浏览器调用者可以访问端点。仅后端调用者时去掉 co。对于 OpenAI 兼容的 vLLM/llama.cpp 端点,调用者使用基础 URL $URL/v1,附带 Authorization: Bearer <TOKEN> —— 但请注意 Pinggy 不会剥离/替换正文中的任何内容,因此模型服务器本身会看到 Pinggy 的令牌;本地服务器应配置为忽略认证(它已经在 127.0.0.1 上),让 Pinggy 负责网关控制。

配方 4 — 使用一次性密码共享开发服务器

最快的”让队友试用我运行的应用程序”模式。随机密码,仅打印一次,Ctrl-C 时终止。

PASS=$(openssl rand -base64 12 | tr -d '+/=' | head -c 12)
echo "Dev server password: $PASS"
ssh -p 443 -o StrictHostKeyChecking=no -o UserKnownHostsFile=/dev/null \
    -o ServerAliveInterval=30 \
    -R0:localhost:3000 "b:dev:$PASS+co+x:https+free@a.pinggy.io"
# URL 打印到终端。分享 URL + 密码。Ctrl-C 拆除。

b:dev:$PASS 使用 HTTP Basic 认证保护 URL。x:https 强制 TLS。co 为 SPA 前端添加 CORS。

验证

# 端到端:启动一个简单源,建立隧道,访问它,拆除
python3 -m http.server 18000 --bind 127.0.0.1 >/tmp/origin.log 2>&1 &
ORIGIN_PID=$!
 
nohup ssh -p 443 \
    -o StrictHostKeyChecking=no \
    -o UserKnownHostsFile=/dev/null \
    -R0:localhost:18000 free@a.pinggy.io >/tmp/pinggy-verify.log 2>&1 &
SSH_PID=$!
 
sleep 5
URL=$(grep -oE 'https://[a-z0-9-]+\.[a-z]+\.pinggy\.link' /tmp/pinggy-verify.log | head -1)
echo "URL: $URL"
curl -sI "$URL/" | head -1
 
kill "$SSH_PID" "$ORIGIN_PID"

预期:一个 pinggy.link URL 和 curl head 返回 HTTP/2 200