跳轉到主要內容
Channels 處於研究預覽階段,需要 Claude Code v2.1.80 或更新版本。它們需要 claude.ai 登入。不支援 Console 和 API 金鑰驗證。Team 和 Enterprise 組織必須明確啟用它們
Channel 是一個 MCP 伺服器,將事件推送到 Claude Code 工作階段,以便 Claude 可以對終端機外發生的事情做出反應。 您可以建立單向或雙向 channel。單向 channel 轉發警報、webhooks 或監控事件供 Claude 處理。雙向 channel(如聊天橋接)也會公開回覆工具,以便 Claude 可以傳送訊息回去。具有受信任寄件者路徑的 channel 也可以選擇加入中繼權限提示,以便您可以遠端批准或拒絕工具使用。 本頁涵蓋: 若要使用現有 channel 而不是建立一個,請參閱 Channels。Telegram、Discord、iMessage 和 fakechat 包含在研究預覽中。

概述

Channel 是在與 Claude Code 相同的機器上執行的 MCP 伺服器。Claude Code 將其作為子程序生成並透過 stdio 進行通訊。您的 channel 伺服器是外部系統和 Claude Code 工作階段之間的橋接:
  • 聊天平台(Telegram、Discord):您的外掛程式在本機執行並輪詢平台的 API 以取得新訊息。當有人傳送 DM 給您的機器人時,外掛程式會接收訊息並將其轉發給 Claude。無需公開 URL。
  • Webhooks(CI、監控):您的伺服器在本機 HTTP 連接埠上監聽。外部系統 POST 到該連接埠,您的伺服器將承載推送給 Claude。
架構圖,顯示外部系統連接到您的本機 channel 伺服器,該伺服器透過 stdio 與 Claude Code 通訊

您需要什麼

唯一的硬性要求是 @modelcontextprotocol/sdk 套件和 Node.js 相容的執行時。BunNodeDeno 都可以運作。研究預覽中的預先建立外掛程式使用 Bun,但您的 channel 不一定要。 您的伺服器需要:
  1. 聲明 claude/channel 功能,以便 Claude Code 註冊通知監聽器
  2. 當發生某事時發出 notifications/claude/channel 事件
  3. 透過 stdio transport 連接(Claude Code 將您的伺服器作為子程序生成)
伺服器選項通知格式部分詳細涵蓋每一項。請參閱範例:建立 webhook 接收器以取得完整逐步解說。 在研究預覽期間,自訂 channels 不在核准允許清單上。使用 --dangerously-load-development-channels 在本機測試。請參閱在研究預覽期間測試以取得詳細資訊。

範例:建立 webhook 接收器

本逐步解說建立一個單一檔案伺服器,該伺服器監聽 HTTP 要求並將其轉發到您的 Claude Code 工作階段。最後,任何可以傳送 HTTP POST 的東西(如 CI 管道、監控警報或 curl 命令)都可以將事件推送給 Claude。 此範例使用 Bun 作為執行時,用於其內建 HTTP 伺服器和 TypeScript 支援。您可以改用 NodeDeno;唯一的要求是 MCP SDK
1

建立專案

建立新目錄並安裝 MCP SDK:
mkdir webhook-channel && cd webhook-channel
bun add @modelcontextprotocol/sdk
2

編寫 channel 伺服器

建立名為 webhook.ts 的檔案。這是您的整個 channel 伺服器:它透過 stdio 連接到 Claude Code,並在連接埠 8788 上監聽 HTTP POST。當要求到達時,它將主體作為 channel 事件推送給 Claude。
webhook.ts
#!/usr/bin/env bun
import { Server } from '@modelcontextprotocol/sdk/server/index.js'
import { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js'

// 建立 MCP 伺服器並將其聲明為 channel
const mcp = new Server(
  { name: 'webhook', version: '0.0.1' },
  {
    // 這個金鑰是使其成為 channel 的原因 — Claude Code 為其註冊監聽器
    capabilities: { experimental: { 'claude/channel': {} } },
    // 新增到 Claude 的系統提示,以便它知道如何處理這些事件
    instructions: 'Events from the webhook channel arrive as <channel source="webhook" ...>. They are one-way: read them and act, no reply expected.',
  },
)

// 透過 stdio 連接到 Claude Code(Claude Code 生成此程序)
await mcp.connect(new StdioServerTransport())

// 啟動 HTTP 伺服器,將每個 POST 轉發給 Claude
Bun.serve({
  port: 8788,  // 任何開放連接埠都可以
  // 僅限 localhost:此機器外的任何東西都無法 POST
  hostname: '127.0.0.1',
  async fetch(req) {
    const body = await req.text()
    await mcp.notification({
      method: 'notifications/claude/channel',
      params: {
        content: body,  // 成為 <channel> 標籤的主體
        // 每個金鑰都成為標籤屬性,例如 <channel path="/" method="POST">
        meta: { path: new URL(req.url).pathname, method: req.method },
      },
    })
    return new Response('ok')
  },
})
該檔案按順序執行三項操作:
  • 伺服器配置:使用 claude/channel 在其功能中建立 MCP 伺服器,這是告訴 Claude Code 這是 channel 的原因。instructions 字串進入 Claude 的系統提示:告訴 Claude 期望什麼事件、是否回覆,以及如果應該回覆,使用哪個工具和傳遞回哪個屬性(如 chat_id)。
  • Stdio 連接:透過 stdin/stdout 連接到 Claude Code。這對任何 MCP 伺服器 都是標準的:Claude Code 將其作為子程序生成。
  • HTTP 監聽器:在連接埠 8788 上啟動本機網頁伺服器。每個 POST 主體都透過 mcp.notification() 作為 channel 事件轉發給 Claude。content 成為事件主體,每個 meta 項目成為 <channel> 標籤上的屬性。監聽器需要存取 mcp 實例,因此它在同一程序中執行。對於較大的專案,您可以將其分割成單獨的模組。
3

向 Claude Code 註冊您的伺服器

將伺服器新增到您的 MCP 配置,以便 Claude Code 知道如何啟動它。對於同一目錄中的專案層級 .mcp.json,使用相對路徑。對於 ~/.claude.json 中的使用者層級配置,使用完整絕對路徑,以便可以從任何專案找到伺服器:
.mcp.json
{
  "mcpServers": {
    "webhook": { "command": "bun", "args": ["./webhook.ts"] }
  }
}
Claude Code 在啟動時讀取您的 MCP 配置並將每個伺服器作為子程序生成。
4

測試它

在研究預覽期間,自訂 channels 不在允許清單上,因此使用開發旗標啟動 Claude Code:
claude --dangerously-load-development-channels server:webhook
當 Claude Code 啟動時,它讀取您的 MCP 配置,將您的 webhook.ts 作為子程序生成,HTTP 監聽器自動在您配置的連接埠上啟動(此範例中為 8788)。您不需要自己執行伺服器。如果您看到’被組織政策阻止’,您的 Team 或 Enterprise 管理員需要先啟用 channels在單獨的終端機中,透過傳送帶有訊息的 HTTP POST 來模擬 webhook 到您的伺服器。此範例將 CI 失敗警報傳送到連接埠 8788(或您配置的任何連接埠):
curl -X POST localhost:8788 -d "build failed on main: https://ci.example.com/run/1234"
承載作為 <channel> 標籤到達您的 Claude Code 工作階段:
<channel source="webhook" path="/" method="POST">build failed on main: https://ci.example.com/run/1234</channel>
在您的 Claude Code 終端機中,您會看到 Claude 接收訊息並開始回應:讀取檔案、執行命令或訊息要求的任何內容。這是一個單向 channel,因此 Claude 在您的工作階段中採取行動,但不會透過 webhook 傳送任何內容回去。若要新增回覆,請參閱公開回覆工具如果事件未到達,診斷取決於 curl 返回的內容:
  • curl 成功但沒有任何內容到達 Claude:在您的工作階段中執行 /mcp 以檢查伺服器的狀態。「無法連接」通常表示伺服器檔案中的相依性或匯入錯誤;檢查 ~/.claude/debug/<session-id>.txt 的偵錯日誌以取得 stderr 追蹤。
  • curl 失敗,出現「連接被拒絕」:連接埠要麼尚未繫結,要麼來自較早執行的過時程序正在佔用它。lsof -i :<port> 顯示正在監聽的內容;在重新啟動工作階段之前 kill 過時程序。
fakechat 伺服器使用網頁 UI、檔案附件和用於雙向聊天的回覆工具擴展此模式。

在研究預覽期間測試

在研究預覽期間,每個 channel 都必須在核准允許清單上才能註冊。開發旗標在確認提示後繞過特定項目的允許清單。此範例顯示兩種項目類型:
# 測試您正在開發的外掛程式
claude --dangerously-load-development-channels plugin:yourplugin@yourmarketplace

# 測試裸 .mcp.json 伺服器(尚無外掛程式包裝)
claude --dangerously-load-development-channels server:webhook
繞過是按項目進行的。將此旗標與 --channels 結合不會將繞過擴展到 --channels 項目。在研究預覽期間,核准允許清單由 Anthropic 策劃,因此您的 channel 在您建立和測試時保持在開發旗標上。
此旗標僅跳過允許清單。channelsEnabled 組織政策仍然適用。不要使用它來執行來自不受信任來源的 channels。

伺服器選項

Channel 在 Server 建構函式中設定這些選項。instructionscapabilities.tools 欄位是標準 MCPcapabilities.experimental['claude/channel']capabilities.experimental['claude/channel/permission'] 是 channel 特定的新增項目:
欄位類型描述
capabilities.experimental['claude/channel']object必需。始終為 {}。存在會註冊通知監聽器。
capabilities.experimental['claude/channel/permission']object選用。始終為 {}。聲明此 channel 可以接收權限中繼要求。聲明時,Claude Code 會將工具批准提示轉發到您的 channel,以便您可以遠端批准或拒絕它們。請參閱中繼權限提示
capabilities.toolsobject僅限雙向。始終為 {}。標準 MCP 工具功能。請參閱公開回覆工具
instructionsstring建議。新增到 Claude 的系統提示。告訴 Claude 期望什麼事件、<channel> 標籤屬性的含義、是否回覆,如果是,使用哪個工具以及傳遞回哪個屬性(如 chat_id)。
若要建立單向 channel,請省略 capabilities.tools。此範例顯示設定了 channel 功能、工具和指示的雙向設定:
import { Server } from '@modelcontextprotocol/sdk/server/index.js'

const mcp = new Server(
  { name: 'your-channel', version: '0.0.1' },
  {
    capabilities: {
      experimental: { 'claude/channel': {} },  // 註冊 channel 監聽器
      tools: {},  // 對於單向 channels 省略
    },
    // 新增到 Claude 的系統提示,以便它知道如何處理您的事件
    instructions: 'Messages arrive as <channel source="your-channel" ...>. Reply with the reply tool.',
  },
)
若要推送事件,請使用方法 notifications/claude/channel 呼叫 mcp.notification()。參數在下一部分中。

通知格式

您的伺服器發出 notifications/claude/channel 和兩個參數:
欄位類型描述
contentstring事件主體。作為 <channel> 標籤的主體傳遞。
metaRecord<string, string>選用。每個項目成為 <channel> 標籤上的屬性,用於路由上下文,如聊天 ID、寄件者名稱或警報嚴重性。金鑰必須是識別碼:僅限字母、數字和底線。包含連字號或其他字元的金鑰會被無聲地捨棄。
您的伺服器透過在 Server 實例上呼叫 mcp.notification() 來推送事件。此範例推送帶有兩個中繼金鑰的 CI 失敗警報:
await mcp.notification({
  method: 'notifications/claude/channel',
  params: {
    content: 'build failed on main: https://ci.example.com/run/1234',
    meta: { severity: 'high', run_id: '1234' },
  },
})
事件在 Claude 的上下文中到達,包裝在 <channel> 標籤中。source 屬性從您的伺服器配置的名稱自動設定:
<channel source="your-channel" severity="high" run_id="1234">
build failed on main: https://ci.example.com/run/1234
</channel>

公開回覆工具

如果您的 channel 是雙向的(如聊天橋接而不是警報轉發器),請公開標準 MCP 工具,Claude 可以呼叫該工具來傳送訊息回去。工具註冊沒有任何 channel 特定的內容。回覆工具有三個元件:
  1. 您的 Server 建構函式功能中的 tools: {} 項目,以便 Claude Code 發現工具
  2. 定義工具架構並實現傳送邏輯的工具處理程式
  3. 您的 Server 建構函式中的 instructions 字串,告訴 Claude 何時以及如何呼叫工具
若要將這些新增到上面的 webhook 接收器
1

啟用工具發現

webhook.ts 中的 Server 建構函式中,將 tools: {} 新增到功能,以便 Claude Code 知道您的伺服器提供工具:
capabilities: {
  experimental: { 'claude/channel': {} },
  tools: {},  // 啟用工具發現
},
2

註冊回覆工具

將以下內容新增到 webhook.tsimport 位於檔案頂部,與您的其他匯入一起;兩個處理程式位於 Server 建構函式和 mcp.connect() 之間。這會註冊一個 reply 工具,Claude 可以使用 chat_idtext 呼叫:
// 在 webhook.ts 頂部新增此匯入
import { ListToolsRequestSchema, CallToolRequestSchema } from '@modelcontextprotocol/sdk/types.js'

// Claude 在啟動時查詢此項以發現您的伺服器提供什麼工具
mcp.setRequestHandler(ListToolsRequestSchema, async () => ({
  tools: [{
    name: 'reply',
    description: 'Send a message back over this channel',
    // inputSchema 告訴 Claude 要傳遞什麼引數
    inputSchema: {
      type: 'object',
      properties: {
        chat_id: { type: 'string', description: 'The conversation to reply in' },
        text: { type: 'string', description: 'The message to send' },
      },
      required: ['chat_id', 'text'],
    },
  }],
}))

// Claude 想要叫用工具時呼叫此項
mcp.setRequestHandler(CallToolRequestSchema, async req => {
  if (req.params.name === 'reply') {
    const { chat_id, text } = req.params.arguments as { chat_id: string; text: string }
    // send() 是您的出站:POST 到您的聊天平台,或用於本機
    // 測試下面完整範例中顯示的 SSE 廣播。
    send(`Reply to ${chat_id}: ${text}`)
    return { content: [{ type: 'text', text: 'sent' }] }
  }
  throw new Error(`unknown tool: ${req.params.name}`)
})
3

更新指示

更新 Server 建構函式中的 instructions 字串,以便 Claude 知道透過工具將回覆路由回去。此範例告訴 Claude 從入站標籤傳遞 chat_id
instructions: 'Messages arrive as <channel source="webhook" chat_id="...">. Reply with the reply tool, passing the chat_id from the tag.'
以下是具有雙向支援的完整 webhook.ts。出站回覆透過 GET /events 使用 Server-Sent Events (SSE) 串流,因此 curl -N localhost:8788/events 可以即時觀看它們;入站聊天到達 POST /
"Full
#!/usr/bin/env bun
import { Server } from '@modelcontextprotocol/sdk/server/index.js'
import { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js'
import { ListToolsRequestSchema, CallToolRequestSchema } from '@modelcontextprotocol/sdk/types.js'

// --- 出站:寫入 /events 上任何 curl -N 監聽器 ---
// 真實橋接會改為 POST 到您的聊天平台。
const listeners = new Set<(chunk: string) => void>()
function send(text: string) {
  const chunk = text.split('\n').map(l => `data: ${l}\n`).join('') + '\n'
  for (const emit of listeners) emit(chunk)
}

const mcp = new Server(
  { name: 'webhook', version: '0.0.1' },
  {
    capabilities: {
      experimental: { 'claude/channel': {} },
      tools: {},
    },
    instructions: 'Messages arrive as <channel source="webhook" chat_id="...">. Reply with the reply tool, passing the chat_id from the tag.',
  },
)

mcp.setRequestHandler(ListToolsRequestSchema, async () => ({
  tools: [{
    name: 'reply',
    description: 'Send a message back over this channel',
    inputSchema: {
      type: 'object',
      properties: {
        chat_id: { type: 'string', description: 'The conversation to reply in' },
        text: { type: 'string', description: 'The message to send' },
      },
      required: ['chat_id', 'text'],
    },
  }],
}))

mcp.setRequestHandler(CallToolRequestSchema, async req => {
  if (req.params.name === 'reply') {
    const { chat_id, text } = req.params.arguments as { chat_id: string; text: string }
    send(`Reply to ${chat_id}: ${text}`)
    return { content: [{ type: 'text', text: 'sent' }] }
  }
  throw new Error(`unknown tool: ${req.params.name}`)
})

await mcp.connect(new StdioServerTransport())

let nextId = 1
Bun.serve({
  port: 8788,
  hostname: '127.0.0.1',
  idleTimeout: 0,  // 不要關閉閒置 SSE 串流
  async fetch(req) {
    const url = new URL(req.url)

    // GET /events:SSE 串流,所以 curl -N 可以即時觀看 Claude 的回覆
    if (req.method === 'GET' && url.pathname === '/events') {
      const stream = new ReadableStream({
        start(ctrl) {
          ctrl.enqueue(': connected\n\n')  // 所以 curl 立即顯示某些內容
          const emit = (chunk: string) => ctrl.enqueue(chunk)
          listeners.add(emit)
          req.signal.addEventListener('abort', () => listeners.delete(emit))
        },
      })
      return new Response(stream, {
        headers: { 'Content-Type': 'text/event-stream', 'Cache-Control': 'no-cache' },
      })
    }

    // POST:作為 channel 事件轉發給 Claude
    const body = await req.text()
    const chat_id = String(nextId++)
    await mcp.notification({
      method: 'notifications/claude/channel',
      params: {
        content: body,
        meta: { chat_id, path: url.pathname, method: req.method },
      },
    })
    return new Response('ok')
  },
})
fakechat 伺服器顯示了一個更完整的範例,具有檔案附件和訊息編輯。

閘道入站訊息

未閘道的 channel 是提示注入向量。任何可以到達您的端點的人都可以在 Claude 前面放置文字。監聽聊天平台或公開端點的 channel 在發出任何內容之前需要真正的寄件者檢查。 在呼叫 mcp.notification() 之前,根據允許清單檢查寄件者。此範例捨棄來自不在集合中的寄件者的任何訊息:
const allowed = new Set(loadAllowlist())  // 從您的 access.json 或等效項

// 在您的訊息處理程式中,在發出之前:
if (!allowed.has(message.from.id)) {  // 寄件者,不是房間
  return  // 無聲地捨棄
}
await mcp.notification({ ... })
根據寄件者的身份而不是聊天或房間身份進行閘道:範例中的 message.from.id,而不是 message.chat.id。在群組聊天中,這些不同,根據房間進行閘道會讓允許清單群組中的任何人將訊息注入到工作階段中。 TelegramDiscord channels 以相同方式根據寄件者允許清單進行閘道。它們透過配對啟動清單:使用者傳送 DM 給機器人,機器人回覆配對代碼,使用者在其 Claude Code 工作階段中批准它,其平台 ID 被新增。請參閱任一實現以取得完整配對流程。iMessage channel 採用不同的方法:它在啟動時從 Messages 資料庫偵測使用者自己的位址並自動讓它們通過,其他寄件者按控制代碼新增。

中繼權限提示

權限中繼需要 Claude Code v2.1.81 或更新版本。較早版本會忽略 claude/channel/permission 功能。
當 Claude 呼叫需要批准的工具時,本機終端機對話框開啟,工作階段等待。雙向 channel 可以選擇加入以在平行接收相同提示,並將其中繼到您在另一台裝置上。兩者都保持活躍:您可以在終端機或手機上回答,Claude Code 應用先到達的答案並關閉另一個。 中繼涵蓋工具使用批准,如 Bash、Write 和 Edit。專案信任和 MCP 伺服器同意對話框不中繼;這些僅在本機終端機中出現。

中繼如何運作

當權限提示開啟時,中繼迴圈有四個步驟:
  1. Claude Code 生成短要求 ID 並通知您的伺服器
  2. 您的伺服器將提示和 ID 轉發到您的聊天應用
  3. 遠端使用者回覆是或否以及該 ID
  4. 您的入站處理程式將回覆解析為判決,Claude Code 僅在 ID 符合開啟要求時應用它
本機終端機對話框在所有這一切中保持開啟。如果終端機上的某人在遠端判決到達之前回答,該答案會改為應用,待處理的遠端要求會被捨棄。 序列圖:Claude Code 傳送 permission_request 通知到 channel 伺服器,伺服器格式化並傳送提示到聊天應用,人類回覆判決,伺服器將該回覆解析為權限通知回到 Claude Code

權限要求欄位

來自 Claude Code 的出站通知是 notifications/claude/channel/permission_request。像頻道通知一樣,傳輸是標準 MCP,但方法和架構是 Claude Code 擴展。params 物件有四個字串欄位,您的伺服器將其格式化為出站提示:
欄位描述
request_id五個小寫字母,從 a-z 中抽取,不包括 l,因此在手機上輸入時永遠不會讀作 1I。將其包含在您的出站提示中,以便可以在回覆中回顯。Claude Code 僅接受帶有其發出的 ID 的判決。本機終端機對話框不顯示此 ID,因此您的出站處理程式是了解它的唯一方式。
tool_nameClaude 想要使用的工具的名稱,例如 BashWrite
description此特定工具呼叫執行的操作的人類可讀摘要,與本機終端機對話框顯示的文字相同。對於 Bash 呼叫,這是 Claude 對命令的描述,或如果未給出,則是命令本身。
input_preview工具的引數作為 JSON 字串,截斷為 200 個字元。對於 Bash,這是命令;對於 Write,它是檔案路徑和內容的前綴。如果您只有一行訊息的空間,請從您的提示中省略它。您的伺服器決定要顯示什麼。
您的伺服器傳送回的判決是 notifications/claude/channel/permission,有兩個欄位:request_id 回顯上面的 ID,behavior 設定為 'allow''deny'。允許讓工具呼叫繼續;拒絕會拒絕它,與在本機對話框中回答’否’相同。兩個判決都不影響未來的呼叫。

將中繼新增到聊天橋接

將權限中繼新增到雙向 channel 需要三個元件:
  1. 您的 Server 建構函式中 experimental 功能下的 claude/channel/permission: {} 項目,以便 Claude Code 知道轉發提示
  2. notifications/claude/channel/permission_request 的通知處理程式,格式化提示並透過您的平台 API 傳送出去
  3. 您的入站訊息處理程式中的檢查,識別 yes <id>no <id> 並發出 notifications/claude/channel/permission 判決通知,而不是將文字轉發給 Claude
僅在您的 channel 驗證寄件者時聲明功能,因為任何可以透過您的 channel 回覆的人都可以批准或拒絕您工作階段中的工具使用。 若要將這些新增到公開回覆工具中組裝的雙向聊天橋接:
1

聲明權限功能

在您的 Server 建構函式中,在 experimental 下的 claude/channel 旁邊新增 claude/channel/permission: {}
capabilities: {
  experimental: {
    'claude/channel': {},
    'claude/channel/permission': {},  // 選擇加入權限中繼
  },
  tools: {},
},
2

處理傳入要求

在您的 Server 建構函式和 mcp.connect() 之間註冊通知處理程式。當權限對話框開啟時,Claude Code 使用四個要求欄位呼叫它。您的處理程式為您的平台格式化提示,並包含使用 ID 回覆的指示:
import { z } from 'zod'

// setNotificationHandler 透過方法欄位上的 z.literal 進行路由,
// 所以此架構既是驗證器又是分派金鑰
const PermissionRequestSchema = z.object({
  method: z.literal('notifications/claude/channel/permission_request'),
  params: z.object({
    request_id: z.string(),     // 五個小寫字母,在您的提示中逐字包含
    tool_name: z.string(),      // 例如 "Bash"、"Write"
    description: z.string(),    // 此呼叫的人類可讀摘要
    input_preview: z.string(),  // 工具引數作為 JSON,截斷至 ~200 個字元
  }),
})

mcp.setNotificationHandler(PermissionRequestSchema, async ({ params }) => {
  // send() 是您的出站:POST 到您的聊天平台,或用於本機
  // 測試下面完整範例中顯示的 SSE 廣播。
  send(
    `Claude wants to run ${params.tool_name}: ${params.description}\n\n` +
    // 指示中的 ID 是您的入站處理程式在步驟 3 中解析的內容
    `Reply "yes ${params.request_id}" or "no ${params.request_id}"`,
  )
})
3

在您的入站處理程式中攔截判決

您的入站處理程式是接收來自您的平台的訊息的迴圈或回呼:與您根據寄件者進行閘道和發出 notifications/claude/channel 以將聊天轉發給 Claude 的地方相同。在聊天轉發呼叫之前新增檢查,識別判決格式並改為發出權限通知。正規表達式符合 Claude Code 生成的 ID 格式:五個字母,永遠不是 l/i 旗標容許手機自動更正將回覆大寫;在傳送回之前將擷取的 ID 小寫。
// 符合 "y abcde"、"yes abcde"、"n abcde"、"no abcde"
// [a-km-z] 是 Claude Code 使用的 ID 字母表(小寫,跳過 'l')
// /i 容許手機自動更正;在傳送前小寫擷取
const PERMISSION_REPLY_RE = /^\s*(y|yes|n|no)\s+([a-km-z]{5})\s*$/i

async function onInbound(message: PlatformMessage) {
  if (!allowed.has(message.from.id)) return  // 首先根據寄件者進行閘道

  const m = PERMISSION_REPLY_RE.exec(message.text)
  if (m) {
    // m[1] 是判決詞,m[2] 是要求 ID
    // 改為將判決通知發出回 Claude Code,而不是聊天
    await mcp.notification({
      method: 'notifications/claude/channel/permission',
      params: {
        request_id: m[2].toLowerCase(),  // 在自動更正大寫的情況下正規化
        behavior: m[1].toLowerCase().startsWith('y') ? 'allow' : 'deny',
      },
    })
    return  // 作為判決處理,不要也轉發為聊天
  }

  // 不符合判決格式:落入正常聊天路徑
  await mcp.notification({
    method: 'notifications/claude/channel',
    params: { content: message.text, meta: { chat_id: String(message.chat.id) } },
  })
}
Claude Code 也保持本機終端機對話框開啟,因此您可以在任一位置回答,先到達的答案會被應用。不完全符合預期格式的遠端回覆以兩種方式之一失敗,在兩種情況下對話框都保持開啟:
  • 不同格式:您的入站處理程式的正規表達式無法符合,因此 approve ityes 之類的文字(沒有 ID)會作為正常訊息落入 Claude。
  • 正確格式,錯誤 ID:您的伺服器發出判決,但 Claude Code 找不到具有該 ID 的開啟要求並無聲地捨棄它。

完整範例

下面組裝的 webhook.ts 結合了本頁的所有三個擴展:回覆工具、寄件者閘道和權限中繼。如果您從這裡開始,您還需要初始逐步解說中的專案設定和 .mcp.json 項目 為了使兩個方向都可以從 curl 測試,HTTP 監聽器提供兩個路徑:
  • GET /events:保持 SSE 串流開啟並將每個出站訊息推送為 data: 行,因此 curl -N 可以即時觀看 Claude 的回覆和權限提示到達。
  • POST /:入站側,與之前相同的處理程式,現在在聊天轉發分支之前插入了判決格式檢查。
"Full
#!/usr/bin/env bun
import { Server } from '@modelcontextprotocol/sdk/server/index.js'
import { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js'
import { ListToolsRequestSchema, CallToolRequestSchema } from '@modelcontextprotocol/sdk/types.js'
import { z } from 'zod'

// --- 出站:寫入 /events 上任何 curl -N 監聽器 ---
// 真實橋接會改為 POST 到您的聊天平台。
const listeners = new Set<(chunk: string) => void>()
function send(text: string) {
  const chunk = text.split('\n').map(l => `data: ${l}\n`).join('') + '\n'
  for (const emit of listeners) emit(chunk)
}

// 寄件者允許清單。對於本機逐步解說,我們信任單一 X-Sender
// 標頭值 "dev";真實橋接會檢查平台的使用者 ID。
const allowed = new Set(['dev'])

const mcp = new Server(
  { name: 'webhook', version: '0.0.1' },
  {
    capabilities: {
      experimental: {
        'claude/channel': {},
        'claude/channel/permission': {},  // 選擇加入權限中繼
      },
      tools: {},
    },
    instructions:
      'Messages arrive as <channel source="webhook" chat_id="...">. ' +
      'Reply with the reply tool, passing the chat_id from the tag.',
  },
)

// --- 回覆工具:Claude 呼叫此項以傳送訊息回去 ---
mcp.setRequestHandler(ListToolsRequestSchema, async () => ({
  tools: [{
    name: 'reply',
    description: 'Send a message back over this channel',
    inputSchema: {
      type: 'object',
      properties: {
        chat_id: { type: 'string', description: 'The conversation to reply in' },
        text: { type: 'string', description: 'The message to send' },
      },
      required: ['chat_id', 'text'],
    },
  }],
}))

mcp.setRequestHandler(CallToolRequestSchema, async req => {
  if (req.params.name === 'reply') {
    const { chat_id, text } = req.params.arguments as { chat_id: string; text: string }
    send(`Reply to ${chat_id}: ${text}`)
    return { content: [{ type: 'text', text: 'sent' }] }
  }
  throw new Error(`unknown tool: ${req.params.name}`)
})

// --- 權限中繼:Claude Code(不是 Claude)在對話框開啟時呼叫此項
const PermissionRequestSchema = z.object({
  method: z.literal('notifications/claude/channel/permission_request'),
  params: z.object({
    request_id: z.string(),
    tool_name: z.string(),
    description: z.string(),
    input_preview: z.string(),
  }),
})

mcp.setNotificationHandler(PermissionRequestSchema, async ({ params }) => {
  send(
    `Claude wants to run ${params.tool_name}: ${params.description}\n\n` +
    `Reply "yes ${params.request_id}" or "no ${params.request_id}"`,
  )
})

await mcp.connect(new StdioServerTransport())

// --- HTTP on :8788:GET /events 串流出站,POST 路由入站 ---
const PERMISSION_REPLY_RE = /^\s*(y|yes|n|no)\s+([a-km-z]{5})\s*$/i
let nextId = 1

Bun.serve({
  port: 8788,
  hostname: '127.0.0.1',
  idleTimeout: 0,  // 不要關閉閒置 SSE 串流
  async fetch(req) {
    const url = new URL(req.url)

    // GET /events:SSE 串流,所以 curl -N 可以即時觀看回覆和提示
    if (req.method === 'GET' && url.pathname === '/events') {
      const stream = new ReadableStream({
        start(ctrl) {
          ctrl.enqueue(': connected\n\n')  // 所以 curl 立即顯示某些內容
          const emit = (chunk: string) => ctrl.enqueue(chunk)
          listeners.add(emit)
          req.signal.addEventListener('abort', () => listeners.delete(emit))
        },
      })
      return new Response(stream, {
        headers: { 'Content-Type': 'text/event-stream', 'Cache-Control': 'no-cache' },
      })
    }

    // 其他所有內容都是入站:首先根據寄件者進行閘道
    const body = await req.text()
    const sender = req.headers.get('X-Sender') ?? ''
    if (!allowed.has(sender)) return new Response('forbidden', { status: 403 })

    // 在將其視為聊天之前檢查判決格式
    const m = PERMISSION_REPLY_RE.exec(body)
    if (m) {
      await mcp.notification({
        method: 'notifications/claude/channel/permission',
        params: {
          request_id: m[2].toLowerCase(),
          behavior: m[1].toLowerCase().startsWith('y') ? 'allow' : 'deny',
        },
      })
      return new Response('verdict recorded')
    }

    // 正常聊天:作為 channel 事件轉發給 Claude
    const chat_id = String(nextId++)
    await mcp.notification({
      method: 'notifications/claude/channel',
      params: { content: body, meta: { chat_id, path: url.pathname } },
    })
    return new Response('ok')
  },
})
在三個終端機中測試判決路徑。第一個是您的 Claude Code 工作階段,使用開發旗標啟動,以便它生成 webhook.ts
claude --dangerously-load-development-channels server:webhook
在第二個中,串流出站側,以便您可以看到 Claude 的回覆和任何權限提示在它們觸發時到達:
curl -N localhost:8788/events
在第三個中,傳送一條訊息,使 Claude 嘗試執行命令:
curl -d "list the files in this directory" -H "X-Sender: dev" localhost:8788
本機權限對話框在您的 Claude Code 終端機中開啟。片刻後,提示出現在 /events 串流中,包括五字母 ID。從遠端側批准它:
curl -d "yes <id>" -H "X-Sender: dev" localhost:8788
本機對話框關閉,工具執行。Claude 的回覆透過 reply 工具回來並也落入串流中。 此檔案中的三個 channel 特定部分:
  • Server 建構函式中的功能claude/channel 註冊通知監聽器,claude/channel/permission 選擇加入權限中繼,tools 讓 Claude 發現回覆工具。
  • 出站路徑reply 工具處理程式是 Claude 為對話回應呼叫的;PermissionRequestSchema 通知處理程式是 Claude Code 在權限對話框開啟時呼叫的。兩者都呼叫 send() 透過 /events 廣播,但它們由系統的不同部分觸發。
  • HTTP 處理程式GET /events 保持 SSE 串流開啟,以便 curl 可以即時觀看出站;POST 是入站,根據 X-Sender 標頭進行閘道。yes <id>no <id> 主體作為判決通知進入 Claude Code,永遠不會到達 Claude;其他任何內容都作為 channel 事件轉發給 Claude。

打包為外掛程式

若要使您的 channel 可安裝和可共享,請將其包裝在外掛程式中並將其發佈到市場。使用者使用 /plugin install 安裝它,然後使用 --channels plugin:<name>@<marketplace> 按工作階段啟用它。 發佈到您自己的市場的 channel 仍然需要 --dangerously-load-development-channels 才能執行,因為它不在核准允許清單上。若要將其新增,請將其提交到官方市場。Channel 外掛程式在被批准之前會進行安全審查。在 Team 和 Enterprise 計劃上,管理員可以改為將您的外掛程式包含在組織自己的 allowedChannelPlugins 清單中,該清單取代預設 Anthropic 允許清單。

另請參閱

  • Channels 安裝並使用 Telegram、Discord、iMessage 或 fakechat 演示,以及為 Team 或 Enterprise 組織啟用 channels
  • 工作 channel 實現以取得具有配對流程、回覆工具和檔案附件的完整伺服器程式碼
  • MCP 用於 channel 伺服器實現的基礎協議
  • 外掛程式 打包您的 channel,以便使用者可以使用 /plugin install 安裝它