agentdev 0.1.10 → 0.2.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/BasicAgent-7TNMYC3X.js +13 -0
- package/dist/ExplorerAgent-L3ZTVQGM.js +13 -0
- package/dist/{chunk-72H6A6NB.js → chunk-35LBACUK.js} +8 -8
- package/dist/{chunk-72H6A6NB.js.map → chunk-35LBACUK.js.map} +1 -1
- package/dist/{chunk-NORTAQIL.js → chunk-4WK7UENZ.js} +1011 -11
- package/dist/chunk-4WK7UENZ.js.map +1 -0
- package/dist/{chunk-BAP2GCYH.js → chunk-7GTVQ55R.js} +1 -1
- package/dist/chunk-7GTVQ55R.js.map +1 -0
- package/dist/{chunk-G5ECPY4K.js → chunk-EK6KGS2M.js} +87 -8
- package/dist/{chunk-G5ECPY4K.js.map → chunk-EK6KGS2M.js.map} +1 -1
- package/dist/{chunk-REOJZCSZ.js → chunk-KE3KYZVJ.js} +21 -8
- package/dist/chunk-KE3KYZVJ.js.map +1 -0
- package/dist/{chunk-EECW6PYP.js → chunk-UL2ZBPBL.js} +60 -4
- package/dist/chunk-UL2ZBPBL.js.map +1 -0
- package/dist/{chunk-3AR3JBW6.js → chunk-XRB6MD2J.js} +5436 -883
- package/dist/chunk-XRB6MD2J.js.map +1 -0
- package/dist/cli/server.js +2 -2
- package/dist/cli/viewer.js +2 -2
- package/dist/features/mcp/templates/mcp-tool.render.js +14 -1
- package/dist/features/mcp/templates/mcp-tool.render.js.map +1 -1
- package/dist/features/shell/templates/bash.render.d.ts +1 -1
- package/dist/features/websearch/templates/web-fetch.render.d.ts +1 -1
- package/dist/index.d.ts +752 -20
- package/dist/index.js +17 -7
- package/dist/index.js.map +1 -1
- package/dist/{notification-NWVOS2WR.js → notification-QPH37BHW.js} +29 -6
- package/dist/notification-QPH37BHW.js.map +1 -0
- package/dist/{tools-LDR3LIJP.js → tools-OKH7SPMP.js} +2 -2
- package/dist/{types-CF5UsxD9.d.ts → types-NVwNUVFR.d.ts} +180 -14
- package/package.json +6 -13
- package/dist/BasicAgent-R7DYGTHF.js +0 -13
- package/dist/ExplorerAgent-DXY3OQ5U.js +0 -13
- package/dist/chunk-3AR3JBW6.js.map +0 -1
- package/dist/chunk-BAP2GCYH.js.map +0 -1
- package/dist/chunk-EECW6PYP.js.map +0 -1
- package/dist/chunk-NORTAQIL.js.map +0 -1
- package/dist/chunk-REOJZCSZ.js.map +0 -1
- package/dist/notification-NWVOS2WR.js.map +0 -1
- /package/dist/{BasicAgent-R7DYGTHF.js.map → BasicAgent-7TNMYC3X.js.map} +0 -0
- /package/dist/{ExplorerAgent-DXY3OQ5U.js.map → ExplorerAgent-L3ZTVQGM.js.map} +0 -0
- /package/dist/{tools-LDR3LIJP.js.map → tools-OKH7SPMP.js.map} +0 -0
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/core/types.ts"],"sourcesContent":["/**\r\n * 基础类型定义\r\n * 所有类型集中在这里,简单直观\r\n */\r\n\r\n// ========== 通知系统类型 ==========\r\n\r\n/**\r\n * 通知分类\r\n * - state: 覆盖式更新(如 LLM 字符计数)\r\n * - event: 追加式记录(如工具开始/完成)\r\n */\r\nexport type NotificationCategory = 'state' | 'event';\r\n\r\n/**\r\n * LLM 生成阶段\r\n */\r\nexport type LLMPhase = 'thinking' | 'content' | 'tool_calling';\r\n\r\n/**\r\n * 通知基础接口\r\n */\r\nexport interface Notification {\r\n type: string;\r\n category: NotificationCategory;\r\n timestamp: number;\r\n data: unknown;\r\n}\r\n\r\nexport type LogLevel = 'trace' | 'debug' | 'info' | 'warn' | 'error';\r\n\r\nexport type DebugLogDeliveryReason =\r\n | 'hub'\r\n | 'hub-unavailable'\r\n | 'no-agent-context';\r\n\r\nexport interface DebugLogDelivery {\r\n hub: boolean;\r\n console: boolean;\r\n reason: DebugLogDeliveryReason;\r\n}\r\n\r\nexport interface LogContextRef {\r\n agentId?: string;\r\n agentName?: string;\r\n parentAgentId?: string;\r\n callIndex?: number;\r\n step?: number;\r\n toolName?: string;\r\n toolCallId?: string;\r\n feature?: string;\r\n lifecycle?: string;\r\n hookMethod?: string;\r\n hookKind?: 'forward' | 'reverse';\r\n sourceFile?: string;\r\n sourceLine?: number;\r\n tags?: string[];\r\n [key: string]: unknown;\r\n}\r\n\r\nexport interface DebugLogEntry {\r\n id: string;\r\n timestamp: number;\r\n level: LogLevel;\r\n message: string;\r\n namespace: string;\r\n context: LogContextRef;\r\n data?: unknown;\r\n delivery: DebugLogDelivery;\r\n}\r\n\r\n/**\r\n * LLM 字符计数通知数据\r\n */\r\nexport interface LLMCharCountData {\r\n charCount: number;\r\n phase: LLMPhase;\r\n}\r\n\r\n/**\r\n * LLM 完成通知数据\r\n */\r\nexport interface LLMCompleteData {\r\n totalChars: number;\r\n}\r\n\r\n/**\r\n * 工具开始通知数据\r\n */\r\nexport interface ToolStartData {\r\n toolName: string;\r\n}\r\n\r\n/**\r\n * 工具完成通知数据\r\n */\r\nexport interface ToolCompleteData {\r\n toolName: string;\r\n success: boolean;\r\n duration: number;\r\n}\r\n\r\n/**\r\n * 通知状态响应(GET /api/agents/:id/notification)\r\n */\r\nexport interface NotificationStateResponse {\r\n state: Notification | null;\r\n hasNewEvents: boolean;\r\n}\r\n\r\nexport interface AgentLogsResponse {\r\n scope: 'current' | 'all';\r\n currentAgentId: string | null;\r\n selectedAgentId: string | null;\r\n total: number;\r\n logs: DebugLogEntry[];\r\n truncation?: {\r\n truncated: boolean;\r\n appliedLimit?: number;\r\n returnedCount: number;\r\n availableCount: number;\r\n nextOffset?: number;\r\n reason?: string;\r\n guidance?: string;\r\n };\r\n collectionPolicy: {\r\n hubConnected: boolean;\r\n includesOnlyHubDeliveredLogs: boolean;\r\n fallbackBehavior: string;\r\n };\r\n}\r\n\r\n/**\r\n * Agent 连接状态响应(GET /api/agents/:id/connection)\r\n */\r\nexport interface AgentConnectionResponse {\r\n connected: boolean;\r\n}\r\n\r\n// ========== 消息类型 ==========\r\n\r\n// 消息角色(支持子代理 ID 作为消息来源)\r\nexport type MessageRole = 'system' | 'user' | 'assistant' | 'tool' | string;\r\n\r\n// 消息结构\r\nexport interface Message {\r\n role: MessageRole;\r\n content: string;\r\n turn?: number;\r\n toolCallId?: string;\r\n toolCalls?: ToolCall[];\r\n reasoning?: string; // 思考内容(GLM-4.7等模型的扩展字段)\r\n thinkingBlocks?: ThinkingBlock[];\r\n}\r\n\r\nexport interface ThinkingBlock {\r\n signature: string;\r\n thinking: string;\r\n}\r\n\r\n// 工具调用\r\nexport interface ToolCall {\r\n id: string;\r\n name: string;\r\n arguments: Record<string, any>;\r\n}\r\n\r\n/**\r\n * 统一用量格式(兼容 Anthropic 和 OpenAI)\r\n */\r\nexport interface UsageInfo {\r\n /** 输入 token 数 */\r\n inputTokens: number;\r\n /** 输出 token 数 */\r\n outputTokens: number;\r\n /** 总 token 数 */\r\n totalTokens: number;\r\n\r\n // ========== Anthropic 特有(可选)==========\r\n /** 创建缓存消耗的 token 数 */\r\n cacheCreationTokens?: number;\r\n /** 从缓存读取的 token 数 */\r\n cacheReadTokens?: number;\r\n\r\n // ========== OpenAI 特有(可选)==========\r\n /** 推理 token 数 */\r\n reasoningTokens?: number;\r\n /** 音频 token 数 */\r\n audioTokens?: number;\r\n}\r\n\r\n// LLM 响应\r\nexport interface LLMResponse {\r\n content: string;\r\n toolCalls?: ToolCall[];\r\n reasoning?: string; // 思考内容(GLM-4.7等模型的扩展字段)\r\n thinkingBlocks?: ThinkingBlock[];\r\n /** 用量统计(可选) */\r\n usage?: UsageInfo;\r\n}\r\n\r\n// ============= 渲染模板类型 =============\r\n/**\r\n * 渲染模板项\r\n * 可以是字符串模板或函数模板\r\n */\r\nexport type RenderTemplateItem =\r\n | string // 字符串模板,使用 {{key}} 插值\r\n | RenderTemplateFn; // 函数模板,处理复杂逻辑\r\n\r\n/**\r\n * 渲染模板函数类型\r\n */\r\nexport type RenderTemplateFn = (data: Record<string, any>, success?: boolean) => string;\r\n\r\n/**\r\n * 内联渲染模板\r\n * 直接定义在工具中的渲染模板(无需引用预设模板)\r\n */\r\nexport interface InlineRenderTemplate {\r\n call: RenderTemplateItem;\r\n result: RenderTemplateItem;\r\n}\r\n\r\n// 工具渲染配置\r\nexport interface ToolRenderConfig {\r\n /** 调用时的渲染模板(字符串引用或内联模板) */\r\n call?: string | InlineRenderTemplate;\r\n /** 结果时的渲染模板(字符串引用或内联模板) */\r\n result?: string | InlineRenderTemplate;\r\n}\r\n\r\n// 工具定义\r\nexport interface Tool {\r\n name: string;\r\n description: string;\r\n parameters?: Record<string, any>;\r\n execute: (args: any, context?: any) => Promise<any>;\r\n /** 可选:渲染配置 */\r\n render?: ToolRenderConfig;\r\n}\r\n\r\n// LLM 接口 - 所有 LLM 适配器都需要实现这个\r\nexport interface LLMClient {\r\n chat(messages: Message[], tools: Tool[]): Promise<LLMResponse>;\r\n}\r\n\r\n// 占位符上下文类型\r\nimport type { PlaceholderContext, TemplateSource } from '../template/types.js';\n\n// MCP 类型导入\nimport type { MCPConfig } from '../mcp/types.js';\nimport type { UsageStatsSnapshot } from './usage.js';\n\r\n// Agent 配置\r\nexport interface AgentConfig {\n llm: LLMClient;\n tools?: Tool[];\n maxTurns?: number;\n systemMessage?: string | TemplateSource;\n name?: string; // Agent 显示名称(用于调试)\n projectRoot?: string;\n workspaceDir?: string;\n\r\n // ========== Feature 系统 ==========\r\n /**\r\n * Feature 配置\r\n *\r\n * 新的声明式 Feature 注册方式\r\n */\r\n features?: {\r\n /** 启用的 Feature 列表 */\r\n enabled?: string[];\r\n /** Feature 特定配置 */\r\n [key: string]: unknown;\r\n };\r\n}\r\n\r\n// 上下文中间件 - 用于处理消息数组\r\nexport type ContextMiddleware = (messages: Message[]) => Message[];\r\n\r\n// ============= 多 Agent 调试支持 =============\r\n\r\n/**\r\n * Agent 注册信息(Hub 端)\r\n */\r\nexport interface AgentInfo {\n id: string; // 唯一标识,如 \"agent-1\"\n name: string; // 显示名称\n registeredAt: number; // 注册时间戳\n projectRoot?: string;\n}\n\r\n/**\r\n * 工具元数据(用于前端渲染)\r\n */\r\nexport interface ToolMetadata {\r\n name: string;\r\n description: string;\r\n render: {\r\n call: string | InlineRenderTemplate; // 模板名称或内联模板\r\n result: string | InlineRenderTemplate; // 模板名称或内联模板\r\n // 内联模板的可选直接存储(用于前端特殊标记)\r\n inlineCall?: InlineRenderTemplate;\r\n inlineResult?: InlineRenderTemplate;\r\n };\r\n}\r\n\r\nexport interface HookSourceLocation {\r\n file?: string;\r\n line?: number;\r\n column?: number;\r\n display: string;\r\n}\r\n\r\nexport interface HookEntryMetadata {\r\n order: number;\r\n featureName: string;\r\n methodName: string;\r\n lifecycle: string;\r\n kind: 'decision' | 'notify';\r\n source?: HookSourceLocation;\r\n description?: string;\r\n}\r\n\r\nexport interface HookLifecycleSnapshot {\r\n lifecycle: string;\r\n kind: 'decision' | 'notify';\r\n entries: HookEntryMetadata[];\r\n}\r\n\r\nexport interface FeatureInspectorSnapshot {\r\n name: string;\r\n enabled: boolean;\r\n status: 'enabled' | 'disabled' | 'partial';\r\n hookCount: number;\r\n toolCount: number;\r\n enabledToolCount: number;\r\n source?: string;\r\n description?: string;\r\n tools: Array<{\r\n name: string;\r\n description: string;\r\n enabled: boolean;\r\n renderCall?: string;\r\n renderResult?: string;\r\n }>;\r\n}\r\n\r\nexport interface HookInspectorSnapshot {\n lifecycleOrder: string[];\n features: FeatureInspectorSnapshot[];\n hooks: HookLifecycleSnapshot[];\n}\n\nexport interface AgentContextMetrics {\n messageCount: number;\n charCount: number;\n toolCallCount: number;\n turnCount: number;\n}\n\nexport interface AgentOverviewSnapshot {\n updatedAt: number;\n context: AgentContextMetrics;\n usageStats: UsageStatsSnapshot;\n}\n\r\n/**\r\n * Agent 会话数据(Worker 端)\r\n */\r\nexport interface AgentSession {\n id: string;\n name: string;\n messages: Message[];\n tools: ToolMetadata[];\n createdAt: number;\r\n lastActive: number;\r\n // 项目根目录(用于定位模板文件)\r\n projectRoot?: string;\r\n // 通知系统扩展\r\n currentState: Notification | null;\r\n events: Notification[];\r\n lastEventCount: number;\r\n logs: DebugLogEntry[];\r\n // 所属 UDS 客户端连接 ID(用于多进程输入响应路由)\r\n clientId?: string;\r\n // 内部:上次最后一条消息的签名(用于推送去重)\n _lastMessageSig?: string;\n hookInspector?: HookInspectorSnapshot;\n overview?: AgentOverviewSnapshot;\n}\n\r\n/**\r\n * DebugHub IPC 消息类型(主进程 → Worker)\r\n * 使用 discriminated union 确保类型安全\r\n */\r\nexport type DebugHubIPCMessage =\n | RegisterAgentMsg\n | UpdateAgentInspectorMsg\n | UpdateAgentOverviewMsg\n | PushMessagesMsg\n | RegisterToolsMsg\n | SetCurrentAgentMsg\n | UnregisterAgentMsg\n | PushNotificationMsg\n | RequestInputMsg\r\n | StopMsg;\r\n\r\n/**\r\n * 注册新 Agent\r\n */\r\nexport interface RegisterAgentMsg {\n type: 'register-agent';\n agentId: string;\n name: string;\n createdAt: number;\n projectRoot?: string; // 项目根目录,用于模板文件加载\r\n featureTemplates?: Record<string, string>; // Feature 模板路径映射\n hookInspector?: HookInspectorSnapshot;\n overview?: AgentOverviewSnapshot;\n activeInputRequest?: ActiveInputRequest; // 活跃的输入请求(用于重连后恢复)\n}\n\nexport interface UpdateAgentInspectorMsg {\n type: 'update-agent-inspector';\n agentId: string;\n hookInspector: HookInspectorSnapshot;\n}\n\nexport interface UpdateAgentOverviewMsg {\n type: 'update-agent-overview';\n agentId: string;\n overview: AgentOverviewSnapshot;\n}\n\r\n/**\r\n * 推送 Agent 消息\r\n */\r\nexport interface PushMessagesMsg {\r\n type: 'push-messages';\r\n agentId: string;\r\n messages: Message[];\r\n}\r\n\r\n/**\r\n * 注册 Agent 工具\r\n */\r\nexport interface RegisterToolsMsg {\r\n type: 'register-tools';\r\n agentId: string;\r\n tools: Tool[];\r\n}\r\n\r\n/**\r\n * 切换当前选中的 Agent\r\n */\r\nexport interface SetCurrentAgentMsg {\r\n type: 'set-current-agent';\r\n agentId: string;\r\n}\r\n\r\n/**\r\n * 活跃的输入请求(用于重连后恢复)\r\n */\r\nexport interface ActiveInputRequest {\r\n requestId: string;\r\n prompt: string;\r\n placeholder?: string;\r\n initialValue?: string;\r\n actions?: UserInputAction[];\r\n timestamp: number;\r\n}\r\n\r\n/**\r\n * 注销 Agent\r\n */\r\nexport interface UnregisterAgentMsg {\r\n type: 'unregister-agent';\r\n agentId: string;\r\n}\r\n\r\n/**\r\n * 停止 Worker\r\n */\r\nexport interface StopMsg {\r\n type: 'stop';\r\n}\r\n\r\n/**\r\n * 推送通知\r\n */\r\nexport interface PushNotificationMsg {\r\n type: 'push-notification';\r\n agentId: string;\r\n notification: Notification;\r\n}\r\n\r\n/**\r\n * 请求用户输入\r\n */\r\nexport interface RequestInputMsg {\r\n type: 'request-input';\r\n agentId: string;\r\n requestId: string;\r\n prompt: string;\r\n timeout?: number;\r\n placeholder?: string;\r\n initialValue?: string;\r\n actions?: UserInputAction[];\r\n}\r\n\r\n/**\r\n * 用户输入响应(Worker → Agent,通过 UDS)\r\n */\r\nexport interface InputResponseMsg {\r\n type: 'input-response';\r\n agentId: string;\r\n requestId: string;\r\n input: string;\r\n response?: UserInputResponse;\r\n}\r\n\r\nexport interface UserInputAction {\r\n id: string;\r\n label: string;\r\n kind?: 'rollback' | 'custom';\r\n variant?: 'primary' | 'secondary' | 'danger';\r\n payload?: Record<string, unknown>;\r\n}\r\n\r\nexport interface UserInputRequest {\r\n prompt: string;\r\n placeholder?: string;\r\n initialValue?: string;\r\n actions?: UserInputAction[];\r\n}\r\n\r\nexport interface UserInputResponse {\r\n kind: 'text' | 'action';\r\n text?: string;\r\n actionId?: string;\r\n payload?: Record<string, unknown>;\r\n}\r\n\r\n/**\r\n * Worker → 主进程 消息\r\n */\r\nexport type WorkerIPCMessage =\r\n | ReadyMsg\r\n | AgentSwitchedMsg\r\n | InputResponseMsg;\r\n\r\n/**\r\n * Worker 就绪\r\n */\r\nexport interface ReadyMsg {\r\n type: 'ready';\r\n}\r\n\r\n/**\r\n * Agent 切换确认\r\n */\r\nexport interface AgentSwitchedMsg {\r\n type: 'agent-switched';\r\n agentId: string;\r\n}\r\n\r\n// ========== 上下文管理类型 ==========\r\n/**\r\n * 消息标签枚举\r\n *\r\n * 用于快速分类和过滤消息,一条消息可能有多个标签\r\n */\r\nexport type MessageTag =\r\n | 'user' // 用户输入消息\r\n | 'system' // 系统消息\r\n | 'assistant' // LLM 响应消息\r\n | 'tool-call' // assistant 消息且包含 toolCalls\r\n | 'tool-result' // role === 'tool' 的工具执行结果\r\n | 'sub-agent' // 来自子代理的消息(与 assistant/tool-result 组合使用)\r\n | 'reminder'; // Feature 注入的提醒消息(与 system 组合使用)\r\n\r\n/**\r\n * 解析结果结构\r\n *\r\n * 从消息 content 中提取的结构化信息\r\n */\r\nexport interface ParsedContent {\r\n /** 从 content 提取的任务 ID(正则匹配 \"taskId\":\"xxx\") */\r\n taskIds: string[];\r\n /** 从 content 提取的工具调用名称(从 toolCalls 或 content 解析) */\r\n toolCalls: string[];\r\n /** @ 提及的内容 */\r\n mentions: string[];\r\n /** 用户可继承扩展更多字段 */\r\n [key: string]: any;\r\n}\r\n\r\n/**\r\n * 消息元数据\r\n *\r\n * 用于 addMessage() 的元数据参数\r\n */\r\nexport interface MessageMeta {\r\n /** ReAct 循环轮次 */\r\n turn: number;\r\n /** 子代理 ID(子代理消息时填写) */\r\n agentId?: string;\r\n /** 来源 Feature(reminder 等消息时填写) */\r\n source?: string;\r\n}\r\n\r\n/**\r\n * 扩展的消息结构\r\n *\r\n * 在原始 Message 基础上添加元数据\r\n * 不破坏现有 Message 类型,保证 LLM 调用兼容性\r\n */\r\nexport interface EnrichedMessage extends Message {\r\n // === 元数据字段 ===\r\n\r\n /** 唯一标识(用于索引关联) */\r\n id: string;\r\n /** 消息产生时间戳(毫秒) */\r\n timestamp: number;\r\n /** 所属 ReAct 循环轮次(从 0 开始) */\r\n turn: number;\r\n /** 全局消息序号(从 0 开始递增) */\r\n sequence: number;\r\n /** 来源 Agent ID(子代理消息) */\r\n agentId?: string;\r\n /** 来源 Feature(如 'todo-feature',仅 reminder 等) */\r\n source?: string;\r\n\r\n // === 分类标签 ===\r\n\r\n /** 消息分类标签(用于快速查询) */\r\n tags: MessageTag[];\r\n\r\n // === 解析结果 ===\r\n\r\n /** 从 content 中提取的结构化信息 */\r\n parsed: ParsedContent;\r\n}\r\n\r\n// ========== 生命周期类型 re-export ==========\r\n// 生命周期类型从 lifecycle.ts 导出,保持类型定义集中管理\r\nexport type {\r\n AgentInitiateContext,\r\n AgentDestroyContext,\r\n CallStartContext,\r\n CallFinishContext,\r\n StepStartContext,\r\n StepFinishedContext,\r\n HookResult,\r\n ToolContext,\r\n ToolResult,\r\n} from './lifecycle.js';\r\n\r\n// ========== 决策上下文类型 ==========\r\n/**\r\n * 决策上下文(反向钩子参数)\r\n *\r\n * 所有决策上下文的联合类型\r\n */\r\nexport type DecisionContext =\r\n | import('./lifecycle.js').AgentInitiateContext\r\n | import('./lifecycle.js').AgentDestroyContext\r\n | import('./lifecycle.js').CallStartContext\r\n | import('./lifecycle.js').CallFinishContext\r\n | import('./lifecycle.js').StepStartContext\r\n | import('./lifecycle.js').StepFinishedContext\r\n | import('./lifecycle.js').ToolContext\r\n | import('./lifecycle.js').ToolResult\r\n | import('./lifecycle.js').StepFinishDecisionContext\r\n | import('./lifecycle.js').ToolFinishedDecisionContext;\r\n\r\n// ========== UDS 通信类型 ==========\r\n\r\n/**\r\n * UDS 配置\r\n */\r\nexport interface UDSConfig {\r\n /** UDS 路径(默认自动检测平台) */\r\n path?: string;\r\n /** HTTP 端口(Web 界面) */\r\n httpPort?: number;\r\n /** 是否自动打开浏览器 */\r\n openBrowser?: boolean;\r\n}\r\n\r\n/**\r\n * 平台检测后的 UDS 路径\r\n */\r\nexport function getDefaultUDSPath(): string {\r\n if (process.platform === 'win32') {\r\n return '\\\\\\\\.\\\\pipe\\\\agentdev-viewer';\r\n }\r\n return '/tmp/agentdev-viewer.sock';\r\n}\r\n"],"mappings":";AAurBO,SAAS,oBAA4B;AAC1C,MAAI,QAAQ,aAAa,SAAS;AAChC,WAAO;AAAA,EACT;AACA,SAAO;AACT;","names":[]}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/core/debug-transport.ts","../src/core/debug-capabilities.ts","../src/core/debug-hub.ts","../src/core/claw-debug-client.ts"],"sourcesContent":["export type DebugTransportMode = 'viewer-worker' | 'claw';\n\nexport function resolveDebugTransportMode(): DebugTransportMode {\n return process.env.AGENTDEV_DEBUG_TRANSPORT === 'claw' ? 'claw' : 'viewer-worker';\n}\n\nexport function getClawRuntimeUrl(): string {\n return process.env.AGENTDEV_CLAW_RUNTIME_URL || 'http://127.0.0.1:3030';\n}\n","import { getClawRuntimeUrl, resolveDebugTransportMode, type DebugTransportMode } from './debug-transport.js';\n\nexport interface DebugCapabilities {\n transportMode: DebugTransportMode;\n interactiveInput: boolean;\n runtimeUrl: string | null;\n viewerCompatibleApi: boolean;\n debuggerMcpMetadata: boolean;\n}\n\nexport function getDebugCapabilities(): DebugCapabilities {\n const transportMode = resolveDebugTransportMode();\n const isClaw = transportMode === 'claw';\n\n return {\n transportMode,\n interactiveInput: true,\n runtimeUrl: isClaw ? getClawRuntimeUrl() : null,\n viewerCompatibleApi: isClaw,\n debuggerMcpMetadata: isClaw,\n };\n}\n","/**\r\n * DebugHub - 全局多 Agent 调试中心\r\n *\r\n * 职责:\r\n * - 管理所有 Agent 的注册和注销\r\n * - 连接到独立的 Viewer Worker UDS 服务器\r\n * - 路由 Agent 消息到 Worker\r\n *\r\n * 设计原则:\r\n * - 单例模式,全局唯一\r\n * - 轻量:只做路由,不存储消息\r\n * - 直观:API 简单明了\r\n */\r\n\r\nimport { connect, Socket } from 'net';\r\nimport {\r\n getDefaultUDSPath,\r\n type Message,\r\n type Tool,\r\n type AgentInfo,\r\n type DebugHubIPCMessage,\r\n type Notification,\r\n type RequestInputMsg,\r\n type HookInspectorSnapshot,\r\n type AgentOverviewSnapshot,\r\n type UserInputRequest,\r\n type UserInputResponse,\r\n type UserInputAction,\r\n} from './types.js';\r\nimport { ClawDebugClient } from './claw-debug-client.js';\r\nimport { getClawRuntimeUrl, resolveDebugTransportMode } from './debug-transport.js';\r\nimport { getDebugCapabilities, type DebugCapabilities } from './debug-capabilities.js';\r\n\r\n// 前向声明 Agent 类型(避免循环依赖)\r\ntype Agent = any;\r\n\r\n/**\r\n * Hub 内部存储的 Agent 数据\r\n */\r\ninterface AgentData {\r\n info: AgentInfo;\r\n agent: Agent;\r\n}\r\n\r\nexport class DebugHub {\r\n private static instance: DebugHub;\r\n private readonly transportMode: 'viewer-worker' | 'claw';\r\n private readonly clawClient?: ClawDebugClient;\r\n\r\n // ========== 状态 ==========\r\n private agents: Map<string, AgentData> = new Map();\r\n private currentAgentId: string | null = null;\r\n private nextId: number = 1;\r\n private readonly processId: string; // 进程唯一标识\r\n\r\n // 输入请求回调映射:requestId → resolver\r\n private pendingInputRequests = new Map<string, (response: UserInputResponse) => void>();\r\n\r\n // 活跃的输入请求元数据(用于重连恢复):agentId → requestInfo\r\n private activeInputRequests = new Map<string, {\r\n requestId: string;\r\n prompt: string;\r\n placeholder?: string;\r\n initialValue?: string;\r\n actions?: UserInputAction[];\r\n timestamp: number;\r\n }>();\r\n\r\n // UDS 客户端连接\r\n private udsClient?: Socket;\r\n private udsPath: string;\r\n private workerPort: number | null = null;\r\n private clientReady: boolean = false;\r\n\r\n // 注册锁(防止并发竞争)\r\n private registrationLock: boolean = false;\r\n\r\n // 待发送的消息队列(连接建立前)\r\n private messageQueue: DebugHubIPCMessage[] = [];\r\n\r\n // 重连机制\r\n private reconnectTimer?: NodeJS.Timeout;\r\n private reconnectAttempts: number = 0;\r\n private readonly MAX_RECONNECT_ATTEMPTS = 10;\r\n private readonly RECONNECT_DELAY = 2000;\r\n\r\n // 缓存每个 Agent 的 featureTemplates(用于重连后重新注册)\r\n private agentFeatureTemplates: Map<string, Record<string, string>> = new Map();\r\n\r\n // ========== 单例 ==========\r\n private constructor() {\r\n this.udsPath = process.env.AGENTDEV_UDS_PATH || getDefaultUDSPath();\r\n // 使用进程 PID 作为唯一标识,确保多进程环境下 Agent ID 不冲突\r\n this.processId = String(process.pid);\r\n this.transportMode = resolveDebugTransportMode();\r\n if (this.transportMode === 'claw') {\r\n this.clawClient = new ClawDebugClient({\r\n processId: this.processId,\r\n projectRoot: process.cwd(),\r\n });\r\n }\r\n }\r\n\r\n static getInstance(): DebugHub {\r\n if (!DebugHub.instance) {\r\n DebugHub.instance = new DebugHub();\r\n }\r\n return DebugHub.instance;\r\n }\r\n\r\n // ========== 公开 API ==========\r\n\r\n /**\r\n * 启动调试服务器\r\n * @param port HTTP 端口(默认 2026,仅用于显示)\r\n * @param openBrowser 是否自动打开浏览器(默认 true,已废弃参数)\r\n */\r\n async start(port: number = 2026, openBrowser: boolean = true): Promise<void> {\r\n if (this.transportMode === 'claw') {\r\n this.workerPort = port;\r\n try {\r\n await this.clawClient?.ping();\r\n this.clientReady = true;\r\n console.log(`[DebugHub] 已连接到 Claw runtime: ${getClawRuntimeUrl()}`);\r\n } catch (err) {\r\n console.warn(`[DebugHub] 无法连接到 Claw runtime: ${(err as Error).message}`);\r\n console.warn('[DebugHub] 调试功能将被禁用。请先启动 AgentDevClaw runtime。');\r\n this.clientReady = false;\r\n }\r\n return;\r\n }\r\n\r\n if (this.udsClient) {\r\n console.log(`[DebugHub] 已连接到 ViewerWorker`);\r\n return;\r\n }\r\n\r\n this.workerPort = port; // 保留用于信息显示\r\n this.openBrowser = openBrowser; // 保存浏览器打开设置\r\n \r\n try {\r\n await this.connectToWorker();\r\n console.log(`[DebugHub] 调试服务器已连接: http://localhost:${port}`);\r\n } catch (err) {\r\n // 连接失败,尝试自动启动 ViewerWorker\r\n console.log(`[DebugHub] ViewerWorker 未运行,正在自动启动...`);\r\n try {\r\n await this.spawnViewerWorker();\r\n // 等待服务器启动\r\n await new Promise(resolve => setTimeout(resolve, 1000));\r\n // 再次尝试连接\r\n await this.connectToWorker();\r\n console.log(`[DebugHub] 调试服务器已连接: http://localhost:${port}`);\r\n } catch (spawnErr) {\r\n console.warn(`[DebugHub] 无法启动 ViewerWorker: ${(spawnErr as Error).message}`);\r\n console.warn(`[DebugHub] 调试功能将被禁用。请手动运行 'agentdev-viewer' 启动调试服务器。`);\r\n this.clientReady = false;\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * 自动启动 ViewerWorker 进程\r\n */\r\n private openBrowser: boolean = true;\r\n private viewerWorkerProcess?: ReturnType<typeof import('child_process').spawn>;\r\n\r\n private async spawnViewerWorker(): Promise<void> {\r\n const { spawn } = await import('child_process');\r\n const { existsSync } = await import('fs');\r\n const { fileURLToPath } = await import('url');\r\n const { dirname, join } = await import('path');\r\n\r\n // 查找 viewer.js 的路径\r\n let viewerPath: string;\r\n\r\n // 方式1: 优先使用 require.resolve(最可靠)\r\n try {\r\n const agentdevPath = require.resolve('agentdev/package.json');\r\n viewerPath = join(dirname(agentdevPath), 'dist', 'cli', 'viewer.js');\r\n } catch {\r\n // 方式2: 从当前模块解析\r\n try {\r\n const currentDir = dirname(fileURLToPath(import.meta.url));\r\n viewerPath = join(currentDir, '..', 'cli', 'viewer.js');\r\n } catch {\r\n // 方式3: 使用 bin 命令\r\n viewerPath = 'agentdev-viewer';\r\n }\r\n }\r\n\r\n // 验证文件存在\r\n if (viewerPath !== 'agentdev-viewer' && !existsSync(viewerPath)) {\r\n // 回退到 bin 命令\r\n viewerPath = 'agentdev-viewer';\r\n }\r\n\r\n return new Promise((resolve, reject) => {\r\n try {\r\n // 设置环境变量\r\n const env = {\r\n ...process.env,\r\n AGENTDEV_PORT: String(this.workerPort || 2026),\r\n AGENTDEV_OPEN_BROWSER: this.openBrowser ? 'true' : 'false',\r\n AGENTDEV_UDS_PATH: this.udsPath,\r\n };\r\n\r\n console.log(`[DebugHub] 启动 ViewerWorker: ${viewerPath}`);\r\n\r\n // 如果是 bin 命令,直接运行命令;否则用 node 执行\r\n const isBinCommand = viewerPath === 'agentdev-viewer' || viewerPath === 'agentdev-server';\r\n const command = isBinCommand ? viewerPath : 'node';\r\n const args = isBinCommand ? [] : [viewerPath];\r\n\r\n this.viewerWorkerProcess = spawn(command, args, {\r\n env,\r\n stdio: ['ignore', 'pipe', 'pipe'],\r\n detached: false,\r\n shell: isBinCommand, // bin 命令需要 shell 来解析\r\n });\r\n\r\n this.viewerWorkerProcess.on('error', (err: Error) => {\r\n console.error('[DebugHub] ViewerWorker 进程错误:', err.message);\r\n reject(err);\r\n });\r\n\r\n // 输出 ViewerWorker 的日志\r\n this.viewerWorkerProcess.stdout?.on('data', (data: Buffer) => {\r\n const lines = data.toString().trim().split('\\n');\r\n for (const line of lines) {\r\n console.log(`[ViewerWorker] ${line}`);\r\n }\r\n });\r\n\r\n this.viewerWorkerProcess.stderr?.on('data', (data: Buffer) => {\r\n const lines = data.toString().trim().split('\\n');\r\n for (const line of lines) {\r\n console.error(`[ViewerWorker] ${line}`);\r\n }\r\n });\r\n\r\n // 给进程一点时间启动\r\n setTimeout(() => {\r\n if (this.viewerWorkerProcess && !this.viewerWorkerProcess.killed) {\r\n resolve();\r\n } else {\r\n reject(new Error('ViewerWorker 进程启动失败'));\r\n }\r\n }, 500);\r\n } catch (err) {\r\n reject(err);\r\n }\r\n });\r\n }\r\n\r\n /**\r\n * 停止调试服务器\r\n */\r\n stop(): void {\r\n if (this.transportMode === 'claw') {\r\n this.clientReady = false;\r\n return;\r\n }\r\n\r\n // 停止重连定时器\r\n if (this.reconnectTimer) {\r\n clearTimeout(this.reconnectTimer);\r\n this.reconnectTimer = undefined;\r\n }\r\n\r\n if (this.udsClient) {\r\n this.sendToWorker({ type: 'stop' });\r\n this.udsClient.end();\r\n this.udsClient = undefined;\r\n this.clientReady = false;\r\n }\r\n }\r\n\r\n /**\r\n * 手动重连(可选)\r\n * 如果已经连接,则不执行任何操作\r\n */\r\n async reconnect(): Promise<void> {\r\n if (this.transportMode === 'claw') {\r\n await this.start(this.workerPort ?? 2026, false);\r\n if (!this.clientReady) {\r\n throw new Error('Claw runtime reconnect failed');\r\n }\r\n return;\r\n }\r\n\r\n if (this.clientReady && this.udsClient) {\r\n console.log('[DebugHub] 已经连接,无需重连');\r\n return;\r\n }\r\n\r\n // 重置重连状态\r\n this.reconnectAttempts = 0;\r\n if (this.reconnectTimer) {\r\n clearTimeout(this.reconnectTimer);\r\n this.reconnectTimer = undefined;\r\n }\r\n\r\n try {\r\n await this.connectToWorker();\r\n console.log('[DebugHub] ✅ 手动重连成功');\r\n } catch (error) {\r\n console.error(`[DebugHub] 手动重连失败: ${(error as Error).message}`);\r\n throw error;\r\n }\r\n }\r\n\r\n /**\r\n * 注册 Agent\r\n * @param agent Agent 实例\r\n * @param name 显示名称(可选,默认使用类名)\r\n * @param featureTemplates Feature 模板路径映射(可选)\r\n * @returns 分配的 agentId\r\n */\r\n registerAgent(\n agent: Agent,\n name?: string,\n featureTemplates?: Record<string, string>,\n hookInspector?: HookInspectorSnapshot,\n overview?: AgentOverviewSnapshot,\n projectRoot?: string\n ): string {\n // 等待注册锁\r\n while (this.registrationLock) {\r\n // 简单的忙等待(实际场景中竞争很少)\r\n }\r\n this.registrationLock = true;\r\n\r\n try {\r\n const resolvedProjectRoot = projectRoot || process.cwd();\n const id = `agent-${this.nextId++}-${this.processId}`;\n const info: AgentInfo = {\n id,\n name: name || agent.constructor.name,\n registeredAt: Date.now(),\n projectRoot: resolvedProjectRoot,\n };\n\r\n this.agents.set(id, { info, agent });\r\n\r\n // 缓存 featureTemplates(用于重连后重新注册)\r\n if (featureTemplates) {\r\n this.agentFeatureTemplates.set(id, featureTemplates);\r\n }\r\n\r\n // 首个注册的 Agent 自动成为当前 Agent\r\n if (this.agents.size === 1) {\r\n this.currentAgentId = id;\r\n }\r\n\r\n // 通知 Worker\r\n if (this.transportMode === 'claw') {\r\n void this.clawClient?.registerAgent({\n agentId: id,\n name: info.name,\n projectRoot: resolvedProjectRoot,\n featureTemplates,\n hookInspector,\n overview,\n }).catch(error => {\r\n console.error(`[DebugHub] Claw registerAgent 失败: ${(error as Error).message}`);\r\n });\r\n } else {\r\n this.sendToWorker({\n type: 'register-agent',\n agentId: id,\n name: info.name,\n createdAt: info.registeredAt,\n projectRoot: resolvedProjectRoot,\n featureTemplates, // 传递 Feature 模板路径映射\n hookInspector,\n overview,\n });\r\n }\r\n\r\n console.log(`[DebugHub] Agent 已注册: ${id} (${info.name})`);\r\n return id;\r\n } finally {\r\n this.registrationLock = false;\r\n }\r\n }\r\n\r\n /**\r\n * 注销 Agent\r\n * @param agentId Agent ID\r\n */\r\n unregisterAgent(agentId: string): void {\r\n const deleted = this.agents.delete(agentId);\r\n if (deleted) {\r\n this.agentFeatureTemplates.delete(agentId);\r\n if (this.transportMode === 'claw') {\r\n void this.clawClient?.unregisterAgent(agentId).catch(error => {\r\n console.error(`[DebugHub] Claw unregisterAgent 失败: ${(error as Error).message}`);\r\n });\r\n } else {\r\n this.sendToWorker({ type: 'unregister-agent', agentId });\r\n }\r\n console.log(`[DebugHub] Agent 已注销: ${agentId}`);\r\n\r\n // 如果注销的是当前 Agent,切换到另一个\r\n if (this.currentAgentId === agentId) {\r\n const remaining = Array.from(this.agents.keys());\r\n this.currentAgentId = remaining.length > 0 ? remaining[0] : null;\r\n if (this.currentAgentId) {\r\n if (this.transportMode === 'claw') {\r\n void this.clawClient?.selectAgent(this.currentAgentId).catch(error => {\r\n console.error(`[DebugHub] Claw selectAgent 失败: ${(error as Error).message}`);\r\n });\r\n } else if (this.transportMode === 'viewer-worker') {\r\n this.sendToWorker({\r\n type: 'set-current-agent',\r\n agentId: this.currentAgentId,\r\n });\r\n }\r\n }\r\n }\r\n }\r\n }\r\n\r\n /**\r\n * 切换当前选中的 Agent\r\n * @param agentId Agent ID\r\n * @returns 是否成功\r\n */\r\n selectAgent(agentId: string): boolean {\r\n if (!this.agents.has(agentId)) {\r\n return false;\r\n }\r\n this.currentAgentId = agentId;\r\n if (this.transportMode === 'claw') {\r\n void this.clawClient?.selectAgent(agentId).catch(error => {\r\n console.error(`[DebugHub] Claw selectAgent 失败: ${(error as Error).message}`);\r\n });\r\n console.log(`[DebugHub] 当前 Agent 已切换: ${agentId}`);\r\n return true;\r\n }\r\n if (this.transportMode === 'viewer-worker') {\r\n this.sendToWorker({\r\n type: 'set-current-agent',\r\n agentId,\r\n });\r\n }\r\n console.log(`[DebugHub] 当前 Agent 已切换: ${agentId}`);\r\n return true;\r\n }\r\n\r\n /**\r\n * 推送 Agent 消息\r\n * @param agentId Agent ID\r\n * @param messages 消息数组\r\n */\r\n pushMessages(agentId: string, messages: Message[]): void {\r\n if (this.transportMode === 'claw') {\r\n void this.clawClient?.pushMessages(agentId, messages).catch(error => {\r\n console.error(`[DebugHub] Claw pushMessages 失败: ${(error as Error).message}`);\r\n });\r\n return;\r\n }\r\n\r\n this.sendToWorker({\r\n type: 'push-messages',\r\n agentId,\r\n messages,\r\n });\r\n }\r\n\r\n /**\r\n * 注册 Agent 工具\r\n * @param agentId Agent ID\r\n * @param tools 工具数组\r\n */\r\n registerAgentTools(agentId: string, tools: Tool[]): void {\r\n if (this.transportMode === 'claw') {\r\n void this.clawClient?.registerTools(agentId, tools).catch(error => {\r\n console.error(`[DebugHub] Claw registerTools 失败: ${(error as Error).message}`);\r\n });\r\n return;\r\n }\r\n\r\n this.sendToWorker({\r\n type: 'register-tools',\r\n agentId,\r\n tools,\r\n });\r\n }\r\n\r\n updateAgentInspector(agentId: string, hookInspector: HookInspectorSnapshot): void {\r\n if (this.transportMode === 'claw') {\r\n void this.clawClient?.updateInspector(agentId, hookInspector).catch(error => {\r\n console.error(`[DebugHub] Claw updateInspector 失败: ${(error as Error).message}`);\r\n });\r\n return;\r\n }\r\n\r\n this.sendToWorker({\r\n type: 'update-agent-inspector',\r\n agentId,\r\n hookInspector,\r\n });\r\n }\r\n\r\n updateAgentOverview(agentId: string, overview: AgentOverviewSnapshot): void {\r\n if (this.transportMode === 'claw') {\r\n void this.clawClient?.updateOverview(agentId, overview).catch(error => {\r\n console.error(`[DebugHub] Claw updateOverview 失败: ${(error as Error).message}`);\r\n });\r\n return;\r\n }\r\n\r\n this.sendToWorker({\r\n type: 'update-agent-overview',\r\n agentId,\r\n overview,\r\n });\r\n }\r\n\r\n /**\r\n * 获取所有已注册的 Agent 信息\r\n */\r\n getAgentList(): AgentInfo[] {\r\n return Array.from(this.agents.values()).map(v => v.info);\r\n }\r\n\r\n /**\r\n * 获取当前选中的 Agent ID\r\n */\r\n getCurrentAgentId(): string | null {\r\n return this.currentAgentId;\r\n }\r\n\r\n getTransportMode(): 'viewer-worker' | 'claw' {\r\n return this.transportMode;\r\n }\r\n\r\n getCapabilities(): DebugCapabilities {\r\n return getDebugCapabilities();\r\n }\r\n\r\n /**\r\n * 根据 Agent 实例获取其 ID\r\n */\r\n getAgentId(agent: Agent): string | undefined {\r\n for (const [id, data] of this.agents) {\r\n if (data.agent === agent) {\r\n return id;\r\n }\r\n }\r\n return undefined;\r\n }\r\n\r\n /**\r\n * 获取 Worker 端口\r\n */\r\n getPort(): number | null {\r\n return this.workerPort;\r\n }\r\n\r\n /**\r\n * 检查是否已连接到 ViewerWorker\r\n */\r\n isConnected(): boolean {\r\n if (this.transportMode === 'claw') {\r\n return this.clientReady;\r\n }\r\n return this.clientReady && !!this.udsClient;\r\n }\r\n\r\n /**\r\n * 推送通知\r\n * @param agentId Agent ID\r\n * @param notification 通知对象\r\n */\r\n pushNotification(agentId: string, notification: Notification): void {\r\n if (this.transportMode === 'claw') {\r\n void this.clawClient?.pushNotification(agentId, notification).catch(error => {\r\n console.error(`[DebugHub] Claw pushNotification 失败: ${(error as Error).message}`);\r\n });\r\n return;\r\n }\r\n\r\n this.sendToWorker({\r\n type: 'push-notification',\r\n agentId,\r\n notification,\r\n });\r\n }\r\n\r\n /**\r\n * 请求用户输入\r\n * @param agentId Agent ID\r\n * @param prompt 提示信息\r\n * @param timeout 超时时间(毫秒),默认 Infinity(无限等待)\r\n * @returns Promise<string> 用户输入内容\r\n */\r\n requestUserInput(agentId: string, prompt: string, timeout: number = Infinity): Promise<string> {\r\n return this.requestUserInputEvent(agentId, { prompt }, timeout).then((response) => {\r\n if (response.kind !== 'text') {\r\n throw new Error(`Expected text user input but received action '${response.actionId ?? 'unknown'}'`);\r\n }\r\n return response.text ?? '';\r\n });\r\n }\r\n\r\n requestUserInputEvent(\r\n agentId: string,\r\n request: UserInputRequest,\r\n timeout: number = Infinity,\r\n ): Promise<UserInputResponse> {\r\n if (this.transportMode === 'claw') {\r\n return this.clawClient?.requestUserInput(agentId, request, timeout)\r\n ?? Promise.reject(new Error('Claw client is not available'));\r\n }\r\n\r\n const requestId = `input-${agentId}-${Date.now()}-${Math.random().toString(36).slice(2)}`;\r\n\r\n return new Promise((resolve, reject) => {\r\n // 设置超时定时器(仅在 timeout 为有限数值时)\r\n let timer: NodeJS.Timeout | undefined;\r\n if (timeout !== Infinity) {\r\n timer = setTimeout(() => {\r\n this.pendingInputRequests.delete(requestId);\r\n this.activeInputRequests.delete(agentId); // 清除活跃请求记录\r\n reject(new Error(`User input timeout after ${timeout}ms`));\r\n }, timeout);\r\n }\r\n\r\n // 存储 resolve 函数\r\n this.pendingInputRequests.set(requestId, (response: UserInputResponse) => {\r\n if (timer) clearTimeout(timer);\r\n this.activeInputRequests.delete(agentId); // 清除活跃请求记录\r\n resolve(response);\r\n });\r\n\r\n // 记录活跃请求(用于重连恢复)\r\n this.activeInputRequests.set(agentId, {\r\n requestId,\r\n prompt: request.prompt,\r\n placeholder: request.placeholder,\r\n initialValue: request.initialValue,\r\n actions: request.actions,\r\n timestamp: Date.now(),\r\n });\r\n\r\n // 发送请求到 ViewerWorker\r\n this.sendToWorker({\r\n type: 'request-input',\r\n agentId,\r\n requestId,\r\n prompt: request.prompt,\r\n placeholder: request.placeholder,\r\n initialValue: request.initialValue,\r\n actions: request.actions,\r\n timeout,\r\n } as RequestInputMsg);\r\n });\r\n }\r\n\r\n // ========== 内部方法 ==========\r\n\r\n /**\r\n * 连接到 UDS 服务器\r\n */\r\n private async connectToWorker(): Promise<void> {\r\n return new Promise((resolve, reject) => {\r\n this.udsClient = connect(this.udsPath);\r\n\r\n this.udsClient.on('connect', () => {\r\n this.clientReady = true;\r\n this.reconnectAttempts = 0; // 重置重连计数\r\n console.log(`[DebugHub] 已连接到 ViewerWorker: ${this.udsPath}`);\r\n\r\n // 发送队列中的消息\r\n for (const msg of this.messageQueue) {\r\n this.sendViaUDS(msg);\r\n }\r\n this.messageQueue = [];\r\n\r\n // 关键:重新注册所有 Agent(用于重连后恢复状态)\r\n this.reregisterAllAgents();\r\n\r\n // 设置当前 Agent\r\n if (this.currentAgentId) {\r\n this.sendToWorker({\r\n type: 'set-current-agent',\r\n agentId: this.currentAgentId,\r\n });\r\n }\r\n\r\n resolve();\r\n });\r\n\r\n this.udsClient.on('data', (data: Buffer) => {\r\n const lines = data.toString().split('\\n');\r\n for (const line of lines) {\r\n if (!line.trim()) continue;\r\n try {\r\n const msg = JSON.parse(line);\r\n this.handleWorkerMessage(msg);\r\n } catch (err) {\r\n console.error('[DebugHub] Worker 消息解析失败:', err);\r\n }\r\n }\r\n });\r\n\r\n this.udsClient.on('error', (err: Error) => {\r\n reject(new Error(`连接 ViewerWorker 失败 (${this.udsPath}): ${err.message}\\n请先启动 ViewerWorker 服务器`));\r\n });\r\n\r\n this.udsClient.on('close', () => {\r\n this.clientReady = false;\r\n console.warn('[DebugHub] 与 ViewerWorker 的连接已断开');\r\n\r\n // 自动重连\r\n this.scheduleReconnect();\r\n });\r\n });\r\n }\r\n\r\n /**\r\n * 处理来自 Worker 的消息\r\n */\r\n private handleWorkerMessage(msg: any): void {\r\n switch (msg.type) {\r\n case 'agent-switched':\r\n console.log(`[DebugHub] 当前 Agent 已切换: ${msg.agentId}`);\r\n break;\r\n\r\n // 处理用户输入响应\r\n case 'input-response':\r\n const resolver = this.pendingInputRequests.get(msg.requestId);\r\n if (resolver) {\r\n resolver(msg.response ?? {\r\n kind: 'text',\r\n text: msg.input,\r\n });\r\n this.pendingInputRequests.delete(msg.requestId);\r\n } else {\r\n console.warn(`[DebugHub] 未知输入响应: ${msg.requestId}`);\r\n }\r\n break;\r\n }\r\n }\r\n\r\n /**\r\n * 重新注册所有 Agent(重连后调用)\r\n * 确保 ViewerWorker 能够恢复所有 Agent 的注册信息\r\n */\r\n private reregisterAllAgents(): void {\r\n if (this.transportMode === 'claw') {\r\n for (const [id, data] of this.agents) {\r\n const hookInspector = (data.agent as any).buildHookInspectorSnapshot?.()\r\n || (data.agent as any).hookInspector;\r\n const overview = (data.agent as any).buildOverviewSnapshot?.();\r\n const featureTemplates = this.agentFeatureTemplates.get(id) || {};\r\n\r\n void this.clawClient?.registerAgent({\n agentId: id,\n name: data.info.name,\n projectRoot: data.info.projectRoot || process.cwd(),\n featureTemplates,\n hookInspector,\n overview,\n }).then(async () => {\r\n const tools = (data.agent as any).tools;\r\n if (tools && typeof tools.getEntries === 'function') {\r\n const entries = tools.getEntries();\r\n const toolList = entries.map((e: any) => e.tool);\r\n if (toolList.length > 0) {\r\n await this.clawClient?.registerTools(id, toolList);\r\n }\r\n }\r\n\r\n const context = (data.agent as any).getContext?.();\r\n if (context && typeof context.getAll === 'function') {\r\n const messages = context.getAll();\r\n if (messages.length > 0) {\r\n await this.clawClient?.pushMessages(id, messages);\r\n }\r\n }\r\n }).catch(error => {\r\n console.error(`[DebugHub] Claw re-register 失败: ${(error as Error).message}`);\r\n });\r\n }\r\n return;\r\n }\r\n\r\n if (this.agents.size === 0) {\r\n return;\r\n }\r\n\r\n console.log(`[DebugHub] 重新注册 ${this.agents.size} 个 Agent...`);\r\n\r\n for (const [id, data] of this.agents) {\r\n // 获取最新的 hookInspector\r\n const hookInspector = (data.agent as any).buildHookInspectorSnapshot?.()\r\n || (data.agent as any).hookInspector;\r\n const overview = (data.agent as any).buildOverviewSnapshot?.();\r\n\r\n // 获取缓存的 featureTemplates\r\n const featureTemplates = this.agentFeatureTemplates.get(id) || {};\r\n\r\n // 获取活跃的输入请求(用于恢复输入框)\r\n const activeInputRequest = this.activeInputRequests.get(id);\r\n if (activeInputRequest) {\r\n console.log(`[DebugHub] 发现活跃输入请求: ${activeInputRequest.requestId}`);\r\n }\r\n\r\n this.sendToWorker({\n type: 'register-agent' as const,\n agentId: id,\n name: data.info.name,\n createdAt: data.info.registeredAt,\n projectRoot: data.info.projectRoot || process.cwd(),\n featureTemplates,\n hookInspector,\n overview,\n activeInputRequest, // 携带活跃输入请求\r\n });\r\n\r\n // 重新注册工具(如果有)\r\n const tools = (data.agent as any).tools;\r\n if (tools && typeof tools.getEntries === 'function') {\r\n const entries = tools.getEntries();\r\n const toolList = entries.map((e: any) => e.tool);\r\n if (toolList.length > 0) {\r\n this.sendToWorker({\r\n type: 'register-tools',\r\n agentId: id,\r\n tools: toolList,\r\n });\r\n }\r\n }\r\n\r\n // 重新发送对话记录(用于重连后恢复消息历史)\r\n const context = (data.agent as any).getContext?.();\r\n if (context && typeof context.getAll === 'function') {\r\n const messages = context.getAll();\r\n if (messages.length > 0) {\r\n this.sendToWorker({\r\n type: 'push-messages',\r\n agentId: id,\r\n messages,\r\n });\r\n console.log(`[DebugHub] 恢复 Agent ${id} 的 ${messages.length} 条消息`);\r\n }\r\n }\r\n }\r\n\r\n console.log(`[DebugHub] ✅ 重新注册完成`);\r\n }\r\n\r\n /**\r\n * 安排重连(指数退避)\r\n */\r\n private scheduleReconnect(): void {\r\n // 清除现有的定时器\r\n if (this.reconnectTimer) {\r\n clearTimeout(this.reconnectTimer);\r\n }\r\n\r\n // 检查是否达到最大重连次数\r\n if (this.reconnectAttempts >= this.MAX_RECONNECT_ATTEMPTS) {\r\n console.error(`[DebugHub] 达到最大重连次数 (${this.MAX_RECONNECT_ATTEMPTS}),停止重连`);\r\n return;\r\n }\r\n\r\n this.reconnectAttempts++;\r\n\r\n // 计算延迟时间(指数退避,最大 30 秒)\r\n const delay = Math.min(\r\n this.RECONNECT_DELAY * Math.pow(2, this.reconnectAttempts - 1),\r\n 30000\r\n );\r\n\r\n console.log(`[DebugHub] ${delay}ms 后尝试第 ${this.reconnectAttempts} 次重连...`);\r\n\r\n this.reconnectTimer = setTimeout(async () => {\r\n try {\r\n await this.connectToWorker();\r\n console.log('[DebugHub] ✅ 重连成功,调试功能已恢复');\r\n } catch (error) {\r\n console.error(`[DebugHub] 重连失败: ${(error as Error).message}`);\r\n // 继续尝试重连\r\n this.scheduleReconnect();\r\n }\r\n }, delay);\r\n }\r\n\r\n /**\r\n * 通过 UDS 发送消息\r\n */\r\n private sendViaUDS(msg: DebugHubIPCMessage): void {\r\n if (this.udsClient && this.clientReady) {\r\n this.udsClient.write(JSON.stringify(msg) + '\\n');\r\n }\r\n // 未连接时丢弃消息,不再队列(避免内存泄漏)\r\n }\r\n\r\n /**\r\n * 发送消息到 Worker\r\n */\r\n private sendToWorker(msg: DebugHubIPCMessage): void {\r\n if (!this.udsClient) {\r\n return;\r\n }\r\n this.sendViaUDS(msg);\r\n }\r\n}\r\n","import type {\n AgentOverviewSnapshot,\n HookInspectorSnapshot,\n Message,\n Notification,\n Tool,\n UserInputRequest,\n UserInputResponse,\n} from './types.js';\nimport { getClawRuntimeUrl } from './debug-transport.js';\n\ntype ClawEventKind =\n | 'message'\n | 'notification'\n | 'snapshot'\n | 'lifecycle'\n | 'tools'\n | 'input';\n\ninterface ClawSessionRegistration {\n sessionId: string;\n runtime: 'agentdev';\n agentName: string;\n projectRoot: string | null;\n metadata: Record<string, unknown>;\n state?: Record<string, unknown>;\n}\n\ninterface ClawEventInput {\n sessionId: string;\n kind: ClawEventKind;\n payload: Record<string, unknown>;\n}\n\nexport interface RegisterClawAgentInput {\n agentId: string;\n name: string;\n projectRoot?: string;\n featureTemplates?: Record<string, string>;\n hookInspector?: HookInspectorSnapshot;\n overview?: AgentOverviewSnapshot;\n}\n\nexport class ClawDebugClient {\n private readonly runtimeUrl: string;\n private readonly processId: string;\n private readonly projectRoot: string;\n private readonly sessionByAgentId = new Map<string, string>();\n private readonly pendingSessionByAgentId = new Map<string, Promise<string>>();\n\n constructor(options: { runtimeUrl?: string; processId?: string; projectRoot?: string } = {}) {\n this.runtimeUrl = (options.runtimeUrl ?? getClawRuntimeUrl()).replace(/\\/$/, '');\n this.processId = options.processId ?? String(process.pid);\n this.projectRoot = options.projectRoot ?? process.cwd();\n }\n\n async ping(): Promise<void> {\n await this.requestJson('/health');\n }\n\n async registerAgent(input: RegisterClawAgentInput): Promise<string> {\n const sessionId = `${this.processId}:${input.agentId}`;\n const existing = this.pendingSessionByAgentId.get(input.agentId);\n if (existing) {\n return existing;\n }\n\n const pending = (async () => {\n const registration: ClawSessionRegistration = {\n sessionId,\n runtime: 'agentdev',\n agentName: input.name,\n projectRoot: input.projectRoot ?? this.projectRoot,\n metadata: {\n adapter: 'agentdev-debug-hub',\n agentId: input.agentId,\n featureTemplates: input.featureTemplates ?? {},\n },\n state: {\n hookInspector: input.hookInspector ?? null,\n overview: input.overview ?? null,\n },\n };\n\n await this.requestJson('/api/sessions/register', {\n method: 'POST',\n body: JSON.stringify(registration),\n });\n\n this.sessionByAgentId.set(input.agentId, sessionId);\n\n await this.pushLifecycle(input.agentId, {\n phase: 'agent-registered',\n agentId: input.agentId,\n name: input.name,\n });\n\n return sessionId;\n })();\n\n this.pendingSessionByAgentId.set(input.agentId, pending);\n\n try {\n return await pending;\n } finally {\n this.pendingSessionByAgentId.delete(input.agentId);\n }\n }\n\n async unregisterAgent(agentId: string): Promise<void> {\n await this.pushLifecycle(agentId, {\n phase: 'agent-unregistered',\n agentId,\n });\n }\n\n async selectAgent(agentId: string): Promise<void> {\n const sessionId = await this.requireSessionId(agentId);\n await this.requestJson('/api/agents/current', {\n method: 'PUT',\n body: JSON.stringify({ agentId: sessionId }),\n });\n }\n\n async pushMessages(agentId: string, messages: Message[]): Promise<void> {\n await this.pushEvent(agentId, 'message', { messages });\n }\n\n async registerTools(agentId: string, tools: Tool[]): Promise<void> {\n await this.pushEvent(agentId, 'tools', { tools });\n }\n\n async updateInspector(agentId: string, hookInspector: HookInspectorSnapshot): Promise<void> {\n await this.pushEvent(agentId, 'snapshot', {\n scope: 'hookInspector',\n hookInspector,\n });\n }\n\n async updateOverview(agentId: string, overview: AgentOverviewSnapshot): Promise<void> {\n await this.pushEvent(agentId, 'snapshot', {\n scope: 'overview',\n overview,\n });\n }\n\n async pushNotification(agentId: string, notification: Notification): Promise<void> {\n await this.pushEvent(agentId, 'notification', { notification });\n }\n\n async requestUserInput(_agentId: string, request: UserInputRequest, timeout: number): Promise<UserInputResponse> {\n const agentId = _agentId;\n const requestId = `input-${agentId}-${Date.now()}-${Math.random().toString(36).slice(2)}`;\n await this.pushEvent(agentId, 'input', {\n requestId,\n request,\n timeout,\n status: 'pending',\n });\n\n const sessionId = await this.requireSessionId(agentId);\n const startedAt = Date.now();\n const pollIntervalMs = 400;\n\n while (true) {\n if (timeout !== Infinity && Date.now() - startedAt > timeout) {\n throw new Error(`User input timeout after ${timeout}ms`);\n }\n\n let response: Response;\n try {\n response = await fetch(\n `${this.runtimeUrl}/api/agents/${encodeURIComponent(sessionId)}/input-response?requestId=${encodeURIComponent(requestId)}`\n );\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error);\n throw new Error(`Unable to reach Claw runtime at ${this.runtimeUrl} while waiting for user input: ${message}`);\n }\n\n if (response.ok) {\n const body = await response.json();\n return body.response;\n }\n\n if (response.status !== 404) {\n const text = await response.text();\n throw new Error(`Input bridge request failed: ${response.status} ${response.statusText} ${text}`);\n }\n\n await new Promise(resolve => setTimeout(resolve, pollIntervalMs));\n }\n }\n\n private async pushLifecycle(agentId: string, payload: Record<string, unknown>): Promise<void> {\n await this.pushEvent(agentId, 'lifecycle', payload);\n }\n\n private async pushEvent(agentId: string, kind: ClawEventKind, payload: Record<string, unknown>): Promise<void> {\n const sessionId = await this.requireSessionId(agentId);\n const event: ClawEventInput = {\n sessionId,\n kind,\n payload,\n };\n\n await this.requestJson('/api/events', {\n method: 'POST',\n body: JSON.stringify(event),\n });\n }\n\n private async requireSessionId(agentId: string): Promise<string> {\n const sessionId = this.sessionByAgentId.get(agentId);\n if (!sessionId) {\n const pending = this.pendingSessionByAgentId.get(agentId);\n if (pending) {\n return pending;\n }\n throw new Error(`No Claw session registered for agentId '${agentId}'`);\n }\n return sessionId;\n }\n\n private async requestJson(path: string, init?: RequestInit): Promise<any> {\n let response: Response;\n try {\n response = await fetch(`${this.runtimeUrl}${path}`, {\n ...init,\n headers: {\n 'content-type': 'application/json',\n ...(init?.headers ?? {}),\n },\n });\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error);\n throw new Error(`Unable to reach Claw runtime at ${this.runtimeUrl}: ${message}`);\n }\n\n if (!response.ok) {\n const body = await response.text();\n throw new Error(`Claw runtime request failed: ${response.status} ${response.statusText} ${body}`);\n }\n\n if (response.status === 204) {\n return null;\n }\n\n return response.json();\n }\n}\n"],"mappings":"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","names":[]}
|