@tencent-ai/cloud-agent-sdk 0.2.11 → 0.2.12
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/index.cjs +95 -39
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.cts +60 -46
- package/dist/index.d.cts.map +1 -1
- package/dist/index.d.mts +60 -46
- package/dist/index.d.mts.map +1 -1
- package/dist/index.mjs +95 -39
- package/dist/index.mjs.map +1 -1
- package/dist/tencent-ai-cloud-agent-sdk-0.2.12.tgz +0 -0
- package/package.json +2 -2
- package/dist/tencent-ai-cloud-agent-sdk-0.2.11.tgz +0 -0
package/dist/index.mjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.mjs","names":["utils","prototype","utils","PlatformFormData","AxiosError","encode","toFormData","utils","AxiosURLSearchParams","AxiosURLSearchParams","utils","platform","toFormData","platform","utils","utils","utils","transitionalDefaults","formDataToJSON","toFormData","AxiosError","platform","utils","utils","parseHeaders","defaults","AxiosHeaders","AxiosError","utils","AxiosError","speedometer","throttle","utils","platform","platform","AxiosHeaders","utils","AxiosHeaders","utils","platform","isURLSameOrigin","cookies","resolveConfig","AxiosHeaders","AxiosError","transitionalDefaults","utils","CanceledError","platform","AxiosError","CanceledError","utils","platform","utils","AxiosError","resolveConfig","composeSignals","AxiosHeaders","httpAdapter","xhrAdapter","fetchAdapter","utils","AxiosError","CanceledError","AxiosHeaders","adapters","defaults","validators","AxiosError","validator","InterceptorManager","utils","AxiosHeaders","CanceledError","utils","Axios","defaults","CanceledError","CancelToken","toFormData","AxiosError","AxiosHeaders","formDataToJSON","utils","adapters","HttpStatusCode","axios"],"sources":["../../agent-provider/src/common/_legacy/tool-schemas.ts","../../agent-provider/src/common/_legacy/MockAgentProvider.ts","../../agent-client-protocol/src/common/types.ts","../../agent-client-protocol/src/common/transport/streamable-http.ts","../../agent-client-protocol/src/common/client/artifacts.ts","../../agent-client-protocol/src/common/client/constants.ts","../../agent-client-protocol/src/common/client/errors.ts","../../agent-client-protocol/src/common/client/events.ts","../../agent-client-protocol/src/common/client/extensions.ts","../../agent-client-protocol/src/common/client/permissions.ts","../../agent-client-protocol/src/common/client/questions.ts","../../agent-client-protocol/src/common/client/client.ts","../../agent-provider/src/common/providers/cloud-agent-provider/cloud-connection.ts","../../agent-provider/node_modules/axios/lib/helpers/bind.js","../../agent-provider/node_modules/axios/lib/utils.js","../../agent-provider/node_modules/axios/lib/core/AxiosError.js","../../agent-provider/node_modules/axios/lib/helpers/null.js","../../agent-provider/node_modules/axios/lib/helpers/toFormData.js","../../agent-provider/node_modules/axios/lib/helpers/AxiosURLSearchParams.js","../../agent-provider/node_modules/axios/lib/helpers/buildURL.js","../../agent-provider/node_modules/axios/lib/core/InterceptorManager.js","../../agent-provider/node_modules/axios/lib/defaults/transitional.js","../../agent-provider/node_modules/axios/lib/platform/browser/classes/URLSearchParams.js","../../agent-provider/node_modules/axios/lib/platform/browser/classes/FormData.js","../../agent-provider/node_modules/axios/lib/platform/browser/classes/Blob.js","../../agent-provider/node_modules/axios/lib/platform/browser/index.js","../../agent-provider/node_modules/axios/lib/platform/common/utils.js","../../agent-provider/node_modules/axios/lib/platform/index.js","../../agent-provider/node_modules/axios/lib/helpers/toURLEncodedForm.js","../../agent-provider/node_modules/axios/lib/helpers/formDataToJSON.js","../../agent-provider/node_modules/axios/lib/defaults/index.js","../../agent-provider/node_modules/axios/lib/helpers/parseHeaders.js","../../agent-provider/node_modules/axios/lib/core/AxiosHeaders.js","../../agent-provider/node_modules/axios/lib/core/transformData.js","../../agent-provider/node_modules/axios/lib/cancel/isCancel.js","../../agent-provider/node_modules/axios/lib/cancel/CanceledError.js","../../agent-provider/node_modules/axios/lib/core/settle.js","../../agent-provider/node_modules/axios/lib/helpers/parseProtocol.js","../../agent-provider/node_modules/axios/lib/helpers/speedometer.js","../../agent-provider/node_modules/axios/lib/helpers/throttle.js","../../agent-provider/node_modules/axios/lib/helpers/progressEventReducer.js","../../agent-provider/node_modules/axios/lib/helpers/isURLSameOrigin.js","../../agent-provider/node_modules/axios/lib/helpers/cookies.js","../../agent-provider/node_modules/axios/lib/helpers/isAbsoluteURL.js","../../agent-provider/node_modules/axios/lib/helpers/combineURLs.js","../../agent-provider/node_modules/axios/lib/core/buildFullPath.js","../../agent-provider/node_modules/axios/lib/core/mergeConfig.js","../../agent-provider/node_modules/axios/lib/helpers/resolveConfig.js","../../agent-provider/node_modules/axios/lib/adapters/xhr.js","../../agent-provider/node_modules/axios/lib/helpers/composeSignals.js","../../agent-provider/node_modules/axios/lib/helpers/trackStream.js","../../agent-provider/node_modules/axios/lib/adapters/fetch.js","../../agent-provider/node_modules/axios/lib/adapters/adapters.js","../../agent-provider/node_modules/axios/lib/core/dispatchRequest.js","../../agent-provider/node_modules/axios/lib/env/data.js","../../agent-provider/node_modules/axios/lib/helpers/validator.js","../../agent-provider/node_modules/axios/lib/core/Axios.js","../../agent-provider/node_modules/axios/lib/cancel/CancelToken.js","../../agent-provider/node_modules/axios/lib/helpers/spread.js","../../agent-provider/node_modules/axios/lib/helpers/isAxiosError.js","../../agent-provider/node_modules/axios/lib/helpers/HttpStatusCode.js","../../agent-provider/node_modules/axios/lib/axios.js","../../agent-provider/src/http/http-service.ts","../../agent-provider/src/http/index.ts","../../agent-provider/src/account/account-service.ts","../../agent-provider/src/common/utils/concurrency.ts","../../agent-provider/src/common/providers/cloud-agent-provider/cos-upload-service.ts","../../agent-provider/src/common/providers/cloud-agent-provider/e2b-filesystem.ts","../../agent-provider/src/common/providers/cloud-agent-provider/cloud-provider.ts","../../agent-provider/src/common/client/session.ts","../../agent-provider/src/common/client/session-manager.ts","../../agent-provider/src/common/client/client.ts","../../agent-provider/src/common/client/types.ts","../../agent-provider/src/backend/types.ts","../../agent-provider/src/backend/backend-provider.ts","../../agent-provider/src/backend/ipc-backend-provider.ts"],"sourcesContent":["/**\n * ACP Tool Input/Output Schema 定义\n *\n * 用于约束 ACP 协议中 ToolCallUpdate 的 rawInput 和 rawOutput 字段\n * 基于 getCraftToolProvider 使用的工具定义\n */\n\nimport { z } from 'zod';\n\n// ============================================================================\n// 基础类型\n// ============================================================================\n\n/**\n * 工具类型标识\n */\nexport type ToolName =\n | 'list_dir'\n | 'search_file'\n | 'read_file'\n | 'read_lints'\n | 'rag_search'\n | 'read_rules'\n | 'mcp_get_tool_description'\n | 'mcp_call_tool'\n | 'fetch_mcp_resource'\n | 'create_rule'\n | 'update_memory'\n | 'search_content'\n | 'write_to_file'\n | 'replace_in_file'\n | 'delete_file'\n | 'execute_command'\n | 'preview_url'\n | 'ask_followup_question'\n | 'invoke_integration'\n | 'call_integration'\n | 'search_integration_tool'\n | 'supabase_get_logs'\n | 'supabase_execute_sql'\n | 'supabase_apply_migration'\n | 'supabase_list_migration'\n | 'supabase_list_tables'\n | 'cloud_studio_fetch_log'\n | 'cloud_studio_execute_command'\n | 'cloud_studio_deploy_sandbox'\n | 'component_get_prompt'\n | 'web_fetch'\n | 'use_skill'\n | 'web_search'\n | 'task'\n | 'codebase_search'\n | 'lsp'\n | 'spec_create'\n | 'spec_update';\n\n// ============================================================================\n// 工具输入类型定义\n// ============================================================================\n\nexport interface ListDirInput {\n target_directory: string;\n ignore_globs?: string;\n}\n\nexport interface SearchFileInput {\n target_directory: string;\n pattern: string;\n recursive: boolean;\n caseSensitive: boolean;\n}\n\nexport interface ReadFileInput {\n filePath: string;\n offset?: number;\n limit?: number;\n}\n\nexport interface ReadLintsInput {\n paths?: string;\n}\n\nexport interface RagSearchInput {\n queryString: string;\n knowledgeBaseNames: string;\n}\n\nexport interface ReadRulesInput {\n ruleNames: string;\n}\n\nexport interface McpGetToolDescriptionInput {\n toolRequests: string;\n}\n\nexport interface McpCallToolInput {\n serverName: string;\n toolName: string;\n arguments: string;\n maxOutputLength?: number;\n}\n\nexport interface FetchMcpResourceInput {\n server: string;\n uri: string;\n arguments?: Record<string, unknown>;\n downloadPath?: string;\n}\n\nexport interface CreateRuleInput {\n ruleScope: string;\n ruleName: string;\n ruleType: string;\n ruleContent: string;\n ruleDescription?: string;\n}\n\nexport interface UpdateMemoryInput {\n action?: 'create' | 'update' | 'delete';\n existing_knowledge_id?: string;\n knowledge_to_store?: string;\n title?: string;\n}\n\nexport interface SearchContentInput {\n pattern: string;\n directory: string;\n fileTypes?: string;\n contextBefore?: number;\n contextAfter?: number;\n contextAround?: number;\n outputMode?: string;\n caseSensitive?: boolean;\n}\n\nexport interface WriteToFileInput {\n filePath: string;\n content: string;\n}\n\nexport interface ReplaceInFileInput {\n filePath: string;\n old_str: string;\n new_str: string;\n}\n\nexport interface DeleteFileInput {\n target_file: string;\n explanation?: string;\n}\n\nexport interface ExecuteCommandInput {\n command: string;\n requires_approval: boolean;\n}\n\nexport interface PreviewUrlInput {\n url: string;\n}\n\nexport interface AskFollowupQuestionInput {\n questions: Array<{\n question: string;\n header: string;\n options: Array<{\n label: string;\n description: string;\n }>;\n multiSelect?: boolean;\n }>;\n}\n\nexport interface InvokeIntegrationInput {\n [key: string]: unknown;\n}\n\nexport interface CallIntegrationInput {\n [key: string]: unknown;\n}\n\nexport interface SearchIntegrationToolInput {\n [key: string]: unknown;\n}\n\nexport interface SupabaseGetLogsInput {\n [key: string]: unknown;\n}\n\nexport interface SupabaseExecuteSqlInput {\n [key: string]: unknown;\n}\n\nexport interface SupabaseApplyMigrationInput {\n [key: string]: unknown;\n}\n\nexport interface SupabaseListMigrationInput {\n [key: string]: unknown;\n}\n\nexport interface SupabaseListTablesInput {\n [key: string]: unknown;\n}\n\nexport interface CloudStudioFetchLogInput {\n [key: string]: unknown;\n}\n\nexport interface CloudStudioExecuteCommandInput {\n [key: string]: unknown;\n}\n\nexport interface CloudStudioDeploySandboxInput {\n [key: string]: unknown;\n}\n\nexport interface ComponentGetPromptInput {\n [key: string]: unknown;\n}\n\nexport interface WebFetchInput {\n url: string;\n fetchInfo: string;\n}\n\nexport interface UseSkillInput {\n command: string;\n}\n\nexport interface WebSearchInput {\n explanation: string;\n searchTerm: string;\n}\n\nexport interface TaskInput {\n subagent_name: string;\n description: string;\n prompt: string;\n subagent_path?: string;\n}\n\nexport interface CodebaseSearchInput {\n query: string;\n path: string;\n limit?: number;\n}\n\nexport interface LspInput {\n [key: string]: unknown;\n}\n\nexport interface SpecCreateInput {\n name: string;\n overview: string;\n relative_history: string;\n}\n\nexport interface SpecUpdateInput {\n status?: 'prepare' | 'ready' | 'building' | 'finished';\n}\n\n/**\n * 工具输入类型映射\n */\nexport interface ToolInputMap {\n list_dir: ListDirInput;\n search_file: SearchFileInput;\n read_file: ReadFileInput;\n read_lints: ReadLintsInput;\n rag_search: RagSearchInput;\n read_rules: ReadRulesInput;\n mcp_get_tool_description: McpGetToolDescriptionInput;\n mcp_call_tool: McpCallToolInput;\n fetch_mcp_resource: FetchMcpResourceInput;\n create_rule: CreateRuleInput;\n update_memory: UpdateMemoryInput;\n search_content: SearchContentInput;\n write_to_file: WriteToFileInput;\n replace_in_file: ReplaceInFileInput;\n delete_file: DeleteFileInput;\n execute_command: ExecuteCommandInput;\n preview_url: PreviewUrlInput;\n ask_followup_question: AskFollowupQuestionInput;\n invoke_integration: InvokeIntegrationInput;\n call_integration: CallIntegrationInput;\n search_integration_tool: SearchIntegrationToolInput;\n supabase_get_logs: SupabaseGetLogsInput;\n supabase_execute_sql: SupabaseExecuteSqlInput;\n supabase_apply_migration: SupabaseApplyMigrationInput;\n supabase_list_migration: SupabaseListMigrationInput;\n supabase_list_tables: SupabaseListTablesInput;\n cloud_studio_fetch_log: CloudStudioFetchLogInput;\n cloud_studio_execute_command: CloudStudioExecuteCommandInput;\n cloud_studio_deploy_sandbox: CloudStudioDeploySandboxInput;\n component_get_prompt: ComponentGetPromptInput;\n web_fetch: WebFetchInput;\n use_skill: UseSkillInput;\n web_search: WebSearchInput;\n task: TaskInput;\n codebase_search: CodebaseSearchInput;\n lsp: LspInput;\n spec_create: SpecCreateInput;\n spec_update: SpecUpdateInput;\n}\n\n// ============================================================================\n// 工具输出类型定义\n// ============================================================================\n\nexport interface ListFilesResult {\n type: 'list_files_result';\n files: Array<{\n filePath: string;\n size: string;\n modifyTime: string;\n }>;\n root: string;\n listing?: string;\n}\n\nexport interface SearchFileResult {\n type: 'search_file_result';\n path: string;\n pattern: string;\n recursive?: boolean;\n caseSensitive?: boolean;\n results: Array<{\n filePath: string;\n size: string;\n modifyTime: string;\n }>;\n}\n\nexport interface ReadFileResult {\n type: 'read_file_result';\n path: string;\n content: string;\n totalLineCount: number;\n hasMore: boolean;\n diagnostic?: string;\n hint?: string;\n image?: {\n data: string;\n mimeType: string;\n };\n}\n\nexport interface ReadLintsResult {\n type: 'read_lints_result';\n diagnostics: string[];\n totalCount?: number;\n hint?: string;\n isTruncated?: boolean;\n}\n\nexport interface KnowledgeSearchResult {\n type: 'knowledge_base_result';\n selectedKnowledgeBases: string;\n queryInput: string;\n}\n\nexport interface RuleMatchResult {\n type: 'rule_match_result';\n ruleDescription: string;\n filePaths: string[];\n}\n\nexport interface McpCallToolResult {\n type: 'mcp_call_tool_result';\n serverName: string;\n toolName: string;\n data: Array<{\n type: 'text' | 'image' | 'resource';\n text?: string;\n data?: string;\n mimeType?: string;\n resource?: {\n uri: string;\n mimeType?: string;\n text?: string;\n blob?: string;\n };\n }>;\n isError?: boolean;\n error?: unknown;\n hint?: string;\n}\n\nexport interface FetchMcpResourceToolResult {\n type: 'fetch_mcp_resource_result';\n server: string;\n uri: string;\n content: string;\n downloadPath?: string;\n}\n\nexport interface RuleCreateResult {\n type: 'rule_create_result';\n ruleName: string;\n createState: 'success' | 'invoke' | 'cancelled';\n hint?: string;\n filePath?: string;\n}\n\nexport interface UpdateMemoryResult {\n type: 'update_memory_result';\n success: boolean;\n message: string;\n action: 'create' | 'update' | 'delete';\n knowledge_id?: string;\n}\n\nexport interface SearchContentResult {\n type: 'search_content_result';\n directory: string;\n pattern: string;\n fileTypes: string;\n matches: Array<{\n filePath: string;\n content: string;\n startLine: number;\n endLine: number;\n size: string;\n modifyTime: string;\n }>;\n totalCount: number;\n hasMore: boolean;\n offset: number;\n limit: number;\n contextBefore: number;\n contextAfter: number;\n contextAround?: number;\n outputMode: string;\n caseSensitive: boolean;\n hint?: string;\n}\n\nexport interface WriteToFileResult {\n type: 'write_to_file_result';\n path: string;\n addLineCount: number;\n removedLines: number;\n addedChars?: number;\n removedChars?: number;\n bytesWritten: number;\n isNewFile: boolean;\n oldContent?: string;\n diagnostic?: string;\n}\n\nexport interface ReplaceInFileResult {\n type: 'replace_in_file_result';\n path: string;\n addLineCount?: number;\n removedLines?: number;\n addedChars?: number;\n removedChars?: number;\n matchCount?: number;\n hint?: string;\n diagnosticChange?: {\n added: string;\n removed: string;\n unchanged: string;\n };\n}\n\nexport interface DeleteFilesResult {\n type: 'delete_file_result';\n path: string;\n recursive: boolean;\n hint?: string;\n}\n\nexport interface ExecuteCommandResult {\n type: 'execute_command_result';\n stdout: string;\n stderr: string;\n exitCode: number;\n hint?: string;\n serviceInfo?: {\n isWatchCommand: boolean;\n isServiceOutput: boolean;\n serviceReady: boolean;\n message: string;\n };\n use_standalone_terminal?: boolean;\n}\n\nexport interface PreviewToolResult {\n type: 'preview_tool_result';\n url: string;\n message: string;\n}\n\nexport interface MultiQuestionResult {\n type: 'multi_question_result';\n questions: Array<{\n id: string;\n question: string;\n options: string[];\n multiSelect?: boolean;\n title?: string;\n }>;\n answers: Record<string, string | string[]>;\n message: string;\n}\n\nexport interface InvokeIntegrationToolResult {\n type: 'invoke_integration_tool_result';\n recommend: {\n id: string;\n type: string;\n status: 'connected' | 'disconnected';\n };\n message: string;\n}\n\nexport interface CallIntegrationToolResult {\n type: 'call_integration_tool_result';\n integrationId: string;\n toolName: string;\n data: {\n type: 'text';\n text: string;\n };\n isError?: boolean;\n error?: unknown;\n}\n\nexport interface SearchIntegrationToolResult {\n type: 'search_integration_tool_result';\n data: Array<{\n integrationId: string;\n integrationName: string;\n toolName: string;\n description: string;\n inputSchema: Record<string, unknown>;\n }>;\n hint?: string;\n}\n\nexport interface SupabaseToolsResult {\n type: 'supabase_get_logs_result' | 'supabase_execute_sql_result' | 'supabase_apply_migration_result' | 'supabase_list_migration_result' | 'supabase_list_tables_result';\n message: string;\n}\n\nexport interface CloudStudioFetchLogResult {\n type: 'cloud_studio_fetch_log_result';\n success: boolean;\n logs: Record<string, string>;\n}\n\nexport interface CloudStudioExecuteCommandResult {\n type: 'cloud_studio_execute_command_result';\n success: boolean;\n message: string;\n}\n\nexport interface CloudStudioDeployResult {\n type: 'cloud_studio_integration_result';\n previewUrl?: string;\n steps: Array<{\n status: 'idle' | 'success' | 'running' | 'error';\n name: 'createSandbox' | 'uploadProject' | 'installDependencies' | 'startService' | 'preview';\n error?: {\n code?: number;\n message?: string;\n };\n }>;\n}\n\nexport interface ComponentGetPromptResult {\n type: 'component_get_prompt_result';\n componentType: string;\n webFramework: string;\n data: {\n type: 'text';\n text: string;\n };\n}\n\nexport interface WebFetchToolResult {\n type: 'web_fetch_tool_result';\n message: string;\n data: string;\n loading?: string;\n title?: string;\n favicon?: string;\n}\n\nexport interface UseSkillToolResult {\n type: 'use_skill_tool_result';\n commandMessage: string;\n message: string;\n}\n\nexport interface WebSearchToolResult {\n type: 'web_search_tool_result';\n data: Array<{\n passage: string;\n uri: string;\n site: string;\n title: string;\n snippets?: string[];\n content?: string;\n }>;\n searchInput?: string;\n}\n\nexport interface TaskToolResult {\n type: 'task_tool_result';\n toolInfo?: Array<{\n name: string;\n info: string;\n needApprove?: boolean;\n toolCallId?: string;\n executeStatus?: 'ing' | 'completed' | 'cancel' | 'fail';\n }>;\n startCallTool?: boolean;\n finalResult?: string;\n toolCallBrief?: string;\n}\n\nexport interface CodebaseSearchResult {\n type: 'codebase_search_result';\n query: string;\n path: string;\n content: string;\n}\n\nexport interface LSPToolResult {\n type: 'lsp_tool_result';\n operation: string;\n result: string;\n resultCount?: number;\n fileCount?: number;\n character?: number;\n}\n\nexport interface PlanCreateToolResult {\n type: 'plan_create_tool_result';\n message: string;\n data: string;\n}\n\nexport interface PlanUpdateToolResult {\n type: 'plan_update_tool_result';\n status: 'prepare' | 'ready' | 'building' | 'finished';\n data: string;\n reminder: string;\n}\n\n/**\n * 工具输出类型映射\n */\nexport interface ToolOutputMap {\n list_dir: ListFilesResult;\n search_file: SearchFileResult;\n read_file: ReadFileResult;\n read_lints: ReadLintsResult;\n rag_search: KnowledgeSearchResult;\n read_rules: RuleMatchResult;\n mcp_get_tool_description: Record<string, unknown>;\n mcp_call_tool: McpCallToolResult;\n fetch_mcp_resource: FetchMcpResourceToolResult;\n create_rule: RuleCreateResult;\n update_memory: UpdateMemoryResult;\n search_content: SearchContentResult;\n write_to_file: WriteToFileResult;\n replace_in_file: ReplaceInFileResult;\n delete_file: DeleteFilesResult;\n execute_command: ExecuteCommandResult;\n preview_url: PreviewToolResult;\n ask_followup_question: MultiQuestionResult;\n invoke_integration: InvokeIntegrationToolResult;\n call_integration: CallIntegrationToolResult;\n search_integration_tool: SearchIntegrationToolResult;\n supabase_get_logs: SupabaseToolsResult;\n supabase_execute_sql: SupabaseToolsResult;\n supabase_apply_migration: SupabaseToolsResult;\n supabase_list_migration: SupabaseToolsResult;\n supabase_list_tables: SupabaseToolsResult;\n cloud_studio_fetch_log: CloudStudioFetchLogResult;\n cloud_studio_execute_command: CloudStudioExecuteCommandResult;\n cloud_studio_deploy_sandbox: CloudStudioDeployResult;\n component_get_prompt: ComponentGetPromptResult;\n web_fetch: WebFetchToolResult;\n use_skill: UseSkillToolResult;\n web_search: WebSearchToolResult;\n task: TaskToolResult;\n codebase_search: CodebaseSearchResult;\n lsp: LSPToolResult;\n spec_create: PlanCreateToolResult;\n spec_update: PlanUpdateToolResult;\n}\n\n// ============================================================================\n// Zod Schema 定义 (用于运行时验证)\n// ============================================================================\n\n/**\n * 工具输入 Schema 定义\n * 用于验证和约束 rawInput\n */\nexport const ToolInputSchemas = {\n list_dir: z.object({\n target_directory: z.string().describe('要列出内容的目录路径'),\n ignore_globs: z.string().optional().describe('可选的 glob 模式数组,用于忽略特定文件'),\n }),\n\n search_file: z.object({\n target_directory: z.string().describe('搜索的目录绝对路径'),\n pattern: z.string().describe('文件模式(如 \"*.js\"),支持通配符'),\n recursive: z.boolean().describe('是否递归搜索子目录'),\n caseSensitive: z.boolean().describe('是否区分大小写'),\n }),\n\n read_file: z.object({\n filePath: z.string().describe('要读取的文件的绝对路径'),\n offset: z.number().optional().describe('开始读取的行号'),\n limit: z.number().optional().describe('要读取的行数'),\n }),\n\n read_lints: z.object({\n paths: z.string().optional().describe('要读取 lint 错误的文件或目录路径'),\n }),\n\n rag_search: z.object({\n queryString: z.string().describe('用户的实际问题或搜索查询'),\n knowledgeBaseNames: z.string().describe('知识库名称,多个用逗号分隔'),\n }),\n\n read_rules: z.object({\n ruleNames: z.string().describe('要读取的规则关键词,用逗号分隔,格式:{ruleName}_{ruleId}'),\n }),\n\n mcp_get_tool_description: z.object({\n toolRequests: z.string().describe('JSON 字符串,二维数组格式:[[\"server1\", \"tool1\"], [\"server2\", \"tool2\"]]'),\n }),\n\n mcp_call_tool: z.object({\n serverName: z.string().describe('MCP 服务器名称'),\n toolName: z.string().describe('要调用的工具名称'),\n arguments: z.string().describe('目标 MCP 工具的参数,JSON 格式字符串'),\n maxOutputLength: z.number().optional().describe('控制工具输出的最大长度,默认 200000'),\n }),\n\n fetch_mcp_resource: z.object({\n server: z.string().describe('MCP 服务器标识符'),\n uri: z.string().describe('要读取的资源 URI'),\n arguments: z.record(z.unknown()).optional().describe('资源模板的参数'),\n downloadPath: z.string().optional().describe('可选的绝对路径,用于保存资源到磁盘'),\n }),\n\n create_rule: z.object({\n ruleScope: z.string().describe('规则范围,project rule 或 user rule'),\n ruleName: z.string().describe('规则文件名,不带扩展名'),\n ruleType: z.string().describe('规则类型,always、manual 或 requested'),\n ruleContent: z.string().describe('规则内容,使用 Markdown 格式'),\n ruleDescription: z.string().optional().describe('规则描述,使用 Markdown 格式'),\n }),\n\n update_memory: z.object({\n action: z.enum(['create', 'update', 'delete']).optional().describe('执行的操作'),\n existing_knowledge_id: z.string().optional().describe('更新或删除时必需,现有记忆的 ID'),\n knowledge_to_store: z.string().optional().describe('要存储的特定记忆'),\n title: z.string().optional().describe('记忆的标题'),\n }),\n\n search_content: z.object({\n pattern: z.string().describe('要搜索的关键字或正则表达式模式'),\n directory: z.string().describe('要搜索的目录的绝对路径'),\n fileTypes: z.string().optional().describe('可选的逗号分隔文件扩展名'),\n contextBefore: z.number().optional().describe('每个匹配前显示的行数'),\n contextAfter: z.number().optional().describe('每个匹配后显示的行数'),\n contextAround: z.number().optional().describe('每个匹配前后显示的行数'),\n outputMode: z.string().optional().describe('输出模式:content、files_with_matches 或 count'),\n caseSensitive: z.boolean().optional().describe('是否区分大小写'),\n }),\n\n write_to_file: z.object({\n filePath: z.string().describe('目标文件的绝对路径'),\n content: z.string().describe('要写入的内容'),\n }),\n\n replace_in_file: z.object({\n filePath: z.string().describe('要修改的文件的绝对路径'),\n old_str: z.string().describe('要替换的文本'),\n new_str: z.string().describe('替换后的文本'),\n }),\n\n delete_file: z.object({\n target_file: z.string().describe('要删除的文件的绝对路径'),\n explanation: z.string().optional().describe('为什么使用此工具的一句话解释'),\n }),\n\n execute_command: z.object({\n command: z.string().describe('要执行的 CLI 命令'),\n requires_approval: z.boolean().describe('命令是否需要用户批准'),\n }),\n\n preview_url: z.object({\n url: z.string().describe('要打开的完整、有效的 HTTP/HTTPS URL'),\n }),\n\n ask_followup_question: z.object({\n questions: z.array(z.object({\n question: z.string(),\n header: z.string().max(12),\n options: z.array(z.object({\n label: z.string().max(50),\n description: z.string(),\n })).min(2).max(4),\n multiSelect: z.boolean().optional(),\n })).min(1).max(4),\n }),\n\n invoke_integration: z.object({}).passthrough(),\n\n call_integration: z.object({}).passthrough(),\n\n search_integration_tool: z.object({}).passthrough(),\n\n supabase_get_logs: z.object({}).passthrough(),\n\n supabase_execute_sql: z.object({}).passthrough(),\n\n supabase_apply_migration: z.object({}).passthrough(),\n\n supabase_list_migration: z.object({}).passthrough(),\n\n supabase_list_tables: z.object({}).passthrough(),\n\n cloud_studio_fetch_log: z.object({}).passthrough(),\n\n cloud_studio_execute_command: z.object({}).passthrough(),\n\n cloud_studio_deploy_sandbox: z.object({}).passthrough(),\n\n component_get_prompt: z.object({}).passthrough(),\n\n web_fetch: z.object({\n url: z.string().describe('要获取内容的 URL'),\n fetchInfo: z.string().describe('用户想要获取的信息描述'),\n }),\n\n use_skill: z.object({\n command: z.string().describe('技能名称(不含参数),如 \"pdf\" 或 \"xlsx\"'),\n }),\n\n web_search: z.object({\n explanation: z.string().describe('为什么使用此工具的一句话解释'),\n searchTerm: z.string().describe('要在网络上搜索的搜索词'),\n }),\n\n task: z.object({\n subagent_name: z.string().describe('要调用的子代理名称'),\n description: z.string().describe('任务的简短描述(3-5 个词)'),\n prompt: z.string().describe('子代理要执行的任务'),\n subagent_path: z.string().optional().describe('子代理定义文件的路径'),\n }),\n\n codebase_search: z.object({\n query: z.string().describe('关于你想理解的内容的完整问题'),\n path: z.string().describe('限制搜索范围的目录路径前缀'),\n limit: z.number().max(100).optional().describe('返回的最大结果数,默认 10'),\n }),\n\n lsp: z.object({}).passthrough(),\n\n spec_create: z.object({\n name: z.string().describe('Plan 名称,用作稳定标识符/文件名'),\n overview: z.string().describe('用一两句话精确概括本次 plan 的主要内容'),\n relative_history: z.string().describe('准备阶段的上下文,包含用户需求、代码位置、额外上下文等'),\n }),\n\n spec_update: z.object({\n status: z.enum(['prepare', 'ready', 'building', 'finished']).optional().describe('Plan 状态'),\n }),\n} satisfies Record<ToolName, z.ZodTypeAny>;\n\n/**\n * 工具输出 Schema 定义\n * 用于验证和约束 rawOutput\n */\nexport const ToolOutputSchemas = {\n list_dir: z.object({\n type: z.literal('list_files_result'),\n files: z.array(z.object({\n filePath: z.string(),\n size: z.string(),\n modifyTime: z.string(),\n })),\n root: z.string(),\n listing: z.string().optional(),\n }),\n\n search_file: z.object({\n type: z.literal('search_file_result'),\n path: z.string(),\n pattern: z.string(),\n recursive: z.boolean().optional(),\n caseSensitive: z.boolean().optional(),\n results: z.array(z.object({\n filePath: z.string(),\n size: z.string(),\n modifyTime: z.string(),\n })),\n }),\n\n read_file: z.object({\n type: z.literal('read_file_result'),\n path: z.string(),\n content: z.string(),\n totalLineCount: z.number(),\n hasMore: z.boolean(),\n diagnostic: z.string().optional(),\n hint: z.string().optional(),\n image: z.object({\n data: z.string(),\n mimeType: z.string(),\n }).optional(),\n }),\n\n read_lints: z.object({\n type: z.literal('read_lints_result'),\n diagnostics: z.array(z.string()),\n totalCount: z.number().optional(),\n hint: z.string().optional(),\n isTruncated: z.boolean().optional(),\n }),\n\n rag_search: z.object({\n type: z.literal('knowledge_base_result'),\n selectedKnowledgeBases: z.string(),\n queryInput: z.string(),\n }),\n\n read_rules: z.object({\n type: z.literal('rule_match_result'),\n ruleDescription: z.string(),\n filePaths: z.array(z.string()),\n }),\n\n mcp_get_tool_description: z.object({}).passthrough(),\n\n mcp_call_tool: z.object({\n type: z.literal('mcp_call_tool_result'),\n serverName: z.string(),\n toolName: z.string(),\n data: z.array(z.union([\n z.object({\n type: z.literal('text'),\n text: z.string(),\n }),\n z.object({\n type: z.literal('image'),\n data: z.string(),\n mimeType: z.string(),\n }),\n z.object({\n type: z.literal('resource'),\n resource: z.object({\n uri: z.string(),\n mimeType: z.string().optional(),\n text: z.string().optional(),\n blob: z.string().optional(),\n }),\n }),\n ])),\n isError: z.boolean().optional(),\n error: z.unknown().optional(),\n hint: z.string().optional(),\n }),\n\n fetch_mcp_resource: z.object({\n type: z.literal('fetch_mcp_resource_result'),\n server: z.string(),\n uri: z.string(),\n content: z.string(),\n downloadPath: z.string().optional(),\n }),\n\n create_rule: z.object({\n type: z.literal('rule_create_result'),\n ruleName: z.string(),\n createState: z.enum(['success', 'invoke', 'cancelled']),\n hint: z.string().optional(),\n filePath: z.string().optional(),\n }),\n\n update_memory: z.object({\n type: z.literal('update_memory_result'),\n success: z.boolean(),\n message: z.string(),\n action: z.enum(['create', 'update', 'delete']),\n knowledge_id: z.string().optional(),\n }),\n\n search_content: z.object({\n type: z.literal('search_content_result'),\n directory: z.string(),\n pattern: z.string(),\n fileTypes: z.string(),\n matches: z.array(z.object({\n filePath: z.string(),\n content: z.string(),\n startLine: z.number(),\n endLine: z.number(),\n size: z.string(),\n modifyTime: z.string(),\n })),\n totalCount: z.number(),\n hasMore: z.boolean(),\n offset: z.number(),\n limit: z.number(),\n contextBefore: z.number(),\n contextAfter: z.number(),\n contextAround: z.number().optional(),\n outputMode: z.string(),\n caseSensitive: z.boolean(),\n hint: z.string().optional(),\n }),\n\n write_to_file: z.object({\n type: z.literal('write_to_file_result'),\n path: z.string(),\n addLineCount: z.number(),\n removedLines: z.number(),\n addedChars: z.number().optional(),\n removedChars: z.number().optional(),\n bytesWritten: z.number(),\n isNewFile: z.boolean(),\n oldContent: z.string().optional(),\n diagnostic: z.string().optional(),\n }),\n\n replace_in_file: z.object({\n type: z.literal('replace_in_file_result'),\n path: z.string(),\n addLineCount: z.number().optional(),\n removedLines: z.number().optional(),\n addedChars: z.number().optional(),\n removedChars: z.number().optional(),\n matchCount: z.number().optional(),\n hint: z.string().optional(),\n diagnosticChange: z.object({\n added: z.string(),\n removed: z.string(),\n unchanged: z.string(),\n }).optional(),\n }),\n\n delete_file: z.object({\n type: z.literal('delete_file_result'),\n path: z.string(),\n recursive: z.boolean(),\n hint: z.string().optional(),\n }),\n\n execute_command: z.object({\n type: z.literal('execute_command_result'),\n stdout: z.string(),\n stderr: z.string(),\n exitCode: z.number(),\n hint: z.string().optional(),\n serviceInfo: z.object({\n isWatchCommand: z.boolean(),\n isServiceOutput: z.boolean(),\n serviceReady: z.boolean(),\n message: z.string(),\n }).optional(),\n use_standalone_terminal: z.boolean().optional(),\n }),\n\n preview_url: z.object({\n type: z.literal('preview_tool_result'),\n url: z.string(),\n message: z.string(),\n }),\n\n ask_followup_question: z.object({\n type: z.literal('multi_question_result'),\n questions: z.array(z.object({\n id: z.string(),\n question: z.string(),\n options: z.array(z.string()),\n multiSelect: z.boolean().optional(),\n title: z.string().optional(),\n })),\n answers: z.record(z.union([z.string(), z.array(z.string())])),\n message: z.string(),\n }),\n\n invoke_integration: z.object({\n type: z.literal('invoke_integration_tool_result'),\n recommend: z.object({\n id: z.string(),\n type: z.string(),\n status: z.enum(['connected', 'disconnected']),\n }),\n message: z.string(),\n }),\n\n call_integration: z.object({\n type: z.literal('call_integration_tool_result'),\n integrationId: z.string(),\n toolName: z.string(),\n data: z.object({\n type: z.literal('text'),\n text: z.string(),\n }),\n isError: z.boolean().optional(),\n error: z.unknown().optional(),\n }),\n\n search_integration_tool: z.object({\n type: z.literal('search_integration_tool_result'),\n data: z.array(z.object({\n integrationId: z.string(),\n integrationName: z.string(),\n toolName: z.string(),\n description: z.string(),\n inputSchema: z.record(z.unknown()),\n })),\n hint: z.string().optional(),\n }),\n\n supabase_get_logs: z.object({\n type: z.enum([\n 'supabase_get_logs_result',\n 'supabase_execute_sql_result',\n 'supabase_apply_migration_result',\n 'supabase_list_migration_result',\n 'supabase_list_tables_result',\n ]),\n message: z.string(),\n }),\n\n supabase_execute_sql: z.object({\n type: z.enum([\n 'supabase_get_logs_result',\n 'supabase_execute_sql_result',\n 'supabase_apply_migration_result',\n 'supabase_list_migration_result',\n 'supabase_list_tables_result',\n ]),\n message: z.string(),\n }),\n\n supabase_apply_migration: z.object({\n type: z.enum([\n 'supabase_get_logs_result',\n 'supabase_execute_sql_result',\n 'supabase_apply_migration_result',\n 'supabase_list_migration_result',\n 'supabase_list_tables_result',\n ]),\n message: z.string(),\n }),\n\n supabase_list_migration: z.object({\n type: z.enum([\n 'supabase_get_logs_result',\n 'supabase_execute_sql_result',\n 'supabase_apply_migration_result',\n 'supabase_list_migration_result',\n 'supabase_list_tables_result',\n ]),\n message: z.string(),\n }),\n\n supabase_list_tables: z.object({\n type: z.enum([\n 'supabase_get_logs_result',\n 'supabase_execute_sql_result',\n 'supabase_apply_migration_result',\n 'supabase_list_migration_result',\n 'supabase_list_tables_result',\n ]),\n message: z.string(),\n }),\n\n cloud_studio_fetch_log: z.object({\n type: z.literal('cloud_studio_fetch_log_result'),\n success: z.boolean(),\n logs: z.record(z.string()),\n }),\n\n cloud_studio_execute_command: z.object({\n type: z.literal('cloud_studio_execute_command_result'),\n success: z.boolean(),\n message: z.string(),\n }),\n\n cloud_studio_deploy_sandbox: z.object({\n type: z.literal('cloud_studio_integration_result'),\n previewUrl: z.string().optional(),\n steps: z.array(z.object({\n status: z.enum(['idle', 'success', 'running', 'error']),\n name: z.enum(['createSandbox', 'uploadProject', 'installDependencies', 'startService', 'preview']),\n error: z.object({\n code: z.number().optional(),\n message: z.string().optional(),\n }).optional(),\n })),\n }),\n\n component_get_prompt: z.object({\n type: z.literal('component_get_prompt_result'),\n componentType: z.string(),\n webFramework: z.string(),\n data: z.object({\n type: z.literal('text'),\n text: z.string(),\n }),\n }),\n\n web_fetch: z.object({\n type: z.literal('web_fetch_tool_result'),\n message: z.string(),\n data: z.string(),\n loading: z.string().optional(),\n title: z.string().optional(),\n favicon: z.string().optional(),\n }),\n\n use_skill: z.object({\n type: z.literal('use_skill_tool_result'),\n commandMessage: z.string(),\n message: z.string(),\n }),\n\n web_search: z.object({\n type: z.literal('web_search_tool_result'),\n data: z.array(z.object({\n passage: z.string(),\n uri: z.string(),\n site: z.string(),\n title: z.string(),\n snippets: z.array(z.string()).optional(),\n content: z.string().optional(),\n })),\n searchInput: z.string().optional(),\n }),\n\n task: z.object({\n type: z.literal('task_tool_result'),\n toolInfo: z.array(z.object({\n name: z.string(),\n info: z.string(),\n needApprove: z.boolean().optional(),\n toolCallId: z.string().optional(),\n executeStatus: z.enum(['ing', 'completed', 'cancel', 'fail']).optional(),\n })).optional(),\n startCallTool: z.boolean().optional(),\n finalResult: z.string().optional(),\n toolCallBrief: z.string().optional(),\n }),\n\n codebase_search: z.object({\n type: z.literal('codebase_search_result'),\n query: z.string(),\n path: z.string(),\n content: z.string(),\n }),\n\n lsp: z.object({\n type: z.literal('lsp_tool_result'),\n operation: z.string(),\n result: z.string(),\n resultCount: z.number().optional(),\n fileCount: z.number().optional(),\n character: z.number().optional(),\n }),\n\n spec_create: z.object({\n type: z.literal('plan_create_tool_result'),\n message: z.string(),\n data: z.string(),\n }),\n\n spec_update: z.object({\n type: z.literal('plan_update_tool_result'),\n status: z.enum(['prepare', 'ready', 'building', 'finished']),\n data: z.string(),\n reminder: z.string(),\n }),\n} satisfies Record<ToolName, z.ZodTypeAny>;\n\n// ============================================================================\n// 工具 Schema 验证工具函数\n// ============================================================================\n\n/**\n * 验证工具输入\n * @param toolName 工具名称\n * @param input 输入数据\n * @returns 验证结果\n */\nexport function validateToolInput<TName extends ToolName>(\n toolName: TName,\n input: unknown\n): { success: true; data: ToolInputMap[TName] } | { success: false; error: z.ZodError } {\n const schema = ToolInputSchemas[toolName];\n const result = schema.safeParse(input);\n if (result.success) {\n return { success: true, data: result.data as ToolInputMap[TName] };\n }\n return { success: false, error: result.error };\n}\n\n/**\n * 验证工具输出\n * @param toolName 工具名称\n * @param output 输出数据\n * @returns 验证结果\n */\nexport function validateToolOutput<TName extends ToolName>(\n toolName: TName,\n output: unknown\n): { success: true; data: ToolOutputMap[TName] } | { success: false; error: z.ZodError } {\n const schema = ToolOutputSchemas[toolName];\n const result = schema.safeParse(output);\n if (result.success) {\n return { success: true, data: result.data as ToolOutputMap[TName] };\n }\n return { success: false, error: result.error };\n}\n","/**\n * Mock Agent Provider\n *\n * 职责:\n * 1. 专门用于 Mock 数据验证链路\n * 2. 返回符合 ACP 协议的模拟数据\n * 3. 用于前端开发和测试阶段\n *\n * Mock 数据包括:\n * - 流式文本消息 (agent_message_chunk)\n * - 嵌入资源内容\n * - 资源链接\n * - 工具调用流程 (tool_call + tool_call_update)\n */\n\nimport type {\n AuthenticateRequest,\n AuthenticateResponse,\n CancelNotification,\n InitializeRequest,\n InitializeResponse,\n LoadSessionRequest,\n LoadSessionResponse,\n NewSessionRequest,\n NewSessionResponse,\n PromptRequest,\n PromptResponse,\n SessionNotification,\n SetSessionModelRequest,\n SetSessionModelResponse,\n SetSessionModeRequest,\n SetSessionModeResponse\n} from '@agentclientprotocol/sdk';\n\nimport type {\n PermissionRequestParams,\n PermissionResponse,\n SessionUpdateParams} from './acp-client-protocol';\nimport {\n DeleteFilesResult,\n ReadFileResult,\n validateToolOutput} from './tool-schemas.js';\nimport type { IAgentProvider } from './types';\n\nexport interface MockAgentProviderConfig {\n useRealProvider?: boolean; // 是否使用真实 Provider(保留向后兼容)\n}\n\n/**\n * ACP 协议 Mock 数据 - 初始化响应\n */\nexport const mockInitializeResponse: InitializeResponse = {\n protocolVersion: 1,\n agentCapabilities: {\n loadSession: true,\n promptCapabilities: {\n image: true,\n audio: false,\n embeddedContext: true\n },\n mcpCapabilities: {\n http: true,\n sse: true\n }\n }\n};\n\n/**\n * ACP 协议 Mock 数据 - 会话创建响应\n */\nexport const mockNewSessionResponse = (sessionId: string): NewSessionResponse => ({\n sessionId\n});\n\n/**\n * Mock 会话数据存储\n * 每个会话都有对应的模拟历史消息\n */\nconst mockSessionHistories = new Map<string, Array<{\n type: 'user' | 'assistant';\n content: string;\n timestamp: number;\n}>>();\n\n// 初始化会话历史数据\nmockSessionHistories.set('1', [\n {\n type: 'user',\n content: '帮我开发一个五子棋游戏,需要包含以下功能:\\n1. 双人对战模式\\n2. 悔棋功能\\n3. 计时器',\n timestamp: Date.now() - 1800000 - 60000,\n },\n {\n type: 'assistant',\n content: '好的,我来帮你开发一个五子棋游戏。我会创建以下文件结构:\\n\\n- `index.html` - 主页面\\n- `game.js` - 游戏逻辑\\n- `style.css` - 样式文件\\n\\n让我开始创建这些文件...',\n timestamp: Date.now() - 1800000 - 50000,\n },\n {\n type: 'assistant',\n content: '我已经完成了所有文件的创建和修改。游戏支持双人对战、悔棋和计时功能。你可以直接在浏览器中打开 index.html 开始游戏。',\n timestamp: Date.now() - 1800000,\n },\n]);\n\nmockSessionHistories.set('2', [\n {\n type: 'user',\n content: '登录页面的样式有问题,按钮颜色不对齐,帮我修复一下',\n timestamp: Date.now() - 7200000 - 120000,\n },\n {\n type: 'assistant',\n content: '我来检查登录页面的样式。让我先看看相关的 CSS 文件...',\n timestamp: Date.now() - 7200000 - 100000,\n },\n {\n type: 'assistant',\n content: '样式问题已修复,请查看效果。主要修改了按钮的 flex 布局和颜色变量。',\n timestamp: Date.now() - 7200000,\n },\n]);\n\nmockSessionHistories.set('3', [\n {\n type: 'user',\n content: 'API 接口响应太慢了,需要添加缓存和错误重试机制',\n timestamp: Date.now() - 14400000 - 180000,\n },\n {\n type: 'assistant',\n content: '好的,我来优化 API 接口。我会实现:\\n1. Redis 缓存层\\n2. 指数退避重试策略\\n3. 请求去重',\n timestamp: Date.now() - 14400000 - 150000,\n },\n {\n type: 'assistant',\n content: '已添加缓存和错误重试机制。性能提升了约 60%。',\n timestamp: Date.now() - 14400000,\n },\n]);\n\nmockSessionHistories.set('4', [\n {\n type: 'user',\n content: '帮我为核心模块添加单元测试,覆盖率要达到 80% 以上',\n timestamp: Date.now() - 86400000 - 300000,\n },\n {\n type: 'assistant',\n content: '好的,我会使用 Jest 来编写单元测试。让我先看看核心模块的代码结构...',\n timestamp: Date.now() - 86400000 - 250000,\n },\n {\n type: 'assistant',\n content: '测试覆盖率已达到 85%,超过了目标。主要测试了:\\n- 用户认证逻辑\\n- 数据验证\\n- 错误处理',\n timestamp: Date.now() - 86400000,\n },\n]);\n\nexport class MockAgentProvider implements IAgentProvider {\n // ========== ACP 原生事件回调集合 ==========\n private sessionUpdateCallbacks: Set<(params: SessionUpdateParams) => void> = new Set();\n private permissionRequestResolvers: Map<string, (params: PermissionRequestParams) => Promise<PermissionResponse>> = new Map();\n private errorCallbacks: Set<(error: Error, sessionId?: string) => void> = new Set();\n\n // Minimal connection tracking to keep API compatibility with previous SSE provider\n private activeConnections: Map<string, unknown> = new Map();\n /**\n * Mock 模式下,等待用户权限决策的解析器集合\n * key: toolCallId -> resolver(outcome)\n */\n private pendingPermissionResolvers: Map<string, (outcome: { outcome: 'selected' | 'cancelled'; optionId?: string; timedOut?: boolean }) => void> = new Map();\n /**\n * Mock 模式下,已初始化的会话集合\n */\n private initializedSessions: Map<string, { cwd: string; mcpServers: any[]; createdAt: number }> = new Map();\n /**\n * 临时缓存当前正在发送的消息的 SessionUpdateParams\n * key: sessionId -> params[]\n * 在 sendMockMessage 开始时初始化,结束时保存到 mockSessionHistories\n */\n private currentSessionUpdates: Map<string, SessionUpdateParams[]> = new Map();\n\n constructor(config: MockAgentProviderConfig = {}) {\n console.log('[MockAgentProvider] Initialized');\n\n // 初始化一些默认的 Mock session 数据\n this.initializeMockSessions();\n }\n\n setSessionMode?(params: SetSessionModeRequest): Promise<SetSessionModeResponse | void> {\n throw new Error('Method not implemented.');\n }\n setSessionModel?(params: SetSessionModelRequest): Promise<SetSessionModelResponse | void> {\n throw new Error('Method not implemented.');\n }\n extMethod?(method: string, params: Record<string, unknown>): Promise<Record<string, unknown>> {\n throw new Error('Method not implemented.');\n }\n extNotification?(method: string, params: Record<string, unknown>): Promise<void> {\n throw new Error('Method not implemented.');\n }\n\n /**\n * 初始化默认的 Mock session 数据\n */\n private initializeMockSessions(): void {\n const mockSessions = [\n {\n sessionId: '1',\n cwd: '/Users/example/Project',\n mcpServers: [],\n createdAt: Date.now() - 3600000, // 1小时前\n },\n {\n sessionId: '2',\n cwd: '/Users/example/Project',\n mcpServers: [],\n createdAt: Date.now() - 7200000, // 2小时前\n },\n {\n sessionId: '3',\n cwd: '/Users/example/Project',\n mcpServers: [],\n createdAt: Date.now() - 86400000, // 1天前\n },\n ];\n\n mockSessions.forEach(session => {\n this.initializedSessions.set(session.sessionId, {\n cwd: session.cwd,\n mcpServers: session.mcpServers,\n createdAt: session.createdAt,\n });\n });\n\n console.log('[MockAgentProvider] Initialized mock sessions:', mockSessions.length);\n }\n\n /**\n * 从 PromptRequest 中提取文本内容\n * Provider 自己决定如何处理 ContentBlock 数组\n */\n private extractTextFromRequest(request: PromptRequest): string {\n if (request.prompt && Array.isArray(request.prompt)) {\n return request.prompt\n .filter(block => block.type === 'text')\n .map(block => ('text' in block ? block.text : ''))\n .join('\\n');\n }\n return '';\n }\n\n /**\n * 发送消息 - 直接使用 Mock 数据\n * 接收 PromptRequest,由 Provider 决定如何处理\n */\n async sendMessage(request: PromptRequest): Promise<void> {\n // 提取文本内容 - 这是 Provider 的职责\n const content = this.extractTextFromRequest(request);\n const sessionId = request.sessionId;\n\n console.log('[MockAgentProvider] Sending mock message:', { sessionId, content });\n\n try {\n // 直接使用 Mock 数据\n await this.sendMockMessage(sessionId, content);\n } catch (error) {\n console.error('[MockAgentProvider] Send message failed:', error);\n\n // 触发错误事件\n this.emitEvent({\n type: 'error',\n sessionId,\n timestamp: Date.now(),\n data: { error: error instanceof Error ? error.message : 'Unknown error' },\n });\n\n throw error;\n }\n }\n\n /**\n * 使用 Mock 数据模拟流式响应\n */\n private async sendMockMessage(sessionId: string, content: string): Promise<void> {\n console.log('[MockAgentProvider] Using mock data for:', content);\n\n // ✅ 初始化临时事件缓存\n this.currentSessionUpdates.set(sessionId, []);\n\n // 发送连接成功事件\n this.emitEvent({\n type: 'connected',\n sessionId,\n timestamp: Date.now(),\n });\n\n // 模拟延迟\n await this.delay(300);\n\n // 1. 发送文本回复消息\n const messageId1 = `m-text-${Date.now()}`;\n const textReply = `我收到了你发送的「${content}」信息,这是符合 ACP 协议的 mock 数据。\\n\\n下面演示不同类型的 ContentBlock:\\n\\n`;\n await this.streamMockTextContent(sessionId, messageId1, textReply);\n\n await this.delay(500);\n\n // 2. 发送包含嵌入资源的回复\n const messageId2 = `m-resource-${Date.now()}`;\n await this.sendMockEmbeddedResource(sessionId, messageId2);\n\n await this.delay(500);\n\n // 3. 发送包含资源链接的回复\n const messageId3 = `m-link-${Date.now()}`;\n await this.sendMockResourceLink(sessionId, messageId3);\n\n await this.delay(500);\n\n // 4. 模拟工具调用流程\n const toolMessageId = `m-tool-${Date.now()}`;\n await this.sendMockEditContentToolCallFlow(sessionId, toolMessageId);\n await this.delay(500);\n\n const diffToolMessageId = `m-tool-d-${Date.now()}`;\n await this.sendMockEditDiffToolCallFlow(sessionId, diffToolMessageId);\n await this.delay(500);\n\n const readToolMessageId = `m-tool-r-${Date.now()}`;\n await this.sendMockReadContentToolCallFlow(sessionId, readToolMessageId);\n await this.delay(300);\n\n // 添加 search_file 和 search_content 的 mock 数据\n const searchContentMessageId = `m-tool-sc-${Date.now()}`;\n await this.sendMockSearchContentToolCallFlow(sessionId, searchContentMessageId);\n\n await this.delay(500);\n\n const searchFileMessageId = `m-tool-sf-${Date.now()}`;\n await this.sendMockSearchFileToolCallFlow(sessionId, searchFileMessageId);\n\n await this.delay(500);\n\n // 6. 发送 list_files 工具调用\n const listFilesMessageId = `m-tool-lf-${Date.now()}`;\n await this.sendMockListFilesToolCallFlow(sessionId, listFilesMessageId);\n\n await this.delay(500);\n\n // 7. 发送 ask_followup_question 工具调用\n const askFollowupMessageId = `m-tool-afq-${Date.now()}`;\n await this.sendMockAskFollowupQuestionToolCallFlow(sessionId, askFollowupMessageId);\n await this.delay(500);\n\n // 添加删除文件工具调用的 mock 数据\n const deleteFileMessageId = `m-tool-df-${Date.now()}`;\n await this.sendMockDeleteFileToolCallFlow(sessionId, deleteFileMessageId);\n\n await this.delay(300);\n\n // 8. 发送 execute_command 工具调用\n const executeCommandMessageId = `m-tool-ec-${Date.now()}`;\n await this.sendMockExecuteCommandToolCallFlow(sessionId, executeCommandMessageId);\n\n await this.delay(300);\n\n // 8.1 发送 execute_command 工具调用(需要权限确认)\n const executeCommandApprovalMessageId = `m-tool-ec-approval-${Date.now()}`;\n await this.sendMockExecuteCommandWithApprovalToolCallFlow(sessionId, executeCommandApprovalMessageId);\n\n await this.delay(300);\n\n // 8.2 发送 write_to_file 工具调用(需要权限确认)\n const writeFileApprovalMessageId = `m-tool-wtf-approval-${Date.now()}`;\n await this.sendMockWriteFileWithApprovalToolCallFlow(sessionId, writeFileApprovalMessageId);\n\n await this.delay(300);\n\n // 9. 发送 preview_url 工具调用 (测试 unknown tool)\n const previewUrlMessageId = `m-tool-pu-${Date.now()}`;\n await this.sendMockPreviewUrlToolCallFlow(sessionId, previewUrlMessageId);\n\n await this.delay(300);\n\n // 10. 发送 list_code_definition_names 工具调用 (测试 unknown tool)\n const listCodeDefMessageId = `m-tool-lcd-${Date.now()}`;\n await this.sendMockListCodeDefinitionNamesToolCallFlow(sessionId, listCodeDefMessageId);\n\n await this.delay(300);\n\n // 添加 web_fetch 工具调用的 mock 数据\n const webFetchMessageId = `m-tool-wf-${Date.now()}`;\n await this.sendMockWebFetchToolCallFlow(sessionId, webFetchMessageId);\n\n await this.delay(300);\n\n // 11. 发送 read_lints 工具调用\n const readLintsMessageId = `m-tool-rl-${Date.now()}`;\n await this.sendMockReadLintsToolCallFlow(sessionId, readLintsMessageId);\n\n await this.delay(300);\n\n // 10. 发送 web_search 工具调用\n const webSearchMessageId = `m-tool-ws-${Date.now()}`;\n await this.sendMockWebSearchToolCallFlow(sessionId, webSearchMessageId);\n\n await this.delay(300);\n\n // 发送完成事件\n this.emitEvent({\n type: 'done',\n sessionId,\n timestamp: Date.now(),\n });\n }\n\n /**\n * 流式发送 Mock 文本消息内容 (符合 ACP 协议的 agent_message_chunk)\n */\n private async streamMockTextContent(sessionId: string, messageId: string, content: string): Promise<void> {\n const chunkSize = 10; // 每次发送 10 个字符\n\n for (let i = 0; i < content.length; i += chunkSize) {\n // 获取当前块的内容\n const chunk = content.substring(i, i + chunkSize);\n\n console.log(`Text Chunk: \"${chunk}\" | messageId: ${messageId} | sessionId: ${sessionId}`);\n\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: chunk,\n }\n }\n } as SessionNotification,\n });\n\n // 模拟网络延迟\n await this.delay(50);\n }\n }\n\n /**\n * 延迟函数\n */\n private delay(ms: number): Promise<void> {\n return new Promise(resolve => setTimeout(resolve, ms));\n }\n\n /**\n * 取消消息 (Mock 模式下直接完成)\n * @deprecated 使用 cancel 代替\n */\n async cancelMessage(sessionId: string): Promise<void> {\n console.log('[MockAgentProvider] Message cancelled:', sessionId);\n\n // 发送完成事件\n this.emitEvent({\n type: 'done',\n sessionId,\n timestamp: Date.now(),\n });\n }\n\n /**\n * ACP 协议: prompt - 发送用户消息\n *\n * Mock 实现: 复用 sendMessage\n */\n async prompt(request: PromptRequest): Promise<PromptResponse> {\n console.log('[MockAgentProvider] prompt called:', request);\n await this.sendMessage(request);\n // Mock 模式下,返回 end_turn 响应\n return {\n stopReason: 'end_turn',\n };\n }\n\n /**\n * ACP 协议: cancel - 取消消息\n *\n * Mock 实现: 复用 cancelMessage\n */\n async cancel(params: CancelNotification): Promise<void> {\n console.log('[MockAgentProvider] cancel called:', params);\n return this.cancelMessage(params.sessionId);\n }\n\n // ========== ACP 原生事件监听器 ==========\n\n /**\n * 监听会话更新事件(ACP 协议原生)\n */\n onSessionUpdate(callback: (params: SessionUpdateParams) => void): () => void {\n this.sessionUpdateCallbacks.add(callback);\n return () => {\n this.sessionUpdateCallbacks.delete(callback);\n };\n }\n\n /**\n * 监听权限请求事件(ACP 协议原生)\n */\n onRequestPermission(\n callback: (params: PermissionRequestParams) => Promise<PermissionResponse>\n ): () => void {\n this.permissionRequestResolvers.set('__global__', callback);\n return () => {\n this.permissionRequestResolvers.delete('__global__');\n };\n }\n\n /**\n * 监听错误事件(扩展事件,非 ACP 标准协议)\n */\n onError(callback: (error: Error, sessionId?: string) => void): () => void {\n this.errorCallbacks.add(callback);\n return () => {\n this.errorCallbacks.delete(callback);\n };\n }\n\n /**\n * 触发会话更新事件(内部辅助方法)\n */\n private emitSessionUpdate(params: SessionUpdateParams): void {\n console.log('[MockAgentProvider] Emitting session update:', params, 'to', this.sessionUpdateCallbacks.size, 'callbacks');\n\n this.sessionUpdateCallbacks.forEach(callback => {\n try {\n callback(params);\n } catch (error) {\n console.error('[MockAgentProvider] Session update callback error:', error);\n }\n });\n }\n\n /**\n * 触发错误事件(内部辅助方法)\n */\n private emitError(error: Error, sessionId?: string): void {\n console.log('[MockAgentProvider] Emitting error:', error, 'for session:', sessionId);\n\n this.errorCallbacks.forEach(callback => {\n try {\n callback(error, sessionId);\n } catch (err) {\n console.error('[MockAgentProvider] Error callback error:', err);\n }\n });\n }\n\n /**\n * 向后兼容的 emitEvent 方法\n * 将旧的事件格式桥接到新的 ACP 原生事件\n */\n private emitEvent(event: any): void {\n // 根据 event.type 路由到对应的新方法\n if (event.type === 'error') {\n const error = event.data?.error ? new Error(event.data.error) : new Error('Unknown error');\n this.emitError(error, event.sessionId);\n } else if (event.type === 'session_update' && event.notification) {\n // 转换为 SessionUpdateParams\n const params: SessionUpdateParams = {\n notification: event.notification,\n messageId: event.messageId,\n };\n this.emitSessionUpdate(params);\n } else if (event.type === 'connected' || event.type === 'done') {\n // 这些事件类型不再需要,忽略\n console.log('[MockAgentProvider] Ignoring deprecated event type:', event.type);\n } else {\n console.warn('[MockAgentProvider] Unknown event type:', event.type);\n }\n }\n\n /**\n * ACP 协议: initialize - 初始化连接\n *\n * Mock 实现: 返回预设的初始化响应\n */\n async initialize(request: InitializeRequest): Promise<InitializeResponse> {\n console.log('[MockAgentProvider] initialize', request);\n\n // 模拟网络延迟\n await this.delay(100);\n\n // 返回 Mock 初始化响应\n return mockInitializeResponse;\n }\n\n /**\n * ACP 协议: authenticate - 身份验证\n *\n * Mock 实现: 直接返回成功\n */\n async authenticate(request: AuthenticateRequest): Promise<AuthenticateResponse | void> {\n console.log('[MockAgentProvider] authenticate', request);\n // Mock 模式下不需要身份验证\n return;\n }\n\n /**\n * ACP 协议: newSession - 创建新会话\n *\n * Mock 实现: 生成唯一的 sessionId 并存储会话信息\n */\n async newSession(request: NewSessionRequest): Promise<NewSessionResponse> {\n console.log('[MockAgentProvider] session/new', request);\n\n // 模拟网络延迟\n await this.delay(100);\n\n // 生成唯一的 sessionId (格式: sess_<timestamp>_<random>)\n const sessionId = `sess_${Date.now()}_${Math.random().toString(36).substring(2, 9)}`;\n\n // 存储会话信息\n this.initializedSessions.set(sessionId, {\n cwd: request.cwd,\n mcpServers: request.mcpServers || [],\n createdAt: Date.now()\n });\n\n console.log('[MockAgentProvider] Created session:', sessionId, 'with config:', {\n cwd: request.cwd,\n mcpServers: request.mcpServers\n });\n\n // 返回 sessionId\n return mockNewSessionResponse(sessionId);\n }\n\n /**\n * ACP 协议: session/load - 加载已有会话\n *\n * Mock 实现: 流式回放会话历史(完整的事件回放)\n */\n async loadSession(request: LoadSessionRequest): Promise<LoadSessionResponse> {\n console.log('[MockAgentProvider] session/load', request);\n\n const { sessionId, cwd, mcpServers } = request;\n\n // 更新或添加会话配置\n this.initializedSessions.set(sessionId, {\n cwd,\n mcpServers: mcpServers || [],\n createdAt: Date.now()\n });\n\n // 从 mockSessionHistories 加载历史\n const history = mockSessionHistories.get(sessionId);\n if (history && history.length > 0) {\n console.log('[MockAgentProvider] Loading session from mockSessionHistories:', sessionId, 'with', history.length, 'entries');\n\n // 模拟流式回放会话历史\n for (let i = 0; i < history.length; i++) {\n const message = history[i];\n\n // 检查是否为事件数组格式(JSON 字符串)\n if (message.type === 'assistant' && message.content && message.content.startsWith('[')) {\n try {\n const events: any[] = JSON.parse(message.content);\n console.log('[MockAgentProvider] Replaying events from history:', {\n index: i,\n eventCount: events.length,\n });\n\n // 回放所有事件(通过 emitEvent 桥接)\n for (const event of events) {\n await this.delay(20); // 每个事件间隔 20ms\n this.emitEvent(event);\n }\n } catch (error) {\n console.error('[MockAgentProvider] Failed to parse events from history:', error);\n }\n } else {\n // 旧格式:普通文本消息\n const messageId = `load-${sessionId}-${i}-${Date.now()}`;\n const sessionUpdateType = message.type === 'user'\n ? 'user_message_chunk'\n : 'agent_message_chunk';\n\n await this.delay(50);\n\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: message.timestamp,\n notification: {\n sessionId,\n update: {\n sessionUpdate: sessionUpdateType,\n content: {\n type: 'text',\n text: message.content,\n }\n }\n } as SessionNotification,\n });\n }\n }\n\n console.log('[MockAgentProvider] Session loaded from mockSessionHistories:', sessionId);\n return {};\n }\n\n // 如果没有任何历史数据\n console.log('[MockAgentProvider] No history found for session:', sessionId);\n return {};\n }\n\n /**\n * ACP 协议: session/request_permission - 请求用户权限\n *\n * Agent 调用此方法来请求用户批准工具调用。\n * 此方法会:\n * 1. 发送 request_permission 事件给 UI\n * 2. 阻塞等待用户响应\n * 3. 返回用户的决策结果\n *\n * @param sessionId ACP 会话 ID\n * @param toolCallId 工具调用 ID\n * @param toolCall 工具调用信息\n * @param options 可用的权限选项\n * @returns 用户的决策结果\n */\n async requestPermission(\n sessionId: string,\n toolCallId: string,\n toolCall: any,\n options: any[]\n ): Promise<{ outcome: 'selected' | 'cancelled'; optionId?: string; timedOut?: boolean }> {\n console.log('[MockAgentProvider] requestPermission called:', { sessionId, toolCallId, toolCall, options });\n\n // 发送权限请求事件给 UI\n this.emitEvent({\n type: 'request_permission',\n sessionId,\n messageId: `permission-${toolCallId}`,\n timestamp: Date.now(),\n data: {\n sessionId,\n toolCall,\n options,\n },\n });\n\n // 阻塞等待用户决策(或超时)\n const permissionOutcome = await new Promise<{\n outcome: 'selected' | 'cancelled';\n optionId?: string;\n timedOut?: boolean;\n }>(resolve => {\n // 保存解析器,respondToPermissionRequest 会调用它\n this.pendingPermissionResolvers.set(toolCallId, resolve);\n\n // 超时降级:如果 60s 内没有用户响应,默认拒绝(避免无限挂起)\n setTimeout(() => {\n if (this.pendingPermissionResolvers.has(toolCallId)) {\n this.pendingPermissionResolvers.delete(toolCallId);\n console.log('[MockAgentProvider] Permission request timed out:', toolCallId);\n resolve({ outcome: 'cancelled', optionId: 'reject-once', timedOut: true });\n }\n }, 60000);\n });\n\n console.log('[MockAgentProvider] Permission outcome:', permissionOutcome);\n return permissionOutcome;\n }\n\n /**\n * 直接释放权限请求的 await(更直接的方法)\n *\n * 这个方法直接从 pendingPermissionResolvers 中获取 resolver 并调用,\n * 不需要经过 respondToPermissionRequest 的中转。\n *\n * @param toolCallId 工具调用 ID\n * @param outcome 用户的决策结果\n */\n releasePermissionRequest(toolCallId: string, outcome: { outcome: 'selected' | 'cancelled'; optionId?: string }): void {\n console.log('[MockAgentProvider] releasePermissionRequest called:', { toolCallId, outcome });\n\n const resolver = this.pendingPermissionResolvers.get(toolCallId);\n if (resolver) {\n // 直接调用 resolver 释放 await\n resolver({ ...outcome, timedOut: false });\n this.pendingPermissionResolvers.delete(toolCallId);\n console.log('[MockAgentProvider] Permission request released:', toolCallId);\n } else {\n console.warn('[MockAgentProvider] No pending permission request found for:', toolCallId);\n }\n }\n\n /**\n * 响应用户的权限决策(兼容旧接口)\n *\n * 这个方法内部调用 releasePermissionRequest。\n * 保留这个方法是为了向后兼容和符合 ACP 协议的命名规范。\n *\n * @param sessionId ACP 会话 ID(未使用,保留以符合接口)\n * @param payload 权限响应数据\n */\n async respondToPermissionRequest(sessionId: string, payload: {\n toolCallId: string;\n outcome: { outcome: 'selected' | 'cancelled'; optionId?: string }\n }): Promise<void> {\n console.log('[MockAgentProvider] respondToPermissionRequest called:', payload);\n\n // 直接调用 releasePermissionRequest\n this.releasePermissionRequest(payload.toolCallId, payload.outcome);\n }\n\n /**\n * 销毁 Provider\n */\n destroy(): void {\n // 清空所有回调解集合\n this.activeConnections.clear();\n this.pendingPermissionResolvers.clear();\n this.permissionRequestResolvers.clear();\n this.initializedSessions.clear();\n this.sessionUpdateCallbacks.clear();\n this.errorCallbacks.clear();\n // ✅ 清空临时事件缓存\n this.currentSessionUpdates.clear();\n console.log('[MockAgentProvider] Destroyed');\n }\n\n /**\n * 获取所有已初始化的会话列表(扩展方法,非 ACP 标准协议)\n * 返回 sessionId 及其配置信息和标题\n */\n getAllSessions(): Array<{ sessionId: string; title: string; cwd: string; mcpServers: any[]; createdAt: number }> {\n const mockTitles = {\n '1': '开发五子棋游戏',\n '2': '修复登录页面样式',\n '3': 'API 接口优化',\n };\n\n const sessions = Array.from(this.initializedSessions.entries()).map(([sessionId, config]) => ({\n sessionId,\n title: mockTitles[sessionId as keyof typeof mockTitles] || `Session ${sessionId}`,\n ...config,\n }));\n console.log('[MockAgentProvider] Getting all sessions:', sessions.length);\n return sessions;\n }\n\n /**\n * 发送嵌入资源消息 (符合 ACP 协议)\n */\n private async sendMockEmbeddedResource(sessionId: string, messageId: string): Promise<void> {\n console.log(`📎 Sending embedded resource: ${messageId}`);\n\n // 发送文本介绍部分\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-intro`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: '这是一个嵌入的 Python 文件示例:\\n\\n',\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(200);\n\n // 发送嵌入资源内容块\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-resource`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'resource',\n resource: {\n uri: 'file:///mock/example.py',\n mimeType: 'text/x-python',\n text: 'def hello_world():\\n print(\"Hello, World!\")\\n return \"success\"\\n\\nif __name__ == \"__main__\":\\n hello_world()'\n }\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(200);\n\n // 发送结尾文本\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-conclusion`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: '\\n\\n以上是直接嵌入的文件内容,无需额外请求即可显示。',\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(100);\n }\n\n /**\n * 发送资源链接消息 (符合 ACP 协议)\n */\n private async sendMockResourceLink(sessionId: string, messageId: string): Promise<void> {\n console.log(`🔗 Sending resource link: ${messageId}`);\n\n // 发送文本介绍部分\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-intro`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: '这是一个指向外部文档的资源链接:\\n\\n',\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(200);\n\n // 发送资源链接内容块\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-link`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'resource_link',\n uri: 'https://agentclientprotocol.com/protocol/content',\n name: 'ACP Content Protocol',\n mimeType: 'text/html',\n description: 'Agent Client Protocol - Content specification',\n size: BigInt(256000)\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(200);\n\n // 发送结尾文本\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-conclusion`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: '\\n\\n点击上面的链接可以访问完整的协议文档。',\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(100);\n }\n\n /**\n * 发送完整的工具调用流程 (符合 ACP 协议)\n */\n private async sendMockEditDiffToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`🔧 Starting tool call flow: ${messageId}`);\n\n // 2. 发送工具调用事件 - 使用 ACP 协议的 tool_call session/update\n const toolCallId = `tool-${Date.now()}`;\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Write File',\n kind: 'edit',\n status: 'pending',\n locations: [{ path: 'example.py', line: 0 }]\n }\n } as SessionNotification,\n });\n\n await this.delay(100);\n\n // 调用 requestPermission 接口请求用户权限\n // 这个方法会:\n // 1. 发送 request_permission 事件给 UI\n // 2. 阻塞等待用户响应\n // 3. 返回用户的决策结果\n // const permissionOutcome = await this.requestPermission(\n // sessionId,\n // toolCallId,\n // {\n // toolCallId,\n // kind: 'edit',\n // arguments: {\n // file_path: 'example.py',\n // content: 'def hello_world():\\n print(\"Hello from ACP tool call!\")...'\n // }\n // },\n // [\n // { optionId: 'allow-once', name: 'Allow once', kind: 'allow_once' },\n // { optionId: 'reject-once', name: 'Reject', kind: 'reject_once' }\n // ]\n // );\n\n // 根据用户决策推进后续 Mock 行为,区分同意与拒绝,发送不同的 tool_call_update 与消息\n // if (permissionOutcome.outcome === 'selected') {\n const content = 'def hello_world():\\n print(\"Hello from ACP tool call!\")\\n return \"success\"\\n\\nif __name__ == \"__main__\":\\n hello_world()';\n const chunkSize = 10; // 每次发送 10 个字符\n\n for (let i = 0; i < content.length; i += chunkSize) {\n // 获取当前块的内容\n const chunk = content.substring(i, i + chunkSize);\n\n console.log(`🐸 tool call Chunk: \"${chunk}\" | messageId: ${messageId} | sessionId: ${sessionId}`);\n\n // ✅ 5.1 开始执行 (in_progress)\n\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress', // ✅ 最终完成状态\n content: [\n {\n type: 'diff', // ✅ 使用 diff 类型显示文件修改\n path: 'example.py',\n oldText: '', // 新文件\n newText: chunk\n }\n ],\n locations: [{ path: 'example.py', line: 0 }]\n }\n } as SessionNotification,\n });\n // 模拟网络延迟\n await this.delay(50);\n }\n\n // 模拟工具执行时间\n await this.delay(1200);\n\n // ✅ 5.2 执行完成 (completed)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed', // ✅ 最终完成状态\n content: [\n {\n type: 'diff', // ✅ 使用 diff 类型显示文件修改\n path: 'example.py',\n oldText: 'def 213', // 新文件\n newText: 'def hello_world():\\n print(\"Hello from ACP tool call!\")\\n return \"success\"\\n\\nif __name__ == \"__main__\":\\n hello_world()'\n }\n ],\n locations: [{ path: 'example.py', line: 0 }]\n }\n } as SessionNotification,\n });\n\n // 发送总结消息\n await this.delay(300);\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-summary`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: '✅ 工具调用完成!我成功创建了一个名为 `example.py` 的文件。\\n\\nACP 协议工具调用流程:\\n1. `tool_call` (pending) - 创建工具调用\\n2. `tool_call_update` (in_progress) - 开始执行\\n3. `tool_call_update` (completed) - 执行完成\\n\\n这就是完整的工具调用生命周期!',\n }\n }\n } as SessionNotification,\n });\n // } else {\n // // ✅ 5.3 权限拒绝 (failed)\n // await this.delay(300);\n // this.emitEvent({\n // type: 'session_update',\n // sessionId,\n // messageId,\n // timestamp: Date.now(),\n // notification: {\n // sessionId,\n // update: {\n // sessionUpdate: 'tool_call_update',\n // toolCallId,\n // status: 'failed', // ✅ 失败状态\n // content: [\n // {\n // type: 'content',\n // content: {\n // type: 'text',\n // text: permissionOutcome.timedOut\n // ? 'Permission request timed out; tool execution cancelled (mock)'\n // : 'User rejected permission; tool execution cancelled (mock)'\n // }\n // }\n // ]\n // }\n // } as SessionNotification,\n // });\n\n // // 发送总结消息\n // await this.delay(300);\n // this.emitEvent({\n // type: 'session_update',\n // sessionId,\n // messageId: `${messageId}-summary`,\n // timestamp: Date.now(),\n // notification: {\n // sessionId,\n // update: {\n // sessionUpdate: 'agent_message_chunk',\n // content: {\n // type: 'text',\n // text: permissionOutcome.timedOut\n // ? '⏰ 权限请求超时,工具调用已取消(mock)。'\n // : '❌ 已拒绝执行写文件操作,工具调用已取消(mock)。',\n // }\n // }\n // } as SessionNotification,\n // });\n // }\n\n await this.delay(100);\n\n // 5. 发送工具调用完成后的总结消息\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-summary`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: '✅ 工具调用完成!我成功创建了一个名为 `example.py` 的文件。\\n\\n工具调用流程包括:\\n1. 发送工具调用请求 (tool_call)\\n2. 返回执行结果 (tool_call_update)\\n\\n这就是 ACP 协议中完整的工具调用生命周期。',\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(100);\n }\n\n private async sendMockEditContentToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`🔧 Starting tool call flow: ${messageId}`);\n\n // 2. 发送工具调用事件 - 使用 ACP 协议的 tool_call session/update\n const toolCallId = `tool-${Date.now()}`;\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'write_to_file',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Write File',\n kind: 'edit',\n status: 'pending',\n locations: [{ path: 'example-edit-content.py', line: 0 }]\n }\n } as SessionNotification,\n });\n\n await this.delay(100);\n\n const content = 'def hello_world():\\n print(\"Hello from ACP tool call!\")\\n return \"success\"\\n\\nif __name__ == \"__main__\":\\n hello_world()';\n const chunkSize = 10; // 每次发送 10 个字符\n\n let accumulatedInput = '';\n for (let i = 0; i < content.length; i += chunkSize) {\n // 获取当前块的内容\n const chunk = content.substring(i, i + chunkSize);\n accumulatedInput += chunk;\n\n console.log(`🐸 tool call Chunk: \"${chunk}\" | messageId: ${messageId} | sessionId: ${sessionId}`);\n\n // ✅ 5.1 开始执行 (in_progress)\n\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'write_to_file',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress', // ✅ 最终完成状态\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: accumulatedInput,\n\n }\n }\n ],\n rawInput: {\n 'filePath': 'example-edit-content.py',\n 'content': accumulatedInput,\n },\n locations: [{ path: 'example-edit-content.py', line: 0 }]\n }\n } as SessionNotification,\n });\n // 模拟网络延迟\n await this.delay(50);\n }\n\n // 模拟工具执行时间\n await this.delay(1200);\n\n // ✅ 5.2 执行完成 (completed)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'write_to_file',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed', // ✅ 最终完成状态\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: content\n }\n }\n ],\n rawOutput: {\n 'type': 'write_to_file_result',\n 'path': 'example-edit-content.py',\n 'addLineCount': 10,\n 'removedLines': 20,\n },\n locations: [{ path: 'example-edit-content.py', line: 0 }]\n }\n } as SessionNotification,\n });\n\n // 发送总结消息\n await this.delay(300);\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-summary`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'write_to_file',\n },\n },\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: '✅ 工具调用完成!我成功创建了一个名为 `example.py` 的文件。\\n\\nACP 协议工具调用流程:\\n1. `tool_call` (pending) - 创建工具调用\\n2. `tool_call_update` (in_progress) - 开始执行\\n3. `tool_call_update` (completed) - 执行完成\\n\\n这就是完整的工具调用生命周期!',\n }\n },\n rawInput: {\n 'filePath': 'example-edit-content.py',\n 'content': accumulatedInput,\n }\n } as SessionNotification,\n });\n\n await this.delay(100);\n\n // 5. 发送工具调用完成后的总结消息\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-summary`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: '✅ 工具调用完成!我成功创建了一个名为 `example.py` 的文件。\\n\\n工具调用流程包括:\\n1. 发送工具调用请求 (tool_call)\\n2. 返回执行结果 (tool_call_update)\\n\\n这就是 ACP 协议中完整的工具调用生命周期。',\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(100);\n }\n\n /**\n * 发送完整的工具调用流程 - kind 为 read (符合 ACP 协议)\n *\n * 模拟 read_file 工具调用流程,流式发送 rawInput 和 rawOutput\n */\n private async sendMockReadContentToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`📖 Starting read tool call flow: ${messageId}`);\n\n // 配置:模拟读取文件\n const filePath = '/Users/liumingyuan/Project/caseTest/subagent/test.js';\n const toolCallId = `tool-${Date.now()}`;\n\n // 1. 发送 tool_call (pending) - 不带 rawInput\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'read_file'\n }\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Read File',\n status: 'pending',\n locations: [{ path: filePath, line: 0 }]\n }\n } as SessionNotification\n });\n\n console.log('[MockAgentProvider] tool_call (pending) sent');\n await this.delay(100);\n\n // 2. 流式发送 rawInput (模拟参数逐个到达)\n // 将 filePath 分块流式发送\n const filePathChunks = [\n '/Users/liumingyuan/',\n 'Project/caseTest',\n '/subagent/',\n 'test.js'\n ];\n\n let accumulatedInput = '';\n for (const chunk of filePathChunks) {\n accumulatedInput += chunk;\n\n // 尝试解析累积的 input\n try {\n // const parsed = JSON.parse(accumulatedInput);\n // const rawInput: ReadFileInput = {\n // filePath: parsed.filePath\n // };\n\n // 验证 rawInput\n // const inputValidation = validateToolInput('read_file', rawInput);\n // if (inputValidation.success) {\n // 发送 tool_call_update with rawInput\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'read_file'\n }\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n rawInput: {\n filePath: accumulatedInput\n }\n }\n } as SessionNotification\n });\n await this.delay(100);\n console.log('[MockAgentProvider] tool_call_update with rawInput:', chunk);\n // }\n } catch {\n // JSON 还未完整,继续累积\n }\n\n }\n\n // Mock file content\n const fileContent = `/**\n * Test file for subagent functionality\n */\n\nfunction helloWorld() {\n console.log('Hello from test.js!');\n return 'success';\n}\n\nclass TestClass {\n constructor(name) {\n this.name = name;\n }\n\n greet() {\n return \\`Hello, \\${this.name}!\\`;\n }\n}\n\n// Export functions\nmodule.exports = {\n helloWorld,\n TestClass\n};\n`;\n\n const totalLineCount = 26;\n const chunkSize = 20;\n let accumulatedContent = '';\n\n // 3. 流式返回结果 with in_progress\n for (let i = 0; i < fileContent.length; i += chunkSize) {\n const chunk = fileContent.substring(i, i + chunkSize);\n accumulatedContent += chunk;\n const progress = Math.round(((i + chunkSize) / fileContent.length) * 100);\n const isLastChunk = (i + chunkSize) >= fileContent.length;\n\n const rawOutput: ReadFileResult = {\n type: 'read_file_result',\n path: filePath,\n content: accumulatedContent,\n totalLineCount,\n hasMore: !isLastChunk,\n hint: `Reading file... ${Math.min(progress, 100)}% complete`\n };\n\n // 验证 rawOutput\n const outputValidation = validateToolOutput('read_file', rawOutput);\n\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'read_file'\n }\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: accumulatedContent\n }\n }\n ],\n locations: [{ path: filePath, line: 0 }],\n rawOutput: (outputValidation.success ? outputValidation.data : rawOutput) as unknown as Record<string, unknown>\n }\n } as SessionNotification\n });\n\n console.log(`📖 Read chunk ${i / chunkSize + 1}: progress ${progress}%`);\n await this.delay(30);\n }\n\n // 4. 发送 completed 事件\n const finalRawOutput: ReadFileResult = {\n type: 'read_file_result',\n path: filePath,\n content: fileContent,\n totalLineCount,\n hasMore: false\n };\n\n const finalOutputValidation = validateToolOutput('read_file', finalRawOutput);\n if (!finalOutputValidation.success) {\n throw new Error(`Invalid rawOutput: ${JSON.stringify(finalOutputValidation.error.errors)}`);\n }\n\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: fileContent\n }\n }\n ],\n locations: [{ path: filePath, line: 0 }],\n rawOutput: finalOutputValidation.data as unknown as Record<string, unknown>\n }\n } as SessionNotification\n });\n\n console.log('[MockAgentProvider] tool_call_update (completed) sent');\n }\n\n /**\n * 发送 search_content 工具调用流程 (符合 ACP 协议)\n *\n * 用于模拟在文件内容中搜索特定文本的工具调用\n * 支持显示匹配的文件路径、行号和上下文内容\n */\n private async sendMockSearchContentToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`🔍 Starting search_content tool call flow: ${messageId}`);\n\n const pattern = '一行工具';\n const directory = '/Users/liumingyuan/Project/caseTest/subagent';\n\n // 1. 发送工具调用事件\n const toolCallId = `tool-sc-${Date.now()}`;\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'search_content',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Search Content',\n kind: 'search', // ✅ kind 为 search\n status: 'pending',\n locations: [{ path: directory, line: 0 }],\n rawInput: {\n pattern,\n directory,\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(100);\n\n // 2. 发送 in_progress 状态和搜索结果\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `Searching for \"${pattern}\" in ${directory}...`,\n }\n }\n ],\n locations: [{ path: directory, line: 0 }]\n }\n } as SessionNotification,\n });\n\n await this.delay(500);\n\n // 3. 发送完成状态和匹配结果\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: 'Found 1 matching result\\n\\nconvert_to_oneline.py\\n 56| print(\"文件内容转一行工具\")\\n',\n }\n }\n ],\n locations: [{ path: directory, line: 0 }],\n rawOutput: {\n pattern,\n directory,\n matches: [\n {\n filePath: 'convert_to_oneline.py',\n fileName: 'convert_to_oneline.py',\n content: ' print(\"文件内容转一行工具\")',\n startLine: 56,\n endLine: 56,\n }\n ],\n totalCount: 1,\n hasMore: false,\n }\n }\n } as SessionNotification,\n });\n\n // 4. 发送总结消息\n await this.delay(300);\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-summary`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: `✅ 内容搜索完成!在 \\`${directory}\\` 目录中找到 1 个匹配结果。\\n\\n**搜索结果:**\\n- \\`convert_to_oneline.py\\` 第 56 行: \\`print(\"文件内容转一行工具\")\\`\\n\\n**工具调用流程:**\\n1. \\`tool_call\\` (pending) - 创建搜索工具调用\\n2. \\`tool_call_update\\` (in_progress) - 执行搜索\\n3. \\`tool_call_update\\` (completed) - 返回匹配结果`,\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(100);\n }\n\n /**\n * 发送 search_file 工具调用流程 (符合 ACP 协议)\n *\n * 用于模拟按文件名模式搜索文件的工具调用\n * 支持递归搜索和文件大小显示\n */\n private async sendMockSearchFileToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`🔍 Starting search_file tool call flow: ${messageId}`);\n\n const pattern = '*.py';\n const directory = '/Users/liumingyuan/Project/caseTest/subagent';\n\n // 1. 发送工具调用事件\n const toolCallId = `tool-sf-${Date.now()}`;\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'search_file',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Search File',\n kind: 'search', // ✅ kind 为 search\n status: 'pending',\n locations: [{ path: directory, line: 0 }],\n rawInput: {\n target_directory: directory,\n pattern,\n recursive: true,\n caseSensitive: false,\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(100);\n\n // 2. 发送 in_progress 状态\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `Searching for files matching \"${pattern}\" in ${directory}...`,\n }\n }\n ],\n locations: [{ path: directory, line: 0 }]\n }\n } as SessionNotification,\n });\n\n await this.delay(500);\n\n // 3. 发送完成状态和搜索结果\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: 'Found 5 files\\n\\nbubble_sort.py (10.38 KB)\\nconvert_to_oneline.py (2.2 KB)\\ntls12_examples.py (4.58 KB)\\n.codebuddy/skills/analyzing-financial-statements/calculate_ratios.py (12.22 KB)\\n.codebuddy/skills/analyzing-financial-statements/interpret_ratios.py (15.83 KB)\\n',\n }\n }\n ],\n locations: [{ path: directory, line: 0 }],\n rawOutput: {\n path: directory,\n pattern,\n recursive: true,\n caseSensitive: false,\n results: [\n { filePath: 'bubble_sort.py', size: '10.38 KB' },\n { filePath: 'convert_to_oneline.py', size: '2.2 KB' },\n { filePath: 'tls12_examples.py', size: '4.58 KB' },\n { filePath: '.codebuddy/skills/analyzing-financial-statements/calculate_ratios.py', size: '12.22 KB' },\n { filePath: '.codebuddy/skills/analyzing-financial-statements/interpret_ratios.py', size: '15.83 KB' },\n ],\n }\n }\n } as SessionNotification,\n });\n\n // 4. 发送总结消息\n await this.delay(300);\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-summary`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: `✅ 文件搜索完成!在 \\`${directory}\\` 目录中找到 5 个匹配的 Python 文件。\\n\\n**搜索结果:**\\n- \\`bubble_sort.py\\` (10.38 KB)\\n- \\`convert_to_oneline.py\\` (2.2 KB)\\n- \\`tls12_examples.py\\` (4.58 KB)\\n- \\`.codebuddy/skills/analyzing-financial-statements/calculate_ratios.py\\` (12.22 KB)\\n- \\`.codebuddy/skills/analyzing-financial-statements/interpret_ratios.py\\` (15.83 KB)\\n\\n**工具调用流程:**\\n1. \\`tool_call\\` (pending) - 创建搜索工具调用\\n2. \\`tool_call_update\\` (in_progress) - 执行搜索\\n3. \\`tool_call_update\\` (completed) - 返回搜索结果`,\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(100);\n }\n\n /**\n * 发送删除文件工具调用流程 (符合 ACP 协议)\n *\n * 用于模拟删除文件的工具调用\n * 支持显示删除操作的进度和结果\n */\n private async sendMockDeleteFileToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`🗑️ Starting delete_file tool call flow: ${messageId}`);\n\n // 配置:模拟删除文件\n const targetFile = '/Users/liumingyuan/Project/caseTest/subagent/temp_file.txt';\n const toolCallId = `tool-df-${Date.now()}`;\n\n // 1. 发送 tool_call (pending) - 不带 rawInput\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'delete_file'\n }\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Delete File',\n status: 'pending',\n locations: [{ path: targetFile, line: 0 }]\n }\n } as SessionNotification\n });\n\n console.log('[MockAgentProvider] delete_file tool_call (pending) sent');\n await this.delay(200);\n\n // 2. 流式发送 rawInput (模拟参数逐个到达)\n // 将 target_file 分块流式发送\n const filePathChunks = [\n '/Users/liumingyuan/',\n 'Project/caseTest',\n '/subagent/',\n 'temp_file.txt'\n ];\n\n let accumulatedInput = '';\n for (const chunk of filePathChunks) {\n accumulatedInput += chunk;\n\n // 发送 tool_call_update with rawInput\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'delete_file'\n }\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n rawInput: {\n target_file: accumulatedInput,\n explanation: 'Deleting temporary test file'\n }\n }\n } as SessionNotification\n });\n await this.delay(100);\n console.log('[MockAgentProvider] delete_file tool_call_update with rawInput:', chunk);\n }\n\n await this.delay(300);\n\n // 3. 模拟删除操作进度\n const progressSteps = [\n { progress: 25, message: 'Checking file permissions...' },\n { progress: 50, message: 'Validating file path...' },\n { progress: 75, message: 'Removing file...' },\n { progress: 100, message: 'File deleted successfully' }\n ];\n\n for (const step of progressSteps) {\n const rawOutput: DeleteFilesResult = {\n type: 'delete_file_result',\n path: targetFile,\n recursive: false,\n hint: step.message\n };\n\n // 验证 rawOutput\n const outputValidation = validateToolOutput('delete_file', rawOutput);\n\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'delete_file'\n }\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: step.progress === 100 ? 'deleted' : 'in_progress',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `🗑️ ${step.message} (${step.progress}%)`\n }\n }\n ],\n locations: [{ path: targetFile, line: 0 }],\n rawOutput: (outputValidation.success ? outputValidation.data : rawOutput) as unknown as Record<string, unknown>\n }\n } as SessionNotification\n });\n\n console.log(`🗑️ Delete progress: ${step.progress}% - ${step.message}`);\n await this.delay(200);\n }\n\n // 4. 发送最终完成事件\n const finalRawOutput: DeleteFilesResult = {\n type: 'delete_file_result',\n path: targetFile,\n recursive: false,\n hint: 'File successfully deleted from filesystem'\n };\n\n const finalOutputValidation = validateToolOutput('delete_file', finalRawOutput);\n if (!finalOutputValidation.success) {\n throw new Error(`Invalid delete_file rawOutput: ${JSON.stringify(finalOutputValidation.error.errors)}`);\n }\n\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `✅ 文件删除完成!\\n\\n**删除的文件:** \\`${targetFile}\\`\\n\\n**操作详情:**\\n- 文件路径验证 ✓\\n- 权限检查 ✓\\n- 文件删除 ✓\\n\\n文件已从文件系统中成功移除。`\n }\n }\n ],\n locations: [{ path: targetFile, line: 0 }],\n rawOutput: finalOutputValidation.data as unknown as Record<string, unknown>\n }\n } as SessionNotification\n });\n\n console.log('[MockAgentProvider] delete_file tool_call_update (completed) sent');\n\n // 5. 发送总结消息\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-summary`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: '🗑️ **删除文件工具调用演示完成**\\n\\n**工具调用流程:**\\n1. `tool_call` (pending) - 创建删除文件工具调用\\n2. `tool_call_update` (in_progress) - 流式发送参数和执行删除操作\\n3. `tool_call_update` (completed) - 返回删除结果\\n\\n**rawInput 包含:**\\n- `target_file`: 要删除的文件路径\\n- `explanation`: 删除原因说明\\n\\n**rawOutput 包含:**\\n- `type`: \\'delete_file_result\\'\\n- `path`: 被删除的文件路径\\n- `recursive`: 是否递归删除\\n- `hint`: 操作提示信息',\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(100);\n }\n\n /**\n * 发送 ask_followup_question 工具调用流程 (符合 ACP 协议)\n *\n * 用于模拟向用户提问并获取回答的工具调用\n * 数据结构与 genie-ide 保持一致:question (单个问题字符串) + options (选项数组)\n */\n private async sendMockAskFollowupQuestionToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`❓ Starting ask_followup_question tool call flow: ${messageId}`);\n\n const toolCallId = `tool-afq-${Date.now()}`;\n\n // 问题数据 - 与 genie-ide 保持一致的数据结构\n const question = '你希望创建什么类型的项目?';\n const options = [\n 'Web 应用(前端 + 后端)',\n '命令行工具(CLI)',\n '移动应用(React Native / Flutter)',\n '桌面应用(Electron)',\n '后端服务(API / 微服务)',\n ];\n\n // 1. 发送 tool_call (pending)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'ask_followup_question',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Ask Followup Question',\n kind: 'other',\n status: 'pending',\n rawInput: {\n question,\n options: JSON.stringify(options),\n },\n },\n } as SessionNotification,\n });\n\n await this.delay(100);\n\n // 2. 发送 tool_call_update (in_progress) - 显示问题\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'ask_followup_question',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n rawInput: {\n question,\n options: JSON.stringify(options),\n },\n },\n } as SessionNotification,\n });\n\n console.log('[MockAgentProvider] ask_followup_question tool displayed, waiting for user interaction...');\n // 工具显示完成,等待用户交互(暂未实现交互逻辑)\n }\n\n /**\n * 发送 list_dir 工具调用流程 (符合 ACP 协议)\n *\n * 用于模拟列出目录中的文件\n * 数据结构符合 tool-schemas.ts 中的 ListDirInput 和 ListFilesResult\n */\n private async sendMockListFilesToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`📁 Starting list_dir tool call flow: ${messageId}`);\n\n const target_directory = '/Users/test/project/src';\n const toolCallId = `tool-ld-${Date.now()}`;\n\n // Mock 文件列表数据 - 符合 ListFilesResult.files 格式\n const mockFiles = [\n { filePath: `${target_directory}/index.ts`, size: '2.5 KB', modifyTime: '2024-01-15 10:30:00' },\n { filePath: `${target_directory}/App.tsx`, size: '8.2 KB', modifyTime: '2024-01-15 09:45:00' },\n { filePath: `${target_directory}/utils.ts`, size: '1.8 KB', modifyTime: '2024-01-14 16:20:00' },\n { filePath: `${target_directory}/components/`, size: '-', modifyTime: '2024-01-15 11:00:00' },\n { filePath: `${target_directory}/hooks/`, size: '-', modifyTime: '2024-01-14 14:30:00' },\n { filePath: `${target_directory}/styles/`, size: '-', modifyTime: '2024-01-13 09:00:00' },\n { filePath: `${target_directory}/types.d.ts`, size: '3.1 KB', modifyTime: '2024-01-12 18:45:00' },\n { filePath: `${target_directory}/constants.ts`, size: '0.9 KB', modifyTime: '2024-01-10 12:00:00' },\n ];\n\n // 生成 listing 格式的文本输出\n const listing = mockFiles\n .map(f => {\n const name = f.filePath.split('/').pop() || '';\n const isDir = f.filePath.endsWith('/');\n return isDir ? `📁 ${name}` : `📄 ${name} (${f.size})`;\n })\n .join('\\n');\n\n // 1. 发送 tool_call (pending)\n // rawInput 符合 ListDirInput: { target_directory, ignore_globs? }\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'list_dir',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'List Directory',\n kind: 'search',\n status: 'pending',\n locations: [{ path: target_directory, line: 0 }],\n rawInput: {\n target_directory,\n // ignore_globs 是可选的,这里不传\n },\n },\n } as SessionNotification,\n });\n\n await this.delay(100);\n\n // 2. 发送 tool_call_update (in_progress)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'list_dir',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `Listing directory ${target_directory}...`,\n },\n },\n ],\n locations: [{ path: target_directory, line: 0 }],\n },\n } as SessionNotification,\n });\n\n await this.delay(300);\n\n // 3. 发送 tool_call_update (completed) 包含文件列表\n // rawOutput 符合 ListFilesResult: { type, files, root, listing? }\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'list_dir',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `Listed ${mockFiles.length} items in ${target_directory}`,\n },\n },\n ],\n locations: [{ path: target_directory, line: 0 }],\n rawOutput: {\n type: 'list_files_result',\n files: mockFiles,\n root: target_directory,\n listing: listing,\n },\n },\n } as SessionNotification,\n });\n\n console.log('[MockAgentProvider] list_dir tool call completed');\n }\n\n /**\n * 发送 preview_url 工具调用流程 (符合 ACP 协议)\n *\n * 用于测试 unknown-tool-renderer 的显示效果\n * preview_url 是一个未知工具,会使用 UnknownToolRenderer 渲染\n */\n private async sendMockPreviewUrlToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`🌐 Starting preview_url tool call flow: ${messageId}`);\n\n const toolCallId = `tool-pu-${Date.now()}`;\n const url = 'https://example.com/preview';\n\n // 1. 发送 tool_call (pending)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'preview_url',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Preview URL',\n kind: 'other',\n status: 'pending',\n rawInput: {\n url,\n },\n },\n } as SessionNotification,\n });\n\n await this.delay(200);\n\n // 2. 发送 tool_call_update (in_progress)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'preview_url',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `Loading preview for ${url}...`,\n },\n },\n ],\n },\n } as SessionNotification,\n });\n\n await this.delay(800);\n\n // 3. 发送 tool_call_update (completed)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'preview_url',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `Preview loaded successfully for ${url}`,\n },\n },\n ],\n rawOutput: {\n url,\n title: 'Example Domain',\n screenshot: 'base64-encoded-screenshot-data...',\n status: 200,\n },\n },\n } as SessionNotification,\n });\n\n console.log('[MockAgentProvider] preview_url tool call completed');\n }\n\n /**\n * 发送 list_code_definition_names 工具调用流程 (符合 ACP 协议)\n *\n * 用于测试 unknown-tool-renderer 的显示效果\n * list_code_definition_names 是一个未知工具,会使用 UnknownToolRenderer 渲染\n */\n private async sendMockListCodeDefinitionNamesToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`📝 Starting list_code_definition_names tool call flow: ${messageId}`);\n\n const toolCallId = `tool-lcd-${Date.now()}`;\n const filePath = '/Users/test/project/src/utils.ts';\n\n // 1. 发送 tool_call (pending)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'list_code_definition_names',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'List Code Definitions',\n kind: 'other',\n status: 'pending',\n rawInput: {\n filePath,\n },\n },\n } as SessionNotification,\n });\n\n await this.delay(200);\n\n // 2. 发送 tool_call_update (in_progress)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'list_code_definition_names',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `Analyzing code definitions in ${filePath}...`,\n },\n },\n ],\n },\n } as SessionNotification,\n });\n\n await this.delay(600);\n\n // 3. 发送 tool_call_update (completed)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'list_code_definition_names',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `Found 5 code definitions in ${filePath}`,\n },\n },\n ],\n rawOutput: {\n filePath,\n definitions: [\n { name: 'formatDate', type: 'function', line: 5 },\n { name: 'parseJSON', type: 'function', line: 15 },\n { name: 'debounce', type: 'function', line: 28 },\n { name: 'Config', type: 'interface', line: 42 },\n { name: 'DEFAULT_OPTIONS', type: 'const', line: 55 },\n ],\n },\n },\n } as SessionNotification,\n });\n\n console.log('[MockAgentProvider] list_code_definition_names tool call completed');\n }\n\n /**\n * 发送 web_fetch 工具调用流程 (符合 ACP 协议)\n *\n * 用于模拟从 URL 获取网页内容的工具调用\n */\n private async sendMockWebFetchToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`🌐 Starting web_fetch tool call flow: ${messageId}`);\n\n const toolCallId = `tool-wf-${Date.now()}`;\n const targetUrl = 'https://github.com/anthropics/claude-code';\n\n // 1. 发送 tool_call (pending)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'web_fetch',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Web Fetch',\n kind: 'other',\n status: 'pending',\n rawInput: {\n url: targetUrl,\n fetchInfo: 'Fetching page content...',\n },\n },\n } as SessionNotification,\n });\n\n await this.delay(200);\n\n // 2. 流式发送 rawInput(模拟 URL 逐步到达)\n const urlChunks = [\n 'https://github.com/',\n 'anthropics/',\n 'claude-code',\n ];\n\n let accumulatedUrl = '';\n for (const chunk of urlChunks) {\n accumulatedUrl += chunk;\n\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'web_fetch',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n rawInput: {\n url: accumulatedUrl,\n fetchInfo: 'Fetching page content...',\n },\n },\n } as SessionNotification,\n });\n await this.delay(100);\n console.log('[MockAgentProvider] web_fetch tool_call_update with rawInput:', chunk);\n }\n\n await this.delay(500);\n\n // 3. 模拟获取进度\n const progressSteps = [\n { loading: 'Connecting to server...' },\n { loading: 'Downloading content...' },\n { loading: 'Parsing HTML...' },\n ];\n\n for (const step of progressSteps) {\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'web_fetch',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n rawInput: {\n url: targetUrl,\n fetchInfo: step.loading,\n },\n rawOutput: {\n loading: step.loading,\n },\n },\n } as SessionNotification,\n });\n console.log(`🌐 Fetch progress: ${step.loading}`);\n await this.delay(300);\n }\n\n // 4. 发送完成事件,包含模拟的网页内容\n const mockWebContent = `# Claude Code\n\nClaude Code is an agentic coding tool that lives in your terminal, understands your codebase, and helps you code faster through natural conversation.\n\n## Features\n\n- **Agentic coding**: Claude Code can read, write, and edit files, run commands, and more.\n- **Context-aware**: Claude Code understands your entire codebase and can answer questions about it.\n- **Natural conversation**: Just describe what you want to do in plain English.\n\n## Installation\n\n\\`\\`\\`bash\nnpm install -g @anthropic-ai/claude-code\n\\`\\`\\`\n\n## Usage\n\n\\`\\`\\`bash\nclaude-code\n\\`\\`\\`\n\nStart a conversation with Claude Code and let it help you with your coding tasks!`;\n\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'web_fetch',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed',\n rawInput: {\n url: targetUrl,\n fetchInfo: 'Fetch web page content',\n },\n rawOutput: {\n title: 'Claude Code - GitHub',\n favicon: 'https://github.githubassets.com/favicons/favicon.svg',\n data: mockWebContent,\n },\n },\n } as SessionNotification,\n });\n\n console.log('[MockAgentProvider] web_fetch tool_call_update (completed) sent');\n\n // 5. 发送总结消息\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-summary`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: '🌐 **Web Fetch 工具调用演示完成**\\n\\n**工具调用流程:**\\n1. `tool_call` (pending) - 创建 web_fetch 工具调用\\n2. `tool_call_update` (in_progress) - 流式发送 URL 和获取进度\\n3. `tool_call_update` (completed) - 返回网页内容\\n\\n**rawInput 包含:**\\n- `url`: 目标 URL\\n- `fetchInfo`: 获取说明\\n\\n**rawOutput 包含:**\\n- `title`: 网页标题\\n- `favicon`: 网站图标\\n- `data`: 网页内容(Markdown 格式)',\n },\n },\n } as SessionNotification,\n });\n\n await this.delay(100);\n }\n\n /**\n * 发送 web_search 工具调用流程 (符合 ACP 协议)\n *\n * 用于模拟网络搜索的工具调用\n * 数据结构符合 tool-schemas.ts 中的 web_search 定义\n */\n private async sendMockWebSearchToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`🔍 Starting web_search tool call flow: ${messageId}`);\n\n const toolCallId = `tool-ws-${Date.now()}`;\n const searchTerm = 'TypeScript 5.0 new features';\n const explanation = 'Searching for the latest TypeScript 5.0 features to provide accurate and up-to-date information.';\n\n // 1. 发送 tool_call (pending)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'web_search',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Web Search',\n kind: 'search',\n status: 'pending',\n rawInput: {\n searchTerm,\n explanation,\n },\n },\n } as SessionNotification,\n });\n\n await this.delay(200);\n\n // 2. 发送 tool_call_update (in_progress) - 流式显示搜索中\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'web_search',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `🔍 Searching the web for \"${searchTerm}\"...`,\n },\n },\n ],\n rawInput: {\n searchTerm,\n explanation,\n },\n },\n } as SessionNotification,\n });\n\n await this.delay(800);\n\n // 3. 发送 tool_call_update (completed) - 包含搜索结果\n const searchResults = [\n {\n title: 'Announcing TypeScript 5.0 - TypeScript Blog',\n url: 'https://devblogs.microsoft.com/typescript/announcing-typescript-5-0/',\n snippet: 'TypeScript 5.0 brings many new features, including decorators, const type parameters, and improvements to enums. This release also includes significant performance improvements...',\n publishedDate: '2023-03-16',\n },\n {\n title: 'TypeScript 5.0: What\\'s New - LogRocket Blog',\n url: 'https://blog.logrocket.com/whats-new-typescript-5-0/',\n snippet: 'TypeScript 5.0 introduces several exciting features: ECMAScript decorators, const type parameters, multiple config extends, all enums are union enums, and more...',\n publishedDate: '2023-04-10',\n },\n {\n title: 'TypeScript 5.0 Release Notes - GitHub',\n url: 'https://github.com/microsoft/TypeScript/releases/tag/v5.0.0',\n snippet: 'Official release notes for TypeScript 5.0. Key highlights include: Decorators, const Type Parameters, Supporting Multiple Configuration Files in extends...',\n publishedDate: '2023-03-16',\n },\n {\n title: 'A Complete Guide to TypeScript 5.0 Features - Medium',\n url: 'https://medium.com/typescript-5-features-guide',\n snippet: 'Explore all the new features in TypeScript 5.0: decorators with metadata, const type parameters for better type inference, improved module resolution...',\n publishedDate: '2023-05-22',\n },\n {\n title: 'TypeScript 5.0 Performance Improvements',\n url: 'https://www.typescriptlang.org/docs/handbook/release-notes/typescript-5-0.html',\n snippet: 'TypeScript 5.0 includes significant performance improvements: faster type checking, reduced memory usage, and improved build times. Package size reduced by 50%...',\n publishedDate: '2023-03-16',\n },\n ];\n\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'web_search',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `Found ${searchResults.length} results for \"${searchTerm}\"`,\n },\n },\n ],\n rawInput: {\n searchTerm,\n explanation,\n },\n rawOutput: {\n query: searchTerm,\n totalResults: searchResults.length,\n results: searchResults,\n searchEngine: 'web',\n timestamp: new Date().toISOString(),\n },\n },\n } as SessionNotification,\n });\n\n // 4. 发送总结消息\n await this.delay(300);\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-summary`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: `🔍 **Web 搜索完成**\\n\\n**搜索词:** \\`${searchTerm}\\`\\n\\n**搜索结果:** 找到 ${searchResults.length} 个相关结果\\n\\n**主要发现:**\\n1. **TypeScript 5.0 新特性** - 包括装饰器、const 类型参数、枚举改进等\\n2. **性能提升** - 类型检查更快,内存使用减少,包大小减少 50%\\n3. **配置增强** - 支持多配置文件继承\\n\\n**工具调用流程:**\\n1. \\`tool_call\\` (pending) - 创建搜索工具调用\\n2. \\`tool_call_update\\` (in_progress) - 执行网络搜索\\n3. \\`tool_call_update\\` (completed) - 返回搜索结果\\n\\n**rawInput 包含:**\\n- \\`searchTerm\\`: 搜索关键词\\n- \\`explanation\\`: 搜索原因说明\\n\\n**rawOutput 包含:**\\n- \\`query\\`: 搜索查询\\n- \\`totalResults\\`: 结果总数\\n- \\`results\\`: 搜索结果数组(包含 title、url、snippet、publishedDate)`,\n },\n },\n } as SessionNotification,\n });\n\n console.log('[MockAgentProvider] web_search tool call completed');\n }\n\n /**\n * 发送 read_lints 工具调用流程 (符合 ACP 协议)\n *\n * 数据结构 (来自 tool-schemas.ts):\n * - Input: ReadLintsInput = { paths?: string }\n * - Output: ReadLintsResult = {\n * type: 'read_lints_result';\n * diagnostics: string[];\n * totalCount?: number;\n * hint?: string;\n * isTruncated?: boolean;\n * }\n */\n private async sendMockReadLintsToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`🔍 Starting read_lints tool call flow: ${messageId}`);\n\n const toolCallId = `tool-rl-${Date.now()}`;\n const targetPath = '/Users/test/project/src/components/Button.tsx';\n\n // 1. 发送 tool_call (pending)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'read_lints',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Read Lints',\n kind: 'read',\n status: 'pending',\n rawInput: {\n paths: targetPath,\n },\n },\n } as SessionNotification,\n });\n\n await this.delay(200);\n\n // 2. 发送 tool_call_update (in_progress)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'read_lints',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n rawInput: {\n paths: targetPath,\n },\n },\n } as SessionNotification,\n });\n\n await this.delay(300);\n\n // 3. 发送 tool_call_update (completed) - 包含 lint 错误\n const diagnosticsText = `Button.tsx - ERROR (3 issues)\n[ERROR] Line 15, Column 8: 'useState' is defined but never used. Consider removing this import.\n[ERROR] Line 23, Column 12: Type 'string' is not assignable to type 'number'. Expected a numeric value.\n[ERROR] Line 45, Column 4: Missing return statement in function that is expected to return 'ReactNode'.`;\n\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'read_lints',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed',\n rawInput: {\n paths: targetPath,\n },\n rawOutput: {\n type: 'read_lints_result',\n diagnostics: [diagnosticsText],\n totalCount: 3,\n hint: 'Found 3 lint errors in Button.tsx',\n isTruncated: false,\n },\n },\n } as SessionNotification,\n });\n\n console.log('[MockAgentProvider] read_lints tool call completed');\n }\n\n /**\n * 发送 execute_command 工具调用流程 (符合 ACP 协议)\n *\n * 数据结构 (来自 tool-schemas.ts):\n * - Input: ExecuteCommandInput = { command: string; requires_approval: boolean; }\n * - Output: ExecuteCommandResult = {\n * type: 'execute_command_result';\n * stdout: string;\n * stderr: string;\n * exitCode: number;\n * hint?: string;\n * serviceInfo?: { isWatchCommand, isServiceOutput, serviceReady, message };\n * use_standalone_terminal?: boolean;\n * }\n */\n private async sendMockExecuteCommandToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`💻 Starting execute_command tool call flow: ${messageId}`);\n const toolCallId = `tool-ec-${Date.now()}`;\n const command = 'npm run build';\n // 1. 发送 tool_call (pending)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'execute_command',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Execute Command',\n kind: 'execute',\n status: 'pending',\n rawInput: {\n command,\n requires_approval: false,\n },\n },\n } as SessionNotification,\n });\n await this.delay(200);\n // 2. 发送 tool_call_update (in_progress) - 模拟命令执行中\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'execute_command',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: '> npm run build\\n\\nRunning build script...',\n },\n },\n ],\n },\n } as SessionNotification,\n });\n await this.delay(1000);\n // 3. 发送 tool_call_update (completed) - 命令执行完成\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'execute_command',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: '> npm run build\\n\\n> genie@1.0.0 build\\n> tsc -b\\n\\nBuild completed successfully.',\n },\n },\n ],\n rawOutput: {\n type: 'execute_command_result',\n stdout: '> genie@1.0.0 build\\n> tsc -b\\n\\nBuild completed successfully.',\n stderr: '',\n exitCode: 0,\n hint: 'Build completed in 2.5s',\n },\n },\n } as SessionNotification,\n });\n console.log('[MockAgentProvider] execute_command tool call completed');\n }\n /**\n * 发送 execute_command 工具调用流程(需要权限确认)(符合 ACP 协议)\n *\n * 这个方法模拟需要用户确认才能执行的命令\n * 用于测试权限确认菜单 (Run/Skip/Reject)\n *\n * 数据结构 (来自 tool-schemas.ts):\n * - Input: ExecuteCommandInput = { command: string; requires_approval: boolean; }\n *\n * 权限请求通过 rawInput.permissionRequest 字段传递:\n * - permissionRequest.options: 可选的权限选项数组\n */\n private async sendMockExecuteCommandWithApprovalToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`💻 Starting execute_command (with approval) tool call flow: ${messageId}`);\n const toolCallId = `tool-ec-approval-${Date.now()}`;\n const command = 'rm -rf node_modules && npm install';\n // 权限请求选项 - 与 ACP 协议的 PermissionOption 结构一致\n const permissionRequest = {\n options: [\n { optionId: 'allow-once', name: '运行', kind: 'allow_once' },\n { optionId: 'skip-once', name: '跳过', kind: 'skip_once' },\n { optionId: 'reject-once', name: '拒绝', kind: 'reject_once' },\n ],\n };\n // 1. 发送 tool_call (pending) - 需要权限确认\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'execute_command',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Execute Command',\n kind: 'execute',\n status: 'pending',\n rawInput: {\n command,\n requires_approval: true, // 保留向后兼容\n permissionRequest, // 新增:权限请求信息\n },\n },\n } as SessionNotification,\n });\n await this.delay(200);\n // 2. 发送 tool_call_update (in_progress) - 等待用户确认\n // 这时 UI 应该显示权限确认菜单 (Run/Skip/Reject)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'execute_command',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'pending', // 保持 pending 状态等待用户审批\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `$ ${command}\\n\\nWaiting for approval...`,\n },\n },\n ],\n rawInput: {\n command,\n requires_approval: true,\n permissionRequest, // 确保 rawInput 中也包含权限请求\n },\n },\n } as SessionNotification,\n });\n console.log('[MockAgentProvider] execute_command (with approval) tool call waiting for user approval');\n // 注意:这里不发送 completed,等待用户在 UI 上点击 Run/Skip/Reject\n }\n /**\n * 发送 write_to_file 工具调用流程(需要权限确认)(符合 ACP 协议)\n *\n * 这个方法模拟需要用户确认才能执行的文件写入\n * 用于测试权限确认菜单 (允许/跳过/拒绝)\n *\n * 权限请求通过 rawInput.permissionRequest 字段传递:\n * - permissionRequest.options: 可选的权限选项数组\n */\n private async sendMockWriteFileWithApprovalToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`📝 Starting write_to_file (with approval) tool call flow: ${messageId}`);\n const toolCallId = `tool-wtf-approval-${Date.now()}`;\n const filePath = '/Users/test/project/src/config.ts';\n const content = `export const config = {\n apiUrl: 'https://api.example.com',\n debug: true,\n timeout: 5000,\n};`;\n // 权限请求选项 - 与 ACP 协议的 PermissionOption 结构一致\n const permissionRequest = {\n options: [\n { optionId: 'allow-once', name: '允许', kind: 'allow_once' },\n { optionId: 'skip-once', name: '跳过', kind: 'skip_once' },\n { optionId: 'reject-once', name: '拒绝', kind: 'reject_once' },\n ],\n };\n // 1. 发送 tool_call (pending) - 需要权限确认\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'write_to_file',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Write File',\n kind: 'edit',\n status: 'pending',\n locations: [{ path: filePath, line: 0 }],\n rawInput: {\n filePath,\n content,\n permissionRequest, // 权限请求信息\n },\n },\n } as SessionNotification,\n });\n await this.delay(200);\n // 2. 发送 tool_call_update (pending) - 等待用户确认\n // 这时 UI 应该显示权限确认菜单\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'write_to_file',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'pending', // 保持 pending 状态等待用户审批\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: content,\n },\n },\n ],\n locations: [{ path: filePath, line: 0 }],\n rawInput: {\n filePath,\n content,\n permissionRequest, // 确保 rawInput 中也包含权限请求\n },\n },\n } as SessionNotification,\n });\n console.log('[MockAgentProvider] write_to_file (with approval) tool call waiting for user approval');\n // 注意:这里不发送 completed,等待用户在 UI 上点击 允许/跳过/拒绝\n }\n}\n","/**\n * Protocol Extension Types for Agent Client Protocol\n *\n * This file contains codebuddy.ai extension types that extend the base ACP protocol.\n * For SDK type re-exports and capability extensions, see ./sdk.ts\n */\n\n// ============================================\n// Extension Method Constants\n// ============================================\n\n/**\n * Extension method names as const for type safety\n */\nexport const ExtensionMethod = {\n ARTIFACT: '_codebuddy.ai/artifact',\n /**\n * Question/ToolInput extension method.\n * Used for tools that require user input (e.g., AskUserQuestion).\n * Request: ToolInputRequest, Response: ToolInputResponse\n */\n QUESTION: '_codebuddy.ai/question',\n CHECKPOINT: '_codebuddy.ai/checkpoint',\n /** Usage notification for token/cost tracking */\n USAGE: '_codebuddy.ai/usage',\n /** Command extension method for executing custom commands */\n COMMAND: '_codebuddy.ai/command',\n /** Auth URL notification for external authentication */\n AUTH_URL: '_codebuddy.ai/authUrl',\n} as const;\n\nexport type ExtensionMethodName = (typeof ExtensionMethod)[keyof typeof ExtensionMethod];\n\n/**\n * All known extension methods\n */\nexport const KNOWN_EXTENSIONS = [\n ExtensionMethod.ARTIFACT,\n ExtensionMethod.QUESTION,\n ExtensionMethod.CHECKPOINT,\n ExtensionMethod.USAGE,\n ExtensionMethod.COMMAND,\n ExtensionMethod.AUTH_URL,\n] as const;\n\n// ============================================\n// Extension Notification Types (Discriminated Union)\n// ============================================\n\n/**\n * Discriminated union for extension notifications\n * Use `method` field as discriminant for type narrowing\n */\nexport type ExtensionNotification = ArtifactExtNotification;\n\nexport interface ArtifactExtNotification {\n method: typeof ExtensionMethod.ARTIFACT;\n params: ArtifactNotificationParams;\n}\n\n// ============================================\n// Artifact Types\n// ============================================\n\n/**\n * Types of artifacts that can be created by the agent\n * Uses discriminant for TypeScript type narrowing\n */\nexport type ArtifactType = 'plan' | 'tasks' | 'media' | 'overview';\n\n/**\n * Artifact lifecycle events\n */\nexport type ArtifactEvent = 'created' | 'updated' | 'deleted';\n\n/**\n * Base Artifact interface\n * Inspired by MCP Resource: https://modelcontextprotocol.io/specification/2025-11-25/server/resources\n */\nexport interface BaseArtifact<T extends ArtifactType = ArtifactType> {\n /** Artifact type (discriminant for type narrowing) */\n type: T;\n /**\n * Unique identifier URI\n * - Cloud Agent: agent:///{path} e.g. agent:///artifacts/plan.md\n * - Local Agent: file:///{path} e.g. file:///Users/xxx/project/plan.md\n */\n uri: string;\n /** Resource name */\n name: string;\n /** Display title */\n title?: string;\n /** Description */\n description?: string;\n}\n\n// ============================================\n// Plan Artifact\n// ============================================\n\n/**\n * Plan Artifact - Markdown document\n * mimeType: text/markdown\n */\nexport interface PlanArtifact extends BaseArtifact<'plan'> {\n /** MIME type fixed as text/markdown */\n mimeType: 'text/markdown';\n /** Markdown text content */\n text: string;\n /** Version number (for diff) */\n version?: number;\n /** Previous version content (for diff display) */\n previousText?: string;\n /** Whether editing is enabled */\n enableEdit?: boolean;\n}\n\n// ============================================\n// Tasks Artifact\n// ============================================\n\n/**\n * Task item status\n */\nexport type TaskItemStatus = 'pending' | 'in_progress' | 'completed' | 'cancelled';\n\n/**\n * Single task item\n */\nexport interface TaskItem {\n id: string;\n content: string;\n status: TaskItemStatus;\n order?: number;\n}\n\n/**\n * Tasks Artifact - Task list\n * mimeType: application/json\n */\nexport interface TasksArtifact extends BaseArtifact<'tasks'> {\n /** MIME type fixed as application/json */\n mimeType: 'application/json';\n /** Task list */\n tasks: TaskItem[];\n /** Whether editing is enabled */\n enableEdit?: boolean;\n}\n\n// ============================================\n// Media Artifact\n// ============================================\n\n/**\n * Media content type (auxiliary classification)\n */\nexport type MediaContentType = 'image' | 'video' | 'audio' | 'document' | 'spreadsheet' | 'presentation' | 'diagram' | 'code';\n\n/**\n * Media Artifact - Media files\n * Supports images, videos, documents and other browser-renderable files\n *\n * URI schemes:\n * - data:image/png;base64,... (inline small files)\n * - agent:///artifacts/... (cloud agent resources)\n * - file:///path/to/file (local agent files)\n */\nexport interface MediaArtifact extends BaseArtifact<'media'> {\n /** MIME type, e.g. image/png, video/mp4, application/pdf */\n mimeType: string;\n /** File size in bytes */\n size?: number;\n /** Media content classification (auxiliary for frontend rendering) */\n contentType?: MediaContentType;\n /** Width (for images/videos) */\n width?: number;\n /** Height (for images/videos) */\n height?: number;\n}\n\n// ============================================\n// Overview Artifact\n// ============================================\n\n/**\n * Overview Artifact - 任务完成后的总结文档\n * mimeType: text/markdown\n */\nexport interface OverviewArtifact extends BaseArtifact<'overview'> {\n /** MIME type fixed as text/markdown */\n mimeType: 'text/markdown';\n /** Markdown text content */\n text?: string;\n}\n\n// ============================================\n// Artifact Union Type\n// ============================================\n\n/**\n * Artifact union type\n * Uses type field for discriminated union\n */\nexport type Artifact = PlanArtifact | TasksArtifact | MediaArtifact | OverviewArtifact;\n\n// ============================================\n// Artifact Notification Types (Discriminated Union by event)\n// ============================================\n\nexport interface ArtifactCreatedParams {\n sessionId: string;\n event: 'created';\n artifact: Artifact;\n}\n\nexport interface ArtifactUpdatedParams {\n sessionId: string;\n event: 'updated';\n artifact: Artifact;\n}\n\nexport interface ArtifactDeletedParams {\n sessionId: string;\n event: 'deleted';\n artifact: Pick<Artifact, 'type' | 'uri'>; // Only type and uri needed for deletion\n}\n\n/**\n * Artifact notification discriminated by event\n */\nexport type ArtifactNotificationParams =\n | ArtifactCreatedParams\n | ArtifactUpdatedParams\n | ArtifactDeletedParams;\n\n// ============================================\n// Question Types\n// ============================================\n\n/**\n * Question option structure\n */\nexport interface QuestionOption {\n /** Display text (1-5 words) */\n label: string;\n /** Option description */\n description: string;\n}\n\n/**\n * Single question structure\n */\nexport interface UserQuestion {\n /** Question ID */\n id: string;\n /** Question text */\n question: string;\n /** Short label (max 12 chars) */\n header?: string;\n /** Available options (2-4) */\n options: QuestionOption[];\n /** Allow multiple selections */\n multiSelect?: boolean;\n}\n\n/**\n * Question request (Server -> Client)\n * Sent via extMethod: _codebuddy.ai/question\n */\nexport interface QuestionRequest {\n /** Session ID */\n sessionId: string;\n /** Associated tool call ID (links extMethod to tool_call for UI) */\n toolCallId: string;\n /** Input type */\n inputType?: string;\n /** Schema containing questions */\n schema?: {\n /** Questions to ask (1-4) */\n questions: UserQuestion[];\n };\n /** Questions to ask (1-4) - legacy format */\n questions?: UserQuestion[];\n /** Request timeout in ms */\n timeout?: number;\n}\n\n/**\n * Question response (Client -> Server)\n */\nexport interface QuestionResponse {\n /** Response outcome */\n outcome: 'submitted' | 'cancelled';\n /** User's answers keyed by question ID (when submitted) */\n answers?: Record<string, string | string[]>;\n /** Cancellation reason (when cancelled) */\n reason?: string;\n}\n\n// ============================================\n// Usage Update Types\n// ============================================\n\n/**\n * Token usage information\n * Sent via extNotification: _codebuddy.ai/usage\n */\nexport interface UsageUpdate {\n sessionId: string;\n inputTokens?: number;\n outputTokens?: number;\n totalTokens?: number;\n cost?: number;\n model?: string;\n _meta?: Record<string, unknown>;\n}\n\n// ============================================\n// Command Types\n// ============================================\n\n/**\n * Command notification parameters\n * Sent via extNotification: _codebuddy.ai/command\n */\nexport interface CommandNotificationParams {\n /** Session ID */\n sessionId: string;\n /** Command action to execute */\n action: string;\n /** Command parameters */\n params?: Record<string, unknown>;\n}\n\n// ============================================\n// Checkpoint Types\n// ============================================\n\n/**\n * Checkpoint event types\n */\nexport type CheckpointEvent = 'created' | 'updated';\n\n/**\n * Checkpoint notification parameters\n * Sent via extNotification: _codebuddy.ai/checkpoint\n */\nexport interface CheckpointNotificationParams {\n /** Session ID */\n sessionId: string;\n /** Event type */\n event: CheckpointEvent;\n /** Checkpoint information */\n checkpoint: CheckpointInfo;\n}\n\n/**\n * Checkpoint information visible to clients\n */\nexport interface CheckpointInfo {\n /** Checkpoint ID */\n id: string;\n /** Label/description */\n label?: string;\n /** Creation timestamp */\n createdAt: number;\n /** File change summary */\n fileChanges: FileChangeSummary;\n}\n\n/**\n * File change summary for a checkpoint\n */\nexport interface FileChangeSummary {\n /** List of changed files */\n files: FileChangeInfo[];\n /** Total lines added */\n totalAdditions: number;\n /** Total lines deleted */\n totalDeletions: number;\n}\n\n/**\n * Change type for a file\n */\nexport type FileChangeType = 'created' | 'modified' | 'deleted';\n\n/**\n * Information about a single file change\n */\nexport interface FileChangeInfo {\n /** File URI (agent://files/{path}) */\n uri: string;\n /** Type of change */\n changeType: FileChangeType;\n /** Lines added */\n additions: number;\n /** Lines deleted */\n deletions: number;\n /** Unified diff format content\n * https://unifiedjs.com/explore/package/unified-diff/\n */\n diff?: string;\n /** File language */\n language?: string;\n}\n\n// ============================================\n// Auth URL Types\n// ============================================\n\n/**\n * Auth URL notification parameters\n * Sent via extNotification: _codebuddy.ai/authUrl\n *\n * Used when the Agent CLI attempts to open an external browser for OAuth authentication.\n * The client displays this URL so users can manually copy and open it if the browser\n * doesn't open automatically.\n */\nexport interface AuthUrlNotificationParams {\n /** The authentication URL to display */\n authUrl: string;\n /** Optional authentication provider identifier (e.g., 'oauth', 'external') */\n provider?: string;\n}\n","// Streamable HTTP Transport for ACP\n// Enables browser-based clients to connect to cloud-hosted ACP agents.\n//\n// Protocol flow:\n// 1. Client establishes GET SSE connection, receives Acp-Connection-Id\n// 2. Client sends POST requests with Acp-Connection-Id header\n// 3. Notifications arrive via GET SSE, responses via POST SSE\n\nimport type { Stream } from '@agentclientprotocol/sdk';\n\ntype StreamMessage = Stream extends { readable: ReadableStream<infer T> } ? T : never;\n\nexport interface StreamableHttpOptions {\n // ACP endpoint URL, e.g. 'https://cloud-agent.example.com/acp'\n endpoint: string;\n // Authorization token (sent as Bearer token)\n authToken?: string;\n // Custom headers to include in all requests\n headers?: Record<string, string>;\n // Reconnect options for SSE connections\n reconnect?: {\n enabled?: boolean; // default: true\n initialDelay?: number; // ms, default: 1000\n maxDelay?: number; // ms, default: 30000\n maxRetries?: number; // default: Infinity\n jitter?: boolean; // default: true, adds ±25% jitter to prevent thundering herd\n };\n // AbortSignal to cancel the connection\n signal?: AbortSignal;\n // Custom fetch implementation (for testing or non-browser environments)\n fetch?: typeof fetch;\n // Callback when connection is established\n onConnect?: (connectionId: string) => void;\n // Callback when connection is closed\n onDisconnect?: (connectionId: string) => void;\n // Callback when an error occurs\n onError?: (error: Error) => void;\n // Heartbeat timeout in ms (default: 60000). If no heartbeat received within this time, reconnect.\n // Set to 0 to disable heartbeat detection.\n heartbeatTimeout?: number;\n // POST request timeout in ms (default: 30000)\n postTimeout?: number;\n // Backpressure options for message queue\n backpressure?: {\n highWaterMark?: number; // default: 100, pause reading SSE when queue reaches this\n lowWaterMark?: number; // default: 50, resume reading SSE when queue drops to this\n pauseTimeout?: number; // default: 5000ms, timeout for backpressure pause to prevent deadlock\n };\n}\n\n/**\n * Extended Stream interface with connection management\n */\nexport interface StreamableHttpTransport extends Stream {\n /**\n * Current connection ID, undefined if not connected\n */\n readonly connectionId: string | undefined;\n /**\n * Promise that resolves when the SSE connection is established\n * and the connectionId is available. Callers should await this\n * before sending messages to ensure the transport is ready.\n */\n readonly ready: Promise<void>;\n /**\n * Close the connection gracefully (sends DELETE request)\n */\n close(): Promise<void>;\n}\n\ninterface SSEEvent {\n type: string;\n data: string;\n id?: string;\n}\n\nfunction parseSSELine(\n line: string,\n currentEvent: Partial<SSEEvent>\n): { event?: SSEEvent; reset: boolean; isComment: boolean } {\n if (line === '') {\n if (currentEvent.data) {\n return {\n event: {\n type: currentEvent.type || 'message',\n data: currentEvent.data,\n id: currentEvent.id,\n },\n reset: true,\n isComment: false,\n };\n }\n return { reset: true, isComment: false };\n }\n\n // SSE comments (including heartbeats) start with ':'\n if (line.startsWith(':')) {\n return { reset: false, isComment: true };\n }\n\n const colonIndex = line.indexOf(':');\n if (colonIndex === -1) {\n return { reset: false, isComment: false };\n }\n\n const field = line.slice(0, colonIndex);\n let value = line.slice(colonIndex + 1);\n if (value.startsWith(' ')) {\n value = value.slice(1);\n }\n\n switch (field) {\n case 'event':\n // Reset data when starting a new event type\n // This prevents data from previous events being concatenated\n if (currentEvent.type && currentEvent.type !== value) {\n currentEvent.data = undefined;\n }\n currentEvent.type = value;\n break;\n case 'data':\n currentEvent.data = (currentEvent.data || '') + value;\n break;\n case 'id':\n currentEvent.id = value;\n break;\n }\n\n return { reset: false, isComment: false };\n}\n\n// Create a Streamable HTTP transport.\n//\n// Example:\n// const transport = streamableHttp({\n// endpoint: 'https://agent.example.com/acp',\n// authToken: 'token123',\n// onConnect: (id) => console.log('Connected:', id),\n// onDisconnect: (id) => console.log('Disconnected:', id),\n// });\n//\n// // Later, close gracefully\n// await transport.close();\nexport function streamableHttp(options: StreamableHttpOptions): StreamableHttpTransport {\n const {\n endpoint,\n authToken,\n headers: customHeaders = {},\n reconnect = {},\n signal: externalSignal,\n fetch: customFetch = globalThis.fetch,\n onConnect,\n onDisconnect,\n onError,\n heartbeatTimeout = 60000,\n postTimeout = 30000,\n backpressure = {},\n } = options;\n\n const {\n enabled: reconnectEnabled = true,\n initialDelay = 1000,\n maxDelay = 30000,\n maxRetries = Infinity,\n jitter: jitterEnabled = true,\n } = reconnect;\n\n const {\n highWaterMark = 100,\n lowWaterMark = 50,\n pauseTimeout = 5000,\n } = backpressure;\n\n // Connection state\n let connectionId: string | undefined;\n let lastEventId: string | undefined;\n let reconnectAttempts = 0;\n let closed = false;\n let isClosing = false;\n\n // Promise that resolves when GET SSE is connected and we have connectionId\n let connectionReady: Promise<void>;\n let resolveConnection: () => void;\n let rejectConnection: (error: Error) => void;\n\n // Connection version to track reconnections\n let connectionVersion = 0;\n\n connectionReady = new Promise((resolve, reject) => {\n resolveConnection = resolve;\n rejectConnection = reject;\n });\n\n const abortController = new AbortController();\n\n // Combine signals - use manual approach for broader compatibility\n function isAborted(): boolean {\n return abortController.signal.aborted || (externalSignal?.aborted ?? false);\n }\n\n function getSignal(): AbortSignal {\n // AbortSignal.any is available in newer Node.js versions\n const anyFn = (AbortSignal as unknown as { any?: (signals: AbortSignal[]) => AbortSignal }).any;\n if (externalSignal && typeof anyFn === 'function') {\n return anyFn([externalSignal, abortController.signal]);\n }\n return abortController.signal;\n }\n\n const combinedSignal = getSignal();\n\n // Message queue for incoming messages with backpressure\n const messageQueue: StreamMessage[] = [];\n const messageResolvers: Array<(value: StreamMessage | null) => void> = [];\n let streamError: Error | null = null;\n let isPaused = false;\n let resumeReading: (() => void) | null = null;\n\n // Track background SSE processors to handle POST responses asynchronously\n const backgroundSSEProcessors: Set<Promise<void>> = new Set();\n\n // Heartbeat tracking\n let lastActivity = Date.now();\n let heartbeatCheckTimer: ReturnType<typeof setInterval> | undefined;\n\n function enqueueMessage(message: StreamMessage): boolean {\n if (messageResolvers.length > 0) {\n const resolver = messageResolvers.shift()!;\n resolver(message);\n return true;\n } else {\n messageQueue.push(message);\n // Check if we should pause reading due to backpressure\n if (messageQueue.length >= highWaterMark) {\n isPaused = true;\n return false;\n }\n return true;\n }\n }\n\n function dequeueMessage(): Promise<StreamMessage | null> {\n if (closed) {\n return Promise.resolve(null);\n }\n if (streamError) {\n return Promise.reject(streamError);\n }\n if (messageQueue.length > 0) {\n const message = messageQueue.shift()!;\n // Check if we should resume reading\n if (isPaused && messageQueue.length <= lowWaterMark) {\n isPaused = false;\n const resume = resumeReading;\n if (resume) {\n // Use queueMicrotask to avoid potential stack issues\n queueMicrotask(() => resume());\n }\n }\n return Promise.resolve(message);\n }\n return new Promise(resolve => {\n messageResolvers.push(resolve);\n });\n }\n\n function updateLastActivity(): void {\n lastActivity = Date.now();\n }\n\n function startHeartbeatCheck(triggerReconnect: () => void): void {\n if (heartbeatTimeout <= 0) {\n return;\n }\n // Check heartbeat every 10 seconds\n heartbeatCheckTimer = setInterval(() => {\n if (Date.now() - lastActivity > heartbeatTimeout) {\n console.warn('[StreamableHTTP] Heartbeat timeout, triggering reconnect');\n triggerReconnect();\n }\n }, 10000);\n }\n\n function stopHeartbeatCheck(): void {\n if (heartbeatCheckTimer) {\n clearInterval(heartbeatCheckTimer);\n heartbeatCheckTimer = undefined;\n }\n }\n\n /**\n * Calculate reconnect delay with optional jitter\n */\n function calculateDelay(attempt: number): number {\n const baseDelay = Math.min(initialDelay * Math.pow(2, attempt - 1), maxDelay);\n if (!jitterEnabled) {\n return baseDelay;\n }\n // Add ±25% jitter to prevent thundering herd\n const jitterFactor = 0.25 * (Math.random() * 2 - 1);\n return Math.round(baseDelay * (1 + jitterFactor));\n }\n\n function closeWithError(error: Error): void {\n streamError = error;\n closed = true;\n stopHeartbeatCheck();\n // Resume any paused reading\n if (resumeReading) {\n resumeReading();\n resumeReading = null;\n }\n rejectConnection(error);\n onError?.(error);\n while (messageResolvers.length > 0) {\n const resolver = messageResolvers.shift()!;\n resolver(null);\n }\n }\n\n function closeNormally(): void {\n closed = true;\n stopHeartbeatCheck();\n // Resume any paused reading\n if (resumeReading) {\n resumeReading();\n resumeReading = null;\n }\n // Clear background SSE processors (they will exit due to closed=true)\n backgroundSSEProcessors.clear();\n while (messageResolvers.length > 0) {\n const resolver = messageResolvers.shift()!;\n resolver(null);\n }\n }\n\n // Send DELETE request to close connection gracefully\n async function sendDelete(): Promise<void> {\n if (!connectionId || isClosing) {\n return;\n }\n\n isClosing = true;\n const currentConnectionId = connectionId;\n\n try {\n const headers = buildHeaders();\n headers['Acp-Connection-Id'] = currentConnectionId;\n\n await customFetch(endpoint, {\n method: 'DELETE',\n headers,\n signal: AbortSignal.timeout(5000), // 5s timeout for DELETE\n });\n } catch {\n // Ignore DELETE errors - connection may already be closed\n } finally {\n if (currentConnectionId) {\n onDisconnect?.(currentConnectionId);\n }\n isClosing = false;\n }\n }\n\n function buildHeaders(): Record<string, string> {\n const headers: Record<string, string> = {\n ...customHeaders,\n };\n\n if (authToken) {\n headers['Authorization'] = `Bearer ${authToken}`;\n }\n\n return headers;\n }\n\n async function processSSEStream(\n reader: ReadableStreamDefaultReader<Uint8Array>\n ): Promise<void> {\n const decoder = new TextDecoder();\n let buffer = '';\n let currentEvent: Partial<SSEEvent> = {};\n\n try {\n while (true) {\n // Check for backpressure - wait if paused with timeout protection\n if (isPaused) {\n await new Promise<void>(resolve => {\n let resolved = false;\n\n // Timeout protection to prevent permanent deadlock\n const timeoutId = setTimeout(() => {\n if (!resolved) {\n resolved = true;\n console.warn('[StreamableHTTP] Backpressure pause timeout, forcing resume');\n resolve();\n }\n }, pauseTimeout);\n\n resumeReading = () => {\n if (!resolved) {\n resolved = true;\n clearTimeout(timeoutId);\n resolve();\n }\n };\n });\n resumeReading = null;\n }\n\n const { value, done } = await reader.read();\n if (done) {break;}\n\n buffer += decoder.decode(value, { stream: true });\n const lines = buffer.split('\\n');\n buffer = lines.pop() || '';\n\n for (const line of lines) {\n const { event, reset, isComment } = parseSSELine(line, currentEvent);\n\n // SSE comments (including heartbeats) update last activity\n if (isComment) {\n updateLastActivity();\n continue;\n }\n\n if (event) {\n // Any event means we're receiving data - update activity\n updateLastActivity();\n\n if (event.id) {\n lastEventId = event.id;\n }\n\n // Skip non-message events (like \"connected\")\n if (event.type !== 'message') {\n continue;\n }\n\n try {\n const message = JSON.parse(event.data) as StreamMessage;\n // Only enqueue valid JSON-RPC messages\n if (message && typeof message === 'object' && 'jsonrpc' in message) {\n enqueueMessage(message);\n }\n } catch {\n console.error('[StreamableHTTP] Failed to parse SSE data:', event.data);\n }\n }\n\n if (reset) {\n currentEvent = {};\n }\n }\n }\n } finally {\n reader.releaseLock();\n }\n }\n\n /**\n * Process SSE stream in background without blocking the caller.\n * This prevents deadlock when POST responses return SSE streams.\n */\n function processSSEStreamBackground(\n reader: ReadableStreamDefaultReader<Uint8Array>\n ): void {\n const promise = processSSEStream(reader)\n .catch(error => {\n console.error('[StreamableHTTP] Background SSE processing error:', error);\n onError?.(error instanceof Error ? error : new Error(String(error)));\n })\n .finally(() => {\n backgroundSSEProcessors.delete(promise);\n });\n backgroundSSEProcessors.add(promise);\n }\n\n // Establish GET SSE connection and get connectionId\n async function startSSEConnection(): Promise<void> {\n // Track current reader for heartbeat-triggered reconnect\n let currentReader: ReadableStreamDefaultReader<Uint8Array> | null = null;\n\n // Function to trigger reconnect (called from heartbeat timeout)\n const triggerReconnect = (): void => {\n if (currentReader) {\n currentReader.cancel().catch(() => {/* ignore */});\n }\n };\n\n while (!closed && !isAborted()) {\n try {\n const headers = buildHeaders();\n headers['Accept'] = 'text/event-stream';\n\n if (lastEventId) {\n headers['Last-Event-ID'] = lastEventId;\n }\n\n const response = await customFetch(endpoint, {\n method: 'GET',\n headers,\n signal: combinedSignal,\n });\n\n if (!response.ok) {\n throw new Error(`HTTP ${response.status}`);\n }\n\n // Get connection ID from response - MUST have it\n const newConnectionId = response.headers.get('Acp-Connection-Id');\n if (!newConnectionId) {\n throw new Error('Server did not return Acp-Connection-Id header');\n }\n\n // Track previous connection for disconnect callback\n const previousConnectionId = connectionId;\n\n // Update connection state atomically\n connectionVersion++;\n connectionId = newConnectionId;\n resolveConnection();\n\n // Notify callbacks\n if (previousConnectionId && previousConnectionId !== newConnectionId) {\n onDisconnect?.(previousConnectionId);\n }\n onConnect?.(newConnectionId);\n\n reconnectAttempts = 0;\n\n // Reset heartbeat tracking and start heartbeat check\n updateLastActivity();\n startHeartbeatCheck(triggerReconnect);\n\n const reader = response.body?.getReader();\n if (reader) {\n currentReader = reader;\n await processSSEStream(reader);\n currentReader = null;\n }\n\n // Stop heartbeat check when connection ends\n stopHeartbeatCheck();\n\n // Connection ended\n const endedConnectionId = connectionId;\n connectionId = undefined;\n\n // Notify disconnect\n if (endedConnectionId) {\n onDisconnect?.(endedConnectionId);\n }\n\n if (!reconnectEnabled || closed) {\n break;\n }\n\n // Reconnect - create new connectionReady promise BEFORE clearing connectionId\n // so that sendMessage waits for new connection\n connectionReady = new Promise((resolve, reject) => {\n resolveConnection = resolve;\n rejectConnection = reject;\n });\n\n // Add delay before reconnecting to prevent tight loop\n // This gives sendMessage a chance to use the connection before it's replaced\n const reconnectDelay = calculateDelay(1); // Use minimal delay with jitter\n await new Promise(resolve => setTimeout(resolve, reconnectDelay));\n } catch (error) {\n // Stop heartbeat check on error\n stopHeartbeatCheck();\n currentReader = null;\n\n if (isAborted() || closed) {\n break;\n }\n\n reconnectAttempts++;\n\n if (reconnectAttempts > maxRetries) {\n closeWithError(new Error(`SSE reconnect failed after ${maxRetries} attempts`));\n break;\n }\n\n // Use calculateDelay with jitter\n const delay = calculateDelay(reconnectAttempts);\n\n console.warn(\n `[StreamableHTTP] SSE error, retrying in ${delay}ms (attempt ${reconnectAttempts}):`,\n error\n );\n\n await new Promise(resolve => setTimeout(resolve, delay));\n }\n }\n }\n\n // Send message via POST\n async function sendMessage(message: StreamMessage): Promise<void> {\n if (closed) {\n throw new Error('Connection is closed');\n }\n\n // Wait for stable connection with retry logic\n // This handles rapid reconnection scenarios\n const maxWaitAttempts = 5;\n let currentConnectionId: string | undefined;\n\n for (let attempt = 0; attempt < maxWaitAttempts; attempt++) {\n // Wait for GET SSE to establish and get connectionId\n const versionBeforeWait = connectionVersion;\n await connectionReady;\n\n // Check if reconnection happened while we were waiting\n if (versionBeforeWait !== connectionVersion && versionBeforeWait > 0) {\n // A reconnection happened - wait for new connection\n await connectionReady;\n }\n\n // Capture connectionId after waiting\n currentConnectionId = connectionId;\n\n if (currentConnectionId) {\n break; // Got a valid connection\n }\n\n // Connection became undefined (reconnecting), wait a bit and retry\n if (attempt < maxWaitAttempts - 1) {\n await new Promise(resolve => setTimeout(resolve, 100));\n }\n }\n\n if (!currentConnectionId) {\n throw new Error('No connection ID available after multiple attempts');\n }\n\n const headers = buildHeaders();\n headers['Content-Type'] = 'application/json';\n headers['Accept'] = 'application/json, text/event-stream';\n headers['Acp-Connection-Id'] = currentConnectionId;\n\n // Create timeout controller for POST request\n const postController = new AbortController();\n let timeoutId: ReturnType<typeof setTimeout> | undefined;\n\n // Combine with external signal if present\n const postSignal = postTimeout > 0\n ? postController.signal\n : combinedSignal;\n\n if (postTimeout > 0) {\n timeoutId = setTimeout(() => postController.abort(), postTimeout);\n // Also abort if external signal is aborted\n if (externalSignal) {\n externalSignal.addEventListener('abort', () => postController.abort(), { once: true });\n }\n abortController.signal.addEventListener('abort', () => postController.abort(), { once: true });\n }\n\n try {\n const response = await customFetch(endpoint, {\n method: 'POST',\n headers,\n body: JSON.stringify(message),\n signal: postSignal,\n });\n\n if (!response.ok) {\n const errorText = await response.text().catch(() => 'Unknown error');\n throw new Error(`HTTP ${response.status}: ${errorText}`);\n }\n\n // Handle response based on content type\n const contentType = response.headers.get('Content-Type') || '';\n\n if (contentType.includes('text/event-stream')) {\n const reader = response.body?.getReader();\n if (reader) {\n // Process SSE in background to avoid blocking sendMessage\n // This prevents deadlock when backpressure causes processSSEStream to pause\n processSSEStreamBackground(reader);\n }\n } else if (contentType.includes('application/json')) {\n const data = await response.json();\n if (data && typeof data === 'object' && 'jsonrpc' in data) {\n enqueueMessage(data as StreamMessage);\n }\n }\n // 202 responses have no body\n } finally {\n if (timeoutId) {\n clearTimeout(timeoutId);\n }\n }\n }\n\n // Start SSE connection immediately\n startSSEConnection().catch(error => {\n console.error('[StreamableHTTP] SSE connection error:', error);\n });\n\n const readable = new ReadableStream<StreamMessage>({\n async pull(controller) {\n const message = await dequeueMessage();\n if (message === null) {\n controller.close();\n } else {\n controller.enqueue(message);\n }\n },\n cancel() {\n closeNormally();\n abortController.abort();\n },\n });\n\n const writable = new WritableStream<StreamMessage>({\n async write(message) {\n await sendMessage(message);\n },\n close() {\n closeNormally();\n abortController.abort();\n },\n abort(reason) {\n closeWithError(reason instanceof Error ? reason : new Error(String(reason)));\n abortController.abort();\n },\n });\n\n // Close the connection gracefully\n async function close(): Promise<void> {\n if (closed) {\n return;\n }\n\n // Send DELETE to server first\n await sendDelete();\n\n // Then close locally\n closeNormally();\n abortController.abort();\n }\n\n return {\n readable,\n writable,\n get connectionId() {\n return connectionId;\n },\n get ready() {\n return connectionReady;\n },\n close,\n };\n}\n\nexport default streamableHttp;\n","/**\n * Artifact Manager for Streamable HTTP ACP Client\n * Handles artifact notification processing\n */\n\nimport type {\n Artifact,\n ArtifactEvent,\n ArtifactNotificationParams,\n} from '../types.js';\nimport type { Logger } from './types.js';\n\n/**\n * Configuration for ArtifactManager\n */\nexport interface ArtifactManagerConfig {\n /** Logger instance */\n logger?: Logger;\n}\n\n/**\n * Callback for artifact events\n */\nexport type ArtifactEventCallback = (artifact: Artifact, event: ArtifactEvent) => void;\n\n/**\n * Manages artifact notifications from the agent\n */\nexport class ArtifactManager {\n private artifacts = new Map<string, Artifact>();\n private logger?: Logger;\n private eventCallbacks: Set<ArtifactEventCallback> = new Set();\n\n constructor(config: ArtifactManagerConfig) {\n this.logger = config.logger;\n }\n\n /**\n * Register a callback for artifact events\n */\n onArtifactEvent(callback: ArtifactEventCallback): () => void {\n this.eventCallbacks.add(callback);\n return () => {\n this.eventCallbacks.delete(callback);\n };\n }\n\n /**\n * Handle an artifact notification from the agent\n */\n handleNotification(notification: ArtifactNotificationParams): void {\n const { event } = notification;\n\n if (event === 'deleted') {\n const { artifact } = notification;\n const existing = this.artifacts.get(artifact.uri);\n this.logger?.debug(`Artifact deleted: ${artifact.uri}`);\n this.artifacts.delete(artifact.uri);\n\n // Notify callbacks with existing artifact if available\n if (existing) {\n this.notifyCallbacks(existing, event);\n }\n } else {\n const { artifact } = notification;\n this.logger?.debug(`Artifact ${event}: ${artifact.uri} (${artifact.type})`);\n this.artifacts.set(artifact.uri, artifact);\n this.notifyCallbacks(artifact, event);\n }\n }\n\n private notifyCallbacks(artifact: Artifact, event: ArtifactEvent): void {\n for (const callback of this.eventCallbacks) {\n try {\n callback(artifact, event);\n } catch (err) {\n this.logger?.error('Error in artifact event callback:', err);\n }\n }\n }\n\n /**\n * Get an artifact by URI (used internally for deleted event handling)\n */\n get(uri: string): Artifact | undefined {\n return this.artifacts.get(uri);\n }\n\n /**\n * Clear all artifacts\n */\n clear(): void {\n this.artifacts.clear();\n this.logger?.debug('Cleared all artifacts');\n }\n}\n","/**\n * Protocol constants for Streamable HTTP ACP Client\n */\n\nimport type { ClientCapabilities } from '../sdk.js';\nimport { ExtensionMethod, KNOWN_EXTENSIONS } from '../types.js';\n\n// Re-export extension constants\nexport { ExtensionMethod, KNOWN_EXTENSIONS };\n\n// ============================================\n// Default Timeouts (in milliseconds)\n// ============================================\n\n/**\n * Default timeout for initialize operation\n */\nexport const DEFAULT_INITIALIZE_TIMEOUT = 30_000; // 30 seconds\n\n/**\n * Default timeout for prompt operation\n */\nexport const DEFAULT_PROMPT_TIMEOUT = 300_000; // 5 minutes\n\n/**\n * Default timeout for permission requests\n */\nexport const DEFAULT_PERMISSION_TIMEOUT = 300_000; // 5 minutes\n\n/**\n * Default timeout for question requests (ask_followup_question)\n */\nexport const DEFAULT_QUESTION_TIMEOUT = 300_000; // 5 minutes\n\n/**\n * Default timeout for tool input requests\n * @deprecated Use DEFAULT_QUESTION_TIMEOUT instead\n */\nexport const DEFAULT_TOOL_INPUT_TIMEOUT = DEFAULT_QUESTION_TIMEOUT;\n\n// ============================================\n// Default Reconnect Configuration\n// ============================================\n\n/**\n * Default reconnect options\n */\nexport const DEFAULT_RECONNECT_OPTIONS = {\n enabled: true,\n initialDelay: 1000, // 1 second\n maxDelay: 30_000, // 30 seconds\n maxRetries: Infinity\n} as const;\n\n// ============================================\n// Client Capabilities\n// ============================================\n\n/**\n * Default client capabilities for cloud-based clients\n * Cloud clients typically have no direct file system access\n */\nexport const CLOUD_CLIENT_CAPABILITIES: ClientCapabilities = {\n fs: {\n readTextFile: false,\n writeTextFile: false\n }\n} as const;\n\n/**\n * Default client capabilities for local (Node.js) clients\n * Local clients have file system access\n */\nexport const LOCAL_CLIENT_CAPABILITIES: ClientCapabilities = {\n fs: {\n readTextFile: true,\n writeTextFile: true\n }\n} as const;\n","/**\n * Custom error classes for Streamable HTTP ACP Client\n */\n\n/**\n * Base error class for all ACP client errors\n */\nexport class ACPClientError extends Error {\n public readonly code: string;\n public readonly cause?: Error;\n\n constructor(message: string, code: string, cause?: Error) {\n super(message);\n this.name = 'ACPClientError';\n this.code = code;\n this.cause = cause;\n\n // Maintain proper stack trace in V8 environments\n if (Error.captureStackTrace) {\n Error.captureStackTrace(this, this.constructor);\n }\n }\n}\n\n/**\n * Error thrown when connection fails\n */\nexport class ConnectionError extends ACPClientError {\n constructor(message: string, cause?: Error) {\n super(message, 'CONNECTION_ERROR', cause);\n this.name = 'ConnectionError';\n }\n}\n\n/**\n * Error thrown when initialization fails\n */\nexport class InitializationError extends ACPClientError {\n constructor(message: string, cause?: Error) {\n super(message, 'INITIALIZATION_ERROR', cause);\n this.name = 'InitializationError';\n }\n}\n\n/**\n * Error thrown for session-related failures\n */\nexport class SessionError extends ACPClientError {\n public readonly sessionId?: string;\n\n constructor(message: string, sessionId?: string, cause?: Error) {\n super(message, 'SESSION_ERROR', cause);\n this.name = 'SessionError';\n this.sessionId = sessionId;\n }\n}\n\n/**\n * Error thrown when prompt operation fails\n */\nexport class PromptError extends ACPClientError {\n public readonly sessionId?: string;\n\n constructor(message: string, sessionId?: string, cause?: Error) {\n super(message, 'PROMPT_ERROR', cause);\n this.name = 'PromptError';\n this.sessionId = sessionId;\n }\n}\n\n/**\n * Error thrown for permission-related failures\n */\nexport class PermissionError extends ACPClientError {\n public readonly requestId?: string;\n\n constructor(message: string, requestId?: string, cause?: Error) {\n super(message, 'PERMISSION_ERROR', cause);\n this.name = 'PermissionError';\n this.requestId = requestId;\n }\n}\n\n/**\n * Error thrown when an operation times out\n */\nexport class TimeoutError extends ACPClientError {\n public readonly operation: string;\n public readonly timeoutMs: number;\n\n constructor(operation: string, timeoutMs: number) {\n super(`Operation '${operation}' timed out after ${timeoutMs}ms`, 'TIMEOUT_ERROR');\n this.name = 'TimeoutError';\n this.operation = operation;\n this.timeoutMs = timeoutMs;\n }\n}\n\n/**\n * Error thrown when client is in invalid state for an operation\n */\nexport class InvalidStateError extends ACPClientError {\n public readonly currentState: string;\n public readonly expectedStates: string[];\n\n constructor(operation: string, currentState: string, expectedStates: string[]) {\n super(\n `Cannot perform '${operation}' in state '${currentState}'. Expected: ${expectedStates.join(' or ')}`,\n 'INVALID_STATE_ERROR'\n );\n this.name = 'InvalidStateError';\n this.currentState = currentState;\n this.expectedStates = expectedStates;\n }\n}\n\n","/**\n * Type-safe event emitter for Streamable HTTP ACP Client\n * Platform-agnostic implementation (no browser dependencies)\n */\n\n/**\n * Event listener function type\n */\nexport type EventListener<T> = (data: T) => void | Promise<void>;\n\n/**\n * Type-safe event emitter implementation\n */\nexport class EventEmitter<TEvents extends Record<string, unknown>> {\n private listeners: Map<keyof TEvents, Set<EventListener<unknown>>> = new Map();\n private onceListeners: Map<keyof TEvents, Set<EventListener<unknown>>> = new Map();\n\n /**\n * Add an event listener\n */\n on<K extends keyof TEvents>(event: K, listener: EventListener<TEvents[K]>): this {\n if (!this.listeners.has(event)) {\n this.listeners.set(event, new Set());\n }\n this.listeners.get(event)!.add(listener as EventListener<unknown>);\n return this;\n }\n\n /**\n * Remove an event listener\n */\n off<K extends keyof TEvents>(event: K, listener: EventListener<TEvents[K]>): this {\n const eventListeners = this.listeners.get(event);\n if (eventListeners) {\n eventListeners.delete(listener as EventListener<unknown>);\n }\n\n const onceEventListeners = this.onceListeners.get(event);\n if (onceEventListeners) {\n onceEventListeners.delete(listener as EventListener<unknown>);\n }\n\n return this;\n }\n\n /**\n * Add a one-time event listener\n */\n once<K extends keyof TEvents>(event: K, listener: EventListener<TEvents[K]>): this {\n if (!this.onceListeners.has(event)) {\n this.onceListeners.set(event, new Set());\n }\n this.onceListeners.get(event)!.add(listener as EventListener<unknown>);\n return this;\n }\n\n /**\n * Emit an event to all registered listeners\n * Returns true if any listeners were invoked\n */\n emit<K extends keyof TEvents>(event: K, data: TEvents[K]): boolean {\n const regularListeners = this.listeners.get(event);\n const onceEventListeners = this.onceListeners.get(event);\n\n let hasListeners = false;\n\n // Call regular listeners\n if (regularListeners && regularListeners.size > 0) {\n hasListeners = true;\n for (const listener of regularListeners) {\n try {\n const result = listener(data);\n // Handle async listeners\n if (result instanceof Promise) {\n result.catch(err => {\n console.error(`Error in async event listener for '${String(event)}':`, err);\n });\n }\n } catch (err) {\n console.error(`Error in event listener for '${String(event)}':`, err);\n }\n }\n }\n\n // Call once listeners and remove them\n if (onceEventListeners && onceEventListeners.size > 0) {\n hasListeners = true;\n const listenersToCall = Array.from(onceEventListeners);\n this.onceListeners.delete(event);\n\n for (const listener of listenersToCall) {\n try {\n const result = listener(data);\n if (result instanceof Promise) {\n result.catch(err => {\n console.error(\n `Error in async once event listener for '${String(event)}':`,\n err\n );\n });\n }\n } catch (err) {\n console.error(`Error in once event listener for '${String(event)}':`, err);\n }\n }\n }\n\n return hasListeners;\n }\n\n /**\n * Remove all listeners for an event, or all listeners if no event specified\n */\n removeAllListeners<K extends keyof TEvents>(event?: K): this {\n if (event !== undefined) {\n this.listeners.delete(event);\n this.onceListeners.delete(event);\n } else {\n this.listeners.clear();\n this.onceListeners.clear();\n }\n return this;\n }\n\n /**\n * Get the number of listeners for an event\n */\n listenerCount<K extends keyof TEvents>(event: K): number {\n const regular = this.listeners.get(event)?.size ?? 0;\n const once = this.onceListeners.get(event)?.size ?? 0;\n return regular + once;\n }\n\n /**\n * Get all event names that have listeners\n */\n eventNames(): Array<keyof TEvents> {\n const names = new Set<keyof TEvents>();\n for (const event of this.listeners.keys()) {\n names.add(event);\n }\n for (const event of this.onceListeners.keys()) {\n names.add(event);\n }\n return Array.from(names);\n }\n}\n","/**\n * Extension Method Handler for Streamable HTTP ACP Client\n * Routes and handles custom extension notifications\n */\n\nimport type { UsageUpdate } from '../types.js';\nimport { ExtensionMethod, KNOWN_EXTENSIONS } from './constants.js';\nimport type { Logger } from './types.js';\n\n/**\n * Configuration for ExtensionManager\n */\nexport interface ExtensionManagerConfig {\n /** Logger instance */\n logger?: Logger;\n}\n\n/**\n * Handler for extension notifications\n */\nexport type ExtensionNotificationHandler = (\n method: string,\n params: Record<string, unknown>\n) => void | Promise<void>;\n\n/**\n * Manages extension methods and notifications\n */\nexport class ExtensionManager {\n private config: ExtensionManagerConfig;\n private handlers = new Map<string, ExtensionNotificationHandler>();\n private fallbackHandler?: ExtensionNotificationHandler;\n\n constructor(config: ExtensionManagerConfig = {}) {\n this.config = config;\n }\n\n /**\n * Register a handler for a specific extension method\n */\n registerHandler(method: string, handler: ExtensionNotificationHandler): () => void {\n this.handlers.set(method, handler);\n return () => {\n this.handlers.delete(method);\n };\n }\n\n /**\n * Set a fallback handler for unknown extensions\n */\n setFallbackHandler(handler: ExtensionNotificationHandler): void {\n this.fallbackHandler = handler;\n }\n\n /**\n * Handle an extension notification\n */\n async handleNotification(method: string, params: Record<string, unknown>): Promise<void> {\n this.config.logger?.debug(`Extension notification: ${method}`);\n\n // Try specific handler first\n const handler = this.handlers.get(method);\n if (handler) {\n await handler(method, params);\n return;\n }\n\n // Try fallback handler\n if (this.fallbackHandler) {\n await this.fallbackHandler(method, params);\n return;\n }\n\n // Log unknown extension\n if (!this.isKnownExtension(method)) {\n this.config.logger?.warn(`Unknown extension notification: ${method}`);\n }\n }\n\n /**\n * Check if a method is a known extension\n */\n isKnownExtension(method: string): boolean {\n return KNOWN_EXTENSIONS.includes(method as typeof KNOWN_EXTENSIONS[number]);\n }\n\n /**\n * Check if method is the artifact extension\n */\n isArtifactExtension(method: string): boolean {\n return method === ExtensionMethod.ARTIFACT;\n }\n\n /**\n * Clear all handlers\n */\n clear(): void {\n this.handlers.clear();\n this.fallbackHandler = undefined;\n }\n}\n\n/**\n * Parse usage update from extension params\n */\nexport function parseUsageUpdate(params: Record<string, unknown>): UsageUpdate {\n return {\n sessionId: params.sessionId as string,\n inputTokens: params.inputTokens as number | undefined,\n outputTokens: params.outputTokens as number | undefined,\n totalTokens: params.totalTokens as number | undefined,\n cost: params.cost as number | undefined,\n model: params.model as string | undefined,\n _meta: params._meta as Record<string, unknown> | undefined\n };\n}\n","/**\n * Permission Manager for Streamable HTTP ACP Client\n * Handles permission requests with timeout support\n */\n\nimport type { RequestPermissionRequest, RequestPermissionResponse } from '@agentclientprotocol/sdk';\n\nimport { DEFAULT_PERMISSION_TIMEOUT } from './constants.js';\nimport { TimeoutError } from './errors.js';\nimport type { Logger, PermissionHandler } from './types.js';\n\n/**\n * Pending permission request state\n */\ninterface PendingPermission {\n params: RequestPermissionRequest;\n resolve: (response: RequestPermissionResponse) => void;\n reject: (error: Error) => void;\n timeoutId?: ReturnType<typeof setTimeout>;\n createdAt: number;\n}\n\n/**\n * Configuration for PermissionManager\n */\nexport interface PermissionManagerConfig {\n /** Default timeout for permission requests (ms) */\n timeout?: number;\n /** Auto-reject permissions on timeout */\n autoRejectOnTimeout?: boolean;\n /** Auto-approve all permissions (for testing) */\n autoApprove?: boolean;\n /** Custom permission handler */\n handler?: PermissionHandler;\n /** Logger instance */\n logger?: Logger;\n}\n\n/**\n * Event callbacks for permission events\n */\nexport interface PermissionEventCallbacks {\n onRequest?: (requestId: string, params: RequestPermissionRequest) => void;\n onResolved?: (requestId: string, optionId: string) => void;\n onRejected?: (requestId: string, reason?: string) => void;\n onTimeout?: (requestId: string) => void;\n}\n\n/**\n * Manages permission requests from the agent\n */\nexport class PermissionManager {\n private pending = new Map<string, PendingPermission>();\n private config: PermissionManagerConfig;\n private callbacks: PermissionEventCallbacks = {};\n\n constructor(config: PermissionManagerConfig = {}) {\n this.config = {\n timeout: DEFAULT_PERMISSION_TIMEOUT,\n autoRejectOnTimeout: true,\n autoApprove: false,\n ...config\n };\n }\n\n /**\n * Set event callbacks\n */\n setCallbacks(callbacks: PermissionEventCallbacks): void {\n this.callbacks = callbacks;\n }\n\n /**\n * Handle a permission request from the agent\n */\n async handleRequest(params: RequestPermissionRequest): Promise<RequestPermissionResponse> {\n const requestId = params.toolCall.toolCallId;\n\n this.config.logger?.debug(`Permission request received: ${requestId}`);\n\n // Auto-approve mode\n if (this.config.autoApprove) {\n const firstOption = params.options[0];\n this.config.logger?.debug(`Auto-approving permission: ${requestId}`);\n return {\n outcome: {\n outcome: 'selected',\n optionId: firstOption?.optionId ?? 'approve'\n }\n };\n }\n\n // Custom handler\n if (this.config.handler) {\n return this.config.handler(params);\n }\n\n // Create pending permission\n return new Promise<RequestPermissionResponse>((resolve, reject) => {\n const pending: PendingPermission = {\n params,\n resolve,\n reject,\n createdAt: Date.now()\n };\n\n // Set up timeout\n if (this.config.timeout && this.config.timeout > 0) {\n pending.timeoutId = setTimeout(() => {\n this.handleTimeout(requestId);\n }, this.config.timeout);\n }\n\n this.pending.set(requestId, pending);\n\n // Notify callback\n this.callbacks.onRequest?.(requestId, params);\n });\n }\n\n /**\n * Handle timeout for a permission request\n */\n private handleTimeout(requestId: string): void {\n const pending = this.pending.get(requestId);\n if (!pending) {return;}\n\n this.config.logger?.warn(`Permission request timed out: ${requestId}`);\n this.callbacks.onTimeout?.(requestId);\n\n if (this.config.autoRejectOnTimeout) {\n pending.resolve({\n outcome: { outcome: 'cancelled' }\n });\n } else {\n pending.reject(new TimeoutError('permission', this.config.timeout ?? DEFAULT_PERMISSION_TIMEOUT));\n }\n\n this.pending.delete(requestId);\n }\n\n /**\n * Resolve a permission request with a selected option\n * Returns true if the permission was found and resolved\n */\n resolve(requestId: string, optionId: string): boolean {\n const pending = this.pending.get(requestId);\n if (!pending) {\n this.config.logger?.warn(`Permission request not found: ${requestId}`);\n return false;\n }\n\n // Clear timeout\n if (pending.timeoutId) {\n clearTimeout(pending.timeoutId);\n }\n\n this.config.logger?.debug(`Permission resolved: ${requestId} -> ${optionId}`);\n\n pending.resolve({\n outcome: { outcome: 'selected', optionId }\n });\n\n this.pending.delete(requestId);\n this.callbacks.onResolved?.(requestId, optionId);\n\n return true;\n }\n\n /**\n * Reject (cancel) a permission request\n * Returns true if the permission was found and rejected\n */\n reject(requestId: string, reason?: string): boolean {\n const pending = this.pending.get(requestId);\n if (!pending) {\n this.config.logger?.warn(`Permission request not found: ${requestId}`);\n return false;\n }\n\n // Clear timeout\n if (pending.timeoutId) {\n clearTimeout(pending.timeoutId);\n }\n\n this.config.logger?.debug(`Permission rejected: ${requestId}${reason ? ` - ${reason}` : ''}`);\n\n pending.resolve({\n outcome: { outcome: 'cancelled' }\n });\n\n this.pending.delete(requestId);\n this.callbacks.onRejected?.(requestId, reason);\n\n return true;\n }\n\n /**\n * Get all pending permissions\n */\n getPending(): Map<string, { params: RequestPermissionRequest; createdAt: number }> {\n const result = new Map<string, { params: RequestPermissionRequest; createdAt: number }>();\n for (const [id, pending] of this.pending) {\n result.set(id, {\n params: pending.params,\n createdAt: pending.createdAt\n });\n }\n return result;\n }\n\n /**\n * Get a specific pending permission\n */\n getPendingById(requestId: string): { params: RequestPermissionRequest; createdAt: number } | undefined {\n const pending = this.pending.get(requestId);\n if (!pending) {return undefined;}\n return {\n params: pending.params,\n createdAt: pending.createdAt\n };\n }\n\n /**\n * Check if there are any pending permissions\n */\n hasPending(): boolean {\n return this.pending.size > 0;\n }\n\n /**\n * Get the count of pending permissions\n */\n get pendingCount(): number {\n return this.pending.size;\n }\n\n /**\n * Clear all pending permissions (reject all)\n */\n clear(): void {\n for (const [requestId, pending] of this.pending) {\n if (pending.timeoutId) {\n clearTimeout(pending.timeoutId);\n }\n pending.resolve({\n outcome: { outcome: 'cancelled' }\n });\n this.callbacks.onRejected?.(requestId, 'cleared');\n }\n this.pending.clear();\n this.config.logger?.debug('Cleared all pending permissions');\n }\n\n /**\n * Update configuration\n */\n updateConfig(config: Partial<PermissionManagerConfig>): void {\n this.config = { ...this.config, ...config };\n }\n}\n","/**\n * Question Manager for Streamable HTTP ACP Client\n * Handles ask_followup_question requests with timeout support\n */\n\nimport type {\n QuestionRequest,\n QuestionResponse,\n} from '../types.js';\nimport { DEFAULT_QUESTION_TIMEOUT } from './constants.js';\nimport { TimeoutError } from './errors.js';\nimport type { Logger } from './types.js';\n\n// Re-export for convenience\nexport type { QuestionRequest, QuestionResponse };\n\n/**\n * Pending question state\n */\ninterface PendingQuestion {\n request: QuestionRequest;\n resolve: (response: QuestionResponse) => void;\n reject: (error: Error) => void;\n timeoutId?: ReturnType<typeof setTimeout>;\n createdAt: number;\n}\n\n/**\n * Configuration for QuestionManager\n */\nexport interface QuestionManagerConfig {\n /** Default timeout for question requests (ms) */\n timeout?: number;\n /** Auto-cancel requests on timeout */\n autoCancelOnTimeout?: boolean;\n /** Logger instance */\n logger?: Logger;\n}\n\n/**\n * User's answers to questions\n */\nexport type QuestionAnswers = Record<string, string | string[]>;\n\n/**\n * Event callbacks for question events\n */\nexport interface QuestionEventCallbacks {\n onRequest?: (toolCallId: string, request: QuestionRequest) => void;\n onAnswered?: (toolCallId: string, answers: QuestionAnswers) => void;\n onCancelled?: (toolCallId: string, reason?: string) => void;\n onTimeout?: (toolCallId: string) => void;\n}\n\n/**\n * Manages question requests from the agent (ask_followup_question tool)\n */\nexport class QuestionManager {\n private pending = new Map<string, PendingQuestion>();\n private config: QuestionManagerConfig;\n private callbacks: QuestionEventCallbacks = {};\n\n constructor(config: QuestionManagerConfig = {}) {\n this.config = {\n timeout: DEFAULT_QUESTION_TIMEOUT,\n autoCancelOnTimeout: true,\n ...config\n };\n }\n\n /**\n * Set event callbacks\n */\n setCallbacks(callbacks: QuestionEventCallbacks): void {\n this.callbacks = callbacks;\n }\n\n /**\n * Handle a question request from the agent\n * Called when receiving _codebuddy.ai/question extMethod\n */\n async handleRequest(request: QuestionRequest): Promise<QuestionResponse> {\n const toolCallId = request.toolCallId;\n\n this.config.logger?.debug(`Question request received: ${toolCallId}`);\n\n return new Promise<QuestionResponse>((resolve, reject) => {\n const pending: PendingQuestion = {\n request,\n resolve,\n reject,\n createdAt: Date.now()\n };\n\n // Set up timeout\n const timeout = request.timeout ?? this.config.timeout;\n if (timeout && timeout > 0) {\n pending.timeoutId = setTimeout(() => {\n this.handleTimeout(toolCallId);\n }, timeout);\n }\n\n this.pending.set(toolCallId, pending);\n\n // Notify callback\n this.callbacks.onRequest?.(toolCallId, request);\n });\n }\n\n /**\n * Handle timeout for a question request\n */\n private handleTimeout(toolCallId: string): void {\n const pending = this.pending.get(toolCallId);\n if (!pending) {return;}\n\n this.config.logger?.warn(`Question request timed out: ${toolCallId}`);\n this.callbacks.onTimeout?.(toolCallId);\n\n if (this.config.autoCancelOnTimeout) {\n pending.resolve({\n outcome: 'cancelled',\n reason: 'timeout'\n });\n } else {\n pending.reject(new TimeoutError('question', this.config.timeout ?? DEFAULT_QUESTION_TIMEOUT));\n }\n\n this.pending.delete(toolCallId);\n }\n\n /**\n * Answer a question request with user's selections\n * Returns true if the request was found and answered\n */\n answer(toolCallId: string, answers: QuestionAnswers): boolean {\n const pending = this.pending.get(toolCallId);\n if (!pending) {\n this.config.logger?.warn(`Question request not found: ${toolCallId}`);\n return false;\n }\n\n // Clear timeout\n if (pending.timeoutId) {\n clearTimeout(pending.timeoutId);\n }\n\n this.config.logger?.debug(`Question answered: ${toolCallId}`);\n\n pending.resolve({\n outcome: 'submitted',\n answers\n });\n\n this.pending.delete(toolCallId);\n this.callbacks.onAnswered?.(toolCallId, answers);\n\n return true;\n }\n\n /**\n * Cancel a question request\n * Returns true if the request was found and cancelled\n */\n cancel(toolCallId: string, reason?: string): boolean {\n const pending = this.pending.get(toolCallId);\n if (!pending) {\n this.config.logger?.warn(`Question request not found: ${toolCallId}`);\n return false;\n }\n\n // Clear timeout\n if (pending.timeoutId) {\n clearTimeout(pending.timeoutId);\n }\n\n this.config.logger?.debug(`Question cancelled: ${toolCallId}${reason ? ` - ${reason}` : ''}`);\n\n pending.resolve({\n outcome: 'cancelled',\n reason\n });\n\n this.pending.delete(toolCallId);\n this.callbacks.onCancelled?.(toolCallId, reason);\n\n return true;\n }\n\n /**\n * Get all pending question requests\n */\n getPending(): Map<string, { request: QuestionRequest; createdAt: number }> {\n const result = new Map<string, { request: QuestionRequest; createdAt: number }>();\n for (const [id, pending] of this.pending) {\n result.set(id, {\n request: pending.request,\n createdAt: pending.createdAt\n });\n }\n return result;\n }\n\n /**\n * Get a specific pending question request\n */\n getPendingById(toolCallId: string): { request: QuestionRequest; createdAt: number } | undefined {\n const pending = this.pending.get(toolCallId);\n if (!pending) {return undefined;}\n return {\n request: pending.request,\n createdAt: pending.createdAt\n };\n }\n\n /**\n * Check if there are any pending question requests\n */\n hasPending(): boolean {\n return this.pending.size > 0;\n }\n\n /**\n * Get the count of pending question requests\n */\n get pendingCount(): number {\n return this.pending.size;\n }\n\n /**\n * Clear all pending question requests (cancel all)\n */\n clear(): void {\n for (const [toolCallId, pending] of this.pending) {\n if (pending.timeoutId) {\n clearTimeout(pending.timeoutId);\n }\n pending.resolve({\n outcome: 'cancelled',\n reason: 'cleared'\n });\n this.callbacks.onCancelled?.(toolCallId, 'cleared');\n }\n this.pending.clear();\n this.config.logger?.debug('Cleared all pending question requests');\n }\n\n /**\n * Update configuration\n */\n updateConfig(config: Partial<QuestionManagerConfig>): void {\n this.config = { ...this.config, ...config };\n }\n}\n","/**\n * Streamable HTTP ACP Client\n * Production-grade client for connecting to cloud-hosted ACP agents\n */\n\nimport {\n type Client,\n ClientSideConnection,\n ContentBlock,\n type InitializeResponse,\n type LoadSessionResponse,\n type NewSessionResponse,\n type PromptResponse,\n PROTOCOL_VERSION,\n type RequestPermissionRequest,\n type RequestPermissionResponse,\n type SessionNotification,\n type SetSessionModelRequest,\n type SetSessionModelResponse,\n type SetSessionModeRequest,\n type SetSessionModeResponse} from '@agentclientprotocol/sdk';\n\nimport { streamableHttp, type StreamableHttpTransport } from '../transport/streamable-http.js';\nimport type {\n ArtifactNotificationParams,\n CheckpointNotificationParams\n} from '../types.js';\nimport { ArtifactManager } from './artifacts.js';\nimport {\n CLOUD_CLIENT_CAPABILITIES,\n DEFAULT_INITIALIZE_TIMEOUT,\n ExtensionMethod\n} from './constants.js';\nimport {\n ConnectionError,\n InitializationError,\n InvalidStateError,\n SessionError\n} from './errors.js';\nimport { EventEmitter } from './events.js';\nimport { ExtensionManager, parseUsageUpdate } from './extensions.js';\nimport { PermissionManager } from './permissions.js';\nimport { type QuestionAnswers,QuestionManager, type QuestionRequest, type QuestionResponse } from './questions.js';\nimport type {\n ClientEvents,\n ClientState,\n PromptOptions,\n StreamableHttpClientOptions,\n} from './types.js';\n\n/**\n * Production-grade Streamable HTTP ACP Client\n *\n * Features:\n * - Full ACP protocol support (initialize, session, prompt, cancel)\n * - Artifact notification handling\n * - Permission handling with timeout support\n * - Extension method support\n * - Type-safe event system\n * - Configurable logging\n */\nexport class StreamableHttpClient {\n private connection!: ClientSideConnection;\n private transport?: StreamableHttpTransport;\n private options: StreamableHttpClientOptions;\n private state: ClientState = 'disconnected';\n private initializeResponse?: InitializeResponse;\n\n // Managers\n private artifactManager: ArtifactManager;\n private permissionManager: PermissionManager;\n private questionManager: QuestionManager;\n private extensionManager: ExtensionManager;\n\n // Event emitter\n private emitter = new EventEmitter<ClientEvents>();\n\n constructor(options: StreamableHttpClientOptions) {\n this.options = options;\n\n // Initialize artifact manager\n this.artifactManager = new ArtifactManager({\n logger: options.logger\n });\n\n // Initialize permission manager\n this.permissionManager = new PermissionManager({\n timeout: options.permissionTimeout,\n autoRejectOnTimeout: options.permissionAutoRejectOnTimeout ?? true,\n autoApprove: options.autoApprove,\n handler: options.requestPermissionHandler,\n logger: options.logger\n });\n\n // Set up permission event callbacks\n this.permissionManager.setCallbacks({\n onRequest: (requestId, params) => {\n this.emitter.emit('permissionRequest', { requestId, params });\n },\n onResolved: (requestId, optionId) => {\n this.emitter.emit('permissionResolved', { requestId, optionId });\n },\n onRejected: (requestId, reason) => {\n this.emitter.emit('permissionRejected', { requestId, reason });\n },\n onTimeout: requestId => {\n this.emitter.emit('permissionTimeout', { requestId });\n }\n });\n\n // Initialize question manager\n this.questionManager = new QuestionManager({\n timeout: options.questionTimeout,\n autoCancelOnTimeout: options.questionAutoCancelOnTimeout ?? true,\n logger: options.logger\n });\n\n // Set up question event callbacks\n this.questionManager.setCallbacks({\n onRequest: (toolCallId: string, request: QuestionRequest) => {\n this.emitter.emit('questionRequest', { toolCallId, request });\n options.onQuestionRequest?.(toolCallId, request);\n },\n onAnswered: (toolCallId: string, answers: QuestionAnswers) => {\n this.emitter.emit('questionAnswered', { toolCallId, answers });\n },\n onCancelled: (toolCallId: string, reason?: string) => {\n this.emitter.emit('questionCancelled', { toolCallId, reason });\n },\n onTimeout: (toolCallId: string) => {\n this.emitter.emit('questionTimeout', { toolCallId });\n }\n });\n\n // Initialize extension manager\n this.extensionManager = new ExtensionManager({\n logger: options.logger\n });\n }\n\n // ============================================\n // State Management\n // ============================================\n\n /**\n * Get current client state\n */\n get currentState(): ClientState {\n return this.state;\n }\n\n /**\n * Check if client is initialized\n */\n get isInitialized(): boolean {\n return this.state === 'initialized';\n }\n\n /**\n * Check if client is connected (but maybe not initialized)\n */\n get isConnected(): boolean {\n return this.state === 'connected' || this.state === 'initialized';\n }\n\n /**\n * Get agent capabilities from initialization response\n */\n get agentCapabilities() {\n return this.initializeResponse?.agentCapabilities;\n }\n\n /**\n * Get full initialization response\n */\n get initializeResult() {\n return this.initializeResponse;\n }\n\n /**\n * Get current transport connection ID\n */\n get connectionId(): string | undefined {\n return this.transport?.connectionId;\n }\n\n private setState(newState: ClientState): void {\n const previous = this.state;\n this.state = newState;\n\n this.options.logger?.debug(`State change: ${previous} -> ${newState}`);\n this.emitter.emit('stateChange', { previous, current: newState });\n\n // Emit specific state events\n switch (newState) {\n case 'connecting':\n this.emitter.emit('connecting', undefined);\n break;\n case 'connected':\n this.emitter.emit('connected', undefined);\n break;\n case 'disconnected':\n this.emitter.emit('disconnected', undefined);\n break;\n case 'error':\n // Error event is emitted separately with the actual error\n break;\n }\n }\n\n // ============================================\n // Connection Management\n // ============================================\n\n /**\n * Connect and initialize the client\n */\n async connect(): Promise<InitializeResponse> {\n if (this.state !== 'disconnected') {\n await this.disconnect();\n }\n if (this.state === 'initialized') {\n return this.initializeResponse!;\n }\n\n if (this.state === 'connecting') {\n throw new ConnectionError('Connection already in progress');\n }\n\n this.setState('connecting');\n\n try {\n // Create transport\n this.transport = streamableHttp({\n endpoint: this.options.endpoint,\n authToken: this.options.authToken,\n headers: this.options.headers,\n reconnect: this.options.reconnect,\n fetch: this.options.fetch,\n onConnect: connectionId => {\n this.options.logger?.debug(`Transport connected: ${connectionId}`);\n },\n onDisconnect: connectionId => {\n this.options.logger?.debug(`Transport disconnected: ${connectionId}`);\n },\n onError: error => {\n this.options.logger?.error('Transport error:', error);\n this.emitter.emit('error', error);\n },\n });\n\n // Create connection\n this.connection = new ClientSideConnection(\n () => this.createClientHandler(),\n this.transport\n );\n\n this.setState('connected');\n\n // Initialize protocol\n // Merge client capabilities with provider defaults (provider takes priority)\n const timeout = this.options.initializeTimeout ?? DEFAULT_INITIALIZE_TIMEOUT;\n const mergedCapabilities = {\n ...this.options.clientCapabilities,\n ...CLOUD_CLIENT_CAPABILITIES,\n _meta: {\n ...this.options.clientCapabilities?._meta,\n ...CLOUD_CLIENT_CAPABILITIES._meta\n }\n };\n const initPromise = this.connection.initialize({\n protocolVersion: PROTOCOL_VERSION,\n clientCapabilities: mergedCapabilities\n });\n\n // Apply timeout\n const timeoutPromise = new Promise<never>((_, reject) => {\n setTimeout(() => {\n reject(new InitializationError(`Initialize timed out after ${timeout}ms`));\n }, timeout);\n });\n\n this.initializeResponse = await Promise.race([initPromise, timeoutPromise]);\n this.setState('initialized');\n\n this.options.logger?.info('Client initialized successfully');\n\n return this.initializeResponse;\n } catch (err) {\n this.setState('error');\n const error = err instanceof Error ? err : new Error(String(err));\n this.emitter.emit('error', error);\n\n if (err instanceof InitializationError || err instanceof ConnectionError) {\n throw err;\n }\n throw new ConnectionError('Failed to connect', error);\n }\n }\n\n /**\n * Disconnect the client gracefully\n * Sends DELETE request to server before closing local resources\n */\n async disconnect(): Promise<void> {\n if (this.state === 'disconnected') {\n return;\n }\n\n this.options.logger?.info('Disconnecting client');\n\n // Close transport gracefully (sends DELETE to server)\n if (this.transport) {\n try {\n await this.transport.close();\n } catch (err) {\n this.options.logger?.warn('Error closing transport:', err);\n }\n this.transport = undefined;\n }\n\n // Clear pending permissions\n this.permissionManager.clear();\n\n // Clear pending question requests\n this.questionManager.clear();\n\n // Clear artifacts\n this.artifactManager.clear();\n\n // Reset state\n this.initializeResponse = undefined;\n this.setState('disconnected');\n }\n\n /**\n * Create the client handler for the connection\n */\n private createClientHandler(): Client {\n return {\n sessionUpdate: async (params: SessionNotification) => {\n await this.handleSessionUpdate(params);\n },\n requestPermission: async (params: RequestPermissionRequest) => this.handleRequestPermission(params),\n extNotification: async (method: string, params: Record<string, unknown>) => {\n console.log('[ACP-Client] extNotification callback invoked:', { method, paramsKeys: Object.keys(params) });\n await this.handleExtNotification(method, params);\n },\n extMethod: async (method: string, params: Record<string, unknown>): Promise<Record<string, unknown>> => {\n // SDK uses generic Record types, cast at boundary to our typed interfaces\n const response = await this.handleExtMethod(method, params as unknown as QuestionRequest);\n return response as unknown as Record<string, unknown>;\n }\n };\n }\n\n // ============================================\n // Session Management\n // ============================================\n\n /**\n * Create a new session\n */\n async createSession(cwd: string): Promise<NewSessionResponse> {\n this.ensureInitialized('createSession');\n\n try {\n const response = await this.connection.newSession({\n cwd,\n mcpServers: []\n });\n\n this.options.logger?.info(`Session created: ${response.sessionId}`);\n return response;\n } catch (err) {\n throw new SessionError(\n `Failed to create session: ${err instanceof Error ? err.message : String(err)}`,\n undefined,\n err instanceof Error ? err : undefined\n );\n }\n }\n\n /**\n * Load an existing session\n * Requires agent to support loadSession capability\n */\n async loadSession(sessionId: string, cwd: string): Promise<LoadSessionResponse> {\n this.ensureInitialized('loadSession');\n\n if (!this.agentCapabilities?.loadSession) {\n throw new SessionError('Agent does not support session loading', sessionId);\n }\n\n try {\n const response = await this.connection.loadSession({\n sessionId,\n cwd,\n mcpServers: []\n });\n\n this.options.logger?.info(`Session loaded: ${sessionId}`);\n return response;\n } catch (err) {\n throw new SessionError(\n `Failed to load session: ${err instanceof Error ? err.message : String(err)}`,\n sessionId,\n err instanceof Error ? err : undefined\n );\n }\n }\n\n /**\n * Set the session mode\n */\n async setSessionMode(params: SetSessionModeRequest): Promise<SetSessionModeResponse> {\n this.ensureInitialized('setSessionMode');\n\n this.options.logger?.debug(`Setting session mode: ${params.sessionId} -> ${params.modeId}`);\n\n return this.connection.setSessionMode(params);\n }\n\n /**\n * Set the session model\n * @experimental This API is unstable and may change\n */\n async setSessionModel(params: SetSessionModelRequest): Promise<SetSessionModelResponse> {\n this.ensureInitialized('setSessionModel');\n\n this.options.logger?.debug(`Setting session model: ${params.sessionId} -> ${params.modelId}`);\n\n return this.connection.unstable_setSessionModel(params);\n }\n\n // ============================================\n // Prompt\n // ============================================\n\n /**\n * Send a prompt to the agent\n */\n async prompt(\n sessionId: string,\n prompt: ContentBlock[],\n options?: PromptOptions\n ): Promise<PromptResponse> {\n this.ensureInitialized('prompt');\n\n this.options.logger?.debug(`Sending prompt to session: ${sessionId}`);\n\n return this.connection.prompt({\n sessionId,\n prompt,\n _meta: options?.planMode ? { planMode: true, ...options._meta } : options?._meta\n });\n }\n\n /**\n * Cancel ongoing operations for a session\n */\n async cancel(sessionId: string): Promise<void> {\n this.ensureInitialized('cancel');\n\n this.options.logger?.debug(`Cancelling session: ${sessionId}`);\n\n await this.connection.cancel({ sessionId });\n }\n\n // ============================================\n // Permission Management\n // ============================================\n\n /**\n * Resolve a pending permission request\n */\n resolvePermission(requestId: string, optionId: string): boolean {\n return this.permissionManager.resolve(requestId, optionId);\n }\n\n /**\n * Reject a pending permission request\n */\n rejectPermission(requestId: string, reason?: string): boolean {\n return this.permissionManager.reject(requestId, reason);\n }\n\n /**\n * Get all pending permissions\n */\n getPendingPermissions() {\n return this.permissionManager.getPending();\n }\n\n /**\n * Check if there are pending permissions\n */\n hasPendingPermissions(): boolean {\n return this.permissionManager.hasPending();\n }\n\n // ============================================\n // Question Management (ask_followup_question)\n // ============================================\n\n /**\n * Answer a pending question request with user's selections\n */\n answerQuestion(toolCallId: string, answers: QuestionAnswers): boolean {\n return this.questionManager.answer(toolCallId, answers);\n }\n\n /**\n * Cancel a pending question request\n */\n cancelQuestion(toolCallId: string, reason?: string): boolean {\n return this.questionManager.cancel(toolCallId, reason);\n }\n\n /**\n * Get all pending question requests\n */\n getPendingQuestions() {\n return this.questionManager.getPending();\n }\n\n /**\n * Check if there are pending question requests\n */\n hasPendingQuestions(): boolean {\n return this.questionManager.hasPending();\n }\n\n // ============================================\n // Extension Methods\n // ============================================\n\n /**\n * Send an extension method request\n */\n async extMethod(method: string, params: Record<string, unknown>): Promise<Record<string, unknown>> {\n this.ensureInitialized('extMethod');\n return this.connection.extMethod(method, params);\n }\n\n /**\n * Send an extension notification\n */\n async extNotification(method: string, params: Record<string, unknown>): Promise<void> {\n this.ensureInitialized('extNotification');\n return this.connection.extNotification(method, params);\n }\n\n // ============================================\n // Event Emitter Implementation\n // ============================================\n\n on<K extends keyof ClientEvents>(\n event: K,\n listener: (data: ClientEvents[K]) => void | Promise<void>\n ): this {\n this.emitter.on(event, listener);\n return this;\n }\n\n off<K extends keyof ClientEvents>(\n event: K,\n listener: (data: ClientEvents[K]) => void | Promise<void>\n ): this {\n this.emitter.off(event, listener);\n return this;\n }\n\n once<K extends keyof ClientEvents>(\n event: K,\n listener: (data: ClientEvents[K]) => void | Promise<void>\n ): this {\n this.emitter.once(event, listener);\n return this;\n }\n\n emit<K extends keyof ClientEvents>(event: K, data: ClientEvents[K]): boolean {\n return this.emitter.emit(event, data);\n }\n\n removeAllListeners<K extends keyof ClientEvents>(event?: K): this {\n this.emitter.removeAllListeners(event);\n return this;\n }\n\n // ============================================\n // Internal Handlers\n // ============================================\n\n private async handleSessionUpdate(params: SessionNotification): Promise<void> {\n // Forward to callback\n await this.options.onSessionUpdate?.(params);\n\n // Emit event\n this.emitter.emit('sessionUpdate', params);\n }\n\n private async handleRequestPermission(\n params: RequestPermissionRequest\n ): Promise<RequestPermissionResponse> {\n return this.permissionManager.handleRequest(params);\n }\n\n private async handleExtNotification(\n method: string,\n params: Record<string, unknown>\n ): Promise<void> {\n // Handle artifact notifications\n if (method === ExtensionMethod.ARTIFACT) {\n const notification = params as unknown as ArtifactNotificationParams;\n const artifactData = notification.artifact as any;\n console.log('[ACP-Client] Received artifact notification:', {\n event: notification.event,\n artifactUri: artifactData?.uri,\n artifactType: artifactData?.type,\n });\n\n // For deleted events, get full artifact before deletion for callbacks\n if (notification.event === 'deleted') {\n const existing = this.artifactManager.get(notification.artifact.uri);\n this.artifactManager.handleNotification(notification);\n if (existing) {\n await this.options.onArtifact?.(existing, 'deleted');\n this.emitter.emit('artifactDeleted', existing);\n }\n } else {\n const { artifact, event } = notification;\n // Let artifactManager handle the notification\n this.artifactManager.handleNotification(notification);\n // Get the artifact from manager or use the original\n const storedArtifact = this.artifactManager.get(artifact.uri) || artifact;\n\n console.log('[ACP-Client] Stored artifact:', {\n event,\n artifactUri: storedArtifact.uri,\n artifactType: storedArtifact.type,\n hasText: storedArtifact.type === 'plan' ? !!(storedArtifact as any).text : undefined,\n });\n\n await this.options.onArtifact?.(storedArtifact, event);\n\n if (event === 'created') {\n console.log('[ACP-Client] Emitting artifactCreated event');\n this.emitter.emit('artifactCreated', storedArtifact);\n if (storedArtifact.type === 'plan') {\n await this.options.onPlanReady?.(storedArtifact);\n }\n } else {\n console.log('[ACP-Client] Emitting artifactUpdated event');\n this.emitter.emit('artifactUpdated', storedArtifact);\n }\n }\n return;\n }\n\n // Handle usage update notifications\n if (method === ExtensionMethod.USAGE) {\n const usage = parseUsageUpdate(params);\n await this.options.onUsageUpdate?.(usage);\n this.emitter.emit('usageUpdate', usage);\n return;\n }\n\n // Handle checkpoint notifications\n if (method === ExtensionMethod.CHECKPOINT) {\n const notification = params as unknown as CheckpointNotificationParams;\n if (notification.event === 'created') {\n this.emitter.emit('checkpointCreated', notification.checkpoint);\n } else if (notification.event === 'updated') {\n this.emitter.emit('checkpointUpdated', notification.checkpoint);\n }\n return;\n }\n\n // Forward other extensions to callback and manager\n await this.options.onExtNotification?.(method, params);\n await this.extensionManager.handleNotification(method, params);\n }\n\n private async handleExtMethod(\n method: string,\n params: QuestionRequest\n ): Promise<QuestionResponse> {\n // Handle question requests (ask_followup_question via question extMethod)\n if (method === ExtensionMethod.QUESTION) {\n return this.questionManager.handleRequest(params);\n }\n\n // Unknown extension method\n this.options.logger?.warn(`Unknown extension method: ${method}`);\n return { outcome: 'cancelled', reason: 'unknown method' };\n }\n\n // ============================================\n // Helpers\n // ============================================\n\n private ensureInitialized(operation: string): void {\n if (this.state !== 'initialized') {\n throw new InvalidStateError(operation, this.state, ['initialized']);\n }\n }\n}\n\nexport default StreamableHttpClient;\n","/**\n * Cloud Agent Connection\n * Wraps StreamableHttpClient to implement AgentConnection interface\n */\n\nimport type {\n ContentBlock,\n InitializeResponse,\n LoadSessionResponse,\n NewSessionResponse,\n PromptResponse,\n RequestPermissionRequest,\n SessionNotification,\n SetSessionModelResponse,\n SetSessionModeResponse\n} from '@agentclientprotocol/sdk';\nimport {\n StreamableHttpClient\n} from '@genie/agent-client-protocol';\n\nimport type { SessionConnectionInfo } from '../../client/types.js';\nimport type {\n AgentCapabilities,\n AgentConnection,\n AgentStatus,\n CloudConnectionConfig,\n ConnectionEventListener,\n ConnectionEvents,\n CreateSessionParams,\n LoadSessionParams,\n PromptParams\n} from '../../types.js';\n\n/**\n * Cloud Agent Connection implementation\n * Uses Streamable HTTP transport to connect to cloud-hosted ACP agents\n * Uses composition pattern - implements event emitter methods internally\n *\n * TODO: Connection Lifecycle Responsibilities\n * CloudAgentProvider caches connections by endpoint link. This class needs to:\n * - Implement connection health checks (detect and handle connection failures/reconnection)\n * - Handle token expiration (refresh or re-authentication when tokens expire)\n * - Emit 'disconnected' event when connection becomes unhealthy so provider can clean up cache\n */\nexport class CloudAgentConnection implements AgentConnection {\n private client: StreamableHttpClient;\n private listeners: Map<keyof ConnectionEvents, Set<ConnectionEventListener<unknown>>> = new Map();\n private onceListeners: Map<keyof ConnectionEvents, Set<ConnectionEventListener<unknown>>> = new Map();\n\n /**\n * Flag to suppress sessionUpdate event emission during streaming.\n * When true, onSessionUpdate callback won't emit to avoid duplicate messages with promptStream.\n */\n private _isStreaming = false;\n\n /**\n * Session connection information (sandboxId, link, token, etc.)\n * Set by CloudAgentProvider.connect() after fetching session data from backend.\n */\n private _sessionConnectionInfo?: SessionConnectionInfo;\n\n readonly agentId: string;\n readonly transport = 'cloud' as const;\n readonly cwd: string;\n\n constructor(agentId: string, config: CloudConnectionConfig, cwd: string = '/workspace') {\n this.agentId = agentId;\n this.cwd = cwd;\n\n // Create the underlying StreamableHttpClient\n this.client = new StreamableHttpClient({\n endpoint: config.endpoint,\n authToken: config.authToken,\n headers: config.headers,\n reconnect: config.reconnect,\n initializeTimeout: config.initializeTimeout,\n permissionTimeout: config.permissionTimeout,\n permissionAutoRejectOnTimeout: config.permissionAutoRejectOnTimeout,\n autoApprove: config.autoApprove,\n logger: config.logger,\n fetch: config.fetch,\n clientCapabilities: config.clientCapabilities,\n // Forward events to our emitter (suppressed during streaming to avoid duplicates)\n onSessionUpdate: update => {\n if (!this._isStreaming) {\n this.emit('sessionUpdate', update);\n }\n },\n onArtifact: (artifact, event) => {\n console.log('[CloudConnection] onArtifact callback:', {\n event,\n artifactUri: artifact.uri,\n artifactType: artifact.type,\n });\n if (event === 'created') {\n this.emit('artifactCreated', artifact);\n } else if (event === 'updated') {\n this.emit('artifactUpdated', artifact);\n } else if (event === 'deleted') {\n this.emit('artifactDeleted', artifact);\n }\n },\n onUsageUpdate: usage => {\n this.emit('usageUpdate', usage);\n }\n });\n\n // Forward client events\n this.setupEventForwarding();\n }\n\n private setupEventForwarding(): void {\n // Forward connection state events\n this.client.on('connecting', () => { this.emit('connecting', undefined); });\n this.client.on('connected', () => { this.emit('connected', undefined); });\n this.client.on('disconnected', () => { this.emit('disconnected', undefined); });\n this.client.on('error', error => { this.emit('error', error); });\n this.client.on('stateChange', change => { this.emit('stateChange', change); });\n\n // Forward permission events\n this.client.on('permissionRequest', data => { this.emit('permissionRequest', data); });\n this.client.on('permissionResolved', data => { this.emit('permissionResolved', data); });\n this.client.on('permissionRejected', data => { this.emit('permissionRejected', data); });\n this.client.on('permissionTimeout', data => { this.emit('permissionTimeout', data); });\n\n // Forward question events\n this.client.on('questionRequest', data => { this.emit('questionRequest', data); });\n this.client.on('questionAnswered', data => { this.emit('questionAnswered', data); });\n this.client.on('questionCancelled', data => { this.emit('questionCancelled', data); });\n this.client.on('questionTimeout', data => { this.emit('questionTimeout', data); });\n\n // Forward checkpoint events\n this.client.on('checkpointCreated', checkpoint => { this.emit('checkpointCreated', checkpoint); });\n this.client.on('checkpointUpdated', checkpoint => { this.emit('checkpointUpdated', checkpoint); });\n }\n\n // ============================================\n // Event Emitter Implementation\n // ============================================\n\n on<K extends keyof ConnectionEvents>(event: K, listener: ConnectionEventListener<ConnectionEvents[K]>): this {\n if (!this.listeners.has(event)) {\n this.listeners.set(event, new Set());\n }\n this.listeners.get(event)!.add(listener as ConnectionEventListener<unknown>);\n return this;\n }\n\n off<K extends keyof ConnectionEvents>(event: K, listener: ConnectionEventListener<ConnectionEvents[K]>): this {\n const eventListeners = this.listeners.get(event);\n if (eventListeners) {\n eventListeners.delete(listener as ConnectionEventListener<unknown>);\n }\n const onceEventListeners = this.onceListeners.get(event);\n if (onceEventListeners) {\n onceEventListeners.delete(listener as ConnectionEventListener<unknown>);\n }\n return this;\n }\n\n once<K extends keyof ConnectionEvents>(event: K, listener: ConnectionEventListener<ConnectionEvents[K]>): this {\n if (!this.onceListeners.has(event)) {\n this.onceListeners.set(event, new Set());\n }\n this.onceListeners.get(event)!.add(listener as ConnectionEventListener<unknown>);\n return this;\n }\n\n emit<K extends keyof ConnectionEvents>(event: K, data: ConnectionEvents[K]): boolean {\n const regularListeners = this.listeners.get(event);\n const onceEventListeners = this.onceListeners.get(event);\n\n let hasListeners = false;\n\n // Call regular listeners\n if (regularListeners && regularListeners.size > 0) {\n hasListeners = true;\n for (const listener of regularListeners) {\n try {\n const result = listener(data);\n if (result instanceof Promise) {\n result.catch(err => {\n console.error(`Error in async event listener for '${String(event)}':`, err);\n });\n }\n } catch (err) {\n console.error(`Error in event listener for '${String(event)}':`, err);\n }\n }\n }\n\n // Call once listeners and remove them\n if (onceEventListeners && onceEventListeners.size > 0) {\n hasListeners = true;\n const listenersToCall = Array.from(onceEventListeners);\n this.onceListeners.delete(event);\n\n for (const listener of listenersToCall) {\n try {\n const result = listener(data);\n if (result instanceof Promise) {\n result.catch(err => {\n console.error(`Error in async once event listener for '${String(event)}':`, err);\n });\n }\n } catch (err) {\n console.error(`Error in once event listener for '${String(event)}':`, err);\n }\n }\n }\n\n return hasListeners;\n }\n\n removeAllListeners<K extends keyof ConnectionEvents>(event?: K): this {\n if (event !== undefined) {\n this.listeners.delete(event);\n this.onceListeners.delete(event);\n } else {\n this.listeners.clear();\n this.onceListeners.clear();\n }\n return this;\n }\n\n // ============================================\n // State Properties\n // ============================================\n\n get state(): AgentStatus {\n return this.client.currentState as AgentStatus;\n }\n\n get isInitialized(): boolean {\n return this.client.isInitialized;\n }\n\n get capabilities(): AgentCapabilities | undefined {\n return this.client.agentCapabilities as AgentCapabilities | undefined;\n }\n\n get initializeResult(): InitializeResponse | undefined {\n return this.client.initializeResult;\n }\n\n // ============================================\n // Connection Lifecycle\n // ============================================\n\n async connect(): Promise<InitializeResponse> {\n return this.client.connect();\n }\n\n async disconnect(): Promise<void> {\n await this.client.disconnect();\n }\n\n // ============================================\n // Session Management\n // ============================================\n\n async createSession(params: CreateSessionParams): Promise<NewSessionResponse> {\n // Cloud Side does not support creating new sessions directly\n // Use this.cwd (provided by CloudAgentProvider) as the working directory\n // params.cwd is ignored because cloud agents already have a fixed cwd from backend\n const loadedSession = await this.client.loadSession(this.agentId, this.cwd);\n return { ...loadedSession, sessionId: this.agentId };\n }\n\n async loadSession(params: LoadSessionParams): Promise<LoadSessionResponse> {\n if (!params.sessionId) {\n throw new Error('sessionId is required for loadSession');\n }\n return this.client.loadSession(params.sessionId, this.cwd);\n }\n\n async setSessionMode(sessionId: string, modeId: string): Promise<SetSessionModeResponse> {\n return this.client.setSessionMode({ sessionId, modeId });\n }\n\n async setSessionModel(sessionId: string, modelId: string): Promise<SetSessionModelResponse> {\n return this.client.setSessionModel({ sessionId, modelId });\n }\n\n // ============================================\n // Prompt Operations\n // ============================================\n\n async prompt(sessionId: string, params: PromptParams): Promise<PromptResponse> {\n const blocks =\n typeof params.content === 'string'\n ? [{ type: 'text', text: params.content }] as ContentBlock[]\n : params.content as ContentBlock[];\n\n return this.client.prompt(sessionId, blocks, {\n planMode: params.planMode,\n _meta: params._meta\n });\n }\n\n async *promptStream(sessionId: string, params: PromptParams): AsyncIterable<SessionNotification> {\n // Suppress sessionUpdate event emission during streaming to avoid duplicates\n this._isStreaming = true;\n\n // For streaming, we need to collect updates via the event system\n const updates: SessionNotification[] = [];\n let resolveUpdate: ((value: SessionNotification | null) => void) | null = null;\n let done = false;\n\n const listener = (update: SessionNotification) => {\n if (resolveUpdate) {\n resolveUpdate(update);\n resolveUpdate = null;\n } else {\n updates.push(update);\n }\n };\n\n this.client.on('sessionUpdate', listener);\n\n try {\n // Start the prompt (non-blocking)\n const promptPromise = this.prompt(sessionId, params);\n\n // Yield updates as they come in\n while (!done) {\n const update = updates.shift();\n if (update) {\n yield update;\n } else {\n // Wait for next update or prompt completion\n const nextUpdate = await new Promise<SessionNotification | null>(resolve => {\n resolveUpdate = resolve;\n // Check if prompt completed while we were setting up\n promptPromise.then(() => {\n if (resolveUpdate === resolve) {\n resolveUpdate = null;\n resolve(null);\n }\n }).catch(() => {\n if (resolveUpdate === resolve) {\n resolveUpdate = null;\n resolve(null);\n }\n });\n });\n\n if (nextUpdate === null) {\n done = true;\n } else {\n yield nextUpdate;\n }\n }\n }\n } finally {\n this._isStreaming = false;\n this.client.off('sessionUpdate', listener);\n }\n }\n\n async cancel(sessionId: string): Promise<void> {\n return this.client.cancel(sessionId);\n }\n\n // ============================================\n // Permission Management\n // ============================================\n\n resolvePermission(requestId: string, optionId: string): boolean {\n return this.client.resolvePermission(requestId, optionId);\n }\n\n rejectPermission(requestId: string, reason?: string): boolean {\n return this.client.rejectPermission(requestId, reason);\n }\n\n getPendingPermissions(): Map<string, { params: RequestPermissionRequest; createdAt: number }> {\n return this.client.getPendingPermissions();\n }\n\n hasPendingPermissions(): boolean {\n return this.client.hasPendingPermissions();\n }\n\n // ============================================\n // Question Management (ask_followup_question)\n // ============================================\n\n answerQuestion(toolCallId: string, answers: import('@genie/agent-client-protocol').QuestionAnswers): boolean {\n return this.client.answerQuestion(toolCallId, answers);\n }\n\n cancelQuestion(toolCallId: string, reason?: string): boolean {\n return this.client.cancelQuestion(toolCallId, reason);\n }\n\n getPendingQuestions() {\n return this.client.getPendingQuestions();\n }\n\n hasPendingQuestions(): boolean {\n return this.client.hasPendingQuestions();\n }\n\n // ============================================\n // Tool Callback Management\n // ============================================\n\n async toolCallback(sessionId: string, toolCallId: string, toolName: string, action: 'skip' | 'cancel'): Promise<{ success: boolean; error?: string }> {\n // Cloud connection does not support toolCallback yet\n return { success: false, error: 'toolCallback not supported for cloud connections' };\n }\n\n // ============================================\n // Session Connection Info\n // ============================================\n\n /**\n * Set session connection information\n * Called by CloudAgentProvider.connect() after fetching session data from backend.\n */\n setSessionConnectionInfo(info: SessionConnectionInfo): void {\n this._sessionConnectionInfo = info;\n }\n\n /**\n * Get session connection information\n * Contains sandboxId, link, token, etc.\n */\n get sessionConnectionInfo(): SessionConnectionInfo | undefined {\n return this._sessionConnectionInfo;\n }\n\n // ============================================\n // Extension Methods\n // ============================================\n\n async extMethod(method: string, params: Record<string, unknown>): Promise<Record<string, unknown>> {\n return this.client.extMethod(method, params);\n }\n}\n\nexport default CloudAgentConnection;\n","'use strict';\n\nexport default function bind(fn, thisArg) {\n return function wrap() {\n return fn.apply(thisArg, arguments);\n };\n}\n","'use strict';\n\nimport bind from './helpers/bind.js';\n\n// utils is a library of generic helper functions non-specific to axios\n\nconst {toString} = Object.prototype;\nconst {getPrototypeOf} = Object;\nconst {iterator, toStringTag} = Symbol;\n\nconst kindOf = (cache => thing => {\n const str = toString.call(thing);\n return cache[str] || (cache[str] = str.slice(8, -1).toLowerCase());\n})(Object.create(null));\n\nconst kindOfTest = (type) => {\n type = type.toLowerCase();\n return (thing) => kindOf(thing) === type\n}\n\nconst typeOfTest = type => thing => typeof thing === type;\n\n/**\n * Determine if a value is an Array\n *\n * @param {Object} val The value to test\n *\n * @returns {boolean} True if value is an Array, otherwise false\n */\nconst {isArray} = Array;\n\n/**\n * Determine if a value is undefined\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if the value is undefined, otherwise false\n */\nconst isUndefined = typeOfTest('undefined');\n\n/**\n * Determine if a value is a Buffer\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is a Buffer, otherwise false\n */\nfunction isBuffer(val) {\n return val !== null && !isUndefined(val) && val.constructor !== null && !isUndefined(val.constructor)\n && isFunction(val.constructor.isBuffer) && val.constructor.isBuffer(val);\n}\n\n/**\n * Determine if a value is an ArrayBuffer\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is an ArrayBuffer, otherwise false\n */\nconst isArrayBuffer = kindOfTest('ArrayBuffer');\n\n\n/**\n * Determine if a value is a view on an ArrayBuffer\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is a view on an ArrayBuffer, otherwise false\n */\nfunction isArrayBufferView(val) {\n let result;\n if ((typeof ArrayBuffer !== 'undefined') && (ArrayBuffer.isView)) {\n result = ArrayBuffer.isView(val);\n } else {\n result = (val) && (val.buffer) && (isArrayBuffer(val.buffer));\n }\n return result;\n}\n\n/**\n * Determine if a value is a String\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is a String, otherwise false\n */\nconst isString = typeOfTest('string');\n\n/**\n * Determine if a value is a Function\n *\n * @param {*} val The value to test\n * @returns {boolean} True if value is a Function, otherwise false\n */\nconst isFunction = typeOfTest('function');\n\n/**\n * Determine if a value is a Number\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is a Number, otherwise false\n */\nconst isNumber = typeOfTest('number');\n\n/**\n * Determine if a value is an Object\n *\n * @param {*} thing The value to test\n *\n * @returns {boolean} True if value is an Object, otherwise false\n */\nconst isObject = (thing) => thing !== null && typeof thing === 'object';\n\n/**\n * Determine if a value is a Boolean\n *\n * @param {*} thing The value to test\n * @returns {boolean} True if value is a Boolean, otherwise false\n */\nconst isBoolean = thing => thing === true || thing === false;\n\n/**\n * Determine if a value is a plain Object\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is a plain Object, otherwise false\n */\nconst isPlainObject = (val) => {\n if (kindOf(val) !== 'object') {\n return false;\n }\n\n const prototype = getPrototypeOf(val);\n return (prototype === null || prototype === Object.prototype || Object.getPrototypeOf(prototype) === null) && !(toStringTag in val) && !(iterator in val);\n}\n\n/**\n * Determine if a value is a Date\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is a Date, otherwise false\n */\nconst isDate = kindOfTest('Date');\n\n/**\n * Determine if a value is a File\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is a File, otherwise false\n */\nconst isFile = kindOfTest('File');\n\n/**\n * Determine if a value is a Blob\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is a Blob, otherwise false\n */\nconst isBlob = kindOfTest('Blob');\n\n/**\n * Determine if a value is a FileList\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is a File, otherwise false\n */\nconst isFileList = kindOfTest('FileList');\n\n/**\n * Determine if a value is a Stream\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is a Stream, otherwise false\n */\nconst isStream = (val) => isObject(val) && isFunction(val.pipe);\n\n/**\n * Determine if a value is a FormData\n *\n * @param {*} thing The value to test\n *\n * @returns {boolean} True if value is an FormData, otherwise false\n */\nconst isFormData = (thing) => {\n let kind;\n return thing && (\n (typeof FormData === 'function' && thing instanceof FormData) || (\n isFunction(thing.append) && (\n (kind = kindOf(thing)) === 'formdata' ||\n // detect form-data instance\n (kind === 'object' && isFunction(thing.toString) && thing.toString() === '[object FormData]')\n )\n )\n )\n}\n\n/**\n * Determine if a value is a URLSearchParams object\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is a URLSearchParams object, otherwise false\n */\nconst isURLSearchParams = kindOfTest('URLSearchParams');\n\nconst [isReadableStream, isRequest, isResponse, isHeaders] = ['ReadableStream', 'Request', 'Response', 'Headers'].map(kindOfTest);\n\n/**\n * Trim excess whitespace off the beginning and end of a string\n *\n * @param {String} str The String to trim\n *\n * @returns {String} The String freed of excess whitespace\n */\nconst trim = (str) => str.trim ?\n str.trim() : str.replace(/^[\\s\\uFEFF\\xA0]+|[\\s\\uFEFF\\xA0]+$/g, '');\n\n/**\n * Iterate over an Array or an Object invoking a function for each item.\n *\n * If `obj` is an Array callback will be called passing\n * the value, index, and complete array for each item.\n *\n * If 'obj' is an Object callback will be called passing\n * the value, key, and complete object for each property.\n *\n * @param {Object|Array} obj The object to iterate\n * @param {Function} fn The callback to invoke for each item\n *\n * @param {Boolean} [allOwnKeys = false]\n * @returns {any}\n */\nfunction forEach(obj, fn, {allOwnKeys = false} = {}) {\n // Don't bother if no value provided\n if (obj === null || typeof obj === 'undefined') {\n return;\n }\n\n let i;\n let l;\n\n // Force an array if not already something iterable\n if (typeof obj !== 'object') {\n /*eslint no-param-reassign:0*/\n obj = [obj];\n }\n\n if (isArray(obj)) {\n // Iterate over array values\n for (i = 0, l = obj.length; i < l; i++) {\n fn.call(null, obj[i], i, obj);\n }\n } else {\n // Iterate over object keys\n const keys = allOwnKeys ? Object.getOwnPropertyNames(obj) : Object.keys(obj);\n const len = keys.length;\n let key;\n\n for (i = 0; i < len; i++) {\n key = keys[i];\n fn.call(null, obj[key], key, obj);\n }\n }\n}\n\nfunction findKey(obj, key) {\n key = key.toLowerCase();\n const keys = Object.keys(obj);\n let i = keys.length;\n let _key;\n while (i-- > 0) {\n _key = keys[i];\n if (key === _key.toLowerCase()) {\n return _key;\n }\n }\n return null;\n}\n\nconst _global = (() => {\n /*eslint no-undef:0*/\n if (typeof globalThis !== \"undefined\") return globalThis;\n return typeof self !== \"undefined\" ? self : (typeof window !== 'undefined' ? window : global)\n})();\n\nconst isContextDefined = (context) => !isUndefined(context) && context !== _global;\n\n/**\n * Accepts varargs expecting each argument to be an object, then\n * immutably merges the properties of each object and returns result.\n *\n * When multiple objects contain the same key the later object in\n * the arguments list will take precedence.\n *\n * Example:\n *\n * ```js\n * var result = merge({foo: 123}, {foo: 456});\n * console.log(result.foo); // outputs 456\n * ```\n *\n * @param {Object} obj1 Object to merge\n *\n * @returns {Object} Result of all merge properties\n */\nfunction merge(/* obj1, obj2, obj3, ... */) {\n const {caseless} = isContextDefined(this) && this || {};\n const result = {};\n const assignValue = (val, key) => {\n const targetKey = caseless && findKey(result, key) || key;\n if (isPlainObject(result[targetKey]) && isPlainObject(val)) {\n result[targetKey] = merge(result[targetKey], val);\n } else if (isPlainObject(val)) {\n result[targetKey] = merge({}, val);\n } else if (isArray(val)) {\n result[targetKey] = val.slice();\n } else {\n result[targetKey] = val;\n }\n }\n\n for (let i = 0, l = arguments.length; i < l; i++) {\n arguments[i] && forEach(arguments[i], assignValue);\n }\n return result;\n}\n\n/**\n * Extends object a by mutably adding to it the properties of object b.\n *\n * @param {Object} a The object to be extended\n * @param {Object} b The object to copy properties from\n * @param {Object} thisArg The object to bind function to\n *\n * @param {Boolean} [allOwnKeys]\n * @returns {Object} The resulting value of object a\n */\nconst extend = (a, b, thisArg, {allOwnKeys}= {}) => {\n forEach(b, (val, key) => {\n if (thisArg && isFunction(val)) {\n a[key] = bind(val, thisArg);\n } else {\n a[key] = val;\n }\n }, {allOwnKeys});\n return a;\n}\n\n/**\n * Remove byte order marker. This catches EF BB BF (the UTF-8 BOM)\n *\n * @param {string} content with BOM\n *\n * @returns {string} content value without BOM\n */\nconst stripBOM = (content) => {\n if (content.charCodeAt(0) === 0xFEFF) {\n content = content.slice(1);\n }\n return content;\n}\n\n/**\n * Inherit the prototype methods from one constructor into another\n * @param {function} constructor\n * @param {function} superConstructor\n * @param {object} [props]\n * @param {object} [descriptors]\n *\n * @returns {void}\n */\nconst inherits = (constructor, superConstructor, props, descriptors) => {\n constructor.prototype = Object.create(superConstructor.prototype, descriptors);\n constructor.prototype.constructor = constructor;\n Object.defineProperty(constructor, 'super', {\n value: superConstructor.prototype\n });\n props && Object.assign(constructor.prototype, props);\n}\n\n/**\n * Resolve object with deep prototype chain to a flat object\n * @param {Object} sourceObj source object\n * @param {Object} [destObj]\n * @param {Function|Boolean} [filter]\n * @param {Function} [propFilter]\n *\n * @returns {Object}\n */\nconst toFlatObject = (sourceObj, destObj, filter, propFilter) => {\n let props;\n let i;\n let prop;\n const merged = {};\n\n destObj = destObj || {};\n // eslint-disable-next-line no-eq-null,eqeqeq\n if (sourceObj == null) return destObj;\n\n do {\n props = Object.getOwnPropertyNames(sourceObj);\n i = props.length;\n while (i-- > 0) {\n prop = props[i];\n if ((!propFilter || propFilter(prop, sourceObj, destObj)) && !merged[prop]) {\n destObj[prop] = sourceObj[prop];\n merged[prop] = true;\n }\n }\n sourceObj = filter !== false && getPrototypeOf(sourceObj);\n } while (sourceObj && (!filter || filter(sourceObj, destObj)) && sourceObj !== Object.prototype);\n\n return destObj;\n}\n\n/**\n * Determines whether a string ends with the characters of a specified string\n *\n * @param {String} str\n * @param {String} searchString\n * @param {Number} [position= 0]\n *\n * @returns {boolean}\n */\nconst endsWith = (str, searchString, position) => {\n str = String(str);\n if (position === undefined || position > str.length) {\n position = str.length;\n }\n position -= searchString.length;\n const lastIndex = str.indexOf(searchString, position);\n return lastIndex !== -1 && lastIndex === position;\n}\n\n\n/**\n * Returns new array from array like object or null if failed\n *\n * @param {*} [thing]\n *\n * @returns {?Array}\n */\nconst toArray = (thing) => {\n if (!thing) return null;\n if (isArray(thing)) return thing;\n let i = thing.length;\n if (!isNumber(i)) return null;\n const arr = new Array(i);\n while (i-- > 0) {\n arr[i] = thing[i];\n }\n return arr;\n}\n\n/**\n * Checking if the Uint8Array exists and if it does, it returns a function that checks if the\n * thing passed in is an instance of Uint8Array\n *\n * @param {TypedArray}\n *\n * @returns {Array}\n */\n// eslint-disable-next-line func-names\nconst isTypedArray = (TypedArray => {\n // eslint-disable-next-line func-names\n return thing => {\n return TypedArray && thing instanceof TypedArray;\n };\n})(typeof Uint8Array !== 'undefined' && getPrototypeOf(Uint8Array));\n\n/**\n * For each entry in the object, call the function with the key and value.\n *\n * @param {Object<any, any>} obj - The object to iterate over.\n * @param {Function} fn - The function to call for each entry.\n *\n * @returns {void}\n */\nconst forEachEntry = (obj, fn) => {\n const generator = obj && obj[iterator];\n\n const _iterator = generator.call(obj);\n\n let result;\n\n while ((result = _iterator.next()) && !result.done) {\n const pair = result.value;\n fn.call(obj, pair[0], pair[1]);\n }\n}\n\n/**\n * It takes a regular expression and a string, and returns an array of all the matches\n *\n * @param {string} regExp - The regular expression to match against.\n * @param {string} str - The string to search.\n *\n * @returns {Array<boolean>}\n */\nconst matchAll = (regExp, str) => {\n let matches;\n const arr = [];\n\n while ((matches = regExp.exec(str)) !== null) {\n arr.push(matches);\n }\n\n return arr;\n}\n\n/* Checking if the kindOfTest function returns true when passed an HTMLFormElement. */\nconst isHTMLForm = kindOfTest('HTMLFormElement');\n\nconst toCamelCase = str => {\n return str.toLowerCase().replace(/[-_\\s]([a-z\\d])(\\w*)/g,\n function replacer(m, p1, p2) {\n return p1.toUpperCase() + p2;\n }\n );\n};\n\n/* Creating a function that will check if an object has a property. */\nconst hasOwnProperty = (({hasOwnProperty}) => (obj, prop) => hasOwnProperty.call(obj, prop))(Object.prototype);\n\n/**\n * Determine if a value is a RegExp object\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is a RegExp object, otherwise false\n */\nconst isRegExp = kindOfTest('RegExp');\n\nconst reduceDescriptors = (obj, reducer) => {\n const descriptors = Object.getOwnPropertyDescriptors(obj);\n const reducedDescriptors = {};\n\n forEach(descriptors, (descriptor, name) => {\n let ret;\n if ((ret = reducer(descriptor, name, obj)) !== false) {\n reducedDescriptors[name] = ret || descriptor;\n }\n });\n\n Object.defineProperties(obj, reducedDescriptors);\n}\n\n/**\n * Makes all methods read-only\n * @param {Object} obj\n */\n\nconst freezeMethods = (obj) => {\n reduceDescriptors(obj, (descriptor, name) => {\n // skip restricted props in strict mode\n if (isFunction(obj) && ['arguments', 'caller', 'callee'].indexOf(name) !== -1) {\n return false;\n }\n\n const value = obj[name];\n\n if (!isFunction(value)) return;\n\n descriptor.enumerable = false;\n\n if ('writable' in descriptor) {\n descriptor.writable = false;\n return;\n }\n\n if (!descriptor.set) {\n descriptor.set = () => {\n throw Error('Can not rewrite read-only method \\'' + name + '\\'');\n };\n }\n });\n}\n\nconst toObjectSet = (arrayOrString, delimiter) => {\n const obj = {};\n\n const define = (arr) => {\n arr.forEach(value => {\n obj[value] = true;\n });\n }\n\n isArray(arrayOrString) ? define(arrayOrString) : define(String(arrayOrString).split(delimiter));\n\n return obj;\n}\n\nconst noop = () => {}\n\nconst toFiniteNumber = (value, defaultValue) => {\n return value != null && Number.isFinite(value = +value) ? value : defaultValue;\n}\n\n/**\n * If the thing is a FormData object, return true, otherwise return false.\n *\n * @param {unknown} thing - The thing to check.\n *\n * @returns {boolean}\n */\nfunction isSpecCompliantForm(thing) {\n return !!(thing && isFunction(thing.append) && thing[toStringTag] === 'FormData' && thing[iterator]);\n}\n\nconst toJSONObject = (obj) => {\n const stack = new Array(10);\n\n const visit = (source, i) => {\n\n if (isObject(source)) {\n if (stack.indexOf(source) >= 0) {\n return;\n }\n\n if(!('toJSON' in source)) {\n stack[i] = source;\n const target = isArray(source) ? [] : {};\n\n forEach(source, (value, key) => {\n const reducedValue = visit(value, i + 1);\n !isUndefined(reducedValue) && (target[key] = reducedValue);\n });\n\n stack[i] = undefined;\n\n return target;\n }\n }\n\n return source;\n }\n\n return visit(obj, 0);\n}\n\nconst isAsyncFn = kindOfTest('AsyncFunction');\n\nconst isThenable = (thing) =>\n thing && (isObject(thing) || isFunction(thing)) && isFunction(thing.then) && isFunction(thing.catch);\n\n// original code\n// https://github.com/DigitalBrainJS/AxiosPromise/blob/16deab13710ec09779922131f3fa5954320f83ab/lib/utils.js#L11-L34\n\nconst _setImmediate = ((setImmediateSupported, postMessageSupported) => {\n if (setImmediateSupported) {\n return setImmediate;\n }\n\n return postMessageSupported ? ((token, callbacks) => {\n _global.addEventListener(\"message\", ({source, data}) => {\n if (source === _global && data === token) {\n callbacks.length && callbacks.shift()();\n }\n }, false);\n\n return (cb) => {\n callbacks.push(cb);\n _global.postMessage(token, \"*\");\n }\n })(`axios@${Math.random()}`, []) : (cb) => setTimeout(cb);\n})(\n typeof setImmediate === 'function',\n isFunction(_global.postMessage)\n);\n\nconst asap = typeof queueMicrotask !== 'undefined' ?\n queueMicrotask.bind(_global) : ( typeof process !== 'undefined' && process.nextTick || _setImmediate);\n\n// *********************\n\n\nconst isIterable = (thing) => thing != null && isFunction(thing[iterator]);\n\n\nexport default {\n isArray,\n isArrayBuffer,\n isBuffer,\n isFormData,\n isArrayBufferView,\n isString,\n isNumber,\n isBoolean,\n isObject,\n isPlainObject,\n isReadableStream,\n isRequest,\n isResponse,\n isHeaders,\n isUndefined,\n isDate,\n isFile,\n isBlob,\n isRegExp,\n isFunction,\n isStream,\n isURLSearchParams,\n isTypedArray,\n isFileList,\n forEach,\n merge,\n extend,\n trim,\n stripBOM,\n inherits,\n toFlatObject,\n kindOf,\n kindOfTest,\n endsWith,\n toArray,\n forEachEntry,\n matchAll,\n isHTMLForm,\n hasOwnProperty,\n hasOwnProp: hasOwnProperty, // an alias to avoid ESLint no-prototype-builtins detection\n reduceDescriptors,\n freezeMethods,\n toObjectSet,\n toCamelCase,\n noop,\n toFiniteNumber,\n findKey,\n global: _global,\n isContextDefined,\n isSpecCompliantForm,\n toJSONObject,\n isAsyncFn,\n isThenable,\n setImmediate: _setImmediate,\n asap,\n isIterable\n};\n","'use strict';\n\nimport utils from '../utils.js';\n\n/**\n * Create an Error with the specified message, config, error code, request and response.\n *\n * @param {string} message The error message.\n * @param {string} [code] The error code (for example, 'ECONNABORTED').\n * @param {Object} [config] The config.\n * @param {Object} [request] The request.\n * @param {Object} [response] The response.\n *\n * @returns {Error} The created error.\n */\nfunction AxiosError(message, code, config, request, response) {\n Error.call(this);\n\n if (Error.captureStackTrace) {\n Error.captureStackTrace(this, this.constructor);\n } else {\n this.stack = (new Error()).stack;\n }\n\n this.message = message;\n this.name = 'AxiosError';\n code && (this.code = code);\n config && (this.config = config);\n request && (this.request = request);\n if (response) {\n this.response = response;\n this.status = response.status ? response.status : null;\n }\n}\n\nutils.inherits(AxiosError, Error, {\n toJSON: function toJSON() {\n return {\n // Standard\n message: this.message,\n name: this.name,\n // Microsoft\n description: this.description,\n number: this.number,\n // Mozilla\n fileName: this.fileName,\n lineNumber: this.lineNumber,\n columnNumber: this.columnNumber,\n stack: this.stack,\n // Axios\n config: utils.toJSONObject(this.config),\n code: this.code,\n status: this.status\n };\n }\n});\n\nconst prototype = AxiosError.prototype;\nconst descriptors = {};\n\n[\n 'ERR_BAD_OPTION_VALUE',\n 'ERR_BAD_OPTION',\n 'ECONNABORTED',\n 'ETIMEDOUT',\n 'ERR_NETWORK',\n 'ERR_FR_TOO_MANY_REDIRECTS',\n 'ERR_DEPRECATED',\n 'ERR_BAD_RESPONSE',\n 'ERR_BAD_REQUEST',\n 'ERR_CANCELED',\n 'ERR_NOT_SUPPORT',\n 'ERR_INVALID_URL'\n// eslint-disable-next-line func-names\n].forEach(code => {\n descriptors[code] = {value: code};\n});\n\nObject.defineProperties(AxiosError, descriptors);\nObject.defineProperty(prototype, 'isAxiosError', {value: true});\n\n// eslint-disable-next-line func-names\nAxiosError.from = (error, code, config, request, response, customProps) => {\n const axiosError = Object.create(prototype);\n\n utils.toFlatObject(error, axiosError, function filter(obj) {\n return obj !== Error.prototype;\n }, prop => {\n return prop !== 'isAxiosError';\n });\n\n AxiosError.call(axiosError, error.message, code, config, request, response);\n\n axiosError.cause = error;\n\n axiosError.name = error.name;\n\n customProps && Object.assign(axiosError, customProps);\n\n return axiosError;\n};\n\nexport default AxiosError;\n","// eslint-disable-next-line strict\nexport default null;\n","'use strict';\n\nimport utils from '../utils.js';\nimport AxiosError from '../core/AxiosError.js';\n// temporary hotfix to avoid circular references until AxiosURLSearchParams is refactored\nimport PlatformFormData from '../platform/node/classes/FormData.js';\n\n/**\n * Determines if the given thing is a array or js object.\n *\n * @param {string} thing - The object or array to be visited.\n *\n * @returns {boolean}\n */\nfunction isVisitable(thing) {\n return utils.isPlainObject(thing) || utils.isArray(thing);\n}\n\n/**\n * It removes the brackets from the end of a string\n *\n * @param {string} key - The key of the parameter.\n *\n * @returns {string} the key without the brackets.\n */\nfunction removeBrackets(key) {\n return utils.endsWith(key, '[]') ? key.slice(0, -2) : key;\n}\n\n/**\n * It takes a path, a key, and a boolean, and returns a string\n *\n * @param {string} path - The path to the current key.\n * @param {string} key - The key of the current object being iterated over.\n * @param {string} dots - If true, the key will be rendered with dots instead of brackets.\n *\n * @returns {string} The path to the current key.\n */\nfunction renderKey(path, key, dots) {\n if (!path) return key;\n return path.concat(key).map(function each(token, i) {\n // eslint-disable-next-line no-param-reassign\n token = removeBrackets(token);\n return !dots && i ? '[' + token + ']' : token;\n }).join(dots ? '.' : '');\n}\n\n/**\n * If the array is an array and none of its elements are visitable, then it's a flat array.\n *\n * @param {Array<any>} arr - The array to check\n *\n * @returns {boolean}\n */\nfunction isFlatArray(arr) {\n return utils.isArray(arr) && !arr.some(isVisitable);\n}\n\nconst predicates = utils.toFlatObject(utils, {}, null, function filter(prop) {\n return /^is[A-Z]/.test(prop);\n});\n\n/**\n * Convert a data object to FormData\n *\n * @param {Object} obj\n * @param {?Object} [formData]\n * @param {?Object} [options]\n * @param {Function} [options.visitor]\n * @param {Boolean} [options.metaTokens = true]\n * @param {Boolean} [options.dots = false]\n * @param {?Boolean} [options.indexes = false]\n *\n * @returns {Object}\n **/\n\n/**\n * It converts an object into a FormData object\n *\n * @param {Object<any, any>} obj - The object to convert to form data.\n * @param {string} formData - The FormData object to append to.\n * @param {Object<string, any>} options\n *\n * @returns\n */\nfunction toFormData(obj, formData, options) {\n if (!utils.isObject(obj)) {\n throw new TypeError('target must be an object');\n }\n\n // eslint-disable-next-line no-param-reassign\n formData = formData || new (PlatformFormData || FormData)();\n\n // eslint-disable-next-line no-param-reassign\n options = utils.toFlatObject(options, {\n metaTokens: true,\n dots: false,\n indexes: false\n }, false, function defined(option, source) {\n // eslint-disable-next-line no-eq-null,eqeqeq\n return !utils.isUndefined(source[option]);\n });\n\n const metaTokens = options.metaTokens;\n // eslint-disable-next-line no-use-before-define\n const visitor = options.visitor || defaultVisitor;\n const dots = options.dots;\n const indexes = options.indexes;\n const _Blob = options.Blob || typeof Blob !== 'undefined' && Blob;\n const useBlob = _Blob && utils.isSpecCompliantForm(formData);\n\n if (!utils.isFunction(visitor)) {\n throw new TypeError('visitor must be a function');\n }\n\n function convertValue(value) {\n if (value === null) return '';\n\n if (utils.isDate(value)) {\n return value.toISOString();\n }\n\n if (utils.isBoolean(value)) {\n return value.toString();\n }\n\n if (!useBlob && utils.isBlob(value)) {\n throw new AxiosError('Blob is not supported. Use a Buffer instead.');\n }\n\n if (utils.isArrayBuffer(value) || utils.isTypedArray(value)) {\n return useBlob && typeof Blob === 'function' ? new Blob([value]) : Buffer.from(value);\n }\n\n return value;\n }\n\n /**\n * Default visitor.\n *\n * @param {*} value\n * @param {String|Number} key\n * @param {Array<String|Number>} path\n * @this {FormData}\n *\n * @returns {boolean} return true to visit the each prop of the value recursively\n */\n function defaultVisitor(value, key, path) {\n let arr = value;\n\n if (value && !path && typeof value === 'object') {\n if (utils.endsWith(key, '{}')) {\n // eslint-disable-next-line no-param-reassign\n key = metaTokens ? key : key.slice(0, -2);\n // eslint-disable-next-line no-param-reassign\n value = JSON.stringify(value);\n } else if (\n (utils.isArray(value) && isFlatArray(value)) ||\n ((utils.isFileList(value) || utils.endsWith(key, '[]')) && (arr = utils.toArray(value))\n )) {\n // eslint-disable-next-line no-param-reassign\n key = removeBrackets(key);\n\n arr.forEach(function each(el, index) {\n !(utils.isUndefined(el) || el === null) && formData.append(\n // eslint-disable-next-line no-nested-ternary\n indexes === true ? renderKey([key], index, dots) : (indexes === null ? key : key + '[]'),\n convertValue(el)\n );\n });\n return false;\n }\n }\n\n if (isVisitable(value)) {\n return true;\n }\n\n formData.append(renderKey(path, key, dots), convertValue(value));\n\n return false;\n }\n\n const stack = [];\n\n const exposedHelpers = Object.assign(predicates, {\n defaultVisitor,\n convertValue,\n isVisitable\n });\n\n function build(value, path) {\n if (utils.isUndefined(value)) return;\n\n if (stack.indexOf(value) !== -1) {\n throw Error('Circular reference detected in ' + path.join('.'));\n }\n\n stack.push(value);\n\n utils.forEach(value, function each(el, key) {\n const result = !(utils.isUndefined(el) || el === null) && visitor.call(\n formData, el, utils.isString(key) ? key.trim() : key, path, exposedHelpers\n );\n\n if (result === true) {\n build(el, path ? path.concat(key) : [key]);\n }\n });\n\n stack.pop();\n }\n\n if (!utils.isObject(obj)) {\n throw new TypeError('data must be an object');\n }\n\n build(obj);\n\n return formData;\n}\n\nexport default toFormData;\n","'use strict';\n\nimport toFormData from './toFormData.js';\n\n/**\n * It encodes a string by replacing all characters that are not in the unreserved set with\n * their percent-encoded equivalents\n *\n * @param {string} str - The string to encode.\n *\n * @returns {string} The encoded string.\n */\nfunction encode(str) {\n const charMap = {\n '!': '%21',\n \"'\": '%27',\n '(': '%28',\n ')': '%29',\n '~': '%7E',\n '%20': '+',\n '%00': '\\x00'\n };\n return encodeURIComponent(str).replace(/[!'()~]|%20|%00/g, function replacer(match) {\n return charMap[match];\n });\n}\n\n/**\n * It takes a params object and converts it to a FormData object\n *\n * @param {Object<string, any>} params - The parameters to be converted to a FormData object.\n * @param {Object<string, any>} options - The options object passed to the Axios constructor.\n *\n * @returns {void}\n */\nfunction AxiosURLSearchParams(params, options) {\n this._pairs = [];\n\n params && toFormData(params, this, options);\n}\n\nconst prototype = AxiosURLSearchParams.prototype;\n\nprototype.append = function append(name, value) {\n this._pairs.push([name, value]);\n};\n\nprototype.toString = function toString(encoder) {\n const _encode = encoder ? function(value) {\n return encoder.call(this, value, encode);\n } : encode;\n\n return this._pairs.map(function each(pair) {\n return _encode(pair[0]) + '=' + _encode(pair[1]);\n }, '').join('&');\n};\n\nexport default AxiosURLSearchParams;\n","'use strict';\n\nimport utils from '../utils.js';\nimport AxiosURLSearchParams from '../helpers/AxiosURLSearchParams.js';\n\n/**\n * It replaces all instances of the characters `:`, `$`, `,`, `+`, `[`, and `]` with their\n * URI encoded counterparts\n *\n * @param {string} val The value to be encoded.\n *\n * @returns {string} The encoded value.\n */\nfunction encode(val) {\n return encodeURIComponent(val).\n replace(/%3A/gi, ':').\n replace(/%24/g, '$').\n replace(/%2C/gi, ',').\n replace(/%20/g, '+').\n replace(/%5B/gi, '[').\n replace(/%5D/gi, ']');\n}\n\n/**\n * Build a URL by appending params to the end\n *\n * @param {string} url The base of the url (e.g., http://www.google.com)\n * @param {object} [params] The params to be appended\n * @param {?(object|Function)} options\n *\n * @returns {string} The formatted url\n */\nexport default function buildURL(url, params, options) {\n /*eslint no-param-reassign:0*/\n if (!params) {\n return url;\n }\n \n const _encode = options && options.encode || encode;\n\n if (utils.isFunction(options)) {\n options = {\n serialize: options\n };\n } \n\n const serializeFn = options && options.serialize;\n\n let serializedParams;\n\n if (serializeFn) {\n serializedParams = serializeFn(params, options);\n } else {\n serializedParams = utils.isURLSearchParams(params) ?\n params.toString() :\n new AxiosURLSearchParams(params, options).toString(_encode);\n }\n\n if (serializedParams) {\n const hashmarkIndex = url.indexOf(\"#\");\n\n if (hashmarkIndex !== -1) {\n url = url.slice(0, hashmarkIndex);\n }\n url += (url.indexOf('?') === -1 ? '?' : '&') + serializedParams;\n }\n\n return url;\n}\n","'use strict';\n\nimport utils from './../utils.js';\n\nclass InterceptorManager {\n constructor() {\n this.handlers = [];\n }\n\n /**\n * Add a new interceptor to the stack\n *\n * @param {Function} fulfilled The function to handle `then` for a `Promise`\n * @param {Function} rejected The function to handle `reject` for a `Promise`\n *\n * @return {Number} An ID used to remove interceptor later\n */\n use(fulfilled, rejected, options) {\n this.handlers.push({\n fulfilled,\n rejected,\n synchronous: options ? options.synchronous : false,\n runWhen: options ? options.runWhen : null\n });\n return this.handlers.length - 1;\n }\n\n /**\n * Remove an interceptor from the stack\n *\n * @param {Number} id The ID that was returned by `use`\n *\n * @returns {Boolean} `true` if the interceptor was removed, `false` otherwise\n */\n eject(id) {\n if (this.handlers[id]) {\n this.handlers[id] = null;\n }\n }\n\n /**\n * Clear all interceptors from the stack\n *\n * @returns {void}\n */\n clear() {\n if (this.handlers) {\n this.handlers = [];\n }\n }\n\n /**\n * Iterate over all the registered interceptors\n *\n * This method is particularly useful for skipping over any\n * interceptors that may have become `null` calling `eject`.\n *\n * @param {Function} fn The function to call for each interceptor\n *\n * @returns {void}\n */\n forEach(fn) {\n utils.forEach(this.handlers, function forEachHandler(h) {\n if (h !== null) {\n fn(h);\n }\n });\n }\n}\n\nexport default InterceptorManager;\n","'use strict';\n\nexport default {\n silentJSONParsing: true,\n forcedJSONParsing: true,\n clarifyTimeoutError: false\n};\n","'use strict';\n\nimport AxiosURLSearchParams from '../../../helpers/AxiosURLSearchParams.js';\nexport default typeof URLSearchParams !== 'undefined' ? URLSearchParams : AxiosURLSearchParams;\n","'use strict';\n\nexport default typeof FormData !== 'undefined' ? FormData : null;\n","'use strict'\n\nexport default typeof Blob !== 'undefined' ? Blob : null\n","import URLSearchParams from './classes/URLSearchParams.js'\nimport FormData from './classes/FormData.js'\nimport Blob from './classes/Blob.js'\n\nexport default {\n isBrowser: true,\n classes: {\n URLSearchParams,\n FormData,\n Blob\n },\n protocols: ['http', 'https', 'file', 'blob', 'url', 'data']\n};\n","const hasBrowserEnv = typeof window !== 'undefined' && typeof document !== 'undefined';\n\nconst _navigator = typeof navigator === 'object' && navigator || undefined;\n\n/**\n * Determine if we're running in a standard browser environment\n *\n * This allows axios to run in a web worker, and react-native.\n * Both environments support XMLHttpRequest, but not fully standard globals.\n *\n * web workers:\n * typeof window -> undefined\n * typeof document -> undefined\n *\n * react-native:\n * navigator.product -> 'ReactNative'\n * nativescript\n * navigator.product -> 'NativeScript' or 'NS'\n *\n * @returns {boolean}\n */\nconst hasStandardBrowserEnv = hasBrowserEnv &&\n (!_navigator || ['ReactNative', 'NativeScript', 'NS'].indexOf(_navigator.product) < 0);\n\n/**\n * Determine if we're running in a standard browser webWorker environment\n *\n * Although the `isStandardBrowserEnv` method indicates that\n * `allows axios to run in a web worker`, the WebWorker will still be\n * filtered out due to its judgment standard\n * `typeof window !== 'undefined' && typeof document !== 'undefined'`.\n * This leads to a problem when axios post `FormData` in webWorker\n */\nconst hasStandardBrowserWebWorkerEnv = (() => {\n return (\n typeof WorkerGlobalScope !== 'undefined' &&\n // eslint-disable-next-line no-undef\n self instanceof WorkerGlobalScope &&\n typeof self.importScripts === 'function'\n );\n})();\n\nconst origin = hasBrowserEnv && window.location.href || 'http://localhost';\n\nexport {\n hasBrowserEnv,\n hasStandardBrowserWebWorkerEnv,\n hasStandardBrowserEnv,\n _navigator as navigator,\n origin\n}\n","import platform from './node/index.js';\nimport * as utils from './common/utils.js';\n\nexport default {\n ...utils,\n ...platform\n}\n","'use strict';\n\nimport utils from '../utils.js';\nimport toFormData from './toFormData.js';\nimport platform from '../platform/index.js';\n\nexport default function toURLEncodedForm(data, options) {\n return toFormData(data, new platform.classes.URLSearchParams(), Object.assign({\n visitor: function(value, key, path, helpers) {\n if (platform.isNode && utils.isBuffer(value)) {\n this.append(key, value.toString('base64'));\n return false;\n }\n\n return helpers.defaultVisitor.apply(this, arguments);\n }\n }, options));\n}\n","'use strict';\n\nimport utils from '../utils.js';\n\n/**\n * It takes a string like `foo[x][y][z]` and returns an array like `['foo', 'x', 'y', 'z']\n *\n * @param {string} name - The name of the property to get.\n *\n * @returns An array of strings.\n */\nfunction parsePropPath(name) {\n // foo[x][y][z]\n // foo.x.y.z\n // foo-x-y-z\n // foo x y z\n return utils.matchAll(/\\w+|\\[(\\w*)]/g, name).map(match => {\n return match[0] === '[]' ? '' : match[1] || match[0];\n });\n}\n\n/**\n * Convert an array to an object.\n *\n * @param {Array<any>} arr - The array to convert to an object.\n *\n * @returns An object with the same keys and values as the array.\n */\nfunction arrayToObject(arr) {\n const obj = {};\n const keys = Object.keys(arr);\n let i;\n const len = keys.length;\n let key;\n for (i = 0; i < len; i++) {\n key = keys[i];\n obj[key] = arr[key];\n }\n return obj;\n}\n\n/**\n * It takes a FormData object and returns a JavaScript object\n *\n * @param {string} formData The FormData object to convert to JSON.\n *\n * @returns {Object<string, any> | null} The converted object.\n */\nfunction formDataToJSON(formData) {\n function buildPath(path, value, target, index) {\n let name = path[index++];\n\n if (name === '__proto__') return true;\n\n const isNumericKey = Number.isFinite(+name);\n const isLast = index >= path.length;\n name = !name && utils.isArray(target) ? target.length : name;\n\n if (isLast) {\n if (utils.hasOwnProp(target, name)) {\n target[name] = [target[name], value];\n } else {\n target[name] = value;\n }\n\n return !isNumericKey;\n }\n\n if (!target[name] || !utils.isObject(target[name])) {\n target[name] = [];\n }\n\n const result = buildPath(path, value, target[name], index);\n\n if (result && utils.isArray(target[name])) {\n target[name] = arrayToObject(target[name]);\n }\n\n return !isNumericKey;\n }\n\n if (utils.isFormData(formData) && utils.isFunction(formData.entries)) {\n const obj = {};\n\n utils.forEachEntry(formData, (name, value) => {\n buildPath(parsePropPath(name), value, obj, 0);\n });\n\n return obj;\n }\n\n return null;\n}\n\nexport default formDataToJSON;\n","'use strict';\n\nimport utils from '../utils.js';\nimport AxiosError from '../core/AxiosError.js';\nimport transitionalDefaults from './transitional.js';\nimport toFormData from '../helpers/toFormData.js';\nimport toURLEncodedForm from '../helpers/toURLEncodedForm.js';\nimport platform from '../platform/index.js';\nimport formDataToJSON from '../helpers/formDataToJSON.js';\n\n/**\n * It takes a string, tries to parse it, and if it fails, it returns the stringified version\n * of the input\n *\n * @param {any} rawValue - The value to be stringified.\n * @param {Function} parser - A function that parses a string into a JavaScript object.\n * @param {Function} encoder - A function that takes a value and returns a string.\n *\n * @returns {string} A stringified version of the rawValue.\n */\nfunction stringifySafely(rawValue, parser, encoder) {\n if (utils.isString(rawValue)) {\n try {\n (parser || JSON.parse)(rawValue);\n return utils.trim(rawValue);\n } catch (e) {\n if (e.name !== 'SyntaxError') {\n throw e;\n }\n }\n }\n\n return (encoder || JSON.stringify)(rawValue);\n}\n\nconst defaults = {\n\n transitional: transitionalDefaults,\n\n adapter: ['xhr', 'http', 'fetch'],\n\n transformRequest: [function transformRequest(data, headers) {\n const contentType = headers.getContentType() || '';\n const hasJSONContentType = contentType.indexOf('application/json') > -1;\n const isObjectPayload = utils.isObject(data);\n\n if (isObjectPayload && utils.isHTMLForm(data)) {\n data = new FormData(data);\n }\n\n const isFormData = utils.isFormData(data);\n\n if (isFormData) {\n return hasJSONContentType ? JSON.stringify(formDataToJSON(data)) : data;\n }\n\n if (utils.isArrayBuffer(data) ||\n utils.isBuffer(data) ||\n utils.isStream(data) ||\n utils.isFile(data) ||\n utils.isBlob(data) ||\n utils.isReadableStream(data)\n ) {\n return data;\n }\n if (utils.isArrayBufferView(data)) {\n return data.buffer;\n }\n if (utils.isURLSearchParams(data)) {\n headers.setContentType('application/x-www-form-urlencoded;charset=utf-8', false);\n return data.toString();\n }\n\n let isFileList;\n\n if (isObjectPayload) {\n if (contentType.indexOf('application/x-www-form-urlencoded') > -1) {\n return toURLEncodedForm(data, this.formSerializer).toString();\n }\n\n if ((isFileList = utils.isFileList(data)) || contentType.indexOf('multipart/form-data') > -1) {\n const _FormData = this.env && this.env.FormData;\n\n return toFormData(\n isFileList ? {'files[]': data} : data,\n _FormData && new _FormData(),\n this.formSerializer\n );\n }\n }\n\n if (isObjectPayload || hasJSONContentType ) {\n headers.setContentType('application/json', false);\n return stringifySafely(data);\n }\n\n return data;\n }],\n\n transformResponse: [function transformResponse(data) {\n const transitional = this.transitional || defaults.transitional;\n const forcedJSONParsing = transitional && transitional.forcedJSONParsing;\n const JSONRequested = this.responseType === 'json';\n\n if (utils.isResponse(data) || utils.isReadableStream(data)) {\n return data;\n }\n\n if (data && utils.isString(data) && ((forcedJSONParsing && !this.responseType) || JSONRequested)) {\n const silentJSONParsing = transitional && transitional.silentJSONParsing;\n const strictJSONParsing = !silentJSONParsing && JSONRequested;\n\n try {\n return JSON.parse(data);\n } catch (e) {\n if (strictJSONParsing) {\n if (e.name === 'SyntaxError') {\n throw AxiosError.from(e, AxiosError.ERR_BAD_RESPONSE, this, null, this.response);\n }\n throw e;\n }\n }\n }\n\n return data;\n }],\n\n /**\n * A timeout in milliseconds to abort a request. If set to 0 (default) a\n * timeout is not created.\n */\n timeout: 0,\n\n xsrfCookieName: 'XSRF-TOKEN',\n xsrfHeaderName: 'X-XSRF-TOKEN',\n\n maxContentLength: -1,\n maxBodyLength: -1,\n\n env: {\n FormData: platform.classes.FormData,\n Blob: platform.classes.Blob\n },\n\n validateStatus: function validateStatus(status) {\n return status >= 200 && status < 300;\n },\n\n headers: {\n common: {\n 'Accept': 'application/json, text/plain, */*',\n 'Content-Type': undefined\n }\n }\n};\n\nutils.forEach(['delete', 'get', 'head', 'post', 'put', 'patch'], (method) => {\n defaults.headers[method] = {};\n});\n\nexport default defaults;\n","'use strict';\n\nimport utils from './../utils.js';\n\n// RawAxiosHeaders whose duplicates are ignored by node\n// c.f. https://nodejs.org/api/http.html#http_message_headers\nconst ignoreDuplicateOf = utils.toObjectSet([\n 'age', 'authorization', 'content-length', 'content-type', 'etag',\n 'expires', 'from', 'host', 'if-modified-since', 'if-unmodified-since',\n 'last-modified', 'location', 'max-forwards', 'proxy-authorization',\n 'referer', 'retry-after', 'user-agent'\n]);\n\n/**\n * Parse headers into an object\n *\n * ```\n * Date: Wed, 27 Aug 2014 08:58:49 GMT\n * Content-Type: application/json\n * Connection: keep-alive\n * Transfer-Encoding: chunked\n * ```\n *\n * @param {String} rawHeaders Headers needing to be parsed\n *\n * @returns {Object} Headers parsed into an object\n */\nexport default rawHeaders => {\n const parsed = {};\n let key;\n let val;\n let i;\n\n rawHeaders && rawHeaders.split('\\n').forEach(function parser(line) {\n i = line.indexOf(':');\n key = line.substring(0, i).trim().toLowerCase();\n val = line.substring(i + 1).trim();\n\n if (!key || (parsed[key] && ignoreDuplicateOf[key])) {\n return;\n }\n\n if (key === 'set-cookie') {\n if (parsed[key]) {\n parsed[key].push(val);\n } else {\n parsed[key] = [val];\n }\n } else {\n parsed[key] = parsed[key] ? parsed[key] + ', ' + val : val;\n }\n });\n\n return parsed;\n};\n","'use strict';\n\nimport utils from '../utils.js';\nimport parseHeaders from '../helpers/parseHeaders.js';\n\nconst $internals = Symbol('internals');\n\nfunction normalizeHeader(header) {\n return header && String(header).trim().toLowerCase();\n}\n\nfunction normalizeValue(value) {\n if (value === false || value == null) {\n return value;\n }\n\n return utils.isArray(value) ? value.map(normalizeValue) : String(value);\n}\n\nfunction parseTokens(str) {\n const tokens = Object.create(null);\n const tokensRE = /([^\\s,;=]+)\\s*(?:=\\s*([^,;]+))?/g;\n let match;\n\n while ((match = tokensRE.exec(str))) {\n tokens[match[1]] = match[2];\n }\n\n return tokens;\n}\n\nconst isValidHeaderName = (str) => /^[-_a-zA-Z0-9^`|~,!#$%&'*+.]+$/.test(str.trim());\n\nfunction matchHeaderValue(context, value, header, filter, isHeaderNameFilter) {\n if (utils.isFunction(filter)) {\n return filter.call(this, value, header);\n }\n\n if (isHeaderNameFilter) {\n value = header;\n }\n\n if (!utils.isString(value)) return;\n\n if (utils.isString(filter)) {\n return value.indexOf(filter) !== -1;\n }\n\n if (utils.isRegExp(filter)) {\n return filter.test(value);\n }\n}\n\nfunction formatHeader(header) {\n return header.trim()\n .toLowerCase().replace(/([a-z\\d])(\\w*)/g, (w, char, str) => {\n return char.toUpperCase() + str;\n });\n}\n\nfunction buildAccessors(obj, header) {\n const accessorName = utils.toCamelCase(' ' + header);\n\n ['get', 'set', 'has'].forEach(methodName => {\n Object.defineProperty(obj, methodName + accessorName, {\n value: function(arg1, arg2, arg3) {\n return this[methodName].call(this, header, arg1, arg2, arg3);\n },\n configurable: true\n });\n });\n}\n\nclass AxiosHeaders {\n constructor(headers) {\n headers && this.set(headers);\n }\n\n set(header, valueOrRewrite, rewrite) {\n const self = this;\n\n function setHeader(_value, _header, _rewrite) {\n const lHeader = normalizeHeader(_header);\n\n if (!lHeader) {\n throw new Error('header name must be a non-empty string');\n }\n\n const key = utils.findKey(self, lHeader);\n\n if(!key || self[key] === undefined || _rewrite === true || (_rewrite === undefined && self[key] !== false)) {\n self[key || _header] = normalizeValue(_value);\n }\n }\n\n const setHeaders = (headers, _rewrite) =>\n utils.forEach(headers, (_value, _header) => setHeader(_value, _header, _rewrite));\n\n if (utils.isPlainObject(header) || header instanceof this.constructor) {\n setHeaders(header, valueOrRewrite)\n } else if(utils.isString(header) && (header = header.trim()) && !isValidHeaderName(header)) {\n setHeaders(parseHeaders(header), valueOrRewrite);\n } else if (utils.isObject(header) && utils.isIterable(header)) {\n let obj = {}, dest, key;\n for (const entry of header) {\n if (!utils.isArray(entry)) {\n throw TypeError('Object iterator must return a key-value pair');\n }\n\n obj[key = entry[0]] = (dest = obj[key]) ?\n (utils.isArray(dest) ? [...dest, entry[1]] : [dest, entry[1]]) : entry[1];\n }\n\n setHeaders(obj, valueOrRewrite)\n } else {\n header != null && setHeader(valueOrRewrite, header, rewrite);\n }\n\n return this;\n }\n\n get(header, parser) {\n header = normalizeHeader(header);\n\n if (header) {\n const key = utils.findKey(this, header);\n\n if (key) {\n const value = this[key];\n\n if (!parser) {\n return value;\n }\n\n if (parser === true) {\n return parseTokens(value);\n }\n\n if (utils.isFunction(parser)) {\n return parser.call(this, value, key);\n }\n\n if (utils.isRegExp(parser)) {\n return parser.exec(value);\n }\n\n throw new TypeError('parser must be boolean|regexp|function');\n }\n }\n }\n\n has(header, matcher) {\n header = normalizeHeader(header);\n\n if (header) {\n const key = utils.findKey(this, header);\n\n return !!(key && this[key] !== undefined && (!matcher || matchHeaderValue(this, this[key], key, matcher)));\n }\n\n return false;\n }\n\n delete(header, matcher) {\n const self = this;\n let deleted = false;\n\n function deleteHeader(_header) {\n _header = normalizeHeader(_header);\n\n if (_header) {\n const key = utils.findKey(self, _header);\n\n if (key && (!matcher || matchHeaderValue(self, self[key], key, matcher))) {\n delete self[key];\n\n deleted = true;\n }\n }\n }\n\n if (utils.isArray(header)) {\n header.forEach(deleteHeader);\n } else {\n deleteHeader(header);\n }\n\n return deleted;\n }\n\n clear(matcher) {\n const keys = Object.keys(this);\n let i = keys.length;\n let deleted = false;\n\n while (i--) {\n const key = keys[i];\n if(!matcher || matchHeaderValue(this, this[key], key, matcher, true)) {\n delete this[key];\n deleted = true;\n }\n }\n\n return deleted;\n }\n\n normalize(format) {\n const self = this;\n const headers = {};\n\n utils.forEach(this, (value, header) => {\n const key = utils.findKey(headers, header);\n\n if (key) {\n self[key] = normalizeValue(value);\n delete self[header];\n return;\n }\n\n const normalized = format ? formatHeader(header) : String(header).trim();\n\n if (normalized !== header) {\n delete self[header];\n }\n\n self[normalized] = normalizeValue(value);\n\n headers[normalized] = true;\n });\n\n return this;\n }\n\n concat(...targets) {\n return this.constructor.concat(this, ...targets);\n }\n\n toJSON(asStrings) {\n const obj = Object.create(null);\n\n utils.forEach(this, (value, header) => {\n value != null && value !== false && (obj[header] = asStrings && utils.isArray(value) ? value.join(', ') : value);\n });\n\n return obj;\n }\n\n [Symbol.iterator]() {\n return Object.entries(this.toJSON())[Symbol.iterator]();\n }\n\n toString() {\n return Object.entries(this.toJSON()).map(([header, value]) => header + ': ' + value).join('\\n');\n }\n\n getSetCookie() {\n return this.get(\"set-cookie\") || [];\n }\n\n get [Symbol.toStringTag]() {\n return 'AxiosHeaders';\n }\n\n static from(thing) {\n return thing instanceof this ? thing : new this(thing);\n }\n\n static concat(first, ...targets) {\n const computed = new this(first);\n\n targets.forEach((target) => computed.set(target));\n\n return computed;\n }\n\n static accessor(header) {\n const internals = this[$internals] = (this[$internals] = {\n accessors: {}\n });\n\n const accessors = internals.accessors;\n const prototype = this.prototype;\n\n function defineAccessor(_header) {\n const lHeader = normalizeHeader(_header);\n\n if (!accessors[lHeader]) {\n buildAccessors(prototype, _header);\n accessors[lHeader] = true;\n }\n }\n\n utils.isArray(header) ? header.forEach(defineAccessor) : defineAccessor(header);\n\n return this;\n }\n}\n\nAxiosHeaders.accessor(['Content-Type', 'Content-Length', 'Accept', 'Accept-Encoding', 'User-Agent', 'Authorization']);\n\n// reserved names hotfix\nutils.reduceDescriptors(AxiosHeaders.prototype, ({value}, key) => {\n let mapped = key[0].toUpperCase() + key.slice(1); // map `set` => `Set`\n return {\n get: () => value,\n set(headerValue) {\n this[mapped] = headerValue;\n }\n }\n});\n\nutils.freezeMethods(AxiosHeaders);\n\nexport default AxiosHeaders;\n","'use strict';\n\nimport utils from './../utils.js';\nimport defaults from '../defaults/index.js';\nimport AxiosHeaders from '../core/AxiosHeaders.js';\n\n/**\n * Transform the data for a request or a response\n *\n * @param {Array|Function} fns A single function or Array of functions\n * @param {?Object} response The response object\n *\n * @returns {*} The resulting transformed data\n */\nexport default function transformData(fns, response) {\n const config = this || defaults;\n const context = response || config;\n const headers = AxiosHeaders.from(context.headers);\n let data = context.data;\n\n utils.forEach(fns, function transform(fn) {\n data = fn.call(config, data, headers.normalize(), response ? response.status : undefined);\n });\n\n headers.normalize();\n\n return data;\n}\n","'use strict';\n\nexport default function isCancel(value) {\n return !!(value && value.__CANCEL__);\n}\n","'use strict';\n\nimport AxiosError from '../core/AxiosError.js';\nimport utils from '../utils.js';\n\n/**\n * A `CanceledError` is an object that is thrown when an operation is canceled.\n *\n * @param {string=} message The message.\n * @param {Object=} config The config.\n * @param {Object=} request The request.\n *\n * @returns {CanceledError} The created error.\n */\nfunction CanceledError(message, config, request) {\n // eslint-disable-next-line no-eq-null,eqeqeq\n AxiosError.call(this, message == null ? 'canceled' : message, AxiosError.ERR_CANCELED, config, request);\n this.name = 'CanceledError';\n}\n\nutils.inherits(CanceledError, AxiosError, {\n __CANCEL__: true\n});\n\nexport default CanceledError;\n","'use strict';\n\nimport AxiosError from './AxiosError.js';\n\n/**\n * Resolve or reject a Promise based on response status.\n *\n * @param {Function} resolve A function that resolves the promise.\n * @param {Function} reject A function that rejects the promise.\n * @param {object} response The response.\n *\n * @returns {object} The response.\n */\nexport default function settle(resolve, reject, response) {\n const validateStatus = response.config.validateStatus;\n if (!response.status || !validateStatus || validateStatus(response.status)) {\n resolve(response);\n } else {\n reject(new AxiosError(\n 'Request failed with status code ' + response.status,\n [AxiosError.ERR_BAD_REQUEST, AxiosError.ERR_BAD_RESPONSE][Math.floor(response.status / 100) - 4],\n response.config,\n response.request,\n response\n ));\n }\n}\n","'use strict';\n\nexport default function parseProtocol(url) {\n const match = /^([-+\\w]{1,25})(:?\\/\\/|:)/.exec(url);\n return match && match[1] || '';\n}\n","'use strict';\n\n/**\n * Calculate data maxRate\n * @param {Number} [samplesCount= 10]\n * @param {Number} [min= 1000]\n * @returns {Function}\n */\nfunction speedometer(samplesCount, min) {\n samplesCount = samplesCount || 10;\n const bytes = new Array(samplesCount);\n const timestamps = new Array(samplesCount);\n let head = 0;\n let tail = 0;\n let firstSampleTS;\n\n min = min !== undefined ? min : 1000;\n\n return function push(chunkLength) {\n const now = Date.now();\n\n const startedAt = timestamps[tail];\n\n if (!firstSampleTS) {\n firstSampleTS = now;\n }\n\n bytes[head] = chunkLength;\n timestamps[head] = now;\n\n let i = tail;\n let bytesCount = 0;\n\n while (i !== head) {\n bytesCount += bytes[i++];\n i = i % samplesCount;\n }\n\n head = (head + 1) % samplesCount;\n\n if (head === tail) {\n tail = (tail + 1) % samplesCount;\n }\n\n if (now - firstSampleTS < min) {\n return;\n }\n\n const passed = startedAt && now - startedAt;\n\n return passed ? Math.round(bytesCount * 1000 / passed) : undefined;\n };\n}\n\nexport default speedometer;\n","/**\n * Throttle decorator\n * @param {Function} fn\n * @param {Number} freq\n * @return {Function}\n */\nfunction throttle(fn, freq) {\n let timestamp = 0;\n let threshold = 1000 / freq;\n let lastArgs;\n let timer;\n\n const invoke = (args, now = Date.now()) => {\n timestamp = now;\n lastArgs = null;\n if (timer) {\n clearTimeout(timer);\n timer = null;\n }\n fn.apply(null, args);\n }\n\n const throttled = (...args) => {\n const now = Date.now();\n const passed = now - timestamp;\n if ( passed >= threshold) {\n invoke(args, now);\n } else {\n lastArgs = args;\n if (!timer) {\n timer = setTimeout(() => {\n timer = null;\n invoke(lastArgs)\n }, threshold - passed);\n }\n }\n }\n\n const flush = () => lastArgs && invoke(lastArgs);\n\n return [throttled, flush];\n}\n\nexport default throttle;\n","import speedometer from \"./speedometer.js\";\nimport throttle from \"./throttle.js\";\nimport utils from \"../utils.js\";\n\nexport const progressEventReducer = (listener, isDownloadStream, freq = 3) => {\n let bytesNotified = 0;\n const _speedometer = speedometer(50, 250);\n\n return throttle(e => {\n const loaded = e.loaded;\n const total = e.lengthComputable ? e.total : undefined;\n const progressBytes = loaded - bytesNotified;\n const rate = _speedometer(progressBytes);\n const inRange = loaded <= total;\n\n bytesNotified = loaded;\n\n const data = {\n loaded,\n total,\n progress: total ? (loaded / total) : undefined,\n bytes: progressBytes,\n rate: rate ? rate : undefined,\n estimated: rate && total && inRange ? (total - loaded) / rate : undefined,\n event: e,\n lengthComputable: total != null,\n [isDownloadStream ? 'download' : 'upload']: true\n };\n\n listener(data);\n }, freq);\n}\n\nexport const progressEventDecorator = (total, throttled) => {\n const lengthComputable = total != null;\n\n return [(loaded) => throttled[0]({\n lengthComputable,\n total,\n loaded\n }), throttled[1]];\n}\n\nexport const asyncDecorator = (fn) => (...args) => utils.asap(() => fn(...args));\n","import platform from '../platform/index.js';\n\nexport default platform.hasStandardBrowserEnv ? ((origin, isMSIE) => (url) => {\n url = new URL(url, platform.origin);\n\n return (\n origin.protocol === url.protocol &&\n origin.host === url.host &&\n (isMSIE || origin.port === url.port)\n );\n})(\n new URL(platform.origin),\n platform.navigator && /(msie|trident)/i.test(platform.navigator.userAgent)\n) : () => true;\n","import utils from './../utils.js';\nimport platform from '../platform/index.js';\n\nexport default platform.hasStandardBrowserEnv ?\n\n // Standard browser envs support document.cookie\n {\n write(name, value, expires, path, domain, secure) {\n const cookie = [name + '=' + encodeURIComponent(value)];\n\n utils.isNumber(expires) && cookie.push('expires=' + new Date(expires).toGMTString());\n\n utils.isString(path) && cookie.push('path=' + path);\n\n utils.isString(domain) && cookie.push('domain=' + domain);\n\n secure === true && cookie.push('secure');\n\n document.cookie = cookie.join('; ');\n },\n\n read(name) {\n const match = document.cookie.match(new RegExp('(^|;\\\\s*)(' + name + ')=([^;]*)'));\n return (match ? decodeURIComponent(match[3]) : null);\n },\n\n remove(name) {\n this.write(name, '', Date.now() - 86400000);\n }\n }\n\n :\n\n // Non-standard browser env (web workers, react-native) lack needed support.\n {\n write() {},\n read() {\n return null;\n },\n remove() {}\n };\n\n","'use strict';\n\n/**\n * Determines whether the specified URL is absolute\n *\n * @param {string} url The URL to test\n *\n * @returns {boolean} True if the specified URL is absolute, otherwise false\n */\nexport default function isAbsoluteURL(url) {\n // A URL is considered absolute if it begins with \"<scheme>://\" or \"//\" (protocol-relative URL).\n // RFC 3986 defines scheme name as a sequence of characters beginning with a letter and followed\n // by any combination of letters, digits, plus, period, or hyphen.\n return /^([a-z][a-z\\d+\\-.]*:)?\\/\\//i.test(url);\n}\n","'use strict';\n\n/**\n * Creates a new URL by combining the specified URLs\n *\n * @param {string} baseURL The base URL\n * @param {string} relativeURL The relative URL\n *\n * @returns {string} The combined URL\n */\nexport default function combineURLs(baseURL, relativeURL) {\n return relativeURL\n ? baseURL.replace(/\\/?\\/$/, '') + '/' + relativeURL.replace(/^\\/+/, '')\n : baseURL;\n}\n","'use strict';\n\nimport isAbsoluteURL from '../helpers/isAbsoluteURL.js';\nimport combineURLs from '../helpers/combineURLs.js';\n\n/**\n * Creates a new URL by combining the baseURL with the requestedURL,\n * only when the requestedURL is not already an absolute URL.\n * If the requestURL is absolute, this function returns the requestedURL untouched.\n *\n * @param {string} baseURL The base URL\n * @param {string} requestedURL Absolute or relative URL to combine\n *\n * @returns {string} The combined full path\n */\nexport default function buildFullPath(baseURL, requestedURL, allowAbsoluteUrls) {\n let isRelativeUrl = !isAbsoluteURL(requestedURL);\n if (baseURL && (isRelativeUrl || allowAbsoluteUrls == false)) {\n return combineURLs(baseURL, requestedURL);\n }\n return requestedURL;\n}\n","'use strict';\n\nimport utils from '../utils.js';\nimport AxiosHeaders from \"./AxiosHeaders.js\";\n\nconst headersToObject = (thing) => thing instanceof AxiosHeaders ? { ...thing } : thing;\n\n/**\n * Config-specific merge-function which creates a new config-object\n * by merging two configuration objects together.\n *\n * @param {Object} config1\n * @param {Object} config2\n *\n * @returns {Object} New object resulting from merging config2 to config1\n */\nexport default function mergeConfig(config1, config2) {\n // eslint-disable-next-line no-param-reassign\n config2 = config2 || {};\n const config = {};\n\n function getMergedValue(target, source, prop, caseless) {\n if (utils.isPlainObject(target) && utils.isPlainObject(source)) {\n return utils.merge.call({caseless}, target, source);\n } else if (utils.isPlainObject(source)) {\n return utils.merge({}, source);\n } else if (utils.isArray(source)) {\n return source.slice();\n }\n return source;\n }\n\n // eslint-disable-next-line consistent-return\n function mergeDeepProperties(a, b, prop , caseless) {\n if (!utils.isUndefined(b)) {\n return getMergedValue(a, b, prop , caseless);\n } else if (!utils.isUndefined(a)) {\n return getMergedValue(undefined, a, prop , caseless);\n }\n }\n\n // eslint-disable-next-line consistent-return\n function valueFromConfig2(a, b) {\n if (!utils.isUndefined(b)) {\n return getMergedValue(undefined, b);\n }\n }\n\n // eslint-disable-next-line consistent-return\n function defaultToConfig2(a, b) {\n if (!utils.isUndefined(b)) {\n return getMergedValue(undefined, b);\n } else if (!utils.isUndefined(a)) {\n return getMergedValue(undefined, a);\n }\n }\n\n // eslint-disable-next-line consistent-return\n function mergeDirectKeys(a, b, prop) {\n if (prop in config2) {\n return getMergedValue(a, b);\n } else if (prop in config1) {\n return getMergedValue(undefined, a);\n }\n }\n\n const mergeMap = {\n url: valueFromConfig2,\n method: valueFromConfig2,\n data: valueFromConfig2,\n baseURL: defaultToConfig2,\n transformRequest: defaultToConfig2,\n transformResponse: defaultToConfig2,\n paramsSerializer: defaultToConfig2,\n timeout: defaultToConfig2,\n timeoutMessage: defaultToConfig2,\n withCredentials: defaultToConfig2,\n withXSRFToken: defaultToConfig2,\n adapter: defaultToConfig2,\n responseType: defaultToConfig2,\n xsrfCookieName: defaultToConfig2,\n xsrfHeaderName: defaultToConfig2,\n onUploadProgress: defaultToConfig2,\n onDownloadProgress: defaultToConfig2,\n decompress: defaultToConfig2,\n maxContentLength: defaultToConfig2,\n maxBodyLength: defaultToConfig2,\n beforeRedirect: defaultToConfig2,\n transport: defaultToConfig2,\n httpAgent: defaultToConfig2,\n httpsAgent: defaultToConfig2,\n cancelToken: defaultToConfig2,\n socketPath: defaultToConfig2,\n responseEncoding: defaultToConfig2,\n validateStatus: mergeDirectKeys,\n headers: (a, b , prop) => mergeDeepProperties(headersToObject(a), headersToObject(b),prop, true)\n };\n\n utils.forEach(Object.keys(Object.assign({}, config1, config2)), function computeConfigValue(prop) {\n const merge = mergeMap[prop] || mergeDeepProperties;\n const configValue = merge(config1[prop], config2[prop], prop);\n (utils.isUndefined(configValue) && merge !== mergeDirectKeys) || (config[prop] = configValue);\n });\n\n return config;\n}\n","import platform from \"../platform/index.js\";\nimport utils from \"../utils.js\";\nimport isURLSameOrigin from \"./isURLSameOrigin.js\";\nimport cookies from \"./cookies.js\";\nimport buildFullPath from \"../core/buildFullPath.js\";\nimport mergeConfig from \"../core/mergeConfig.js\";\nimport AxiosHeaders from \"../core/AxiosHeaders.js\";\nimport buildURL from \"./buildURL.js\";\n\nexport default (config) => {\n const newConfig = mergeConfig({}, config);\n\n let {data, withXSRFToken, xsrfHeaderName, xsrfCookieName, headers, auth} = newConfig;\n\n newConfig.headers = headers = AxiosHeaders.from(headers);\n\n newConfig.url = buildURL(buildFullPath(newConfig.baseURL, newConfig.url, newConfig.allowAbsoluteUrls), config.params, config.paramsSerializer);\n\n // HTTP basic authentication\n if (auth) {\n headers.set('Authorization', 'Basic ' +\n btoa((auth.username || '') + ':' + (auth.password ? unescape(encodeURIComponent(auth.password)) : ''))\n );\n }\n\n let contentType;\n\n if (utils.isFormData(data)) {\n if (platform.hasStandardBrowserEnv || platform.hasStandardBrowserWebWorkerEnv) {\n headers.setContentType(undefined); // Let the browser set it\n } else if ((contentType = headers.getContentType()) !== false) {\n // fix semicolon duplication issue for ReactNative FormData implementation\n const [type, ...tokens] = contentType ? contentType.split(';').map(token => token.trim()).filter(Boolean) : [];\n headers.setContentType([type || 'multipart/form-data', ...tokens].join('; '));\n }\n }\n\n // Add xsrf header\n // This is only done if running in a standard browser environment.\n // Specifically not if we're in a web worker, or react-native.\n\n if (platform.hasStandardBrowserEnv) {\n withXSRFToken && utils.isFunction(withXSRFToken) && (withXSRFToken = withXSRFToken(newConfig));\n\n if (withXSRFToken || (withXSRFToken !== false && isURLSameOrigin(newConfig.url))) {\n // Add xsrf header\n const xsrfValue = xsrfHeaderName && xsrfCookieName && cookies.read(xsrfCookieName);\n\n if (xsrfValue) {\n headers.set(xsrfHeaderName, xsrfValue);\n }\n }\n }\n\n return newConfig;\n}\n\n","import utils from './../utils.js';\nimport settle from './../core/settle.js';\nimport transitionalDefaults from '../defaults/transitional.js';\nimport AxiosError from '../core/AxiosError.js';\nimport CanceledError from '../cancel/CanceledError.js';\nimport parseProtocol from '../helpers/parseProtocol.js';\nimport platform from '../platform/index.js';\nimport AxiosHeaders from '../core/AxiosHeaders.js';\nimport {progressEventReducer} from '../helpers/progressEventReducer.js';\nimport resolveConfig from \"../helpers/resolveConfig.js\";\n\nconst isXHRAdapterSupported = typeof XMLHttpRequest !== 'undefined';\n\nexport default isXHRAdapterSupported && function (config) {\n return new Promise(function dispatchXhrRequest(resolve, reject) {\n const _config = resolveConfig(config);\n let requestData = _config.data;\n const requestHeaders = AxiosHeaders.from(_config.headers).normalize();\n let {responseType, onUploadProgress, onDownloadProgress} = _config;\n let onCanceled;\n let uploadThrottled, downloadThrottled;\n let flushUpload, flushDownload;\n\n function done() {\n flushUpload && flushUpload(); // flush events\n flushDownload && flushDownload(); // flush events\n\n _config.cancelToken && _config.cancelToken.unsubscribe(onCanceled);\n\n _config.signal && _config.signal.removeEventListener('abort', onCanceled);\n }\n\n let request = new XMLHttpRequest();\n\n request.open(_config.method.toUpperCase(), _config.url, true);\n\n // Set the request timeout in MS\n request.timeout = _config.timeout;\n\n function onloadend() {\n if (!request) {\n return;\n }\n // Prepare the response\n const responseHeaders = AxiosHeaders.from(\n 'getAllResponseHeaders' in request && request.getAllResponseHeaders()\n );\n const responseData = !responseType || responseType === 'text' || responseType === 'json' ?\n request.responseText : request.response;\n const response = {\n data: responseData,\n status: request.status,\n statusText: request.statusText,\n headers: responseHeaders,\n config,\n request\n };\n\n settle(function _resolve(value) {\n resolve(value);\n done();\n }, function _reject(err) {\n reject(err);\n done();\n }, response);\n\n // Clean up request\n request = null;\n }\n\n if ('onloadend' in request) {\n // Use onloadend if available\n request.onloadend = onloadend;\n } else {\n // Listen for ready state to emulate onloadend\n request.onreadystatechange = function handleLoad() {\n if (!request || request.readyState !== 4) {\n return;\n }\n\n // The request errored out and we didn't get a response, this will be\n // handled by onerror instead\n // With one exception: request that using file: protocol, most browsers\n // will return status as 0 even though it's a successful request\n if (request.status === 0 && !(request.responseURL && request.responseURL.indexOf('file:') === 0)) {\n return;\n }\n // readystate handler is calling before onerror or ontimeout handlers,\n // so we should call onloadend on the next 'tick'\n setTimeout(onloadend);\n };\n }\n\n // Handle browser request cancellation (as opposed to a manual cancellation)\n request.onabort = function handleAbort() {\n if (!request) {\n return;\n }\n\n reject(new AxiosError('Request aborted', AxiosError.ECONNABORTED, config, request));\n\n // Clean up request\n request = null;\n };\n\n // Handle low level network errors\n request.onerror = function handleError() {\n // Real errors are hidden from us by the browser\n // onerror should only fire if it's a network error\n reject(new AxiosError('Network Error', AxiosError.ERR_NETWORK, config, request));\n\n // Clean up request\n request = null;\n };\n\n // Handle timeout\n request.ontimeout = function handleTimeout() {\n let timeoutErrorMessage = _config.timeout ? 'timeout of ' + _config.timeout + 'ms exceeded' : 'timeout exceeded';\n const transitional = _config.transitional || transitionalDefaults;\n if (_config.timeoutErrorMessage) {\n timeoutErrorMessage = _config.timeoutErrorMessage;\n }\n reject(new AxiosError(\n timeoutErrorMessage,\n transitional.clarifyTimeoutError ? AxiosError.ETIMEDOUT : AxiosError.ECONNABORTED,\n config,\n request));\n\n // Clean up request\n request = null;\n };\n\n // Remove Content-Type if data is undefined\n requestData === undefined && requestHeaders.setContentType(null);\n\n // Add headers to the request\n if ('setRequestHeader' in request) {\n utils.forEach(requestHeaders.toJSON(), function setRequestHeader(val, key) {\n request.setRequestHeader(key, val);\n });\n }\n\n // Add withCredentials to request if needed\n if (!utils.isUndefined(_config.withCredentials)) {\n request.withCredentials = !!_config.withCredentials;\n }\n\n // Add responseType to request if needed\n if (responseType && responseType !== 'json') {\n request.responseType = _config.responseType;\n }\n\n // Handle progress if needed\n if (onDownloadProgress) {\n ([downloadThrottled, flushDownload] = progressEventReducer(onDownloadProgress, true));\n request.addEventListener('progress', downloadThrottled);\n }\n\n // Not all browsers support upload events\n if (onUploadProgress && request.upload) {\n ([uploadThrottled, flushUpload] = progressEventReducer(onUploadProgress));\n\n request.upload.addEventListener('progress', uploadThrottled);\n\n request.upload.addEventListener('loadend', flushUpload);\n }\n\n if (_config.cancelToken || _config.signal) {\n // Handle cancellation\n // eslint-disable-next-line func-names\n onCanceled = cancel => {\n if (!request) {\n return;\n }\n reject(!cancel || cancel.type ? new CanceledError(null, config, request) : cancel);\n request.abort();\n request = null;\n };\n\n _config.cancelToken && _config.cancelToken.subscribe(onCanceled);\n if (_config.signal) {\n _config.signal.aborted ? onCanceled() : _config.signal.addEventListener('abort', onCanceled);\n }\n }\n\n const protocol = parseProtocol(_config.url);\n\n if (protocol && platform.protocols.indexOf(protocol) === -1) {\n reject(new AxiosError('Unsupported protocol ' + protocol + ':', AxiosError.ERR_BAD_REQUEST, config));\n return;\n }\n\n\n // Send the request\n request.send(requestData || null);\n });\n}\n","import CanceledError from \"../cancel/CanceledError.js\";\nimport AxiosError from \"../core/AxiosError.js\";\nimport utils from '../utils.js';\n\nconst composeSignals = (signals, timeout) => {\n const {length} = (signals = signals ? signals.filter(Boolean) : []);\n\n if (timeout || length) {\n let controller = new AbortController();\n\n let aborted;\n\n const onabort = function (reason) {\n if (!aborted) {\n aborted = true;\n unsubscribe();\n const err = reason instanceof Error ? reason : this.reason;\n controller.abort(err instanceof AxiosError ? err : new CanceledError(err instanceof Error ? err.message : err));\n }\n }\n\n let timer = timeout && setTimeout(() => {\n timer = null;\n onabort(new AxiosError(`timeout ${timeout} of ms exceeded`, AxiosError.ETIMEDOUT))\n }, timeout)\n\n const unsubscribe = () => {\n if (signals) {\n timer && clearTimeout(timer);\n timer = null;\n signals.forEach(signal => {\n signal.unsubscribe ? signal.unsubscribe(onabort) : signal.removeEventListener('abort', onabort);\n });\n signals = null;\n }\n }\n\n signals.forEach((signal) => signal.addEventListener('abort', onabort));\n\n const {signal} = controller;\n\n signal.unsubscribe = () => utils.asap(unsubscribe);\n\n return signal;\n }\n}\n\nexport default composeSignals;\n","\nexport const streamChunk = function* (chunk, chunkSize) {\n let len = chunk.byteLength;\n\n if (!chunkSize || len < chunkSize) {\n yield chunk;\n return;\n }\n\n let pos = 0;\n let end;\n\n while (pos < len) {\n end = pos + chunkSize;\n yield chunk.slice(pos, end);\n pos = end;\n }\n}\n\nexport const readBytes = async function* (iterable, chunkSize) {\n for await (const chunk of readStream(iterable)) {\n yield* streamChunk(chunk, chunkSize);\n }\n}\n\nconst readStream = async function* (stream) {\n if (stream[Symbol.asyncIterator]) {\n yield* stream;\n return;\n }\n\n const reader = stream.getReader();\n try {\n for (;;) {\n const {done, value} = await reader.read();\n if (done) {\n break;\n }\n yield value;\n }\n } finally {\n await reader.cancel();\n }\n}\n\nexport const trackStream = (stream, chunkSize, onProgress, onFinish) => {\n const iterator = readBytes(stream, chunkSize);\n\n let bytes = 0;\n let done;\n let _onFinish = (e) => {\n if (!done) {\n done = true;\n onFinish && onFinish(e);\n }\n }\n\n return new ReadableStream({\n async pull(controller) {\n try {\n const {done, value} = await iterator.next();\n\n if (done) {\n _onFinish();\n controller.close();\n return;\n }\n\n let len = value.byteLength;\n if (onProgress) {\n let loadedBytes = bytes += len;\n onProgress(loadedBytes);\n }\n controller.enqueue(new Uint8Array(value));\n } catch (err) {\n _onFinish(err);\n throw err;\n }\n },\n cancel(reason) {\n _onFinish(reason);\n return iterator.return();\n }\n }, {\n highWaterMark: 2\n })\n}\n","import platform from \"../platform/index.js\";\nimport utils from \"../utils.js\";\nimport AxiosError from \"../core/AxiosError.js\";\nimport composeSignals from \"../helpers/composeSignals.js\";\nimport {trackStream} from \"../helpers/trackStream.js\";\nimport AxiosHeaders from \"../core/AxiosHeaders.js\";\nimport {progressEventReducer, progressEventDecorator, asyncDecorator} from \"../helpers/progressEventReducer.js\";\nimport resolveConfig from \"../helpers/resolveConfig.js\";\nimport settle from \"../core/settle.js\";\n\nconst isFetchSupported = typeof fetch === 'function' && typeof Request === 'function' && typeof Response === 'function';\nconst isReadableStreamSupported = isFetchSupported && typeof ReadableStream === 'function';\n\n// used only inside the fetch adapter\nconst encodeText = isFetchSupported && (typeof TextEncoder === 'function' ?\n ((encoder) => (str) => encoder.encode(str))(new TextEncoder()) :\n async (str) => new Uint8Array(await new Response(str).arrayBuffer())\n);\n\nconst test = (fn, ...args) => {\n try {\n return !!fn(...args);\n } catch (e) {\n return false\n }\n}\n\nconst supportsRequestStream = isReadableStreamSupported && test(() => {\n let duplexAccessed = false;\n\n const hasContentType = new Request(platform.origin, {\n body: new ReadableStream(),\n method: 'POST',\n get duplex() {\n duplexAccessed = true;\n return 'half';\n },\n }).headers.has('Content-Type');\n\n return duplexAccessed && !hasContentType;\n});\n\nconst DEFAULT_CHUNK_SIZE = 64 * 1024;\n\nconst supportsResponseStream = isReadableStreamSupported &&\n test(() => utils.isReadableStream(new Response('').body));\n\n\nconst resolvers = {\n stream: supportsResponseStream && ((res) => res.body)\n};\n\nisFetchSupported && (((res) => {\n ['text', 'arrayBuffer', 'blob', 'formData', 'stream'].forEach(type => {\n !resolvers[type] && (resolvers[type] = utils.isFunction(res[type]) ? (res) => res[type]() :\n (_, config) => {\n throw new AxiosError(`Response type '${type}' is not supported`, AxiosError.ERR_NOT_SUPPORT, config);\n })\n });\n})(new Response));\n\nconst getBodyLength = async (body) => {\n if (body == null) {\n return 0;\n }\n\n if(utils.isBlob(body)) {\n return body.size;\n }\n\n if(utils.isSpecCompliantForm(body)) {\n const _request = new Request(platform.origin, {\n method: 'POST',\n body,\n });\n return (await _request.arrayBuffer()).byteLength;\n }\n\n if(utils.isArrayBufferView(body) || utils.isArrayBuffer(body)) {\n return body.byteLength;\n }\n\n if(utils.isURLSearchParams(body)) {\n body = body + '';\n }\n\n if(utils.isString(body)) {\n return (await encodeText(body)).byteLength;\n }\n}\n\nconst resolveBodyLength = async (headers, body) => {\n const length = utils.toFiniteNumber(headers.getContentLength());\n\n return length == null ? getBodyLength(body) : length;\n}\n\nexport default isFetchSupported && (async (config) => {\n let {\n url,\n method,\n data,\n signal,\n cancelToken,\n timeout,\n onDownloadProgress,\n onUploadProgress,\n responseType,\n headers,\n withCredentials = 'same-origin',\n fetchOptions\n } = resolveConfig(config);\n\n responseType = responseType ? (responseType + '').toLowerCase() : 'text';\n\n let composedSignal = composeSignals([signal, cancelToken && cancelToken.toAbortSignal()], timeout);\n\n let request;\n\n const unsubscribe = composedSignal && composedSignal.unsubscribe && (() => {\n composedSignal.unsubscribe();\n });\n\n let requestContentLength;\n\n try {\n if (\n onUploadProgress && supportsRequestStream && method !== 'get' && method !== 'head' &&\n (requestContentLength = await resolveBodyLength(headers, data)) !== 0\n ) {\n let _request = new Request(url, {\n method: 'POST',\n body: data,\n duplex: \"half\"\n });\n\n let contentTypeHeader;\n\n if (utils.isFormData(data) && (contentTypeHeader = _request.headers.get('content-type'))) {\n headers.setContentType(contentTypeHeader)\n }\n\n if (_request.body) {\n const [onProgress, flush] = progressEventDecorator(\n requestContentLength,\n progressEventReducer(asyncDecorator(onUploadProgress))\n );\n\n data = trackStream(_request.body, DEFAULT_CHUNK_SIZE, onProgress, flush);\n }\n }\n\n if (!utils.isString(withCredentials)) {\n withCredentials = withCredentials ? 'include' : 'omit';\n }\n\n // Cloudflare Workers throws when credentials are defined\n // see https://github.com/cloudflare/workerd/issues/902\n const isCredentialsSupported = \"credentials\" in Request.prototype;\n request = new Request(url, {\n ...fetchOptions,\n signal: composedSignal,\n method: method.toUpperCase(),\n headers: headers.normalize().toJSON(),\n body: data,\n duplex: \"half\",\n credentials: isCredentialsSupported ? withCredentials : undefined\n });\n\n let response = await fetch(request, fetchOptions);\n\n const isStreamResponse = supportsResponseStream && (responseType === 'stream' || responseType === 'response');\n\n if (supportsResponseStream && (onDownloadProgress || (isStreamResponse && unsubscribe))) {\n const options = {};\n\n ['status', 'statusText', 'headers'].forEach(prop => {\n options[prop] = response[prop];\n });\n\n const responseContentLength = utils.toFiniteNumber(response.headers.get('content-length'));\n\n const [onProgress, flush] = onDownloadProgress && progressEventDecorator(\n responseContentLength,\n progressEventReducer(asyncDecorator(onDownloadProgress), true)\n ) || [];\n\n response = new Response(\n trackStream(response.body, DEFAULT_CHUNK_SIZE, onProgress, () => {\n flush && flush();\n unsubscribe && unsubscribe();\n }),\n options\n );\n }\n\n responseType = responseType || 'text';\n\n let responseData = await resolvers[utils.findKey(resolvers, responseType) || 'text'](response, config);\n\n !isStreamResponse && unsubscribe && unsubscribe();\n\n return await new Promise((resolve, reject) => {\n settle(resolve, reject, {\n data: responseData,\n headers: AxiosHeaders.from(response.headers),\n status: response.status,\n statusText: response.statusText,\n config,\n request\n })\n })\n } catch (err) {\n unsubscribe && unsubscribe();\n\n if (err && err.name === 'TypeError' && /Load failed|fetch/i.test(err.message)) {\n throw Object.assign(\n new AxiosError('Network Error', AxiosError.ERR_NETWORK, config, request),\n {\n cause: err.cause || err\n }\n )\n }\n\n throw AxiosError.from(err, err && err.code, config, request);\n }\n});\n\n\n","import utils from '../utils.js';\nimport httpAdapter from './http.js';\nimport xhrAdapter from './xhr.js';\nimport fetchAdapter from './fetch.js';\nimport AxiosError from \"../core/AxiosError.js\";\n\nconst knownAdapters = {\n http: httpAdapter,\n xhr: xhrAdapter,\n fetch: fetchAdapter\n}\n\nutils.forEach(knownAdapters, (fn, value) => {\n if (fn) {\n try {\n Object.defineProperty(fn, 'name', {value});\n } catch (e) {\n // eslint-disable-next-line no-empty\n }\n Object.defineProperty(fn, 'adapterName', {value});\n }\n});\n\nconst renderReason = (reason) => `- ${reason}`;\n\nconst isResolvedHandle = (adapter) => utils.isFunction(adapter) || adapter === null || adapter === false;\n\nexport default {\n getAdapter: (adapters) => {\n adapters = utils.isArray(adapters) ? adapters : [adapters];\n\n const {length} = adapters;\n let nameOrAdapter;\n let adapter;\n\n const rejectedReasons = {};\n\n for (let i = 0; i < length; i++) {\n nameOrAdapter = adapters[i];\n let id;\n\n adapter = nameOrAdapter;\n\n if (!isResolvedHandle(nameOrAdapter)) {\n adapter = knownAdapters[(id = String(nameOrAdapter)).toLowerCase()];\n\n if (adapter === undefined) {\n throw new AxiosError(`Unknown adapter '${id}'`);\n }\n }\n\n if (adapter) {\n break;\n }\n\n rejectedReasons[id || '#' + i] = adapter;\n }\n\n if (!adapter) {\n\n const reasons = Object.entries(rejectedReasons)\n .map(([id, state]) => `adapter ${id} ` +\n (state === false ? 'is not supported by the environment' : 'is not available in the build')\n );\n\n let s = length ?\n (reasons.length > 1 ? 'since :\\n' + reasons.map(renderReason).join('\\n') : ' ' + renderReason(reasons[0])) :\n 'as no adapter specified';\n\n throw new AxiosError(\n `There is no suitable adapter to dispatch the request ` + s,\n 'ERR_NOT_SUPPORT'\n );\n }\n\n return adapter;\n },\n adapters: knownAdapters\n}\n","'use strict';\n\nimport transformData from './transformData.js';\nimport isCancel from '../cancel/isCancel.js';\nimport defaults from '../defaults/index.js';\nimport CanceledError from '../cancel/CanceledError.js';\nimport AxiosHeaders from '../core/AxiosHeaders.js';\nimport adapters from \"../adapters/adapters.js\";\n\n/**\n * Throws a `CanceledError` if cancellation has been requested.\n *\n * @param {Object} config The config that is to be used for the request\n *\n * @returns {void}\n */\nfunction throwIfCancellationRequested(config) {\n if (config.cancelToken) {\n config.cancelToken.throwIfRequested();\n }\n\n if (config.signal && config.signal.aborted) {\n throw new CanceledError(null, config);\n }\n}\n\n/**\n * Dispatch a request to the server using the configured adapter.\n *\n * @param {object} config The config that is to be used for the request\n *\n * @returns {Promise} The Promise to be fulfilled\n */\nexport default function dispatchRequest(config) {\n throwIfCancellationRequested(config);\n\n config.headers = AxiosHeaders.from(config.headers);\n\n // Transform request data\n config.data = transformData.call(\n config,\n config.transformRequest\n );\n\n if (['post', 'put', 'patch'].indexOf(config.method) !== -1) {\n config.headers.setContentType('application/x-www-form-urlencoded', false);\n }\n\n const adapter = adapters.getAdapter(config.adapter || defaults.adapter);\n\n return adapter(config).then(function onAdapterResolution(response) {\n throwIfCancellationRequested(config);\n\n // Transform response data\n response.data = transformData.call(\n config,\n config.transformResponse,\n response\n );\n\n response.headers = AxiosHeaders.from(response.headers);\n\n return response;\n }, function onAdapterRejection(reason) {\n if (!isCancel(reason)) {\n throwIfCancellationRequested(config);\n\n // Transform response data\n if (reason && reason.response) {\n reason.response.data = transformData.call(\n config,\n config.transformResponse,\n reason.response\n );\n reason.response.headers = AxiosHeaders.from(reason.response.headers);\n }\n }\n\n return Promise.reject(reason);\n });\n}\n","export const VERSION = \"1.10.0\";","'use strict';\n\nimport {VERSION} from '../env/data.js';\nimport AxiosError from '../core/AxiosError.js';\n\nconst validators = {};\n\n// eslint-disable-next-line func-names\n['object', 'boolean', 'number', 'function', 'string', 'symbol'].forEach((type, i) => {\n validators[type] = function validator(thing) {\n return typeof thing === type || 'a' + (i < 1 ? 'n ' : ' ') + type;\n };\n});\n\nconst deprecatedWarnings = {};\n\n/**\n * Transitional option validator\n *\n * @param {function|boolean?} validator - set to false if the transitional option has been removed\n * @param {string?} version - deprecated version / removed since version\n * @param {string?} message - some message with additional info\n *\n * @returns {function}\n */\nvalidators.transitional = function transitional(validator, version, message) {\n function formatMessage(opt, desc) {\n return '[Axios v' + VERSION + '] Transitional option \\'' + opt + '\\'' + desc + (message ? '. ' + message : '');\n }\n\n // eslint-disable-next-line func-names\n return (value, opt, opts) => {\n if (validator === false) {\n throw new AxiosError(\n formatMessage(opt, ' has been removed' + (version ? ' in ' + version : '')),\n AxiosError.ERR_DEPRECATED\n );\n }\n\n if (version && !deprecatedWarnings[opt]) {\n deprecatedWarnings[opt] = true;\n // eslint-disable-next-line no-console\n console.warn(\n formatMessage(\n opt,\n ' has been deprecated since v' + version + ' and will be removed in the near future'\n )\n );\n }\n\n return validator ? validator(value, opt, opts) : true;\n };\n};\n\nvalidators.spelling = function spelling(correctSpelling) {\n return (value, opt) => {\n // eslint-disable-next-line no-console\n console.warn(`${opt} is likely a misspelling of ${correctSpelling}`);\n return true;\n }\n};\n\n/**\n * Assert object's properties type\n *\n * @param {object} options\n * @param {object} schema\n * @param {boolean?} allowUnknown\n *\n * @returns {object}\n */\n\nfunction assertOptions(options, schema, allowUnknown) {\n if (typeof options !== 'object') {\n throw new AxiosError('options must be an object', AxiosError.ERR_BAD_OPTION_VALUE);\n }\n const keys = Object.keys(options);\n let i = keys.length;\n while (i-- > 0) {\n const opt = keys[i];\n const validator = schema[opt];\n if (validator) {\n const value = options[opt];\n const result = value === undefined || validator(value, opt, options);\n if (result !== true) {\n throw new AxiosError('option ' + opt + ' must be ' + result, AxiosError.ERR_BAD_OPTION_VALUE);\n }\n continue;\n }\n if (allowUnknown !== true) {\n throw new AxiosError('Unknown option ' + opt, AxiosError.ERR_BAD_OPTION);\n }\n }\n}\n\nexport default {\n assertOptions,\n validators\n};\n","'use strict';\n\nimport utils from './../utils.js';\nimport buildURL from '../helpers/buildURL.js';\nimport InterceptorManager from './InterceptorManager.js';\nimport dispatchRequest from './dispatchRequest.js';\nimport mergeConfig from './mergeConfig.js';\nimport buildFullPath from './buildFullPath.js';\nimport validator from '../helpers/validator.js';\nimport AxiosHeaders from './AxiosHeaders.js';\n\nconst validators = validator.validators;\n\n/**\n * Create a new instance of Axios\n *\n * @param {Object} instanceConfig The default config for the instance\n *\n * @return {Axios} A new instance of Axios\n */\nclass Axios {\n constructor(instanceConfig) {\n this.defaults = instanceConfig || {};\n this.interceptors = {\n request: new InterceptorManager(),\n response: new InterceptorManager()\n };\n }\n\n /**\n * Dispatch a request\n *\n * @param {String|Object} configOrUrl The config specific for this request (merged with this.defaults)\n * @param {?Object} config\n *\n * @returns {Promise} The Promise to be fulfilled\n */\n async request(configOrUrl, config) {\n try {\n return await this._request(configOrUrl, config);\n } catch (err) {\n if (err instanceof Error) {\n let dummy = {};\n\n Error.captureStackTrace ? Error.captureStackTrace(dummy) : (dummy = new Error());\n\n // slice off the Error: ... line\n const stack = dummy.stack ? dummy.stack.replace(/^.+\\n/, '') : '';\n try {\n if (!err.stack) {\n err.stack = stack;\n // match without the 2 top stack lines\n } else if (stack && !String(err.stack).endsWith(stack.replace(/^.+\\n.+\\n/, ''))) {\n err.stack += '\\n' + stack\n }\n } catch (e) {\n // ignore the case where \"stack\" is an un-writable property\n }\n }\n\n throw err;\n }\n }\n\n _request(configOrUrl, config) {\n /*eslint no-param-reassign:0*/\n // Allow for axios('example/url'[, config]) a la fetch API\n if (typeof configOrUrl === 'string') {\n config = config || {};\n config.url = configOrUrl;\n } else {\n config = configOrUrl || {};\n }\n\n config = mergeConfig(this.defaults, config);\n\n const {transitional, paramsSerializer, headers} = config;\n\n if (transitional !== undefined) {\n validator.assertOptions(transitional, {\n silentJSONParsing: validators.transitional(validators.boolean),\n forcedJSONParsing: validators.transitional(validators.boolean),\n clarifyTimeoutError: validators.transitional(validators.boolean)\n }, false);\n }\n\n if (paramsSerializer != null) {\n if (utils.isFunction(paramsSerializer)) {\n config.paramsSerializer = {\n serialize: paramsSerializer\n }\n } else {\n validator.assertOptions(paramsSerializer, {\n encode: validators.function,\n serialize: validators.function\n }, true);\n }\n }\n\n // Set config.allowAbsoluteUrls\n if (config.allowAbsoluteUrls !== undefined) {\n // do nothing\n } else if (this.defaults.allowAbsoluteUrls !== undefined) {\n config.allowAbsoluteUrls = this.defaults.allowAbsoluteUrls;\n } else {\n config.allowAbsoluteUrls = true;\n }\n\n validator.assertOptions(config, {\n baseUrl: validators.spelling('baseURL'),\n withXsrfToken: validators.spelling('withXSRFToken')\n }, true);\n\n // Set config.method\n config.method = (config.method || this.defaults.method || 'get').toLowerCase();\n\n // Flatten headers\n let contextHeaders = headers && utils.merge(\n headers.common,\n headers[config.method]\n );\n\n headers && utils.forEach(\n ['delete', 'get', 'head', 'post', 'put', 'patch', 'common'],\n (method) => {\n delete headers[method];\n }\n );\n\n config.headers = AxiosHeaders.concat(contextHeaders, headers);\n\n // filter out skipped interceptors\n const requestInterceptorChain = [];\n let synchronousRequestInterceptors = true;\n this.interceptors.request.forEach(function unshiftRequestInterceptors(interceptor) {\n if (typeof interceptor.runWhen === 'function' && interceptor.runWhen(config) === false) {\n return;\n }\n\n synchronousRequestInterceptors = synchronousRequestInterceptors && interceptor.synchronous;\n\n requestInterceptorChain.unshift(interceptor.fulfilled, interceptor.rejected);\n });\n\n const responseInterceptorChain = [];\n this.interceptors.response.forEach(function pushResponseInterceptors(interceptor) {\n responseInterceptorChain.push(interceptor.fulfilled, interceptor.rejected);\n });\n\n let promise;\n let i = 0;\n let len;\n\n if (!synchronousRequestInterceptors) {\n const chain = [dispatchRequest.bind(this), undefined];\n chain.unshift.apply(chain, requestInterceptorChain);\n chain.push.apply(chain, responseInterceptorChain);\n len = chain.length;\n\n promise = Promise.resolve(config);\n\n while (i < len) {\n promise = promise.then(chain[i++], chain[i++]);\n }\n\n return promise;\n }\n\n len = requestInterceptorChain.length;\n\n let newConfig = config;\n\n i = 0;\n\n while (i < len) {\n const onFulfilled = requestInterceptorChain[i++];\n const onRejected = requestInterceptorChain[i++];\n try {\n newConfig = onFulfilled(newConfig);\n } catch (error) {\n onRejected.call(this, error);\n break;\n }\n }\n\n try {\n promise = dispatchRequest.call(this, newConfig);\n } catch (error) {\n return Promise.reject(error);\n }\n\n i = 0;\n len = responseInterceptorChain.length;\n\n while (i < len) {\n promise = promise.then(responseInterceptorChain[i++], responseInterceptorChain[i++]);\n }\n\n return promise;\n }\n\n getUri(config) {\n config = mergeConfig(this.defaults, config);\n const fullPath = buildFullPath(config.baseURL, config.url, config.allowAbsoluteUrls);\n return buildURL(fullPath, config.params, config.paramsSerializer);\n }\n}\n\n// Provide aliases for supported request methods\nutils.forEach(['delete', 'get', 'head', 'options'], function forEachMethodNoData(method) {\n /*eslint func-names:0*/\n Axios.prototype[method] = function(url, config) {\n return this.request(mergeConfig(config || {}, {\n method,\n url,\n data: (config || {}).data\n }));\n };\n});\n\nutils.forEach(['post', 'put', 'patch'], function forEachMethodWithData(method) {\n /*eslint func-names:0*/\n\n function generateHTTPMethod(isForm) {\n return function httpMethod(url, data, config) {\n return this.request(mergeConfig(config || {}, {\n method,\n headers: isForm ? {\n 'Content-Type': 'multipart/form-data'\n } : {},\n url,\n data\n }));\n };\n }\n\n Axios.prototype[method] = generateHTTPMethod();\n\n Axios.prototype[method + 'Form'] = generateHTTPMethod(true);\n});\n\nexport default Axios;\n","'use strict';\n\nimport CanceledError from './CanceledError.js';\n\n/**\n * A `CancelToken` is an object that can be used to request cancellation of an operation.\n *\n * @param {Function} executor The executor function.\n *\n * @returns {CancelToken}\n */\nclass CancelToken {\n constructor(executor) {\n if (typeof executor !== 'function') {\n throw new TypeError('executor must be a function.');\n }\n\n let resolvePromise;\n\n this.promise = new Promise(function promiseExecutor(resolve) {\n resolvePromise = resolve;\n });\n\n const token = this;\n\n // eslint-disable-next-line func-names\n this.promise.then(cancel => {\n if (!token._listeners) return;\n\n let i = token._listeners.length;\n\n while (i-- > 0) {\n token._listeners[i](cancel);\n }\n token._listeners = null;\n });\n\n // eslint-disable-next-line func-names\n this.promise.then = onfulfilled => {\n let _resolve;\n // eslint-disable-next-line func-names\n const promise = new Promise(resolve => {\n token.subscribe(resolve);\n _resolve = resolve;\n }).then(onfulfilled);\n\n promise.cancel = function reject() {\n token.unsubscribe(_resolve);\n };\n\n return promise;\n };\n\n executor(function cancel(message, config, request) {\n if (token.reason) {\n // Cancellation has already been requested\n return;\n }\n\n token.reason = new CanceledError(message, config, request);\n resolvePromise(token.reason);\n });\n }\n\n /**\n * Throws a `CanceledError` if cancellation has been requested.\n */\n throwIfRequested() {\n if (this.reason) {\n throw this.reason;\n }\n }\n\n /**\n * Subscribe to the cancel signal\n */\n\n subscribe(listener) {\n if (this.reason) {\n listener(this.reason);\n return;\n }\n\n if (this._listeners) {\n this._listeners.push(listener);\n } else {\n this._listeners = [listener];\n }\n }\n\n /**\n * Unsubscribe from the cancel signal\n */\n\n unsubscribe(listener) {\n if (!this._listeners) {\n return;\n }\n const index = this._listeners.indexOf(listener);\n if (index !== -1) {\n this._listeners.splice(index, 1);\n }\n }\n\n toAbortSignal() {\n const controller = new AbortController();\n\n const abort = (err) => {\n controller.abort(err);\n };\n\n this.subscribe(abort);\n\n controller.signal.unsubscribe = () => this.unsubscribe(abort);\n\n return controller.signal;\n }\n\n /**\n * Returns an object that contains a new `CancelToken` and a function that, when called,\n * cancels the `CancelToken`.\n */\n static source() {\n let cancel;\n const token = new CancelToken(function executor(c) {\n cancel = c;\n });\n return {\n token,\n cancel\n };\n }\n}\n\nexport default CancelToken;\n","'use strict';\n\n/**\n * Syntactic sugar for invoking a function and expanding an array for arguments.\n *\n * Common use case would be to use `Function.prototype.apply`.\n *\n * ```js\n * function f(x, y, z) {}\n * var args = [1, 2, 3];\n * f.apply(null, args);\n * ```\n *\n * With `spread` this example can be re-written.\n *\n * ```js\n * spread(function(x, y, z) {})([1, 2, 3]);\n * ```\n *\n * @param {Function} callback\n *\n * @returns {Function}\n */\nexport default function spread(callback) {\n return function wrap(arr) {\n return callback.apply(null, arr);\n };\n}\n","'use strict';\n\nimport utils from './../utils.js';\n\n/**\n * Determines whether the payload is an error thrown by Axios\n *\n * @param {*} payload The value to test\n *\n * @returns {boolean} True if the payload is an error thrown by Axios, otherwise false\n */\nexport default function isAxiosError(payload) {\n return utils.isObject(payload) && (payload.isAxiosError === true);\n}\n","const HttpStatusCode = {\n Continue: 100,\n SwitchingProtocols: 101,\n Processing: 102,\n EarlyHints: 103,\n Ok: 200,\n Created: 201,\n Accepted: 202,\n NonAuthoritativeInformation: 203,\n NoContent: 204,\n ResetContent: 205,\n PartialContent: 206,\n MultiStatus: 207,\n AlreadyReported: 208,\n ImUsed: 226,\n MultipleChoices: 300,\n MovedPermanently: 301,\n Found: 302,\n SeeOther: 303,\n NotModified: 304,\n UseProxy: 305,\n Unused: 306,\n TemporaryRedirect: 307,\n PermanentRedirect: 308,\n BadRequest: 400,\n Unauthorized: 401,\n PaymentRequired: 402,\n Forbidden: 403,\n NotFound: 404,\n MethodNotAllowed: 405,\n NotAcceptable: 406,\n ProxyAuthenticationRequired: 407,\n RequestTimeout: 408,\n Conflict: 409,\n Gone: 410,\n LengthRequired: 411,\n PreconditionFailed: 412,\n PayloadTooLarge: 413,\n UriTooLong: 414,\n UnsupportedMediaType: 415,\n RangeNotSatisfiable: 416,\n ExpectationFailed: 417,\n ImATeapot: 418,\n MisdirectedRequest: 421,\n UnprocessableEntity: 422,\n Locked: 423,\n FailedDependency: 424,\n TooEarly: 425,\n UpgradeRequired: 426,\n PreconditionRequired: 428,\n TooManyRequests: 429,\n RequestHeaderFieldsTooLarge: 431,\n UnavailableForLegalReasons: 451,\n InternalServerError: 500,\n NotImplemented: 501,\n BadGateway: 502,\n ServiceUnavailable: 503,\n GatewayTimeout: 504,\n HttpVersionNotSupported: 505,\n VariantAlsoNegotiates: 506,\n InsufficientStorage: 507,\n LoopDetected: 508,\n NotExtended: 510,\n NetworkAuthenticationRequired: 511,\n};\n\nObject.entries(HttpStatusCode).forEach(([key, value]) => {\n HttpStatusCode[value] = key;\n});\n\nexport default HttpStatusCode;\n","'use strict';\n\nimport utils from './utils.js';\nimport bind from './helpers/bind.js';\nimport Axios from './core/Axios.js';\nimport mergeConfig from './core/mergeConfig.js';\nimport defaults from './defaults/index.js';\nimport formDataToJSON from './helpers/formDataToJSON.js';\nimport CanceledError from './cancel/CanceledError.js';\nimport CancelToken from './cancel/CancelToken.js';\nimport isCancel from './cancel/isCancel.js';\nimport {VERSION} from './env/data.js';\nimport toFormData from './helpers/toFormData.js';\nimport AxiosError from './core/AxiosError.js';\nimport spread from './helpers/spread.js';\nimport isAxiosError from './helpers/isAxiosError.js';\nimport AxiosHeaders from \"./core/AxiosHeaders.js\";\nimport adapters from './adapters/adapters.js';\nimport HttpStatusCode from './helpers/HttpStatusCode.js';\n\n/**\n * Create an instance of Axios\n *\n * @param {Object} defaultConfig The default config for the instance\n *\n * @returns {Axios} A new instance of Axios\n */\nfunction createInstance(defaultConfig) {\n const context = new Axios(defaultConfig);\n const instance = bind(Axios.prototype.request, context);\n\n // Copy axios.prototype to instance\n utils.extend(instance, Axios.prototype, context, {allOwnKeys: true});\n\n // Copy context to instance\n utils.extend(instance, context, null, {allOwnKeys: true});\n\n // Factory for creating new instances\n instance.create = function create(instanceConfig) {\n return createInstance(mergeConfig(defaultConfig, instanceConfig));\n };\n\n return instance;\n}\n\n// Create the default instance to be exported\nconst axios = createInstance(defaults);\n\n// Expose Axios class to allow class inheritance\naxios.Axios = Axios;\n\n// Expose Cancel & CancelToken\naxios.CanceledError = CanceledError;\naxios.CancelToken = CancelToken;\naxios.isCancel = isCancel;\naxios.VERSION = VERSION;\naxios.toFormData = toFormData;\n\n// Expose AxiosError class\naxios.AxiosError = AxiosError;\n\n// alias for CanceledError for backward compatibility\naxios.Cancel = axios.CanceledError;\n\n// Expose all/spread\naxios.all = function all(promises) {\n return Promise.all(promises);\n};\n\naxios.spread = spread;\n\n// Expose isAxiosError\naxios.isAxiosError = isAxiosError;\n\n// Expose mergeConfig\naxios.mergeConfig = mergeConfig;\n\naxios.AxiosHeaders = AxiosHeaders;\n\naxios.formToJSON = thing => formDataToJSON(utils.isHTMLForm(thing) ? new FormData(thing) : thing);\n\naxios.getAdapter = adapters.getAdapter;\n\naxios.HttpStatusCode = HttpStatusCode;\n\naxios.default = axios;\n\n// this module should only have a default export\nexport default axios\n","import axios, { AxiosError,AxiosInstance, AxiosRequestConfig } from 'axios';\n\nimport {\n HttpServiceConfig,\n RequestErrorInterceptor,\n RequestInterceptor,\n ResponseErrorInterceptor,\n ResponseInterceptor,\n UnauthorizedCallback,\n} from './types';\n\n/**\n * HttpService - 单例 HTTP 客户端\n *\n * 特性:\n * - 单例模式,全局唯一实例,延迟初始化(首次使用时自动创建)\n * - 支持拦截器注册(其他模块可注入 header)\n * - 统一 401/403 处理\n * - 自动携带凭证(withCredentials)\n * - 类型安全\n *\n * @example\n * ```typescript\n * // 直接使用导出的 httpService(首次使用时自动初始化)\n * import { httpService } from '@genie/agent-provider';\n *\n * // 按需设置 baseURL\n * httpService.setBaseURL('https://api.example.com');\n *\n * // 按需设置 authToken\n * httpService.setAuthToken('your-token');\n *\n * // 注册请求拦截器(注入 header)\n * httpService.registerRequestInterceptor((config) => {\n * config.headers['X-Enterprise-Id'] = 'enterprise-123';\n * return config;\n * });\n *\n * // 注册 401 回调\n * httpService.onUnauthorized(() => {\n * console.log('User unauthorized');\n * window.location.href = '/login';\n * });\n *\n * // 发起请求\n * const data = await httpService.get<Account>('/console/accounts');\n * ```\n */\nexport class HttpService {\n private static instance: HttpService | null = null;\n private axiosInstance: AxiosInstance;\n private unauthorizedCallbacks: Set<UnauthorizedCallback> = new Set();\n private config: HttpServiceConfig;\n\n /**\n * 私有构造函数(单例模式)\n */\n private constructor(config: HttpServiceConfig = {}) {\n this.config = config;\n\n // 创建 axios 实例\n this.axiosInstance = axios.create({\n baseURL: config.baseURL?.replace(/\\/$/, '') || '',\n timeout: config.timeout || 60000,\n withCredentials: config.withCredentials !== false, // 默认 true\n headers: {\n 'Content-Type': 'application/json',\n 'Accept': 'application/json',\n ...config.headers,\n },\n });\n\n // 注册默认请求拦截器(添加 Authorization)\n this.registerDefaultRequestInterceptor();\n\n // 注册默认响应拦截器(处理 401/403)\n this.registerDefaultResponseInterceptor();\n }\n\n /**\n * 获取单例实例(延迟初始化)\n */\n public static getInstance(): HttpService {\n if (!HttpService.instance) {\n HttpService.instance = new HttpService();\n }\n return HttpService.instance;\n }\n\n /**\n * 重置单例实例(主要用于测试)\n */\n public static resetInstance(): void {\n HttpService.instance = null;\n }\n\n /**\n * 注册默认请求拦截器(添加 Authorization header)\n */\n private registerDefaultRequestInterceptor(): void {\n this.axiosInstance.interceptors.request.use(\n requestConfig => {\n // 添加 Authorization header\n if (this.config.authToken) {\n requestConfig.headers = requestConfig.headers || {};\n requestConfig.headers['Authorization'] = `Bearer ${this.config.authToken}`;\n }\n return requestConfig;\n },\n error => Promise.reject(error)\n );\n }\n\n /**\n * 注册默认响应拦截器(处理 401/403)\n */\n private registerDefaultResponseInterceptor(): void {\n this.axiosInstance.interceptors.response.use(\n response => response,\n (error: AxiosError) => {\n // 统一处理 401/403\n if (error.response?.status === 401 || error.response?.status === 403) {\n console.warn('[HttpService] Unauthorized (401/403), triggering callbacks');\n this.triggerUnauthorizedCallbacks();\n }\n return Promise.reject(error);\n }\n );\n }\n\n /**\n * 注册请求拦截器\n * @param onFulfilled 请求成功拦截器\n * @param onRejected 请求失败拦截器\n * @returns 拦截器 ID(用于移除)\n *\n * @example\n * ```typescript\n * const id = httpService.registerRequestInterceptor((config) => {\n * config.headers['X-Custom-Header'] = 'value';\n * return config;\n * });\n *\n * // 移除拦截器\n * httpService.ejectRequestInterceptor(id);\n * ```\n */\n public registerRequestInterceptor(\n onFulfilled: RequestInterceptor,\n onRejected?: RequestErrorInterceptor\n ): number {\n return this.axiosInstance.interceptors.request.use(onFulfilled, onRejected);\n }\n\n /**\n * 注册响应拦截器\n * @param onFulfilled 响应成功拦截器\n * @param onRejected 响应失败拦截器\n * @returns 拦截器 ID(用于移除)\n *\n * @example\n * ```typescript\n * const id = httpService.registerResponseInterceptor((response) => {\n * console.log('Response:', response.data);\n * return response;\n * });\n *\n * // 移除拦截器\n * httpService.ejectResponseInterceptor(id);\n * ```\n */\n public registerResponseInterceptor(\n onFulfilled: ResponseInterceptor,\n onRejected?: ResponseErrorInterceptor\n ): number {\n return this.axiosInstance.interceptors.response.use(onFulfilled, onRejected);\n }\n\n /**\n * 移除请求拦截器\n * @param id 拦截器 ID\n */\n public ejectRequestInterceptor(id: number): void {\n this.axiosInstance.interceptors.request.eject(id);\n }\n\n /**\n * 移除响应拦截器\n * @param id 拦截器 ID\n */\n public ejectResponseInterceptor(id: number): void {\n this.axiosInstance.interceptors.response.eject(id);\n }\n\n /**\n * 注册 401 未授权回调\n * @param callback 回调函数\n *\n * @example\n * ```typescript\n * httpService.onUnauthorized(() => {\n * console.log('User logged out');\n * window.location.href = '/login';\n * });\n * ```\n */\n public onUnauthorized(callback: UnauthorizedCallback): void {\n this.unauthorizedCallbacks.add(callback);\n }\n\n /**\n * 移除 401 未授权回调\n * @param callback 回调函数\n */\n public offUnauthorized(callback: UnauthorizedCallback): void {\n this.unauthorizedCallbacks.delete(callback);\n }\n\n /**\n * 触发所有 401 回调\n */\n private triggerUnauthorizedCallbacks(): void {\n this.unauthorizedCallbacks.forEach(callback => {\n try {\n callback();\n } catch (error) {\n console.error('[HttpService] Error in unauthorized callback:', error);\n }\n });\n }\n\n /**\n * 更新 authToken\n * @param token 新的 token\n */\n public setAuthToken(token: string | undefined): void {\n this.config.authToken = token;\n }\n\n /**\n * 更新 baseURL\n * @param baseURL 新的 baseURL\n */\n public setBaseURL(baseURL: string): void {\n this.config.baseURL = baseURL;\n this.axiosInstance.defaults.baseURL = baseURL.replace(/\\/$/, '');\n }\n\n /**\n * GET 请求\n * @param url 请求路径\n * @param config axios 配置\n * @returns 响应数据\n */\n public async get<T = any>(url: string, config?: AxiosRequestConfig): Promise<T> {\n const response = await this.axiosInstance.get<T>(url, config);\n return response.data;\n }\n\n /**\n * POST 请求\n * @param url 请求路径\n * @param data 请求体\n * @param config axios 配置\n * @returns 响应数据\n */\n public async post<T = any>(url: string, data?: any, config?: AxiosRequestConfig): Promise<T> {\n const response = await this.axiosInstance.post<T>(url, data, config);\n return response.data;\n }\n\n /**\n * PATCH 请求\n * @param url 请求路径\n * @param data 请求体\n * @param config axios 配置\n * @returns 响应数据\n */\n public async patch<T = any>(url: string, data?: any, config?: AxiosRequestConfig): Promise<T> {\n const response = await this.axiosInstance.patch<T>(url, data, config);\n return response.data;\n }\n\n /**\n * DELETE 请求\n * @param url 请求路径\n * @param config axios 配置\n * @returns 响应数据\n */\n public async delete<T = any>(url: string, config?: AxiosRequestConfig): Promise<T> {\n const response = await this.axiosInstance.delete<T>(url, config);\n return response.data;\n }\n\n /**\n * PUT 请求\n * @param url 请求路径\n * @param data 请求体\n * @param config axios 配置\n * @returns 响应数据\n */\n public async put<T = any>(url: string, data?: any, config?: AxiosRequestConfig): Promise<T> {\n const response = await this.axiosInstance.put<T>(url, data, config);\n return response.data;\n }\n\n /**\n * 获取原始 axios 实例(用于高级场景)\n */\n public getAxiosInstance(): AxiosInstance {\n return this.axiosInstance;\n }\n}\n","import { HttpService } from './http-service';\n\nexport * from './http-service';\nexport * from './types';\n\n/**\n * 导出 httpService 单例实例\n * 首次访问时自动初始化,无需手动调用初始化方法\n *\n * @example\n * ```typescript\n * import { httpService } from '@genie/agent-provider';\n *\n * // 按需设置 baseURL\n * httpService.setBaseURL('https://api.example.com');\n *\n * // 按需设置 authToken\n * httpService.setAuthToken('your-token');\n *\n * // 直接使用\n * const data = await httpService.get('/api/endpoint');\n * ```\n */\nexport const httpService = new Proxy({} as HttpService, {\n get(target, prop) {\n const instance = HttpService.getInstance();\n return (instance as any)[prop];\n },\n});\n\n","/**\n * AccountService - 账号状态管理单例\n *\n * 提供全局的账号状态管理,任何模块都可以:\n * - 获取当前账号: accountService.getAccount()\n * - 设置账号: accountService.setAccount(account)\n * - 订阅变化: accountService.subscribe(callback)\n * - 初始化时自动注册 HTTP 拦截器,注入账号相关 headers\n */\n\nimport type { Account } from '../backend/types';\nimport { httpService } from '../http';\n\n/**\n * 账号变化回调函数类型\n */\nexport type AccountChangeCallback = (account: Account | null) => void;\n\n/**\n * AccountService 类\n *\n * 单例模式,管理全局账号状态\n */\nclass AccountService {\n /** 当前账号 */\n private account: Account | null = null;\n\n /** 订阅者列表 */\n private listeners = new Set<AccountChangeCallback>();\n\n /** 是否已初始化(首次加载完成) */\n private initialized = false;\n\n /** 初始化等待队列 */\n private initPromise: Promise<Account | null> | null = null;\n private initResolve: ((account: Account | null) => void) | null = null;\n\n /** HTTP 请求拦截器 ID(未来可用于清理拦截器) */\n private requestInterceptorId: number | null = null;\n\n constructor() {\n // 创建初始化 Promise,允许其他模块等待首次加载完成\n this.initPromise = new Promise(resolve => {\n this.initResolve = resolve;\n });\n\n // 注册 HTTP 拦截器,注入账号相关 headers\n this.registerHttpInterceptor();\n }\n\n /**\n * 注册 HTTP 请求拦截器\n * 自动注入账号相关的 headers (X-User-Id, X-Enterprise-Id, X-Tenant-Id)\n */\n private registerHttpInterceptor(): void {\n try {\n this.requestInterceptorId = httpService.registerRequestInterceptor(config => {\n const account = this.getAccount();\n\n if (account?.uid) {\n config.headers = config.headers || {};\n config.headers['X-User-Id'] = account.uid;\n }\n\n if (account?.enterpriseId) {\n config.headers = config.headers || {};\n config.headers['X-Enterprise-Id'] = account.enterpriseId;\n config.headers['X-Tenant-Id'] = account.enterpriseId;\n }\n\n return config;\n });\n\n console.log('[AccountService] HTTP interceptor registered');\n } catch (error) {\n // HttpService 可能还未初始化,延迟注册\n console.warn('[AccountService] Failed to register HTTP interceptor, will retry on first request:', error);\n }\n }\n\n /**\n * 获取当前账号\n * @returns 当前账号,未登录或未加载时返回 null\n */\n getAccount(): Account | null {\n return this.account;\n }\n\n /**\n * 设置账号\n * @param account 账号信息,登出时传 null\n */\n setAccount(account: Account | null): void {\n const prev = this.account;\n const wasInitialized = this.initialized;\n this.account = account;\n\n // 首次设置时,标记为已初始化并 resolve 等待的 Promise\n if (!this.initialized) {\n this.initialized = true;\n this.initResolve?.(account);\n }\n\n // 首次初始化时总是通知订阅者,之后只有账号真正变化时才通知\n if (!wasInitialized || prev?.uid !== account?.uid) {\n this.notifyListeners();\n }\n }\n\n /**\n * 清除账号(登出)\n */\n clearAccount(): void {\n this.setAccount(null);\n }\n\n /**\n * 订阅账号变化\n * @param callback 变化时的回调函数\n * @returns 取消订阅函数\n */\n subscribe(callback: AccountChangeCallback): () => void {\n this.listeners.add(callback);\n return () => {\n this.listeners.delete(callback);\n };\n }\n\n /**\n * 等待首次账号加载完成\n * @returns Promise<Account | null>\n */\n waitForInit(): Promise<Account | null> {\n if (this.initialized) {\n return Promise.resolve(this.account);\n }\n return this.initPromise!;\n }\n\n /**\n * 是否已初始化\n */\n isInitialized(): boolean {\n return this.initialized;\n }\n\n /**\n * 是否已登录\n */\n isLoggedIn(): boolean {\n return this.account !== null;\n }\n\n /**\n * 通知所有订阅者\n */\n private notifyListeners(): void {\n this.listeners.forEach(callback => {\n try {\n callback(this.account);\n } catch (error) {\n console.error('[AccountService] Listener error:', error);\n }\n });\n }\n\n /**\n * 重置服务状态(仅用于测试)\n */\n _reset(): void {\n this.account = null;\n this.listeners.clear();\n this.initialized = false;\n this.initPromise = new Promise(resolve => {\n this.initResolve = resolve;\n });\n\n // 清理拦截器\n if (this.requestInterceptorId !== null) {\n try {\n httpService.ejectRequestInterceptor(this.requestInterceptorId);\n this.requestInterceptorId = null;\n } catch (error) {\n console.warn('[AccountService] Failed to eject interceptor during reset:', error);\n }\n }\n }\n}\n\n/**\n * 导出单例实例\n */\nexport const accountService = new AccountService();\n","/**\n * 并发控制工具\n *\n * 提供限制并发数量的工具函数,用于控制并行请求数量\n */\n\n/**\n * 并发执行任务,限制同时运行的任务数量\n *\n * @param tasks - 任务函数数组,每个函数返回 Promise\n * @param concurrency - 最大并发数,默认 5\n * @returns 所有任务的结果数组,顺序与输入一致\n *\n * @example\n * ```typescript\n * const urls = ['url1', 'url2', 'url3', 'url4', 'url5'];\n * const tasks = urls.map(url => () => fetch(url));\n * const results = await runWithConcurrency(tasks, 2); // 最多同时 2 个请求\n * ```\n */\nexport async function runWithConcurrency<T>(\n tasks: Array<() => Promise<T>>,\n concurrency: number = 5\n): Promise<T[]> {\n if (tasks.length === 0) {\n return [];\n }\n\n // 确保并发数至少为 1\n const limit = Math.max(1, concurrency);\n\n const results: T[] = new Array(tasks.length);\n let currentIndex = 0;\n\n async function runNext(): Promise<void> {\n while (currentIndex < tasks.length) {\n const index = currentIndex++;\n const task = tasks[index];\n results[index] = await task();\n }\n }\n\n // 启动 limit 个 worker\n const workers = Array(Math.min(limit, tasks.length))\n .fill(null)\n .map(() => runNext());\n\n await Promise.all(workers);\n\n return results;\n}\n\n/**\n * 并发执行任务,返回包含成功/失败状态的结果\n *\n * @param tasks - 任务函数数组\n * @param concurrency - 最大并发数,默认 5\n * @returns 所有任务的结果数组,包含状态信息\n *\n * @example\n * ```typescript\n * const results = await runWithConcurrencySettled(tasks, 3);\n * const successes = results.filter(r => r.status === 'fulfilled');\n * const failures = results.filter(r => r.status === 'rejected');\n * ```\n */\nexport async function runWithConcurrencySettled<T>(\n tasks: Array<() => Promise<T>>,\n concurrency: number = 5\n): Promise<PromiseSettledResult<T>[]> {\n if (tasks.length === 0) {\n return [];\n }\n\n const limit = Math.max(1, concurrency);\n const results: PromiseSettledResult<T>[] = new Array(tasks.length);\n let currentIndex = 0;\n\n async function runNext(): Promise<void> {\n while (currentIndex < tasks.length) {\n const index = currentIndex++;\n const task = tasks[index];\n try {\n const value = await task();\n results[index] = { status: 'fulfilled', value };\n } catch (reason) {\n results[index] = { status: 'rejected', reason };\n }\n }\n }\n\n const workers = Array(Math.min(limit, tasks.length))\n .fill(null)\n .map(() => runNext());\n\n await Promise.all(workers);\n\n return results;\n}\n\n/**\n * 创建一个并发限制器,可复用于多次调用\n *\n * @param concurrency - 最大并发数\n * @returns 并发限制器实例\n *\n * @example\n * ```typescript\n * const limiter = createConcurrencyLimiter(3);\n *\n * // 多个地方可以共用同一个限制器\n * const result1 = await limiter.run(() => fetch(url1));\n * const result2 = await limiter.run(() => fetch(url2));\n *\n * // 或者批量执行\n * const results = await limiter.runAll([\n * () => fetch(url1),\n * () => fetch(url2),\n * ]);\n * ```\n */\nexport function createConcurrencyLimiter(concurrency: number = 5) {\n const limit = Math.max(1, concurrency);\n let running = 0;\n const queue: Array<{\n task: () => Promise<unknown>;\n resolve: (value: unknown) => void;\n reject: (reason: unknown) => void;\n }> = [];\n\n async function processQueue(): Promise<void> {\n if (running >= limit || queue.length === 0) {\n return;\n }\n\n running++;\n const item = queue.shift()!;\n\n try {\n const result = await item.task();\n item.resolve(result);\n } catch (error) {\n item.reject(error);\n } finally {\n running--;\n processQueue();\n }\n }\n\n return {\n /**\n * 执行单个任务,受并发限制\n */\n run<T>(task: () => Promise<T>): Promise<T> {\n return new Promise<T>((resolve, reject) => {\n queue.push({\n task: task as () => Promise<unknown>,\n resolve: resolve as (value: unknown) => void,\n reject,\n });\n processQueue();\n });\n },\n\n /**\n * 批量执行任务,受并发限制\n */\n async runAll<T>(tasks: Array<() => Promise<T>>): Promise<T[]> {\n return Promise.all(tasks.map(task => this.run(task)));\n },\n\n /**\n * 获取当前正在运行的任务数\n */\n get runningCount(): number {\n return running;\n },\n\n /**\n * 获取队列中等待的任务数\n */\n get pendingCount(): number {\n return queue.length;\n },\n };\n}\n","/**\n * COS Upload Service\n *\n * 负责通过预签名 URL 上传文件到腾讯云 COS\n *\n * 上传流程:\n * 1. 生成 objectKey (uploads/{timestamp}-{random}-{filename})\n * 2. 批量请求后端获取预签名 URL (同时包含 upload_url 和 download_url)\n * 3. 使用 upload_url 并发上传文件到 COS\n * 4. 返回 download_url 作为访问地址\n *\n * 注意:文件级缓存(避免重复上传相同文件)由上层 agent-ui 的 upload-cache 处理\n */\n\nimport { httpService } from '../../../http/index.js';\nimport type { Logger } from '../../client/types.js';\nimport { runWithConcurrencySettled } from '../../utils/index.js';\n\n/**\n * 预签名 URL 请求参数\n */\nexport interface PresignedURLRequest {\n /** 对象路径数组 */\n object_keys: string[];\n}\n\n/**\n * 预签名 URL 响应项\n */\nexport interface PresignedURLItem {\n /** 对象路径 */\n object_key: string;\n /** 上传 URL (PUT) */\n upload_url: string;\n /** 下载 URL (GET) */\n download_url: string;\n}\n\n/**\n * 预签名 URL 响应\n */\nexport interface PresignedURLResponse {\n /** 过期时间 (秒) */\n expire: number;\n /** 预签名 URL 列表 */\n items: PresignedURLItem[];\n}\n\n/**\n * API 响应包装类型\n */\ninterface ApiResponse<T> {\n code: number;\n message: string;\n data?: T;\n}\n\n/**\n * 上传结果\n */\nexport interface UploadResult {\n /** 是否成功 */\n success: boolean;\n /** 访问 URL (成功时返回) */\n url?: string;\n /** 对象路径 */\n objectKey?: string;\n /** 错误信息 (失败时返回) */\n error?: string;\n}\n\n/**\n * COS Upload Service 配置\n */\nexport interface CosUploadServiceOptions {\n /** Logger 实例 */\n logger?: Logger;\n /** 上传并发数,默认 3 */\n uploadConcurrency?: number;\n}\n\n/**\n * COS 上传服务\n *\n * @example\n * ```typescript\n * const service = new CosUploadService({\n * logger: console,\n * uploadConcurrency: 3,\n * });\n *\n * const result = await service.uploadFile(file);\n * if (result.success) {\n * console.log('File URL:', result.url);\n * }\n * ```\n */\nexport class CosUploadService {\n private logger?: Logger;\n private uploadConcurrency: number;\n\n constructor(options: CosUploadServiceOptions = {}) {\n this.logger = options.logger;\n this.uploadConcurrency = options.uploadConcurrency ?? 3;\n }\n\n /**\n * 生成唯一的 objectKey\n *\n * 格式: uploads/{timestamp}-{randomId}-{encodedFilename}\n */\n private generateObjectKey(filename: string): string {\n const timestamp = Date.now();\n const randomId = Math.random().toString(36).substring(2, 10);\n const encodedFilename = encodeURIComponent(filename);\n return `uploads/${timestamp}-${randomId}-${encodedFilename}`;\n }\n\n /**\n * 批量获取预签名 URL\n *\n * POST /console/as/support/presigned_url\n */\n private async getPresignedUrls(objectKeys: string[]): Promise<PresignedURLResponse> {\n const apiResponse = await httpService.post<ApiResponse<PresignedURLResponse>>(\n '/console/as/support/presigned_url',\n { object_keys: objectKeys }\n );\n\n if (!apiResponse.data) {\n throw new Error('No data in presigned URL response');\n }\n\n return apiResponse.data;\n }\n\n /**\n * 上传单个文件到 COS\n *\n * @param file - 要上传的文件\n * @returns 上传结果,包含访问 URL 或错误信息\n */\n async uploadFile(file: File): Promise<UploadResult> {\n const filename = file.name;\n this.logger?.info(`[CosUploadService] Uploading file: ${filename}`);\n\n try {\n // 1. 生成 objectKey\n const objectKey = this.generateObjectKey(filename);\n this.logger?.debug(`[CosUploadService] Generated objectKey: ${objectKey}`);\n\n // 2. 获取预签名 URL\n const presignedResponse = await this.getPresignedUrls([objectKey]);\n const presignedItem = presignedResponse.items[0];\n if (!presignedItem) {\n throw new Error('No presigned URL item returned');\n }\n\n // 3. 上传文件到 COS(使用原生 fetch,因为这是上传到外部 COS)\n const uploadResponse = await fetch(presignedItem.upload_url, {\n method: 'PUT',\n body: file,\n headers: {\n 'Content-Type': file.type || 'application/octet-stream',\n },\n });\n\n if (!uploadResponse.ok) {\n const errorText = await uploadResponse.text().catch(() => uploadResponse.statusText);\n throw new Error(`COS upload failed: ${uploadResponse.status} ${errorText}`);\n }\n this.logger?.debug('[CosUploadService] File uploaded to COS');\n\n this.logger?.info(`[CosUploadService] Upload success: ${filename}`);\n return {\n success: true,\n url: presignedItem.download_url,\n objectKey,\n };\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : 'Unknown error';\n this.logger?.error(`[CosUploadService] Upload failed: ${filename}`, error);\n return {\n success: false,\n error: errorMessage,\n };\n }\n }\n\n /**\n * 批量上传文件到 COS\n *\n * 使用并发控制,限制同时上传的文件数量\n *\n * @param files - 要上传的文件数组\n * @returns 所有文件的上传结果\n */\n async uploadFiles(files: File[]): Promise<{\n success: boolean;\n urls?: string[];\n expireSeconds?: number;\n error?: string;\n results: UploadResult[];\n }> {\n if (files.length === 0) {\n return { success: true, urls: [], results: [] };\n }\n\n this.logger?.info(`[CosUploadService] Uploading ${files.length} file(s) with concurrency ${this.uploadConcurrency}`);\n\n try {\n // 1. 为所有文件生成 objectKey\n const fileInfos = files.map(file => ({\n file,\n objectKey: this.generateObjectKey(file.name),\n }));\n\n // 2. 批量获取预签名 URL\n const objectKeys = fileInfos.map(info => info.objectKey);\n const presignedResponse = await this.getPresignedUrls(objectKeys);\n this.logger?.debug(`[CosUploadService] Got ${presignedResponse.items.length} presigned URLs`);\n\n // 3. 验证返回的数量\n if (presignedResponse.items.length !== fileInfos.length) {\n throw new Error(`Expected ${fileInfos.length} presigned URLs, got ${presignedResponse.items.length}`);\n }\n\n // 4. 创建上传任务(使用后端返回的顺序来匹配)\n // 后端会在 objectKey 前加上 userID,所以我们使用数组索引来匹配\n const uploadTasks = fileInfos.map(({ file }, index) => async (): Promise<UploadResult> => {\n const presignedItem = presignedResponse.items[index];\n if (!presignedItem) {\n return {\n success: false,\n error: `No presigned URL for ${file.name}`,\n objectKey: fileInfos[index].objectKey,\n };\n }\n\n try {\n // 使用原生 fetch 上传到 COS\n const uploadResponse = await fetch(presignedItem.upload_url, {\n method: 'PUT',\n body: file,\n headers: {\n 'Content-Type': file.type || 'application/octet-stream',\n },\n });\n\n if (!uploadResponse.ok) {\n const errorText = await uploadResponse.text().catch(() => uploadResponse.statusText);\n return {\n success: false,\n error: `COS upload failed: ${uploadResponse.status} ${errorText}`,\n objectKey: presignedItem.object_key,\n };\n }\n\n this.logger?.debug(`[CosUploadService] Uploaded: ${file.name}`);\n return {\n success: true,\n url: presignedItem.download_url,\n objectKey: presignedItem.object_key,\n };\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : 'Unknown error';\n return {\n success: false,\n error: errorMessage,\n objectKey: presignedItem.object_key,\n };\n }\n });\n\n // 5. 并发执行上传任务\n const settledResults = await runWithConcurrencySettled(uploadTasks, this.uploadConcurrency);\n\n // 6. 处理结果\n const results: UploadResult[] = settledResults.map((result, index) => {\n if (result.status === 'fulfilled') {\n return result.value;\n }\n // rejected 的情况\n return {\n success: false,\n error: result.reason instanceof Error ? result.reason.message : 'Unknown error',\n objectKey: fileInfos[index].objectKey,\n };\n });\n\n const urls = results.filter(r => r.success && r.url).map(r => r.url!);\n const failedResults = results.filter(r => !r.success);\n\n if (failedResults.length > 0) {\n const failedErrors = failedResults.map(r => r.error).join('; ');\n return {\n success: false,\n error: `${failedResults.length} file(s) failed: ${failedErrors}`,\n expireSeconds: presignedResponse.expire,\n results,\n };\n }\n\n this.logger?.info(`[CosUploadService] All ${files.length} file(s) uploaded successfully`);\n return {\n success: true,\n urls,\n expireSeconds: presignedResponse.expire,\n results,\n };\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : 'Unknown error';\n this.logger?.error('[CosUploadService] Batch upload failed', error);\n return {\n success: false,\n error: errorMessage,\n results: files.map(() => ({ success: false, error: errorMessage })),\n };\n }\n }\n}\n\nexport default CosUploadService;\n","/**\n * E2B Filesystem Implementation\n *\n * Provides FilesResource implementation using E2B Sandbox SDK.\n * Directly uses e2b SDK types.\n *\n * @see https://e2b.dev/docs/filesystem/read-write\n * @see https://e2b.dev/docs/filesystem/watch\n */\n\nimport { type EntryInfo, Sandbox, type WatchHandle,type WriteInfo } from 'e2b';\n\nimport type {\n E2BSandboxConnectionInfo,\n FilesResource,\n FilesystemEvent,\n FilesystemListOpts,\n FilesystemRequestOpts,\n WatchOpts,\n WriteEntry\n} from '../../types.js';\n\n/**\n * E2B Filesystem Implementation\n *\n * Wraps E2B Sandbox SDK's filesystem operations to implement FilesResource interface.\n *\n * @example\n * ```typescript\n * const fs = await E2BFilesystem.connect({\n * sandboxId: 'sandbox-123',\n * apiKey: 'e2b_xxx'\n * });\n *\n * // Read/write files\n * await fs.write('/test.txt', 'Hello World');\n * const content = await fs.read('/test.txt');\n *\n * // Watch for changes\n * const handle = await fs.watchDir('/workspace', (event) => {\n * console.log('File changed:', event);\n * });\n * ```\n */\nexport class E2BFilesystem implements FilesResource {\n private sandbox: Sandbox;\n\n constructor(sandbox: Sandbox) {\n this.sandbox = sandbox;\n }\n\n /**\n * Connect to an E2B Sandbox and create filesystem instance\n */\n static async connect(info: E2BSandboxConnectionInfo): Promise<E2BFilesystem> {\n const sandbox = await Sandbox.connect(info.sandboxId, {\n domain: info.domain,\n apiUrl: info.apiUrl,\n requestTimeoutMs: info.requestTimeoutMs,\n debug: info.debug,\n headers: info.headers\n });\n return new E2BFilesystem(sandbox);\n }\n\n /**\n * Get the underlying E2B Sandbox instance\n */\n getSandbox(): Sandbox {\n return this.sandbox;\n }\n\n // ============================================\n // Read 方法重载实现\n // ============================================\n\n read(path: string, opts?: FilesystemRequestOpts & { format?: 'text' }): Promise<string>;\n read(path: string, opts: FilesystemRequestOpts & { format: 'bytes' }): Promise<Uint8Array>;\n read(path: string, opts: FilesystemRequestOpts & { format: 'blob' }): Promise<Blob>;\n read(path: string, opts: FilesystemRequestOpts & { format: 'stream' }): Promise<ReadableStream<Uint8Array>>;\n read(path: string, opts?: FilesystemRequestOpts & { format?: string }): Promise<string | Uint8Array | Blob | ReadableStream<Uint8Array>> {\n return this.sandbox.files.read(path, opts as any);\n }\n\n // ============================================\n // Write 方法重载实现\n // ============================================\n\n write(path: string, data: string | ArrayBuffer | Blob | ReadableStream, opts?: FilesystemRequestOpts): Promise<WriteInfo>;\n write(files: WriteEntry[], opts?: FilesystemRequestOpts): Promise<WriteInfo[]>;\n write(pathOrFiles: string | WriteEntry[], dataOrOpts?: string | ArrayBuffer | Blob | ReadableStream | FilesystemRequestOpts, opts?: FilesystemRequestOpts): Promise<WriteInfo | WriteInfo[]> {\n if (Array.isArray(pathOrFiles)) {\n // Batch write: write(files: WriteEntry[], opts?: FilesystemRequestOpts)\n return this.sandbox.files.write(pathOrFiles, dataOrOpts as FilesystemRequestOpts);\n }\n // Single file write: write(path: string, data: ..., opts?: FilesystemRequestOpts)\n return this.sandbox.files.write(pathOrFiles, dataOrOpts as string | ArrayBuffer | Blob | ReadableStream, opts);\n }\n\n // ============================================\n // 其他方法实现\n // ============================================\n\n async list(path: string, opts?: FilesystemListOpts): Promise<EntryInfo[]> {\n return this.sandbox.files.list(path, opts);\n }\n\n async exists(path: string, opts?: FilesystemRequestOpts): Promise<boolean> {\n return this.sandbox.files.exists(path, opts);\n }\n\n async makeDir(path: string, opts?: FilesystemRequestOpts): Promise<boolean> {\n return this.sandbox.files.makeDir(path, opts);\n }\n\n async remove(path: string, opts?: FilesystemRequestOpts): Promise<void> {\n return this.sandbox.files.remove(path, opts);\n }\n\n async rename(oldPath: string, newPath: string, opts?: FilesystemRequestOpts): Promise<EntryInfo> {\n return this.sandbox.files.rename(oldPath, newPath, opts);\n }\n\n async getInfo(path: string, opts?: FilesystemRequestOpts): Promise<EntryInfo> {\n return this.sandbox.files.getInfo(path, opts);\n }\n\n async watchDir(\n path: string,\n onEvent: (event: FilesystemEvent) => void | Promise<void>,\n opts?: WatchOpts & { onExit?: (err?: Error) => void | Promise<void> }\n ): Promise<WatchHandle> {\n return this.sandbox.files.watchDir(path, onEvent, opts);\n }\n}\n\nexport default E2BFilesystem;\n","/**\n * Cloud Agent Provider\n *\n * REST API based agent provider that manages cloud-hosted agents.\n * Implements the AgentProvider interface for the unified agent API.\n */\n\nimport { accountService } from '../../../account/index.js';\nimport { httpService } from '../../../http/index.js';\nimport type {\n AgentProvider,\n CloudAgentState,\n FilesResource,\n FilesystemProvider,\n ListAgentOptions,\n ListAgentResult,\n Logger,\n PaginationInfo,\n PickFileParams,\n PickFileResponse,\n UploadFileParams,\n UploadFileResponse,\n} from '../../client/types.js';\nimport type {\n AgentStatus,\n ClientCapabilities,\n E2BSandboxConnectionInfo,\n ModelInfo\n} from '../../types.js';\nimport { ApiResponse, ArchiveConversationResponse, Conversation as ConversationDto, CreateConversationRequest, CreateConversationResponse, DeleteConversationRequest, GetConversationResponse, GetConversationSessionResponse, ListConversationRequest, ListConversationResponse, PatchConversationRequest, PatchConversationResponse, ProductConfigResponse } from './api-types.js';\nimport { CloudAgentConnection } from './cloud-connection.js';\nimport { CosUploadService } from './cos-upload-service.js';\nimport { E2BFilesystem } from './e2b-filesystem.js';\n\n/**\n * Agent data stored in cloud backend\n */\n// @ts-expect-error 没对齐\ninterface CloudAgentData {\n id: string;\n name?: string;\n description?: string;\n endpoint: string;\n authToken?: string;\n headers?: Record<string, string>;\n metadata?: Record<string, unknown>;\n createdAt?: string;\n updatedAt?: string;\n}\n\n/**\n * Configuration for CloudAgentProvider\n */\nexport interface CloudAgentProviderOptions {\n /** Base endpoint URL for agent management API (e.g., 'https://api.example.com') */\n endpoint: string;\n /** Authorization token */\n authToken?: string;\n /** Custom headers */\n headers?: Record<string, string>;\n /** Logger instance */\n logger?: Logger;\n /** Client capabilities (sent during agent initialization) */\n clientCapabilities?: ClientCapabilities;\n}\n\n/**\n * Normalize a path by resolving `.` and `..` segments\n * This is a simplified version that works in browser environment\n */\nfunction normalizePath(path: string): string {\n const segments = path.split('/');\n const result: string[] = [];\n\n for (const segment of segments) {\n if (segment === '..') {\n // Don't pop if we're at root level\n if (result.length > 0 && result[result.length - 1] !== '') {\n result.pop();\n }\n } else if (segment !== '.' && segment !== '') {\n result.push(segment);\n }\n }\n\n // Preserve leading slash for absolute paths\n return (path.startsWith('/') ? '/' : '') + result.join('/');\n}\n\n/**\n * Resolve agent:/// URI to filesystem path\n *\n * 只处理 agent:// 协议的 URI,raw path 直接透传不做任何处理\n *\n * Supported formats:\n * - `agent:///workspace/{path}` → `/workspace/{path}`\n * - `agent:///plans/{path}` → `/root/.codebuddy/plans/{path}`\n * - `agent:///{path}` → `/{path}`\n * - Raw paths (e.g. `/foo/bar`) → 直接透传,不处理\n *\n * Security: Path traversal attacks are prevented by normalizing paths\n * and verifying they stay within expected boundaries.\n */\nfunction resolveAgentUri(input: string): string {\n // 只处理 agent:// 协议,raw path 直接返回\n if (!input.startsWith('agent://')) {\n return input;\n }\n\n const path = input.slice('agent://'.length);\n if (!path.startsWith('/')) {\n return input;\n }\n\n // Normalize path to prevent traversal attacks\n const normalizedPath = normalizePath(path);\n\n // 路径映射: /plans/{path} → /root/.codebuddy/plans/{path}\n if (normalizedPath.startsWith('/plans/') || normalizedPath === '/plans') {\n const mappedPath = '/root/.codebuddy' + normalizedPath;\n // Verify the mapped path stays within /root/.codebuddy/plans\n const finalPath = normalizePath(mappedPath);\n if (!finalPath.startsWith('/root/.codebuddy/plans')) {\n throw new Error(`Invalid path: path traversal detected in ${input}`);\n }\n return finalPath;\n }\n\n return normalizedPath;\n}\n\n/**\n * Create a FilesResource wrapper that resolves agent:/// URIs\n */\nfunction createAgentFilesystem(fs: E2BFilesystem): FilesResource {\n return {\n read: (path: string, opts?: any) => fs.read(resolveAgentUri(path), opts),\n write: (pathOrFiles: any, dataOrOpts?: any, opts?: any) => {\n if (Array.isArray(pathOrFiles)) {\n const resolved = pathOrFiles.map(f => ({ ...f, path: resolveAgentUri(f.path) }));\n return fs.write(resolved, dataOrOpts);\n }\n return fs.write(resolveAgentUri(pathOrFiles), dataOrOpts, opts);\n },\n list: (path: string, opts?: any) => fs.list(resolveAgentUri(path), opts),\n exists: (path: string, opts?: any) => fs.exists(resolveAgentUri(path), opts),\n makeDir: (path: string, opts?: any) => fs.makeDir(resolveAgentUri(path), opts),\n remove: (path: string, opts?: any) => fs.remove(resolveAgentUri(path), opts),\n rename: (oldPath: string, newPath: string, opts?: any) =>\n fs.rename(resolveAgentUri(oldPath), resolveAgentUri(newPath), opts),\n getInfo: (path: string, opts?: any) => fs.getInfo(resolveAgentUri(path), opts),\n watchDir: (path: string, onEvent: any, opts?: any) =>\n fs.watchDir(resolveAgentUri(path), onEvent, opts),\n } as FilesResource;\n}\n\n/**\n * CloudAgentProvider - Manages cloud-hosted agents via REST API\n *\n * API Endpoints (base path: /console/as):\n * - POST {endpoint}/console/as/conversations - Create new conversation\n * - GET {endpoint}/console/as/conversations/{id} - Get conversation details\n * - GET {endpoint}/console/as/conversations - List all conversations\n * - POST {endpoint}/console/as/conversations/{id}/delete - Delete conversation\n * - POST {endpoint}/console/as/conversations/{id}/archive - Archive conversation\n * - POST {endpoint}/console/as/conversations/{id} - Update conversation (rename)\n * - GET {endpoint}/console/as/conversations/{id}/session - Get conversation session (includes sandboxId)\n * - GET {endpoint}/console/enterprises/{id}/models - Get available models\n *\n * The provider stores agent endpoint configurations in the cloud backend.\n * When connect() is called, it creates a CloudAgentConnection to the agent's\n * endpoint and returns an Agent instance.\n *\n * @example\n * ```typescript\n * const provider = new CloudAgentProvider({\n * endpoint: 'https://staging-copilot.tencent.com',\n * authToken: 'token'\n * });\n *\n * // List all agents (uses default pagination and sorting)\n * const allAgents = await provider.list();\n *\n * // List agents with custom pagination\n * const page2 = await provider.list({\n * page: 2,\n * size: 50\n * });\n *\n * // List agents with filtering\n * const runningAgents = await provider.list({\n * filters: [\n * { field: 'status', value: 'running' }\n * ]\n * });\n *\n * // List agents with custom sorting\n * const sortedAgents = await provider.list({\n * sort: {\n * orderBy: 'createdAt',\n * order: 'desc'\n * }\n * });\n *\n * // List agents created in last 14 days with multiple filters\n * const recentAgents = await provider.list({\n * dayRange: 14,\n * filters: [\n * { field: 'status', value: 'running,stopped' }\n * ],\n * page: 1,\n * size: 20\n * });\n *\n * // Get agent state\n * const state = await provider.get('agent-id');\n *\n * // Connect to agent\n * const agent = await provider.connect('agent-id');\n *\n * // Use agent\n * const session = await agent.sessions.create({ cwd: '/workspace' });\n *\n * // Get available models\n * const models = await provider.getModels('my-repo');\n * ```\n */\nexport class CloudAgentProvider implements AgentProvider<CloudAgentConnection>, FilesystemProvider {\n private options: CloudAgentProviderOptions;\n private logger?: Logger;\n private requestInterceptorId: number | null = null;\n\n /** Cache for filesystem instances (keyed by agentId) */\n private filesystemCache: Map<string, FilesResource> = new Map();\n\n /** Cache for agent connections (keyed by endpoint link) */\n private connectionCache: Map<string, CloudAgentConnection> = new Map();\n\n /** COS upload service instance */\n private cosUploadService: CosUploadService;\n\n /** Event listeners for provider-level events */\n private eventListeners: Map<string, Set<(...args: any[]) => void>> = new Map();\n\n constructor(options: CloudAgentProviderOptions) {\n this.options = options;\n this.logger = options.logger;\n\n // 设置 baseURL(如果提供)\n if (options.endpoint) {\n httpService.setBaseURL(options.endpoint);\n }\n\n // 设置 authToken(如果提供)\n if (options.authToken) {\n httpService.setAuthToken(options.authToken);\n }\n\n // 注册请求拦截器,只注入 options 中的 headers\n if (options.headers && Object.keys(options.headers).length > 0) {\n this.requestInterceptorId = httpService.registerRequestInterceptor(config => {\n config.headers = config.headers || {};\n // 注入 options 中的自定义 headers\n Object.entries(options.headers!).forEach(([key, value]) => {\n config.headers[key] = value;\n });\n return config;\n });\n }\n\n // Initialize COS upload service\n this.cosUploadService = new CosUploadService({\n logger: this.logger,\n });\n }\n\n /**\n * Dispose the provider and clean up resources\n */\n dispose(): void {\n this.filesystemCache.clear();\n this.connectionCache.clear();\n\n // 移除注册的拦截器\n if (this.requestInterceptorId !== null) {\n httpService.ejectRequestInterceptor(this.requestInterceptorId);\n }\n }\n\n // ============================================\n // FilesystemProvider Implementation\n // ============================================\n\n /**\n * Get the filesystem provider (returns self)\n */\n get filesystem(): FilesystemProvider {\n return this;\n }\n\n /**\n * Get filesystem resource for an agent\n *\n * Creates or returns cached filesystem instance for the agent's sandbox.\n * The filesystem supports both `agent:///` URIs and raw paths.\n *\n * @param agentId - Agent ID to get filesystem for\n * @returns FilesResource instance for the agent's sandbox (with URI support)\n *\n * @example\n * ```typescript\n * const fs = await provider.getFilesystem(agentId);\n *\n * // Use agent:/// URIs\n * const content = await fs.read('agent:///files/src/app.ts');\n * await fs.write('agent:///artifacts/output.txt', 'Hello');\n *\n * // Raw paths still work (backward compatible)\n * const content2 = await fs.read('/src/app.ts');\n * ```\n */\n async getFilesystem(agentId: string): Promise<FilesResource> {\n // Check cache first\n const cached = this.filesystemCache.get(agentId);\n if (cached) {\n return cached;\n }\n\n // Get sandbox info from backend\n const info = await this.getSandboxInfo(agentId);\n\n // Create E2BFilesystem and wrap with URI support\n const e2bFilesystem = await E2BFilesystem.connect(info);\n const filesystem = createAgentFilesystem(e2bFilesystem);\n\n this.filesystemCache.set(agentId, filesystem);\n\n this.logger?.debug(`Created filesystem for agent: ${agentId}`);\n return filesystem;\n }\n\n /**\n * Get sandbox information from backend\n *\n * Uses GET {endpoint}/console/as/conversations/{agentId}/session\n * to retrieve sandbox information. Extracts sandboxId from the session response\n * and constructs the apiUrl for E2B proxy.\n *\n * @param agentId - Agent ID\n * @returns E2B Sandbox connection information with sandboxId and apiUrl\n */\n private async getSandboxInfo(agentId: string): Promise<E2BSandboxConnectionInfo> {\n const apiResponse = await httpService.get<ApiResponse<GetConversationSessionResponse>>(\n `/console/as/conversations/${agentId}/session`\n );\n\n if (!apiResponse.data) {\n throw new Error('No data in API response');\n }\n\n // Build E2BSandboxConnectionInfo from session response\n // apiUrl is from response endpoint with /v2 replaced by /console\n const apiUrl = apiResponse.data.e2bEndpoint.replace(/^http:\\/\\//, 'https://').replace('/v2', '/console');\n const currentEnterpriseId = localStorage.getItem('currentEnterpriseId');\n\n return {\n sandboxId: apiResponse.data.sandboxId,\n apiUrl,\n accessToken: apiResponse.data.token,\n headers: {\n // 只有当 enterpriseId 存在时才添加该 header\n ...(currentEnterpriseId && { 'X-Enterprise-Id': currentEnterpriseId }),\n },\n };\n }\n\n /**\n * Get agent state by ID\n */\n async get(agentId: string): Promise<CloudAgentState | undefined> {\n try {\n const apiResponse = await httpService.get<ApiResponse<GetConversationResponse>>(\n `/console/as/conversations/${agentId}`\n );\n\n if (!apiResponse.data) {\n throw new Error('No data in API response');\n }\n return this.toAgentState(apiResponse.data);\n } catch (error: any) {\n // axios will throw error on 404\n if (error.response?.status === 404) {\n return undefined;\n }\n this.logger?.error(`Failed to get agent ${agentId}:`, error);\n throw error;\n }\n }\n\n /**\n * List all agent states with pagination information\n *\n * @param options - Optional query parameters for filtering, sorting, and pagination\n * @returns Object containing agents array and pagination info\n */\n async list(options?: ListAgentOptions): Promise<ListAgentResult<CloudAgentState>> {\n try {\n // Build request parameters with defaults and user overrides\n const params: ListConversationRequest = {\n // Default values\n page: 1,\n size: 30,\n sort: {\n order: 'desc',\n orderBy: 'status'\n },\n // User overrides\n ...options && {\n ...(options.dayRange !== undefined && { dayRange: options.dayRange }),\n ...(options.page !== undefined && { page: options.page }),\n ...(options.size !== undefined && { size: options.size }),\n ...(options.sort !== undefined && { sort: options.sort }),\n ...(options.filters !== undefined && { filters: options.filters }),\n ...(options.title !== undefined && { title: options.title }),\n }\n };\n\n const url = this.buildGetUrl('/console/as/conversations/', params);\n const apiResponse = await httpService.get<ApiResponse<ListConversationResponse>>(url);\n\n if (!apiResponse.data) {\n throw new Error('No data in API response');\n }\n\n const agents = apiResponse.data.conversations.map(a => this.toAgentState(a));\n const pagination: PaginationInfo = apiResponse.data.pagination;\n\n console.log('[CloudAgentProvider] API response:', {\n agentsCount: agents.length,\n pagination\n });\n\n return { agents, pagination };\n } catch (error) {\n this.logger?.error('Failed to list agents:', error);\n throw error;\n }\n }\n\n /**\n * Create a new conversation\n * POST {endpoint}/console/as/conversations\n */\n async create(): Promise<string> {\n try {\n // All fields are optional in CreateConversationRequest\n const createPayload: CreateConversationRequest = {\n prompt: '',\n model: 'deepseek-r1',\n };\n const apiResponse = await httpService.post<ApiResponse<CreateConversationResponse>>(\n '/console/as/conversations/',\n createPayload\n );\n\n if (!apiResponse.data) {\n throw new Error('No data in API response');\n }\n\n this.logger?.info(`Created conversation: ${apiResponse.data.id}`);\n return apiResponse.data.id;\n } catch (error) {\n this.logger?.error('Failed to create conversation:', error);\n throw error;\n }\n }\n\n /**\n * Connect to an agent and return the connection\n *\n * This method:\n * 1. Fetches the agent configuration from the backend\n * 2. Checks if an existing connection can be reused (based on endpoint link)\n * 3. Creates a CloudAgentConnection to the agent's endpoint if not cached\n * 4. Saves session connection info to the connection\n * 5. Connects and initializes the connection\n * 6. Returns the connected CloudAgentConnection\n *\n * Connection caching:\n * - Connections are cached by endpoint link to enable reuse\n * - CloudAgentConnection is responsible for handling connection health checks\n * and token expiration internally\n */\n async connect(agentId: string): Promise<CloudAgentConnection> {\n // Fetch conversation session data from backend\n let sessionApiResponse: ApiResponse<GetConversationSessionResponse>;\n try {\n sessionApiResponse = await httpService.get<ApiResponse<GetConversationSessionResponse>>(\n `/console/as/conversations/${agentId}/session`\n );\n\n if (!sessionApiResponse.data) {\n throw new Error('No data in API response');\n }\n } catch (error: any) {\n // axios will throw error on 404\n if (error.response?.status === 404) {\n throw new Error(`Agent not found: ${agentId}`);\n }\n throw new Error(`Failed to get agent for connection: ${error.message}`);\n }\n\n const sessionData = sessionApiResponse.data;\n const endpoint = sessionData.link.replace(/^http:\\/\\//, 'https://');\n const cwd = sessionApiResponse.data.cwd || '/workspace';\n\n // Fetch conversation details to get name, createdAt and status\n let agentName: string | undefined;\n let agentCreatedAt: Date | undefined;\n let agentStatus: string | undefined;\n\n try {\n const agentApiResponse = await httpService.get<ApiResponse<GetConversationResponse>>(\n `/console/as/conversations/${agentId}`\n );\n if (agentApiResponse.data) {\n agentName = agentApiResponse.data.name;\n agentCreatedAt = agentApiResponse.data.createdAt ? new Date(agentApiResponse.data.createdAt) : undefined;\n // 优先使用 sessionStatus(会话状态),fallback 到 status(Agent 状态)\n agentStatus = agentApiResponse.data.sessionStatus || agentApiResponse.data.status;\n }\n } catch (error) {\n // Ignore errors when fetching conversation details (non-critical)\n this.logger?.debug(`Failed to fetch conversation details for ${agentId}:`, error);\n }\n\n // TEMP FIX: Disable connection cache to ensure GET long-polling reconnects\n // TODO: Remove this after backend implements history message push on every session/load\n const existingConnection = this.connectionCache.get(endpoint);\n if (existingConnection) {\n this.connectionCache.delete(endpoint);\n }\n\n // Prepare client capabilities with cwd in _meta (consistent with LocalAgentProvider)\n const clientCapabilities = {\n ...this.options.clientCapabilities,\n _meta: {\n ...this.options.clientCapabilities?._meta,\n 'codebuddy.ai': {\n ...this.options.clientCapabilities?._meta?.['codebuddy.ai'],\n cwd,\n }\n }\n };\n\n // Create connection to agent's endpoint\n // Note: CloudAgentConnection is responsible for:\n // - Connection health checks (reconnection on disconnect)\n // - Token expiration handling (refresh or re-authentication)\n const connection = new CloudAgentConnection(sessionData.sessionId || agentId, {\n endpoint,\n authToken: sessionData.token,\n // Backend does not provide custom headers currently\n // headers: data.headers,\n logger: this.logger,\n clientCapabilities\n }, cwd);\n\n // Save session connection info to the connection\n connection.setSessionConnectionInfo({\n sessionId: sessionData.sessionId,\n agentId: sessionData.id,\n link: sessionData.link,\n token: sessionData.token,\n sandboxId: sessionData.sandboxId,\n expireAt: sessionData.expireAt,\n cwd\n });\n\n // Connect and initialize\n try {\n await connection.connect();\n } catch (error) {\n this.logger?.error(`Failed to connect to agent ${agentId}:`, error);\n throw error;\n }\n\n // Cache the connection\n this.connectionCache.set(endpoint, connection);\n\n // Clean up cache when connection is disconnected\n connection.once('disconnected', () => {\n this.connectionCache.delete(endpoint);\n this.logger?.debug(`Connection removed from cache: ${endpoint}`);\n });\n\n this.logger?.info(`Connected to agent: ${agentId}`);\n\n // Emit sessionCreated event for agent-new-adapter to create conversation\n // SessionInfo format: Use agentId as both id and agentId (1:1 mapping per SessionManager)\n this.emitEvent('sessionCreated', {\n id: agentId, // Use agentId as sessionId (1:1 mapping in current design)\n agentId: agentId,\n name: agentName,\n status: agentStatus as AgentStatus,\n cwd: cwd,\n createdAt: agentCreatedAt\n });\n\n return connection;\n }\n\n /**\n * Delete a conversation by ID\n * POST {endpoint}/console/as/conversations/{agentId}/delete\n */\n async delete(agentId: string): Promise<boolean> {\n try {\n const requestBody: DeleteConversationRequest = { id: agentId };\n await httpService.post(\n `/console/as/conversations/${agentId}/delete`,\n requestBody\n );\n\n return true;\n } catch (error: any) {\n // axios will throw error on 404\n if (error.response?.status === 404) {\n return false;\n }\n this.logger?.error(`Failed to delete conversation ${agentId}:`, error);\n throw error;\n }\n }\n\n /**\n * Archive a conversation by ID\n * POST {endpoint}/console/as/conversations/{agentId}/archive\n *\n * @param agentId - Conversation ID to archive\n * @returns ArchiveConversationResponse containing the archived conversation ID\n *\n * @example\n * ```typescript\n * const result = await provider.archive('agent-123');\n * console.log('Archived conversation:', result.id);\n * ```\n */\n async archive(agentId: string): Promise<ArchiveConversationResponse> {\n try {\n const apiResponse = await httpService.post<ApiResponse<ArchiveConversationResponse>>(\n `/console/as/conversations/${agentId}/archive`\n );\n\n if (!apiResponse.data) {\n // 如果后端没有返回 data,使用传入的 agentId\n this.logger?.info(`Archived conversation: ${agentId}`);\n return { id: agentId };\n }\n\n this.logger?.info(`Archived conversation: ${apiResponse.data.id}`);\n return apiResponse.data;\n } catch (error) {\n this.logger?.error(`Failed to archive conversation ${agentId}:`, error);\n throw error;\n }\n }\n\n /**\n * Rename a conversation by ID\n * POST {endpoint}/console/as/conversations/{agentId}\n *\n * @param agentId - Conversation ID to rename\n * @param title - New title for the conversation\n * @returns PatchConversationResponse containing the renamed conversation ID\n *\n * @example\n * ```typescript\n * const result = await provider.rename('agent-123', 'New Title');\n * console.log('Renamed conversation:', result.id);\n * ```\n */\n async rename(agentId: string, title: string): Promise<PatchConversationResponse> {\n try {\n const body: PatchConversationRequest = { title };\n const apiResponse = await httpService.post<ApiResponse<PatchConversationResponse>>(\n `/console/as/conversations/${agentId}`,\n body\n );\n\n if (!apiResponse.data) {\n // 如果后端没有返回 data,使用传入的 agentId\n this.logger?.info(`Renamed conversation: ${agentId} to \"${title}\"`);\n return { id: agentId };\n }\n\n this.logger?.info(`Renamed conversation: ${apiResponse.data.id} to \"${title}\"`);\n return apiResponse.data;\n } catch (error) {\n this.logger?.error(`Failed to rename conversation ${agentId}:`, error);\n throw error;\n }\n }\n\n /**\n * Get available models from product configuration\n *\n * GET {endpoint}/console/enterprises/{personal|enterpriseId}/models?repos[]={repo}\n *\n * This method fetches the models from /console/enterprises API\n * and extracts the models array from the response.\n *\n * @param repo - Optional repository URL for context-specific config\n * @returns Array of ModelInfo with full model details\n */\n async getModels(repo?: string): Promise<ModelInfo[]> {\n try {\n // Get user info from accountService to determine personal vs enterprise account\n const account = accountService.getAccount();\n\n // Check if user is logged in\n if (!account) {\n this.logger?.warn('[CloudAgentProvider] No account info available, cannot get models');\n return [];\n }\n\n // Determine account type: personal or enterprise\n const accountIdentifier = account.enterpriseId || 'personal';\n\n // Build URL with new console API format\n let url = `/console/enterprises/${accountIdentifier}/models`;\n if (repo) {\n // Keep same parameter format: ?repos[]=encoded_repo_url\n url += `?repos[]=${encodeURIComponent(repo)}`;\n }\n\n // Build custom headers specific to this request\n // Note: Authorization, X-User-Id, X-Enterprise-Id, X-Tenant-Id, X-Product, X-User-Agent\n // are already added by the request interceptor\n const headers: Record<string, string> = {\n 'Accept': 'application/json, text/plain, */*',\n 'X-Requested-With': 'XMLHttpRequest',\n 'X-Request-ID': this.generateRequestId(),\n };\n\n this.logger?.debug(`[CloudAgentProvider] GET ${url}`);\n\n // Parse response - expecting { data: ProductConfiguration }\n const apiResponse = await httpService.get<ApiResponse<ProductConfigResponse>>(url, {\n headers\n });\n if (!apiResponse.data) {\n this.logger?.warn('[CloudAgentProvider] No data in config response, returning empty models');\n return [];\n }\n\n // Extract models from product configuration\n const productConfig = apiResponse.data;\n const models = productConfig.models ?? [];\n\n this.logger?.info(`[CloudAgentProvider] Retrieved ${models.length} models from /console/enterprises API`);\n\n // Convert LanguageModel to ModelInfo\n return models.map((model): ModelInfo => ({\n id: model.id,\n name: model.name ?? model.id,\n description: model.description,\n credits: model.credits,\n configurable: model.configurable,\n configured: model.configured,\n isDefault: model.isDefault,\n supportsImages: model.supportsImages,\n supportsReasoning: model.supportsReasoning,\n onlyReasoning: model.onlyReasoning,\n disabledMultimodal: model.disabledMultimodal,\n disabled: model.disabled,\n disabledReason: model.disabledReason,\n disabledAction: model.disabledAction,\n }));\n } catch (error) {\n this.logger?.error('[CloudAgentProvider] Failed to get models:', error);\n throw error;\n }\n }\n\n /**\n * Generate a unique request ID\n */\n private generateRequestId(): string {\n // Generate UUID without dashes\n return 'xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx'.replace(/x/g, () =>\n Math.floor(Math.random() * 16).toString(16)\n );\n }\n\n // ============================================\n // File Picker (Browser Environment)\n // ============================================\n\n /**\n * Common image MIME types for filtering\n */\n private static readonly IMAGE_MIME_TYPES = [\n 'image/png',\n 'image/jpeg',\n 'image/jpg',\n 'image/gif',\n 'image/webp',\n 'image/svg+xml',\n 'image/bmp',\n ];\n\n /**\n * Pick files using browser's native file input\n *\n * @param params - File picker parameters\n * @returns Response with selected file paths (filenames in browser)\n */\n async pickFile(params?: PickFileParams): Promise<PickFileResponse> {\n return new Promise(resolve => {\n const input = document.createElement('input');\n input.type = 'file';\n input.style.display = 'none';\n\n // Set accept attribute\n if (params?.filters && params.filters.length > 0) {\n const acceptTypes: string[] = [];\n for (const filter of params.filters) {\n for (const ext of filter.extensions) {\n const mimeType = this.extensionToMimeType(ext);\n acceptTypes.push(mimeType || `.${ext}`);\n }\n }\n input.accept = acceptTypes.join(',');\n } else {\n input.accept = CloudAgentProvider.IMAGE_MIME_TYPES.join(',');\n }\n\n input.multiple = params?.canSelectMany ?? false;\n\n input.onchange = () => {\n const files = input.files;\n if (!files || files.length === 0) {\n resolve({ files: [], canceled: true });\n } else {\n const fileArray = Array.from(files);\n this.logger?.info(`Picked ${fileArray.length} file(s)`);\n resolve({ files: fileArray, canceled: false });\n }\n document.body.removeChild(input);\n };\n\n input.oncancel = () => {\n resolve({ files: [], canceled: true });\n document.body.removeChild(input);\n };\n\n document.body.appendChild(input);\n input.click();\n });\n }\n\n /**\n * Convert file extension to MIME type\n */\n private extensionToMimeType(ext: string): string | null {\n const extLower = ext.toLowerCase().replace(/^\\./, '');\n const mimeMap: Record<string, string> = {\n 'png': 'image/png',\n 'jpg': 'image/jpeg',\n 'jpeg': 'image/jpeg',\n 'gif': 'image/gif',\n 'webp': 'image/webp',\n 'svg': 'image/svg+xml',\n 'bmp': 'image/bmp',\n 'ico': 'image/x-icon',\n };\n return mimeMap[extLower] || null;\n }\n\n // ============================================\n // File Upload (Browser Environment)\n // ============================================\n\n /**\n * Upload files to cloud storage via COS presigned URL\n *\n * @param params - files array (File objects in browser)\n * @returns Response with corresponding cloud URLs\n */\n async uploadFile(params: UploadFileParams): Promise<UploadFileResponse> {\n this.logger?.info(`[CloudAgentProvider] uploadFile called for ${params.files.length} file(s)`);\n\n // Filter out string paths (only File objects are supported in browser)\n const files = params.files.filter((f): f is File => typeof f !== 'string');\n if (files.length === 0) {\n return {\n success: false,\n error: 'No valid File objects provided',\n };\n }\n\n // Use CosUploadService for actual upload\n const result = await this.cosUploadService.uploadFiles(files);\n\n return {\n success: result.success,\n urls: result.urls,\n expireSeconds: result.expireSeconds,\n error: result.error,\n };\n }\n\n // ============================================\n // Event Emitter Implementation\n // ============================================\n\n /**\n * Register event listener\n * @param event - Event name\n * @param handler - Event handler function\n */\n on(event: string, handler: (...args: any[]) => void): () => void {\n if (!this.eventListeners.has(event)) {\n this.eventListeners.set(event, new Set());\n }\n this.eventListeners.get(event)!.add(handler);\n\n // Return unsubscribe function\n return () => {\n this.off(event, handler);\n };\n }\n\n /**\n * Unregister event listener\n * @param event - Event name\n * @param handler - Event handler function\n */\n off(event: string, handler: (...args: any[]) => void): void {\n const listeners = this.eventListeners.get(event);\n if (listeners) {\n listeners.delete(handler);\n }\n }\n\n /**\n * Emit event to all registered listeners\n * @param event - Event name\n * @param args - Event arguments\n */\n private emitEvent(event: string, ...args: any[]): void {\n const listeners = this.eventListeners.get(event);\n if (listeners && listeners.size > 0) {\n this.logger?.debug(`Emitting event: ${event}`, args);\n for (const handler of listeners) {\n try {\n handler(...args);\n } catch (error) {\n this.logger?.error(`Error in event handler for ${event}:`, error);\n }\n }\n }\n }\n\n // ============================================\n // Helpers\n // ============================================\n\n private toAgentState(data: ConversationDto): CloudAgentState {\n // 优先使用 sessionStatus(会话状态),fallback 到 status(Agent 状态)\n const status = data.sessionStatus || data.status;\n return {\n id: data.id,\n name: data.name,\n description: data.summary,\n type: 'cloud',\n status: status as AgentStatus,\n createdAt: data.createdAt ? new Date(data.createdAt) : undefined,\n capabilities: this.options.clientCapabilities,\n };\n }\n\n /**\n * Helper: 将 GET 请求的 body 转换为 URL 查询参数\n */\n private buildGetUrl(path: string, params?: unknown): string {\n if (!params) {\n return path;\n }\n\n const searchParams = new URLSearchParams();\n for (const [key, value] of Object.entries(params as Record<string, unknown>)) {\n if (value !== undefined && value !== null) {\n const stringValue = typeof value === 'object'\n ? JSON.stringify(value)\n : String(value);\n searchParams.append(key, stringValue);\n }\n }\n\n const queryString = searchParams.toString();\n return queryString ? `${path}?${queryString}` : path;\n }\n}\n\nexport default CloudAgentProvider;\n","/**\n * ActiveSessionImpl - Implements the ActiveSession interface\n *\n * Represents an active session with its resources and operations.\n * Session is the primary API surface for client interactions.\n */\n\nimport type { PromptResponse as SdkPromptResponse,SessionNotification } from '@agentclientprotocol/sdk';\n\nimport type { AgentStatus,FilesResource, PromptParams } from '../types.js';\nimport type {\n ActiveSession,\n AgentCapabilities,\n AgentConnection,\n AgentState,\n Artifact,\n ArtifactsResource,\n ArtifactType,\n AvailableCommand,\n Logger,\n ModelInfo,\n PromptResponse,\n PromptsResource,\n QuestionAnswers,\n SessionAgentOperations,\n SessionConnectionInfo,\n SessionEventHandler,\n SessionEvents,\n SessionMode} from './types.js';\n\n/**\n * Event listener type\n */\ntype EventListener<T> = (data: T) => void | Promise<void>;\n\n/**\n * Filesystem getter function type\n * Returns a FilesResource instance for file operations\n */\nexport type FilesystemGetter = () => Promise<FilesResource>;\n\n/**\n * Options for creating an ActiveSessionImpl instance\n */\nexport interface ActiveSessionImplOptions {\n /** Logger instance */\n logger?: Logger;\n /** Getter function for filesystem resource (provided by SessionManager) */\n getFilesystem?: FilesystemGetter;\n /** Session connection information (for cloud sessions) */\n connectionInfo?: SessionConnectionInfo;\n}\n\n/**\n * ActiveSessionImpl - Implements the ActiveSession interface\n *\n * This class wraps an AgentConnection and provides the session-centric API.\n * It is created by SessionManager when creating or loading sessions.\n *\n * @example\n * ```typescript\n * // Created by client.sessions.new() or client.sessions.load()\n * const session = await client.sessions.new({ cwd: '/workspace' });\n *\n * // Access agent state\n * console.log(session.agentState.status);\n *\n * // Send prompt\n * const response = await session.prompts.send({ content: 'Hello!' });\n *\n * // Cleanup\n * session.disconnect();\n * ```\n */\nexport class ActiveSessionImpl implements ActiveSession {\n private _id: string;\n private _agentId: string;\n private _availableModes?: SessionMode[];\n private _currentMode?: string;\n private _availableModels?: ModelInfo[];\n private _currentModelId?: string;\n private _availableCommands: AvailableCommand[] = [];\n private logger?: Logger;\n private connection: AgentConnection;\n private _getFilesystem?: FilesystemGetter;\n private _connectionInfo?: SessionConnectionInfo;\n\n // Event emitter storage\n private listeners: Map<keyof SessionEvents, Set<EventListener<unknown>>> = new Map();\n private onceListeners: Map<keyof SessionEvents, Set<EventListener<unknown>>> = new Map();\n\n /**\n * Agent operations namespace\n */\n readonly agent: SessionAgentOperations;\n\n /**\n * Prompts resource namespace\n */\n readonly prompts: PromptsResource;\n\n /**\n * Artifacts resource namespace\n */\n readonly artifacts: ArtifactsResource;\n\n /**\n * Files resource namespace (lazily loaded via getter)\n */\n readonly files: FilesResource;\n\n /**\n * Create an ActiveSessionImpl instance\n *\n * @param sessionId - Session ID\n * @param agentId - Agent ID\n * @param connection - Already connected AgentConnection\n * @param options - Additional options\n */\n constructor(\n sessionId: string,\n agentId: string,\n connection: AgentConnection,\n options: ActiveSessionImplOptions = {}\n ) {\n this._id = sessionId;\n this._agentId = agentId;\n this.connection = connection;\n this.logger = options.logger;\n this._getFilesystem = options.getFilesystem;\n this._connectionInfo = options.connectionInfo;\n\n // Set up event forwarding from connection\n this.setupConnectionEvents(connection);\n\n // Initialize agent operations namespace\n this.agent = this.createAgentOperations();\n\n // Initialize resource namespaces\n this.prompts = this.createPromptsResource();\n this.artifacts = this.createArtifactsResource();\n this.files = this.createFilesResource();\n }\n\n // ============================================\n // Properties\n // ============================================\n\n /**\n * Session ID\n */\n get id(): string {\n return this._id;\n }\n\n /**\n * Agent ID\n */\n get agentId(): string {\n return this._agentId;\n }\n\n /**\n * Agent state (live connection state)\n * Returns LocalAgentState or CloudAgentState based on transport type\n */\n get agentState(): AgentState {\n return {\n id: this._agentId,\n status: this.connection.state as AgentStatus,\n capabilities: this.connection.capabilities,\n type: this.connection.transport,\n cwd: this.connection.cwd || ''\n };\n }\n\n /**\n * Get agent capabilities (available after connection)\n */\n get capabilities(): AgentCapabilities | undefined {\n return this.connection.capabilities;\n }\n\n /**\n * Available session modes\n */\n get availableModes(): SessionMode[] | undefined {\n return this._availableModes;\n }\n\n /**\n * Current session mode\n */\n get currentMode(): string | undefined {\n return this._currentMode;\n }\n\n /**\n * Available models for this session\n */\n get availableModels(): ModelInfo[] | undefined {\n return this._availableModels;\n }\n\n /**\n * Current model ID\n */\n get currentModelId(): string | undefined {\n return this._currentModelId;\n }\n\n /**\n * Available slash commands\n *\n * When Agent sends available_commands_update, this list is automatically updated.\n * Commands can be accessed directly without waiting for events.\n */\n get availableCommands(): AvailableCommand[] {\n return this._availableCommands;\n }\n\n /**\n * Set available commands (called when available_commands_update is received)\n */\n setAvailableCommands(commands: AvailableCommand[]): void {\n this._availableCommands = commands;\n this.logger?.info(`Session ${this._id}: Available commands updated, count: ${commands.length}`);\n }\n\n /**\n * Check if the session is active\n */\n get isActive(): boolean {\n return this.connection.isInitialized;\n }\n\n /**\n * Session connection information (only available for cloud sessions)\n * 会话连接信息,包括sandboxId、link、token等\n */\n get connectionInfo(): SessionConnectionInfo | undefined {\n return this._connectionInfo;\n }\n\n /**\n * Set session modes (called after create/load)\n */\n setModes(availableModes?: SessionMode[], currentMode?: string): void {\n this._availableModes = availableModes;\n this._currentMode = currentMode;\n }\n\n /**\n * Set available models (called after create/load)\n */\n setModels(availableModels?: ModelInfo[], currentModelId?: string): void {\n this._availableModels = availableModels;\n this._currentModelId = currentModelId;\n }\n\n // ============================================\n // Agent Operations Namespace\n // ============================================\n\n private createAgentOperations(): SessionAgentOperations {\n // eslint-disable-next-line @typescript-eslint/no-this-alias\n const self = this;\n return {\n get id(): string {\n return self._agentId;\n },\n get state(): AgentState {\n return self.agentState;\n },\n get isConnected(): boolean {\n return self.connection.isInitialized;\n },\n get capabilities(): AgentCapabilities | undefined {\n return self.connection.capabilities;\n }\n };\n }\n\n // ============================================\n // Prompts Resource\n // ============================================\n\n private createPromptsResource(): PromptsResource {\n return {\n send: async (params: PromptParams): Promise<PromptResponse> => {\n const connection = this.getConnectionOrThrow();\n const response = await connection.prompt(this._id, params);\n return this.mapPromptResponse(response);\n },\n\n stream: (params: PromptParams): AsyncIterable<SessionNotification> => {\n const connection = this.getConnectionOrThrow();\n return connection.promptStream(this._id, params);\n },\n\n cancel: async (): Promise<void> => {\n const connection = this.getConnectionOrThrow();\n await connection.cancel(this._id);\n }\n };\n }\n\n // ============================================\n // Artifacts Resource\n // ============================================\n\n private createArtifactsResource(): ArtifactsResource {\n // Artifact management has been simplified - these methods are no longer supported\n const notSupported = () => {\n throw new Error('Artifact management is no longer supported through this API');\n };\n\n return {\n list: async (_params?: { type?: ArtifactType }): Promise<Artifact[]> => {\n notSupported();\n return [];\n },\n\n retrieve: async (_artifactId: string): Promise<Artifact> => {\n notSupported();\n return undefined as unknown as Artifact;\n },\n\n content: async (_artifactId: string): Promise<string> => {\n notSupported();\n return '';\n }\n };\n }\n\n // ============================================\n // Files Resource\n // ============================================\n\n /**\n * Create files resource with lazy-loaded filesystem\n *\n * The filesystem is lazily loaded on first use to avoid unnecessary\n * connections to the sandbox. The actual filesystem instance is obtained\n * via the getter function provided by SessionManager.\n */\n private createFilesResource(): FilesResource {\n // eslint-disable-next-line @typescript-eslint/no-this-alias\n const self = this;\n let filesPromise: Promise<FilesResource> | null = null;\n\n /**\n * Get or create the filesystem instance\n */\n const getFs = async (): Promise<FilesResource> => {\n if (!self._getFilesystem) {\n throw new Error('Filesystem not available: provider does not support filesystem operations');\n }\n if (!filesPromise) {\n filesPromise = self._getFilesystem();\n }\n return filesPromise;\n };\n\n return {\n // Read operations - 支持多种格式重载\n read: (async (path: string, opts?: any) => (await getFs()).read(path, opts)) as FilesResource['read'],\n\n // Write operations - 支持单文件和批量写入重载\n write: (async (pathOrFiles: any, dataOrOpts?: any, opts?: any) => {\n const fs = await getFs();\n if (Array.isArray(pathOrFiles)) {\n // Batch write\n return fs.write(pathOrFiles, dataOrOpts);\n }\n // Single file write\n return fs.write(pathOrFiles, dataOrOpts, opts);\n }) as FilesResource['write'],\n\n // List with depth support\n list: async (path, opts) => (await getFs()).list(path, opts),\n exists: async (path, opts) => (await getFs()).exists(path, opts),\n makeDir: async (path, opts) => (await getFs()).makeDir(path, opts),\n remove: async (path, opts) => (await getFs()).remove(path, opts),\n rename: async (oldPath, newPath, opts) => (await getFs()).rename(oldPath, newPath, opts),\n\n // 新增 getInfo 方法\n getInfo: async (path, opts) => (await getFs()).getInfo(path, opts),\n\n // Watch operations - 支持扩展的 opts\n watchDir: async (path, onEvent, opts) => (await getFs()).watchDir(path, onEvent, opts)\n };\n }\n\n // ============================================\n // Permission Management\n // ============================================\n\n /**\n * Resolve a permission request\n */\n resolvePermission(requestId: string, optionId: string): boolean {\n return this.connection.resolvePermission(requestId, optionId);\n }\n\n /**\n * Reject a permission request\n */\n rejectPermission(requestId: string, reason?: string): boolean {\n return this.connection.rejectPermission(requestId, reason);\n }\n\n // ============================================\n // Question Management (ask_followup_question)\n // ============================================\n\n /**\n * Answer a question request with user's selections\n */\n answerQuestion(toolCallId: string, answers: QuestionAnswers): boolean {\n return this.connection.answerQuestion(toolCallId, answers);\n }\n\n /**\n * Cancel a question request\n */\n cancelQuestion(toolCallId: string, reason?: string): boolean {\n return this.connection.cancelQuestion(toolCallId, reason);\n }\n\n // ============================================\n // Tool Callback Management\n // ============================================\n\n /**\n * Callback for tool operations (skip or cancel)\n * @param toolCallId Tool call ID\n * @param toolName Tool name\n * @param action Action to perform ('skip' or 'cancel')\n */\n async toolCallback(toolCallId: string, toolName: string, action: 'skip' | 'cancel'): Promise<{ success: boolean; error?: string }> {\n const connection = this.getConnectionOrThrow();\n return await connection.toolCallback(this._id, toolCallId, toolName, action);\n }\n\n // ============================================\n // Session Mode Management\n // ============================================\n\n /**\n * Set the current session mode\n *\n * @param modeId - The mode ID to switch to (must be in availableModes)\n * @throws Error if modeId is not in availableModes or connection fails\n *\n * @example\n * ```typescript\n * // Switch to 'code' mode\n * await session.setMode('code');\n *\n * // Switch to 'architect' mode\n * await session.setMode('architect');\n * ```\n */\n async setMode(modeId: string): Promise<void> {\n // Validate modeId if availableModes is set\n if (this._availableModes) {\n const modeExists = this._availableModes.some(m => m.id === modeId);\n if (!modeExists) {\n const availableIds = this._availableModes.map(m => m.id).join(', ');\n throw new Error(`Invalid modeId: \"${modeId}\". Available modes: ${availableIds}`);\n }\n }\n\n const connection = this.getConnectionOrThrow();\n await connection.setSessionMode(this._id, modeId);\n\n // Update internal state\n this._currentMode = modeId;\n }\n\n /**\n * Set the current session model\n *\n * @param modelId - The model ID to switch to\n * @example\n * ```typescript\n * // Switch to Claude Sonnet 4\n * await session.setSessionModel('claude-sonnet-4-20250514');\n *\n * // Switch to GPT-4o\n * await session.setSessionModel('gpt-4o');\n * ```\n */\n async setSessionModel(modelId: string): Promise<void> {\n const connection = this.getConnectionOrThrow();\n await connection.setSessionModel(this._id, modelId);\n }\n\n // ============================================\n // Event Subscription\n // ============================================\n\n /**\n * Subscribe to session events\n */\n on<K extends keyof SessionEvents>(event: K, handler: SessionEventHandler<K>): this {\n if (!this.listeners.has(event)) {\n this.listeners.set(event, new Set());\n }\n this.listeners.get(event)!.add(handler as EventListener<unknown>);\n return this;\n }\n\n /**\n * Unsubscribe from session events\n */\n off<K extends keyof SessionEvents>(event: K, handler: SessionEventHandler<K>): this {\n const eventListeners = this.listeners.get(event);\n if (eventListeners) {\n eventListeners.delete(handler as EventListener<unknown>);\n }\n const onceEventListeners = this.onceListeners.get(event);\n if (onceEventListeners) {\n onceEventListeners.delete(handler as EventListener<unknown>);\n }\n return this;\n }\n\n /**\n * Subscribe to a session event once\n */\n once<K extends keyof SessionEvents>(event: K, handler: SessionEventHandler<K>): this {\n if (!this.onceListeners.has(event)) {\n this.onceListeners.set(event, new Set());\n }\n this.onceListeners.get(event)!.add(handler as EventListener<unknown>);\n return this;\n }\n\n /**\n * Emit an event to all registered listeners\n */\n private emit<K extends keyof SessionEvents>(event: K, data: SessionEvents[K]): boolean {\n const regularListeners = this.listeners.get(event);\n const onceEventListeners = this.onceListeners.get(event);\n\n let hasListeners = false;\n\n // Call regular listeners\n if (regularListeners && regularListeners.size > 0) {\n hasListeners = true;\n for (const listener of regularListeners) {\n try {\n const result = listener(data);\n if (result instanceof Promise) {\n result.catch(err => {\n console.error(`Error in async event listener for '${String(event)}':`, err);\n });\n }\n } catch (err) {\n console.error(`Error in event listener for '${String(event)}':`, err);\n }\n }\n }\n\n // Call once listeners and remove them\n if (onceEventListeners && onceEventListeners.size > 0) {\n hasListeners = true;\n const listenersToCall = Array.from(onceEventListeners);\n this.onceListeners.delete(event);\n\n for (const listener of listenersToCall) {\n try {\n const result = listener(data);\n if (result instanceof Promise) {\n result.catch(err => {\n console.error(`Error in async once event listener for '${String(event)}':`, err);\n });\n }\n } catch (err) {\n console.error(`Error in once event listener for '${String(event)}':`, err);\n }\n }\n }\n\n return hasListeners;\n }\n\n /**\n * Remove all listeners for an event\n */\n private removeAllListeners<K extends keyof SessionEvents>(event?: K): void {\n if (event !== undefined) {\n this.listeners.delete(event);\n this.onceListeners.delete(event);\n } else {\n this.listeners.clear();\n this.onceListeners.clear();\n }\n }\n\n // ============================================\n // Lifecycle\n // ============================================\n\n /**\n * Disconnect from the session/agent\n */\n disconnect(): void {\n this.connection.disconnect();\n this.removeAllListeners();\n this.logger?.info(`Session ${this._id}: Disconnected`);\n }\n\n /**\n * Symbol.dispose for 'using' keyword support\n * Automatically disconnects and cleans up when session goes out of scope\n *\n * @example\n * ```typescript\n * {\n * using session = await client.sessions.new({ cwd: '/workspace' });\n * // ... use session\n * } // session automatically disposed\n * ```\n */\n [Symbol.dispose](): void {\n this.disconnect();\n }\n\n // ============================================\n // Helpers\n // ============================================\n\n private getConnectionOrThrow(): AgentConnection {\n if (!this.connection.isInitialized) {\n throw new Error(`Session ${this._id}: Connection not initialized.`);\n }\n return this.connection;\n }\n\n private setupConnectionEvents(connection: AgentConnection): void {\n // Connection lifecycle events\n connection.on('connected', () => {\n this.emit('connected', undefined as never);\n });\n\n connection.on('disconnected', () => {\n this.emit('disconnected', undefined as never);\n });\n\n connection.on('error', error => {\n this.emit('error', error);\n });\n\n // Session updates\n connection.on('sessionUpdate', update => {\n this.emit('sessionUpdate', update);\n });\n\n // Artifact events\n connection.on('artifactCreated', artifact => {\n console.log('[Session] Forwarding artifactCreated:', {\n artifactUri: artifact.uri,\n artifactType: artifact.type,\n });\n this.emit('artifactCreated', artifact);\n });\n\n connection.on('artifactUpdated', artifact => {\n console.log('[Session] Forwarding artifactUpdated:', {\n artifactUri: artifact.uri,\n artifactType: artifact.type,\n });\n this.emit('artifactUpdated', artifact);\n });\n\n connection.on('artifactDeleted', artifact => {\n console.log('[Session] Forwarding artifactDeleted:', { artifactUri: artifact.uri });\n this.emit('artifactDeleted', artifact);\n });\n\n // Permission requests\n connection.on('permissionRequest', request => {\n this.emit('permissionRequest', request);\n });\n\n // Question requests\n connection.on('questionRequest', request => {\n this.emit('questionRequest', request);\n });\n\n connection.on('questionCancelled', () => {\n this.prompts.cancel();\n });\n\n // Usage updates\n connection.on('usageUpdate', usage => {\n this.emit('usageUpdate', usage);\n });\n\n // Checkpoint events\n connection.on('checkpointCreated', checkpoint => {\n this.emit('checkpointCreated', checkpoint);\n });\n\n connection.on('checkpointUpdated', checkpoint => {\n this.emit('checkpointUpdated', checkpoint);\n });\n\n // Command events\n connection.on('command', command => {\n console.log('[Session] Forwarding command:', {\n action: command.action,\n paramsKeys: command.params ? Object.keys(command.params) : []\n });\n this.emit('command', command);\n });\n }\n\n private mapPromptResponse(response: SdkPromptResponse): PromptResponse {\n return {\n stopReason: response.stopReason as PromptResponse['stopReason'],\n _meta: response._meta ?? undefined\n };\n }\n}\n","/**\n * SessionManager - Manages session lifecycle and connections\n *\n * Provides the core implementation for session-centric API operations:\n * - list() - Lists sessions (mapped from agents)\n * - createSession() - Creates new session (auto-creates agent)\n * - loadSession() - Loads existing session (finds agent by sessionId)\n */\n\nimport { ActiveSessionImpl } from './session.js';\nimport type {\n ActiveSession,\n AgentProvider,\n CreateSessionParams,\n ListAgentOptions,\n ListAgentResult,\n LoadSessionParams,\n Logger,\n ModelInfo,\n SessionInfo} from './types.js';\n\n/**\n * Options for creating a SessionManager instance\n */\nexport interface SessionManagerOptions {\n /** Agent provider (required) */\n provider: AgentProvider;\n /** Logger instance */\n logger?: Logger;\n}\n\n/**\n * SessionManager - Session lifecycle management\n *\n * This class manages the relationship between sessions and agents.\n * Since the backend is agent-centric, SessionManager handles the mapping:\n * - Sessions are views over agents\n * - sessionId may equal agentId in simple cases\n *\n * Features:\n * - Session caching: reuses existing ActiveSession instances\n * - Automatic cleanup on session disconnect\n *\n * @example\n * ```typescript\n * const manager = new SessionManager({ provider, logger });\n *\n * // List sessions\n * const sessions = await manager.listSessions();\n *\n * // Create new session\n * const session = await manager.createSession({ cwd: '/workspace' });\n *\n * // Load existing session (returns cached instance if available)\n * const loaded = await manager.loadSession({ sessionId: 'xxx', cwd: '/workspace' });\n * ```\n */\nexport class SessionManager {\n private provider: AgentProvider;\n private logger?: Logger;\n\n constructor(options: SessionManagerOptions) {\n this.provider = options.provider;\n this.logger = options.logger;\n }\n\n /**\n * List all sessions with pagination info (mapped from agents)\n *\n * Each agent maps to a session. The sessionId is derived from the agent.\n * Cloud: Returns server-side filtered/sorted/paginated results\n * Local: Returns client-side filtered/sorted results (synthetic pagination)\n *\n * @param options - Optional query parameters for filtering, sorting, and pagination\n */\n async listSessions(options?: ListAgentOptions): Promise<ListAgentResult<SessionInfo>> {\n const result = await this.provider.list(options);\n const sessions = result.agents.map(agent => ({\n // Use agentId as sessionId (since sessions are 1:1 with agents in current design)\n id: agent.id,\n agentId: agent.id,\n name: agent.name,\n status: agent.status,\n createdAt: agent.createdAt,\n lastActivityAt: agent.updatedAt,\n // cwd is only available for local agents\n cwd: agent.type === 'local' ? agent.cwd : undefined,\n isPlayground: agent.isPlayground\n }));\n\n console.log('[SessionManager] Returning sessions:', { count: sessions.length, pagination: result.pagination });\n return {\n agents: sessions,\n pagination: result.pagination\n };\n }\n\n /**\n * Create a new session\n *\n * Steps:\n * 1. Create new agent (if provider supports it) or use existing\n * 2. Connect to agent\n * 3. Call ACP newSession\n * 4. Register session mapping (for LocalAgentProvider)\n * 5. Return ActiveSession instance\n */\n async createSession(params: CreateSessionParams): Promise<ActiveSession> {\n this.logger?.info('Creating new session');\n\n // Step 1: Create new agent if provider supports it\n let agentId: string;\n\n if (this.provider.create) {\n // Pass params to create() - LocalAgentProvider uses cwd as agentId\n agentId = await this.provider.create(params);\n this.logger?.debug(`Created new agent: ${agentId}`);\n } else {\n // If provider doesn't support create, we need to throw\n // The provider must support creating agents for sessions.new()\n throw new Error('Provider does not support creating agents. Use sessions.load() with an existing sessionId.');\n }\n\n // Step 2: Connect to agent\n const connection = await this.provider.connect(agentId);\n this.logger?.debug(`Connected to agent: ${agentId}`);\n\n // Step 3: Create session via ACP\n const response = await connection.createSession({\n _meta: params._meta,\n cwd: params.cwd,\n mcpServers: params.mcpServers\n });\n\n // Step 4: Register session mapping (for LocalAgentProvider to support loadSession)\n if (this.provider.registerSession) {\n this.provider.registerSession(response.sessionId, agentId);\n this.logger?.debug(`Registered session mapping: ${response.sessionId} → ${agentId}`);\n }\n\n // Step 5: Create and return ActiveSession\n // Get connectionInfo from CloudAgentConnection if available\n const connectionInfo = (connection as any).sessionConnectionInfo;\n\n const session = new ActiveSessionImpl(\n response.sessionId,\n agentId,\n connection,\n {\n logger: this.logger,\n // Bind sessionId (not agentId) to create a getter function for filesystem\n // Note: conversation-channel-router binds sessionId to windowId,\n // so we must use sessionId for filesystem routing\n getFilesystem: this.provider.filesystem\n ? () => this.provider.filesystem!.getFilesystem(response.sessionId)\n : undefined,\n // Pass connectionInfo from CloudAgentConnection (if available)\n connectionInfo\n }\n );\n\n // Set modes from response\n session.setModes(\n response.modes?.availableModes,\n response.modes?.currentModeId\n );\n\n // Set models from response (convert ACP ModelInfo to local ModelInfo)\n if (response.models?.availableModels) {\n const localModels: ModelInfo[] = response.models.availableModels.map(m => ({\n id: m.modelId,\n name: m.name,\n description: m.description ?? undefined\n }));\n session.setModels(localModels, response.models?.currentModelId);\n }\n\n this.logger?.info(`Session created: ${response.sessionId}`);\n return session;\n }\n\n /**\n * Load an existing session\n *\n * Steps:\n * 1. Check cache for existing session\n * 2. Find agent by sessionId (sessionId === agentId in current design)\n * 3. Connect to agent\n * 4. Create ActiveSession instance\n * 5. Execute onSessionCreated callback (if provided) to allow early setup (e.g., event listeners)\n * 6. Call ACP loadSession\n * 7. Return ActiveSession instance (cached)\n */\n async loadSession(params: LoadSessionParams): Promise<ActiveSession> {\n this.logger?.info(`Loading session: ${params.sessionId}`);\n\n // Step 2: Find agent by sessionId\n // In current design, sessionId === agentId\n const agentId = params.sessionId;\n\n // Verify agent exists\n const agentState = await this.provider.get(agentId);\n if (!agentState) {\n throw new Error(`Session not found: ${params.sessionId}`);\n }\n\n // Step 3: Connect to agent\n const connection = await this.provider.connect(agentId);\n this.logger?.debug(`Connected to agent: ${agentId}`);\n\n // Step 4: Create ActiveSession instance (BEFORE loadSession call)\n // Get connectionInfo from CloudAgentConnection if available\n const connectionInfo = (connection as any).sessionConnectionInfo;\n\n const session = new ActiveSessionImpl(\n params.sessionId,\n agentId,\n connection,\n {\n logger: this.logger,\n // Bind sessionId (not agentId) to create a getter function for filesystem\n // Note: conversation-channel-router binds sessionId to windowId,\n // so we must use sessionId for filesystem routing\n getFilesystem: this.provider.filesystem\n ? () => this.provider.filesystem!.getFilesystem(params.sessionId)\n : undefined,\n // Pass connectionInfo from CloudAgentConnection (if available)\n connectionInfo\n }\n );\n\n // Step 5: Execute onSessionCreated callback (BEFORE loadSession)\n // This allows callers to set up event listeners before historical messages arrive\n if (params.onSessionCreated) {\n await params.onSessionCreated(session);\n }\n\n // Step 6: Load session via ACP (may trigger historical message push)\n const response = await connection.loadSession({\n sessionId: params.sessionId,\n cwd: agentState.type === 'local' ? agentState.cwd : params.cwd,\n mcpServers: params.mcpServers\n });\n\n // Step 7: Set modes and models from response\n session.setModes(\n response.modes?.availableModes,\n response.modes?.currentModeId\n );\n\n // Set models from response (convert ACP ModelInfo to local ModelInfo)\n if (response.models?.availableModels) {\n const localModels: ModelInfo[] = response.models.availableModels.map(m => ({\n id: m.modelId,\n name: m.name,\n description: m.description ?? undefined\n }));\n session.setModels(localModels, response.models?.currentModelId);\n }\n\n this.logger?.info(`Session loaded: ${params.sessionId}`);\n return session;\n }\n}\n","/**\n * AgentClient - Session-centric API\n *\n * Provides a unified entry point for managing sessions.\n * Sessions are the primary API surface; agents are internal implementation.\n */\n\nimport type { ModelInfo } from '../types.js';\nimport { SessionManager } from './session-manager.js';\nimport type {\n AgentClientOptions,\n AgentProvider,\n BatchPluginOperationRequest,\n BatchPluginOperationResult,\n ClientSessionsResource,\n InitializeWorkspaceParams,\n InitializeWorkspaceResponse,\n ListAgentOptions,\n Logger,\n ModelsResource,\n PickFileParams,\n PickFileResponse,\n PickFolderParams,\n PickFolderResponse,\n SearchFileParams,\n SearchFileResponse,\n SessionsResourceEventHandler,\n SessionsResourceEvents,\n UploadFileParams,\n UploadFileResponse,\n WorkspaceInfo} from './types.js';\n\n// ============================================\n// AgentClient - Session-Centric API\n// ============================================\n\n/**\n * AgentClient - Session-centric client\n *\n * Provides a session-centric API that internally manages agents.\n * Users interact with sessions; the agent lifecycle is handled internally.\n *\n * @example\n * ```typescript\n * // Create client with a provider\n * const provider = new CloudAgentProvider({\n * endpoint: 'https://api.example.com',\n * authToken: 'token'\n * });\n *\n * const client = new AgentClient({\n * provider,\n * logger: console\n * });\n *\n * // List all sessions\n * const sessions = await client.sessions.list();\n *\n * // Create new session (auto-creates agent and connects)\n * const session = await client.sessions.create({ cwd: '/workspace' });\n * console.log(session.agentState.status); // agent status\n * console.log(session.agentState.id); // agent ID\n *\n * // Send prompt\n * await session.prompts.send({ content: 'Hello' });\n *\n * // Get available models\n * const models = await client.sessions.models.list('my-repo');\n *\n * // Use 'using' keyword for automatic cleanup\n * {\n * using session = await client.sessions.create({ cwd: '/workspace' });\n * // ... use session\n * } // session automatically disposed\n *\n * // Or manually disconnect\n * session.disconnect();\n *\n * // Load existing session\n * const loadedSession = await client.sessions.load({\n * sessionId: 'xxx',\n * cwd: '/workspace'\n * });\n * ```\n */\nexport class AgentClient {\n private logger?: Logger;\n private provider: AgentProvider;\n private sessionManager: SessionManager;\n\n /**\n * Sessions resource namespace (primary API entry point)\n */\n readonly sessions: ClientSessionsResource;\n\n /**\n * 运行环境类型\n * - 'local': IDE 本地环境\n * - 'cloud': 云端环境\n */\n readonly environmentType: 'local' | 'cloud';\n\n constructor(options: AgentClientOptions) {\n this.logger = options.logger;\n this.provider = options.provider;\n this.environmentType = options.environmentType ?? 'cloud';\n\n // Initialize session manager\n this.sessionManager = new SessionManager({\n provider: this.provider,\n logger: this.logger\n });\n\n // Initialize sessions resource\n this.sessions = this.createSessionsResource();\n }\n\n // ============================================\n // Sessions Resource\n // ============================================\n\n private createSessionsResource(): ClientSessionsResource {\n return {\n list: async (options?: ListAgentOptions) => this.sessionManager.listSessions(options),\n\n create: async params => this.sessionManager.createSession(params),\n\n load: async params => {\n console.log('[AgentClient] sessions.load called:', params.sessionId);\n return this.sessionManager.loadSession(params);\n },\n\n archive: async (sessionId: string): Promise<{ id: string }> => {\n this.logger?.debug('AgentClient.sessions.archive called', { sessionId });\n\n try {\n // Check if provider supports archive\n if (this.provider.archive) {\n const result = await this.provider.archive(sessionId);\n this.logger?.info('Session archived successfully', { sessionId });\n return result;\n }\n\n // If provider does not support archive, throw error\n throw new Error('Provider does not support archive method');\n } catch (error) {\n this.logger?.error('Failed to archive session', error);\n throw error;\n }\n },\n\n rename: async (sessionId: string, title: string): Promise<{ id: string }> => {\n this.logger?.debug('AgentClient.sessions.rename called', { sessionId, title });\n\n try {\n // Check if provider supports rename\n if (this.provider.rename) {\n const result = await this.provider.rename(sessionId, title);\n this.logger?.info('Session renamed successfully', { sessionId, title });\n return result;\n }\n\n // If provider does not support rename, throw error\n throw new Error('Provider does not support rename method');\n } catch (error) {\n this.logger?.error('Failed to rename session', error);\n throw error;\n }\n },\n\n move: async (sessionId: string): Promise<{ id: string }> => {\n this.logger?.debug('AgentClient.sessions.move called', { sessionId });\n\n try {\n // Check if provider supports move\n if (this.provider.move) {\n const result = await this.provider.move(sessionId);\n this.logger?.info('Session moved successfully', { sessionId });\n return result;\n }\n\n // If provider does not support move, throw error\n throw new Error('Provider does not support move method');\n } catch (error) {\n this.logger?.error('Failed to move session', error);\n throw error;\n }\n },\n\n // Initialize workspace for future sessions\n initializeWorkspace: async (params: InitializeWorkspaceParams): Promise<InitializeWorkspaceResponse> => {\n this.logger?.debug('AgentClient.sessions.initializeWorkspace called', params);\n\n try {\n // openWorkspace 是 LocalAgentProvider 特有的能力\n if (this.provider.openWorkspace) {\n const result = await this.provider.openWorkspace(params);\n this.logger?.info('Workspace opened successfully', { cwd: params.cwd });\n return result;\n }\n\n // 如果 provider 不支持 openWorkspace,返回成功(向后兼容)\n this.logger?.warn('Provider does not support openWorkspace');\n return { success: true };\n\n } catch (error) {\n this.logger?.error('Failed to initialize workspace', error);\n return {\n success: false,\n error: error instanceof Error ? error.message : 'Unknown error'\n };\n }\n },\n\n // Get current workspaces list\n getCurrentWorkspaces: async (filter?: { activeOnly?: boolean }): Promise<WorkspaceInfo[]> => {\n this.logger?.debug('AgentClient.sessions.getCurrentWorkspaces called', filter);\n\n try {\n // getCurrentWorkspaces 是 LocalAgentProvider 特有的能力\n if ('getCurrentWorkspaces' in this.provider && typeof this.provider.getCurrentWorkspaces === 'function') {\n const result = await this.provider.getCurrentWorkspaces(filter);\n this.logger?.info('Current workspaces retrieved', { count: result.length });\n return result;\n }\n\n // 如果 provider 不支持 getCurrentWorkspaces,返回空数组(向后兼容)\n this.logger?.warn('Provider does not support getCurrentWorkspaces');\n return [];\n\n } catch (error) {\n this.logger?.error('Failed to get current workspaces', error);\n return [];\n }\n },\n\n // Event methods - forward to provider if supported\n on: <K extends keyof SessionsResourceEvents>(\n event: K,\n handler: SessionsResourceEventHandler<K>\n ): void => {\n if (this.provider.on) {\n this.provider.on(event as string, handler as (...args: any[]) => void);\n } else {\n this.logger?.warn(`Provider does not support event registration: ${String(event)}`);\n }\n },\n\n off: <K extends keyof SessionsResourceEvents>(\n event: K,\n handler: SessionsResourceEventHandler<K>\n ): void => {\n if (this.provider.off) {\n this.provider.off(event as string, handler as (...args: any[]) => void);\n } else {\n this.logger?.warn(`Provider does not support event unregistration: ${String(event)}`);\n }\n },\n\n openWorkspace: async (params: InitializeWorkspaceParams): Promise<InitializeWorkspaceResponse> => {\n try {\n if (this.provider && this.provider.openWorkspace) {\n const result = await this.provider.openWorkspace(params);\n this.logger?.info('Workspace opened successfully', { cwd: params.cwd });\n return result;\n }\n return { success: false, error: 'Provider does not support openWorkspace' };\n } catch (error) {\n this.logger?.error('Failed to open workspace', error);\n return {\n success: false,\n error: error instanceof Error ? error.message : 'Unknown error'\n };\n }\n },\n\n pickFile: async (params?: PickFileParams): Promise<PickFileResponse> => {\n try {\n if (this.provider && this.provider.pickFile) {\n const result = await this.provider.pickFile(params);\n this.logger?.info('File picker completed', { fileCount: result.files.length, canceled: result.canceled });\n return result;\n }\n return { files: [], canceled: true, error: 'Provider does not support pickFile' };\n } catch (error) {\n this.logger?.error('Failed to pick file', error);\n return {\n files: [],\n canceled: true,\n error: error instanceof Error ? error.message : 'Unknown error'\n };\n }\n },\n\n pickFolder: async (params?: PickFolderParams): Promise<PickFolderResponse> => {\n try {\n if (this.provider && this.provider.pickFolder) {\n const result = await this.provider.pickFolder(params);\n this.logger?.info('Folder picker completed', { folderPaths: result.folderPaths, canceled: result.canceled });\n return result;\n }\n return { folderPaths: [], canceled: true, error: 'Provider does not support pickFolder' };\n } catch (error) {\n this.logger?.error('Failed to pick folder', error);\n return {\n folderPaths: [],\n canceled: true,\n error: error instanceof Error ? error.message : 'Unknown error'\n };\n }\n },\n\n uploadFile: async (params: UploadFileParams): Promise<UploadFileResponse> => {\n try {\n if (this.provider && this.provider.uploadFile) {\n const result = await this.provider.uploadFile(params);\n this.logger?.info('File upload completed', { count: params.files.length, success: result.success });\n return result;\n }\n return { success: false, error: 'Provider does not support uploadFile' };\n } catch (error) {\n this.logger?.error('Failed to upload file', error);\n return {\n success: false,\n error: error instanceof Error ? error.message : 'Unknown error'\n };\n }\n },\n\n searchFile: async (params: SearchFileParams): Promise<SearchFileResponse> => {\n try {\n if (this.provider && this.provider.searchFile) {\n const result = await this.provider.searchFile(params);\n this.logger?.info('File search completed', { resultCount: result.results.length, hasError: !!result.error });\n return result;\n }\n return { results: [], error: 'Provider does not support searchFile' };\n } catch (error) {\n this.logger?.error('Failed to search file', error);\n return {\n results: [],\n error: error instanceof Error ? error.message : 'Unknown error'\n };\n }\n },\n\n batchTogglePlugins: async (request: BatchPluginOperationRequest): Promise<BatchPluginOperationResult> => {\n try {\n if (this.provider && this.provider.batchTogglePlugins) {\n const result = await this.provider.batchTogglePlugins(request);\n this.logger?.info('Batch toggle plugins completed', {\n succeededCount: result.succeededPlugins.length,\n failedCount: result.failedPlugins.length\n });\n return result;\n }\n return {\n success: false,\n succeededPlugins: [],\n failedPlugins: request.items.map(item => ({\n ...item,\n error: 'Provider does not support batchTogglePlugins'\n }))\n };\n } catch (error) {\n this.logger?.error('Failed to batch toggle plugins', error);\n return {\n success: false,\n succeededPlugins: [],\n failedPlugins: request.items.map(item => ({\n ...item,\n error: error instanceof Error ? error.message : 'Unknown error'\n }))\n };\n }\n },\n\n getInstalledPlugins: async (forceRefresh?: boolean) => {\n try {\n if (this.provider && 'getInstalledPlugins' in this.provider && typeof this.provider.getInstalledPlugins === 'function') {\n const result = await this.provider.getInstalledPlugins(forceRefresh);\n this.logger?.info('Got installed plugins', { count: result?.length ?? 0 });\n return result;\n }\n this.logger?.warn('Provider does not support getInstalledPlugins');\n return [];\n } catch (error) {\n this.logger?.error('Failed to get installed plugins', error);\n return [];\n }\n },\n\n installPlugins: async (pluginNames, marketplaceName, installScope) => {\n try {\n if (this.provider && 'installPlugins' in this.provider && typeof this.provider.installPlugins === 'function') {\n const result = await this.provider.installPlugins(pluginNames, marketplaceName, installScope);\n this.logger?.info('Install plugins', { pluginNames, marketplaceName, success: result.success });\n return result;\n }\n this.logger?.warn('Provider does not support installPlugins');\n return { success: false, error: 'Provider does not support installPlugins' };\n } catch (error) {\n this.logger?.error('Failed to install plugins', error);\n return {\n success: false,\n error: error instanceof Error ? error.message : 'Unknown error'\n };\n }\n },\n\n getSupportPlugins: async () => {\n try {\n if (this.provider && 'getSupportPlugins' in this.provider && typeof this.provider.getSupportPlugins === 'function') {\n const result = await this.provider.getSupportPlugins();\n this.logger?.info('Got support plugins', { count: result?.length ?? 0 });\n return result;\n }\n this.logger?.warn('Provider does not support getSupportPlugins');\n return [];\n } catch (error) {\n this.logger?.error('Failed to get support plugins', error);\n return [];\n }\n },\n\n // Models resource - delegates to provider's getModels method\n models: this.createModelsResource(),\n };\n }\n\n private createModelsResource(): ModelsResource {\n return {\n list: async (repo: string): Promise<ModelInfo[]> => {\n // Check if provider supports getModels\n if (this.provider.getModels) {\n return this.provider.getModels(repo);\n }\n throw new Error('Provider does not support getModels method');\n }\n };\n }\n\n // ============================================\n // Lifecycle\n // ============================================\n\n /**\n * Dispose the client\n *\n * Note: Active sessions are not automatically disposed.\n * The caller is responsible for disconnecting sessions they created.\n */\n dispose(): void {\n this.logger?.info('AgentClient disposed');\n }\n}\n\nexport default AgentClient;\n","/**\n * API type definitions for AgentClient\n * Session-centric API design\n */\n\nimport type { RequestPermissionRequest,SessionNotification } from '@agentclientprotocol/sdk';\nimport type {\n Artifact,\n ArtifactEvent,\n ArtifactType,\n CheckpointInfo,\n ClientEvents,\n EventListener,\n QuestionAnswers,\n QuestionRequest,\n UsageUpdate} from '@genie/agent-client-protocol';\n\nimport type { BatchPluginOperationRequest, BatchPluginOperationResult } from '../../backend/types.js';\nimport type { AvailableCommand } from '../providers/local-agent-provider/acp/types.js';\nimport type {\n Agent as AgentInfo,\n AgentCapabilities,\n AgentConnection,\n AgentStatus,\n ArtifactsConfig,\n ArtifactTypeConfig,\n ClientCapabilities,\n CodebuddyAgentMeta,\n CodebuddyClientMeta,\n E2BSandboxConnectionInfo,\n // e2b SDK types (re-exported from types.js)\n EntryInfo,\n FilesResource,\n Filesystem,\n FilesystemProvider,\n McpServerConfig,\n // Model types\n ModelInfo,\n PromptContentBlock,\n PromptParams,\n Session,\n SessionMode} from '../types.js';\n\n// Re-export commonly used types\nexport type {\n AgentInfo,\n AgentConnection,\n AgentStatus,\n Session,\n SessionMode,\n PromptParams,\n PromptContentBlock,\n AgentCapabilities,\n ClientCapabilities,\n // codebuddy.ai extension types\n ArtifactTypeConfig,\n ArtifactsConfig,\n CodebuddyClientMeta,\n CodebuddyAgentMeta,\n // Other types\n Artifact,\n ArtifactType,\n ArtifactEvent,\n UsageUpdate,\n ClientEvents,\n EventListener,\n SessionNotification,\n RequestPermissionRequest,\n QuestionRequest,\n QuestionAnswers,\n // Filesystem types\n FilesResource,\n FilesystemProvider,\n McpServerConfig,\n E2BSandboxConnectionInfo,\n // e2b SDK types (re-exported from types.js)\n EntryInfo,\n Filesystem,\n // Model types\n ModelInfo,\n // ACP types\n AvailableCommand,\n // Plugin types\n BatchPluginOperationRequest,\n BatchPluginOperationResult\n};\n\n// ============================================\n// Session Connection Info (for cloud sessions)\n// ============================================\n\n/**\n * Session connection information\n * 包含连接到Agent会话所需的所有信息,包括sandbox连接凭证\n */\nexport interface SessionConnectionInfo {\n /** Session ID */\n sessionId: string;\n /** Agent ID */\n agentId: string;\n /** Session endpoint URL (Agent的WebSocket/HTTP端点) */\n link: string;\n /** Session token (JWT格式的认证令牌) */\n token: string;\n /** Sandbox ID (E2B沙箱的唯一标识) */\n sandboxId: string;\n /** Session expiration timestamp (unix timestamp) */\n expireAt: number;\n /** Current working directory (optional) */\n cwd?: string;\n}\n\n// ============================================\n// Agent State (Unified agent state object)\n// ============================================\n\n/**\n * Agent 来源类型\n */\nexport type AgentStateType = 'local' | 'cloud';\n\n/**\n * 云端 Agent 可见性\n */\nexport type CloudAgentVisibility = 'PRIVATE' | 'PUBLIC' | 'TEAM';\n\n/**\n * 云端 Agent 来源信息\n */\nexport interface CloudAgentSourceInfo {\n /** 提供商: github, gitlab 等 */\n provider: string;\n /** 分支/引用 */\n ref: string;\n /** 仓库路径 */\n repository: string;\n}\n\n/**\n * 云端 Agent 目标信息\n */\nexport interface CloudAgentTarget {\n /** 是否自动创建 PR */\n autoCreatePr: boolean;\n /** 分支名称 */\n branchName?: string;\n /** PR URL */\n prUrl?: string;\n /** Agent URL */\n url?: string;\n}\n\n/**\n * AgentState 基础接口\n * 所有类型的 AgentState 都必须实现此接口\n */\nexport interface BaseAgentState {\n /** Unique agent ID */\n id: string;\n /** Display name */\n name?: string;\n /** Description */\n description?: string;\n /** Agent type */\n type: AgentStateType;\n /** Current connection status */\n status: AgentStatus;\n /** Agent capabilities (available after connection) */\n capabilities?: AgentCapabilities;\n /** When the agent was created */\n createdAt?: Date;\n /** When the agent was last updated */\n updatedAt?: Date;\n /** 是否为 playground */\n isPlayground?: boolean;\n}\n\n/**\n * LocalAgentState - 本地 Agent 状态\n * 来自本地 IPC 通信的 Agent\n */\nexport interface LocalAgentState extends BaseAgentState {\n type: 'local';\n /** 工作目录 */\n cwd: string;\n}\n\n/**\n * CloudAgentState - 云端 Agent 状态\n * 来自远程 API 的云端实例\n */\nexport interface CloudAgentState extends BaseAgentState {\n type: 'cloud';\n}\n\n/**\n * AgentState - Unified agent state object exposed to client users\n *\n * This is the primary way clients access agent information.\n * Uses discriminated union pattern to distinguish between local and cloud agents.\n */\nexport type AgentState = LocalAgentState | CloudAgentState;\n\n/**\n * 类型守卫:判断是否为 LocalAgentState\n */\nexport function isLocalAgentState(state: AgentState): state is LocalAgentState {\n return state.type === 'local';\n}\n\n/**\n * 类型守卫:判断是否为 CloudAgentState\n */\nexport function isCloudAgentState(state: AgentState): state is CloudAgentState {\n return state.type === 'cloud';\n}\n\n/**\n * Logger interface\n */\nexport interface Logger {\n debug(message: string, ...args: unknown[]): void;\n info(message: string, ...args: unknown[]): void;\n warn(message: string, ...args: unknown[]): void;\n error(message: string, ...args: unknown[]): void;\n}\n\n// ============================================\n// Agent List Query Options\n// ============================================\n\n/**\n * Filter condition for listing agents\n */\nexport interface ListAgentFilter {\n /** Filter field name (e.g., 'status', 'name') */\n field: string;\n /** Filter value (comma-separated for multiple values) */\n value: string;\n}\n\n/**\n * Sort options for listing agents\n */\nexport interface ListAgentSort {\n /** Sort field (e.g., 'createdAt', 'status') */\n orderBy: string;\n /** Sort direction */\n order?: 'asc' | 'desc';\n}\n\n/**\n * Query options for listing agents\n *\n * These options are supported by both CloudAgentProvider and LocalAgentProvider.\n * Cloud: Server-side filtering, sorting, and pagination\n * Local: Client-side filtering and sorting, no pagination (returns all)\n */\nexport interface ListAgentOptions {\n /**\n * Page number (starts from 1)\n * Cloud: Used for API pagination\n * Local: Ignored (returns all sessions)\n */\n page?: number;\n\n /**\n * Page size\n * Cloud: Number of items per page (default 20, max 100)\n * Local: Ignored (returns all sessions)\n */\n size?: number;\n\n /**\n * Sort options\n * Cloud: Sorts results by specified field and order\n * Local: Sorts results by specified field and order\n */\n sort?: ListAgentSort;\n\n /**\n * Filter conditions\n * Cloud: Filters results by specified field values\n * Local: Filters results by specified field values\n */\n filters?: ListAgentFilter[];\n\n /**\n * Day range filter (e.g., agents created in last N days)\n * Cloud: Filters by creation date\n * Local: Filters by creation date\n */\n dayRange?: number;\n\n /**\n * Title search keyword (matches agent title)\n * Cloud: Server-side search\n * Local: Client-side search\n */\n title?: string;\n}\n\n/**\n * Pagination metadata returned from list operations\n */\nexport interface PaginationInfo {\n /** Current page number (starts from 1) */\n page: number;\n /** Page size */\n size: number;\n /** Total number of items */\n total: number;\n /** Total number of pages */\n totalPages: number;\n /** Whether there is a next page */\n hasNext: boolean;\n /** Whether there is a previous page */\n hasPrev: boolean;\n}\n\n/**\n * Response from list operations that includes pagination\n */\nexport interface ListAgentResult<T = AgentState> {\n /** List of agent states or session info */\n agents: T[];\n /** Pagination information */\n pagination: PaginationInfo;\n}\n\n// ============================================\n// Session-Centric Types\n// ============================================\n\n/**\n * Session information (returned by list, mapped from Agent)\n */\nexport interface SessionInfo {\n /** Session ID (from agent.session) */\n id: string;\n /** Associated agent ID */\n agentId: string;\n /** Agent name */\n name?: string;\n /** Agent status */\n status: AgentStatus;\n /** When the session/agent was created */\n createdAt?: Date;\n /** Last activity timestamp */\n lastActivityAt?: Date;\n /** Working directory (for local agents) */\n cwd?: string;\n /** Whether the session is a playground */\n isPlayground?: boolean;\n}\n\n/**\n * Parameters for creating a new session\n */\nexport interface CreateSessionParams {\n /** Working directory */\n cwd: string;\n /** MCP server configurations */\n mcpServers?: McpServerConfig[];\n _meta?: Record<string, unknown>;\n}\n\n/**\n * Parameters for loading an existing session\n */\nexport interface LoadSessionParams {\n /** Session ID to load (required) */\n sessionId: string;\n /** Working directory */\n cwd: string;\n /** MCP server configurations */\n mcpServers?: McpServerConfig[];\n /** Callback executed right after session instance is created, before connection.loadSession() is called */\n onSessionCreated?: (session: ActiveSession) => Promise<void> | void;\n}\n\n/**\n * Parameters for initializing a workspace\n */\nexport interface InitializeWorkspaceParams {\n /** Working directory */\n cwd: string;\n /** MCP server configurations */\n mcpServers?: McpServerConfig[];\n /** Whether to activate the workspace window to foreground (default true) */\n needActivated?: boolean;\n}\n\n/**\n * Response for workspace initialization\n */\nexport interface InitializeWorkspaceResponse {\n /** Whether initialization was successful */\n success: boolean;\n /** Error message (if failed) */\n error?: string;\n}\n\n// ============================================\n// Resource Interfaces\n// ============================================\n\n/**\n * Prompts resource interface (ACP verbs)\n * Operations use the current session automatically\n */\nexport interface PromptsResource {\n /** Send a prompt and wait for completion */\n send(params: PromptParams): Promise<PromptResponse>;\n\n /** Stream a prompt (yields session updates) */\n stream(params: PromptParams): AsyncIterable<SessionNotification>;\n\n /** Cancel an ongoing prompt */\n cancel(): Promise<void>;\n}\n\n/**\n * Artifacts resource interface\n */\nexport interface ArtifactsResource {\n /** List all artifacts */\n list(params?: { type?: ArtifactType }): Promise<Artifact[]>;\n\n /** Get a single artifact */\n retrieve(artifactId: string): Promise<Artifact>;\n\n /** Get artifact content */\n content(artifactId: string): Promise<string>;\n}\n\n/**\n * Models resource interface\n */\nexport interface ModelsResource {\n /** Get available models for a repository */\n list(repo?: string): Promise<ModelInfo[]>;\n}\n\n// ============================================\n// Response Types\n// ============================================\n\n/**\n * Prompt response\n */\nexport interface PromptResponse {\n /** Stop reason */\n stopReason: 'end_turn' | 'max_tokens' | 'tool_use' | 'cancelled' | 'error';\n /** Response metadata */\n _meta?: Record<string, unknown>;\n}\n\n// ============================================\n// Sessions Resource Events (for ClientSessionsResource)\n// ============================================\n\n/**\n * Sessions resource events for monitoring session list changes\n */\nexport interface SessionsResourceEvents {\n /** Emitted when the sessions list changes (create, delete, update) */\n sessionsChanged: SessionInfo[];\n /** Emitted when a new session is created */\n sessionCreated: SessionInfo;\n /** Emitted when a session is deleted */\n sessionDeleted: { sessionId: string };\n /** Emitted when a session is updated (status change, etc.) */\n sessionUpdated: SessionInfo;\n}\n\n/**\n * Event handler type for sessions resource events\n */\nexport type SessionsResourceEventHandler<K extends keyof SessionsResourceEvents> = (\n data: SessionsResourceEvents[K]\n) => void | Promise<void>;\n\n// ============================================\n// Session Events (for ActiveSession)\n// ============================================\n\n/**\n * Session events for event subscription\n */\nexport interface SessionEvents {\n /** Emitted when session updates occur */\n sessionUpdate: SessionNotification;\n /** Emitted when an artifact is created */\n artifactCreated: Artifact;\n /** Emitted when an artifact is updated */\n artifactUpdated: Artifact;\n /** Emitted when an artifact is deleted */\n artifactDeleted: Artifact;\n /** Emitted when a permission request is received */\n permissionRequest: { requestId: string; params: RequestPermissionRequest };\n /** Emitted when a question request is received (ask_followup_question) */\n questionRequest: { toolCallId: string; request: QuestionRequest };\n /** Emitted when usage data is updated */\n usageUpdate: UsageUpdate;\n /** Emitted when a checkpoint is created */\n checkpointCreated: CheckpointInfo;\n /** Emitted when a checkpoint is updated */\n checkpointUpdated: CheckpointInfo;\n /** Emitted when a command is received */\n command: { action: string; params?: Record<string, unknown> };\n /** Emitted when connected to agent */\n connected: void;\n /** Emitted when disconnected from agent */\n disconnected: void;\n /** Emitted when an error occurs */\n error: Error;\n}\n\n/**\n * Event handler type for session events\n */\nexport type SessionEventHandler<K extends keyof SessionEvents> = (data: SessionEvents[K]) => void | Promise<void>;\n\n// ============================================\n// Active Session Interface\n// ============================================\n\n/**\n * Agent operations (accessed via session.agent)\n */\nexport interface SessionAgentOperations {\n /** Agent ID */\n readonly id: string;\n /** Agent state */\n readonly state: AgentState;\n /** Whether the agent is connected */\n readonly isConnected: boolean;\n /** Agent capabilities */\n readonly capabilities?: AgentCapabilities;\n}\n\n/**\n * Active Session interface\n * Represents an active session with its resources and operations\n *\n * Key design:\n * - Session is the primary API surface\n * - agentState provides direct access to underlying agent state\n * - disconnect() is called directly on session (not session.agent)\n */\nexport interface ActiveSession {\n /** Session ID */\n readonly id: string;\n /** Agent ID */\n readonly agentId: string;\n /** Agent state (direct access to underlying agent state) */\n readonly agentState: AgentState;\n /** Agent capabilities (available after connection) */\n readonly capabilities?: AgentCapabilities;\n /** Available session modes */\n readonly availableModes?: SessionMode[];\n /** Current session mode */\n readonly currentMode?: string;\n /** Available models for this session */\n readonly availableModels?: ModelInfo[];\n /** Current model ID */\n readonly currentModelId?: string;\n /** Available slash commands (updated via available_commands_update) */\n readonly availableCommands: AvailableCommand[];\n /** Whether the session is active */\n readonly isActive: boolean;\n /**\n * Session connection information (only available for cloud sessions)\n * 会话连接信息,包括sandboxId、link、token等\n */\n readonly connectionInfo?: SessionConnectionInfo;\n\n // Agent operations namespace (optional agent-level access)\n /** Agent operations */\n readonly agent: SessionAgentOperations;\n\n // ACP Resources\n /** Prompts resource */\n readonly prompts: PromptsResource;\n /** Artifacts resource */\n readonly artifacts: ArtifactsResource;\n /** Files resource */\n readonly files: FilesResource;\n\n // Permission management\n /** Resolve a permission request */\n resolvePermission(requestId: string, optionId: string): boolean;\n /** Reject a permission request */\n rejectPermission(requestId: string, reason?: string): boolean;\n\n // Question management (ask_followup_question)\n /** Answer a question request with user's selections */\n answerQuestion(toolCallId: string, answers: QuestionAnswers): boolean;\n /** Cancel a question request */\n cancelQuestion(toolCallId: string, reason?: string): boolean;\n\n // Tool callback management\n /** Callback for tool operations (skip or cancel) */\n toolCallback(toolCallId: string, toolName: string, action: 'skip' | 'cancel'): Promise<{ success: boolean; error?: string }>;\n\n // Session mode management\n /** Set the current session mode */\n setMode(modeId: string): Promise<void>;\n /** Set the current session model */\n setSessionModel(modelId: string): Promise<void>;\n /** Set available commands (called when available_commands_update is received) */\n setAvailableCommands(commands: AvailableCommand[]): void;\n /** Set available modes (called after create/load) */\n setModes(availableModes?: SessionMode[], currentMode?: string): void;\n /** Set available models (called after create/load) */\n setModels(availableModels?: ModelInfo[], currentModelId?: string): void;\n\n // Event subscription\n /** Subscribe to an event */\n on<K extends keyof SessionEvents>(event: K, handler: SessionEventHandler<K>): this;\n /** Unsubscribe from an event */\n off<K extends keyof SessionEvents>(event: K, handler: SessionEventHandler<K>): this;\n /** Subscribe to an event once */\n once<K extends keyof SessionEvents>(event: K, handler: SessionEventHandler<K>): this;\n\n // Lifecycle - disconnect directly on session\n /** Disconnect from the session/agent */\n disconnect(): void;\n\n /** Symbol.dispose for 'using' keyword support */\n [Symbol.dispose](): void;\n}\n\n// ============================================\n// Provider & Client Types\n// ============================================\n\n/**\n * 环境类型\n */\nexport type EnvironmentType = 'local' | 'cloud';\n\n/**\n * Agent provider interface\n *\n * Responsible for:\n * - Managing agent state/configuration storage\n * - Creating connections to agents\n * - Abstracting away transport details (cloud/local)\n *\n * The provider.connect() method returns an AgentConnection.\n * The client wraps the connection in an ActiveSession instance.\n *\n * @typeParam C - Connection type used by this provider (e.g., CloudAgentConnection, LocalAgentConnection)\n */\nexport interface AgentProvider<C extends AgentConnection = AgentConnection> {\n /**\n * Create a new agent and return its ID\n *\n * @param params - Optional session params (used by LocalAgentProvider to get cwd)\n * @returns Agent ID (Cloud: UUID, Local: cwd)\n */\n create?(params?: CreateSessionParams): Promise<string>;\n /** Get agent state by ID */\n get(agentId: string): Promise<AgentState | undefined>;\n\n /**\n * List all agent states with pagination information\n *\n * @param options - Optional query parameters for filtering, sorting, and pagination\n * Cloud providers use these for API queries and return server pagination\n * Local providers apply client-side filtering and return synthetic pagination\n * @returns Object containing agents array and pagination info\n */\n list(options?: ListAgentOptions): Promise<ListAgentResult<AgentState>>;\n\n /** Connect to an agent and return the connection */\n connect(agentId: string): Promise<C>;\n /** Delete an agent by ID */\n delete(agentId: string): Promise<boolean>;\n\n /**\n * Archive an agent by ID (optional)\n * Used by CloudAgentProvider for archiving agents\n *\n * @param agentId - Agent ID to archive\n * @returns Object containing the archived agent ID\n */\n archive?(agentId: string): Promise<{ id: string }>;\n\n /**\n * Rename an agent by ID (optional)\n * Used by CloudAgentProvider and LocalAgentProvider for renaming agents\n *\n * @param agentId - Agent ID to rename\n * @param title - New title for the agent\n * @returns Object containing the renamed agent ID\n */\n rename?(agentId: string, title: string): Promise<{ id: string }>;\n\n /**\n * Move an agent by ID (optional)\n * Used by LocalAgentProvider for moving Playground sessions to Workspace\n *\n * @param agentId - Agent ID to move\n * @returns Object containing the moved agent ID\n */\n move?(agentId: string): Promise<{ id: string }>;\n\n /** Filesystem provider (optional - some providers may not support filesystem operations) */\n readonly filesystem?: FilesystemProvider;\n\n /**\n * Get available models for a repository (optional)\n * Implementation varies by provider type\n * @param repo - Repository identifier\n * @returns Array of model information\n */\n getModels?(repo?: string): Promise<ModelInfo[]>;\n\n /**\n * Register sessionId → agentId mapping (optional, used by LocalAgentProvider)\n * Called after session creation to maintain the mapping for loadSession\n *\n * @param sessionId - Session ID returned by connection.createSession()\n * @param agentId - Agent ID (cwd for Local)\n */\n registerSession?(sessionId: string, agentId: string): void;\n\n /**\n * Open a workspace window (optional, used by LocalAgentProvider)\n *\n * @param params - Workspace params including cwd\n * @returns Response with success status\n */\n openWorkspace?(params: InitializeWorkspaceParams): Promise<InitializeWorkspaceResponse>;\n\n /**\n * Pick files from file dialog (optional, used by LocalAgentProvider)\n *\n * @param params - File picker params including filters\n * @returns Response with file paths and cancel status\n */\n pickFile?(params?: PickFileParams): Promise<PickFileResponse>;\n\n /**\n * Pick folders from folder dialog (optional, used by LocalAgentProvider)\n *\n * @param params - Folder picker params\n * @returns Response with folder paths and cancel status\n */\n pickFolder?(params?: PickFolderParams): Promise<PickFolderResponse>;\n\n /**\n * Upload a file to cloud storage (optional)\n *\n * @param params - Upload parameters including file content\n * @returns Response with cloud URL after successful upload\n */\n uploadFile?(params: UploadFileParams): Promise<UploadFileResponse>;\n\n /**\n * Search for files in the workspace (optional, used by LocalAgentProvider)\n *\n * @param params - Search parameters including options\n * @returns Response with search results\n */\n searchFile?(params: SearchFileParams): Promise<SearchFileResponse>;\n\n /**\n * Batch toggle plugins (optional, used by LocalAgentProvider)\n *\n * @param request - Batch plugin operation request\n * @returns Batch operation result\n */\n batchTogglePlugins?(request: BatchPluginOperationRequest): Promise<BatchPluginOperationResult>;\n\n /**\n * Get installed plugins (optional, used by LocalAgentProvider)\n *\n * @param forceRefresh - Whether to force refresh the cache\n * @returns Array of installed plugins\n */\n getInstalledPlugins?(forceRefresh?: boolean): Promise<Array<{\n name: string;\n marketplaceName: string;\n status: string;\n description?: string;\n version?: string;\n }>>;\n\n /**\n * Install plugins (optional, used by LocalAgentProvider)\n *\n * @param pluginNames - Array of plugin names to install\n * @param marketplaceName - Marketplace name\n * @param installScope - Install scope ('user' | 'project')\n * @returns Result of the operation\n */\n installPlugins?(\n pluginNames: string[],\n marketplaceName: string,\n installScope?: 'user' | 'project'\n ): Promise<{ success: boolean; error?: string }>;\n\n /**\n * Register an event listener\n * Provider implementations should forward events to the underlying transport\n *\n * @param event - Event name\n * @param handler - Event handler function\n */\n on?(event: string, handler: (...args: any[]) => void): void;\n\n /**\n * Unregister an event listener\n *\n * @param event - Event name\n * @param handler - Event handler function to remove\n */\n off?(event: string, handler: (...args: any[]) => void): void;\n}\n\n/**\n * AgentClient initialization options\n */\nexport interface AgentClientOptions {\n /** Agent provider (required) */\n provider: AgentProvider;\n /** Logger instance */\n logger?: Logger;\n /** Client capabilities (sent during initialization) */\n clientCapabilities?: ClientCapabilities;\n /**\n * 运行环境类型\n * - 'local': IDE 本地环境\n * - 'cloud': 云端环境\n */\n environmentType?: EnvironmentType;\n}\n\n/**\n * Client sessions resource interface\n * Top-level API for session management\n *\n * Key design:\n * - list() returns sessions with pagination info (mapped from agents)\n * - create() creates a new session (auto-creates agent and connects)\n * - load() loads an existing session (finds agent by sessionId and connects)\n * - archive() archives a session/agent\n * - initializeWorkspace() initializes a workspace for future sessions\n */\nexport interface ClientSessionsResource {\n /**\n * List all sessions with pagination info\n * Cloud: Returns server-side filtered/sorted/paginated results\n * Local: Returns client-side filtered/sorted results (synthetic pagination)\n */\n list(options?: ListAgentOptions): Promise<ListAgentResult<SessionInfo>>;\n\n /** Create a new session (auto-creates agent and connects) */\n create(params: CreateSessionParams): Promise<ActiveSession>;\n\n /** Load an existing session (finds agent by sessionId and connects) */\n load(params: LoadSessionParams): Promise<ActiveSession>;\n\n /**\n * Archive a session/agent\n * @param sessionId - Session ID to archive\n * @returns Object containing the archived session ID\n */\n archive(sessionId: string): Promise<{ id: string }>;\n\n /**\n * Rename a session/agent\n * @param sessionId - Session ID to rename\n * @param title - New title for the session\n * @returns Object containing the renamed session ID\n */\n rename(sessionId: string, title: string): Promise<{ id: string }>;\n\n /**\n * Move a session (Playground → Workspace)\n * @param sessionId - Session ID to move\n * @returns Object containing the moved session ID\n */\n move(sessionId: string): Promise<{ id: string }>;\n\n /** Initialize a workspace for future sessions */\n initializeWorkspace(params: InitializeWorkspaceParams): Promise<InitializeWorkspaceResponse>;\n\n /** Models resource for getting available models */\n readonly models: ModelsResource;\n\n /** Get current workspaces list */\n getCurrentWorkspaces(filter?: { activeOnly?: boolean }): Promise<WorkspaceInfo[]>;\n\n // Event subscription for sessions list changes\n /** Subscribe to sessions resource events */\n on<K extends keyof SessionsResourceEvents>(\n event: K,\n handler: SessionsResourceEventHandler<K>\n ): void;\n\n /** Unsubscribe from sessions resource events */\n off<K extends keyof SessionsResourceEvents>(\n event: K,\n handler: SessionsResourceEventHandler<K>\n ): void;\n\n /** Open a workspace (for LocalAgentProvider) */\n openWorkspace(params: InitializeWorkspaceParams): Promise<InitializeWorkspaceResponse>;\n\n /** Pick files from file dialog (for LocalAgentProvider) */\n pickFile(params?: PickFileParams): Promise<PickFileResponse>;\n\n /** Pick folders from folder dialog (for LocalAgentProvider) */\n pickFolder(params?: PickFolderParams): Promise<PickFolderResponse>;\n\n /** Upload a file to cloud storage */\n uploadFile(params: UploadFileParams): Promise<UploadFileResponse>;\n\n /** Search for files in the workspace (for LocalAgentProvider) */\n searchFile(params: SearchFileParams): Promise<SearchFileResponse>;\n\n /** Batch toggle plugins (for LocalAgentProvider) */\n batchTogglePlugins(request: BatchPluginOperationRequest): Promise<BatchPluginOperationResult>;\n\n /** Get installed plugins (for LocalAgentProvider) */\n getInstalledPlugins?(forceRefresh?: boolean): Promise<Array<{\n name: string;\n marketplaceName: string;\n status: string;\n description?: string;\n version?: string;\n }>>;\n\n /** Install plugins (for LocalAgentProvider) */\n installPlugins?(\n pluginNames: string[],\n marketplaceName: string,\n installScope?: 'user' | 'project'\n ): Promise<{ success: boolean; error?: string }>;\n\n /** Get support plugins from backend API (for LocalAgentProvider) */\n getSupportPlugins?(): Promise<Array<{\n id: string;\n name: string;\n description?: string;\n icon?: string;\n marketplace?: string;\n tags?: string[];\n installed?: boolean;\n prompt?: string;\n pluginMarketplaceName?: string;\n pluginName?: string;\n quickAction?: {\n title?: string;\n description?: string;\n icon?: string;\n promptTemplate?: string;\n };\n }>>;\n\n}\n\n// ============================================\n// Workspace Types\n// ============================================\n\n/**\n * Workspace information (aligned with FolderSelectResult)\n */\nexport interface WorkspaceInfo {\n /** Folder path */\n path: string;\n /** Folder display name */\n label: string;\n}\n\n// ============================================\n// File Picker Types\n// ============================================\n\n/**\n * File filter for pickFile dialog\n */\nexport interface FileFilter {\n /** Display name for the filter */\n readonly name: string;\n /** File extensions (without dot) */\n readonly extensions: string[];\n}\n\n/**\n * Parameters for picking files\n */\nexport interface PickFileParams {\n /** Default path for the dialog */\n readonly defaultPath?: string;\n /** File type filters */\n readonly filters?: FileFilter[];\n /** Whether to allow multiple selection (default false) */\n readonly canSelectMany?: boolean;\n}\n\n/**\n * Response from picking files\n */\nexport interface PickFileResponse {\n /** Selected files - File objects in browser, absolute path strings in IDE */\n readonly files: Array<File | string>;\n /** Whether user cancelled the dialog */\n readonly canceled: boolean;\n /** Error message (if failed) */\n readonly error?: string;\n}\n\n// ============================================\n// Folder Picker Types\n// ============================================\n\n/**\n * Parameters for picking folders\n */\nexport interface PickFolderParams {\n /** Default path for the dialog */\n readonly defaultPath?: string;\n}\n\n/**\n * Response from picking folders\n */\nexport interface PickFolderResponse {\n /** Selected folder paths */\n readonly folderPaths: string[];\n /** Whether user cancelled the dialog */\n readonly canceled: boolean;\n /** Error message (if failed) */\n readonly error?: string;\n}\n\n// ============================================\n// File Upload Types\n// ============================================\n\n/**\n * Parameters for uploading files\n */\nexport interface UploadFileParams {\n /** Files to upload - File objects in browser, absolute path strings in IDE */\n readonly files: Array<File | string>;\n}\n\n/**\n * Response from uploading files\n */\nexport interface UploadFileResponse {\n /** Whether upload was successful */\n readonly success: boolean;\n /** Cloud URLs corresponding to each uploaded file (same order as input files) */\n readonly urls?: string[];\n /** URL expiration time in seconds (from backend) */\n readonly expireSeconds?: number;\n /** Error message (if upload failed) */\n readonly error?: string;\n}\n\n// ============================================\n// File Search Types\n// ============================================\n\n/**\n * Mention type for file/folder\n */\nexport enum MentionType {\n file = 'file',\n folder = 'folder'\n}\n\n/**\n * Search options for file search\n */\nexport interface SearchOptions {\n /** Search keyword */\n readonly search?: string;\n /** Number of results to return */\n readonly resultNum: number;\n}\n\n/**\n * File search result\n */\nexport interface SearchFileResult {\n /** Full path */\n path: string;\n /** Relative path */\n relativePath: string;\n /** File name */\n fileName?: string;\n /** Folder name */\n folderName?: string;\n /** Type (file or folder) */\n type: MentionType.file | MentionType.folder;\n}\n\n/**\n * Parameters for searching files\n */\nexport interface SearchFileParams {\n /** Search options */\n readonly options: SearchOptions;\n /** Search path*/\n readonly cwd?: string;\n}\n\n/**\n * Response from searching files\n */\nexport interface SearchFileResponse {\n /** Search results */\n readonly results: SearchFileResult[];\n /** Error message (if failed) */\n readonly error?: string;\n}\n","/**\n * Backend Provider 类型定义\n *\n * 定义 IBackendProvider 接口和配置\n */\n\n// ============================================================================\n// Account 相关类型\n// ============================================================================\n\n/**\n * 账号版本类型\n */\nexport type Edition = 'pro' | 'personal' | 'ultimate' | 'exclusive';\n\n/**\n * 版本展示类型(用于 UI 展示)\n * - free: 免费版(个人版未订阅 Pro)\n * - pro: Pro 版(个人版已订阅 Pro)\n * - ultimate: 旗舰版(团队版)\n * - exclusive: 专享版(企业版)\n */\nexport type EditionDisplayType = 'free' | 'pro' | 'ultimate' | 'exclusive';\n\n/**\n * 部署状态\n */\nexport interface DeployStatus {\n statusCode: number;\n statusMsg: string;\n detailMsg: string;\n}\n\n/**\n * 套餐代码\n */\n\n/**\n * TCACA_code_001_PqouKr6QWV CodeBuddy海外版免费包\n * TCACA_code_002_AkiJS3ZHF5 CodeBuddy海外版Pro版本包-包月/CodeBuddy Pro Plan - Monthly:\n * TCACA_code_006_DbXS0lrypC CodeBuddy海外版一次性免费赠送2周的Pro版本包/CodeBuddy One-time Free 2-Week Pro Plan Trial\n * TCACA_code_007_nzdH5h4Nl0 CodeBuddy海外版运营裂变包/CodeBuddy Growth Plan\n * TCACA_code_003_FAnt7lcmRT CodeBuddy海外版Pro版本包-包年/CodeBuddy Pro Plan - Yearly\n * TCACA_code_008_cfWoLwvjU4 赠送月包\n */\nexport enum CommodityCode {\n free = 'TCACA_code_001_PqouKr6QWV', // free\n proMon = 'TCACA_code_002_AkiJS3ZHF5',\n // 国内月包(国际pro+)\n proMonPlus = 'TCACA_code_005_maRGyrHhw1',\n // 免费赠送2周\n gift = 'TCACA_code_006_DbXS0lrypC',\n activity = 'TCACA_code_007_nzdH5h4Nl0',\n proYear = 'TCACA_code_003_FAnt7lcmRT',\n // 国际(free 月包、国内试用包)\n freeMon = 'TCACA_code_008_cfWoLwvjU4', // free\n // 加量包\n extra = 'TCACA_code_009_0XmEQc2xOf',\n}\n\n/**\n * 套餐资源项(用于展示列表)\n */\nexport interface PlanResource {\n /** 资源 ID */\n id: string;\n /** 套餐名称(i18n key) */\n name: string;\n /** 套餐代码 */\n packageCode: CommodityCode;\n /** 是否是每日刷新的套餐 */\n isDaily: boolean;\n /** 总量 */\n total: number;\n /** 已用 */\n used: number;\n /** 剩余 */\n left: number;\n /** 到期时间戳 */\n expireAt?: number;\n /** 刷新时间戳 */\n refreshAt?: number;\n}\n\n/**\n * 账号套餐信息\n */\nexport interface AccountPlan {\n /** 是否是 Pro 版本 */\n isPro: boolean;\n // 是否是试用版本\n isTria?: boolean;\n /** 到期时间戳 */\n expireAt?: number;\n // 刷新时间(年套餐下、本周期结束日期)\n refreshAt?: number;\n /** 自动续费标志 0-关闭 1-开启 */\n renewFlag: 0 | 1;\n /** 套餐代码 */\n PackageCode?: CommodityCode;\n /** 套餐名称 */\n name: string;\n usageTotal?: string;\n usageUsed?: string;\n usageLeft?: string;\n /** 所有套餐资源列表 */\n resources?: PlanResource[];\n}\n\n/**\n * 账号信息\n */\nexport interface Account {\n /** 用户ID(唯一标识) */\n uid: string;\n /** 用户昵称 */\n nickname: string;\n /** 版本类型 */\n type: Edition;\n /** 版本展示类型(用于 UI 展示) */\n editionType: EditionDisplayType;\n /** 是否最后一次登录 */\n lastLogin: boolean;\n /** 企业ID */\n enterpriseId?: string;\n /** 企业名称 */\n enterpriseName?: string;\n /** 企业LOGO */\n enterpriseLogo?: string;\n /** 企业内用户名 */\n enterpriseUserName?: string;\n /** 插件是否启用 */\n pluginEnabled?: boolean;\n /** 部署状态 */\n deployStatus?: DeployStatus;\n /** 是否是 Pro 版本 */\n isPro?: boolean;\n /** 到期时间戳 */\n expireAt?: string | number;\n /** 刷新时间(年套餐下、本周期结束日期) */\n refreshAt?: number;\n /** 自动续费标志 0-关闭 1-开启 */\n renewFlag?: 0 | 1;\n /** 套餐代码 */\n PackageCode?: CommodityCode;\n /** 套餐名称 */\n name?: string;\n email?: string;\n /** 套餐总用量 */\n usageTotal?: string;\n /** 套餐已用量 */\n usageUsed?: string;\n /** 套餐剩余用量 */\n usageLeft?: string;\n /** 所有套餐资源列表(个人用户) */\n resources?: PlanResource[];\n}\n\n/** 账户状态 */\nexport enum AccountStatus {\n /** 有效 */\n valid = 0,\n /** 已退款 */\n refund = 1,\n /** 已过期 */\n expired = 2,\n /** 已用完 */\n usedUp = 3,\n}\n\n// https://iwiki.woa.com/p/1151041572#7%E3%80%81%E8%B4%B9%E7%94%A8%E4%B8%AD%E5%BF%83%E6%8E%A7%E5%88%B6%E5%8F%B0-%E8%B5%84%E6%BA%90%E5%8C%85%E7%AE%A1%E7%90%86%E6%9F%A5%E8%AF%A2%E6%8E%A5%E5%8F%A3\nexport interface UserResource {\n AccountId: number;\n ResourceId: string;\n // 账户类型: 1-RI 2-资源包 9 - 组合包\n // ResourceType: 1 | 2 | 9;\n // 总周期数量\n TotalCycles: 1 | 12;\n // 剩余周期数量\n RemainCycles: number;\n Status: AccountStatus;\n // 费用类型:免费1/付费2\n FeeType: 1 | 2;\n PackageCode: CommodityCode;\n PackageName: string;\n // // 周期容量(使用下方精确值)\n // CycleCapacitySize: number;\n // // 剩余周期数量\n // CycleCapacityRemain: number;\n SupportAutoRenew: 0 | 1;\n SupportManualRenew: 0 | 1;\n AutoRenewFlag: 0 | 1;\n // AutoRenewTimeUnit: string;\n // AutoRenewTimeSpan: 1;\n // ProductCode: string;\n // SubProductCode: string;\n // 单周期开始时间\n CycleStartTime: number; // 时间戳 单位为毫秒\n // 单周期结束时间\n CycleEndTime: number; // 时间戳 单位为毫秒\n // CapacityType: string;\n CreateTime: string;\n ExpiredTime: string; // 2025-11-27 14:36:00\n DeductionStartTime: number; // 时间戳 单位为毫秒\n // 抵扣结束时间戳 单位为毫秒\n DeductionEndTime: number; // 时间戳 单位为毫秒\n CapacityUsedPrecise: string;\n // 当前周期余量精确值\n CycleCapacityRemainPrecise: string;\n // 当前周期总量精确值\n CycleCapacitySizePrecise: string;\n // 剩余容量精确值\n CapacityRemainPrecise: string;\n // 用量精确值\n CapacitySizePrecise: string;\n}\n\n// ============================================================================\n// Model 相关类型\n// ============================================================================\n\n/**\n * 推理配置\n */\nexport interface ReasoningConfig {\n /** 推理努力程度 */\n effort: 'low' | 'medium' | 'high';\n /** 摘要模式 */\n summary: 'auto' | 'always' | 'never';\n}\n\n/**\n * 模型信息\n */\nexport interface ModelInfo {\n /** 模型ID */\n id: string;\n /** 模型名称 */\n name: string;\n /** 供应商 */\n vendor: string;\n /** 最大输出 token 数 */\n maxOutputTokens: number;\n /** 最大输入 token 数 */\n maxInputTokens: number;\n /** 是否支持工具调用 */\n supportsToolCall: boolean;\n /** 是否支持图像 */\n supportsImages: boolean;\n /** 是否禁用多模态 */\n disabledMultimodal: boolean;\n /** 最大允许大小 */\n maxAllowedSize: number;\n /** 是否支持推理 */\n supportsReasoning: boolean;\n /** 是否仅推理模式 */\n onlyReasoning: boolean;\n /** 温度参数 */\n temperature: number;\n /** 推理配置 */\n reasoning: ReasoningConfig;\n /** 英文描述 */\n descriptionEn: string;\n /** 中文描述 */\n descriptionZh: string;\n}\n\n// ============================================================================\n// UserConnector 相关类型\n// ============================================================================\n\n/**\n * 用户连接器信息\n */\nexport interface UserConnector {\n /** 用户ID */\n user_id: string;\n /** 连接器名称 */\n name: ConnectorType;\n displayName: string;\n /** 连接时间 */\n connect_at: string;\n /** 连接状态,0未连接,1已连接 */\n connectStatus: 0 | 1;\n /** 勾选仓库列表,逗号分隔 */\n repos: string;\n /** 激活状态,0未激活,1已激活 */\n activeStatus: 0 | 1;\n /** 提示词 */\n prompt: string;\n /** 跳转URL */\n url: string;\n /** 描述 */\n description: string;\n /** OAuth Client ID */\n oauthClientId: string;\n /** OAuth 重定向 URL */\n oauthRedirectUrl: string;\n}\n\n/**\n * 获取用户连接器列表响应\n */\nexport interface ListUserConnectorResponse {\n /** 连接器列表 */\n connectors: UserConnector[];\n}\n\n/**\n * 修改用户连接器连接状态请求\n */\nexport interface ModifyUserConnectorConnectStatusRequest {\n /** 连接器名称 */\n name: ConnectorType;\n /** 连接状态,0未连接,1已连接 */\n connectStatus: 0 | 1;\n /** 激活状态,0未激活,1已激活 */\n activeStatus?: 0 | 1;\n /** 勾选仓库列表 */\n repos?: string[];\n}\n\n/**\n * 修改用户连接器仓库请求\n */\nexport interface ModifyUserConnectorRepoRequest {\n /** 连接器名称 */\n name: ConnectorType;\n /** 仓库列表 */\n repo?: string[];\n}\n\n/**\n * 修改用户连接器激活状态请求\n */\nexport interface ModifyUserConnectorActiveStatusRequest {\n /** 连接器名称 */\n name: ConnectorType;\n /** 激活状态,0未激活,1已激活 */\n activeStatus: 0 | 1;\n}\n\n// 连接器类型\nexport type ConnectorType = 'github' | 'gongfeng' | 'cnb' | 'figma';\n\n// ============================================================================\n// TaskConnector 相关类型\n// ============================================================================\n\n/**\n * 任务连接器信息\n */\nexport interface TaskConnector {\n /** 用户ID */\n user_id: string;\n /** 任务ID */\n task_id: string;\n /** 连接器名称(唯一标识,如github/gongfeng/figma/cnb) */\n name: string;\n displayName: string;\n /** 勾选仓库列表,逗号分隔 */\n repos: string;\n /** 激活状态:0未激活 / 1已激活 */\n activeStatus: 0 | 1;\n /** 创建时间 */\n created_at: string;\n /** 更新时间 */\n updated_at: string;\n /** 提示词 */\n prompt: string;\n /** 跳转URL */\n url: string;\n /** 描述 */\n description: string;\n /** OAuth Client ID */\n oauthClientId?: string;\n /** OAuth 重定向 URL */\n oauthRedirectUrl?: string;\n /** 连接时间 */\n connect_at: string;\n /** 连接状态,0未连接,1已连接 */\n connectStatus: 0 | 1;\n}\n\n/**\n * 添加任务请求\n */\nexport interface AddTaskRequest {\n /** 任务ID */\n taskId: string;\n /** 连接器列表 */\n connectors?: Array<{\n name: string;\n repos: string;\n activeStatus?: 0 | 1;\n }>;\n}\n\n/**\n * 添加任务响应\n */\nexport interface AddTaskResponse {\n /** 任务ID */\n taskId: string;\n}\n\n/**\n * 获取任务连接器列表响应\n */\nexport interface ListTaskConnectorResponse {\n /** 连接器列表 */\n connectors: TaskConnector[];\n}\n\n/**\n * 修改任务连接器激活状态请求\n */\nexport interface ModifyTaskConnectorActiveStatusRequest {\n /** 任务ID */\n taskId: string;\n /** 连接器名称 */\n name: string;\n /** 激活状态,0未激活,1已激活 */\n activeStatus: 0 | 1;\n}\n\n/**\n * 修改任务连接器激活状态响应\n */\nexport interface ModifyTaskConnectorActiveStatusResponse {\n /** 任务ID */\n taskId: string;\n}\n\n/**\n * 修改任务连接器仓库请求\n */\nexport interface ModifyTaskConnectorRepoRequest {\n /** 任务ID */\n taskId: string;\n /** 连接器名称 */\n name: string;\n /** 仓库列表 */\n repo: string[];\n}\n\n/**\n * 修改任务连接器仓库响应\n */\nexport interface ModifyTaskConnectorRepoResponse {\n /** 任务ID */\n taskId: string;\n}\n\n/**\n * oauth回调请求参数\n */\nexport interface SaveOauthTokenRequest {\n /** 连接器名称 */\n name: ConnectorType;\n /** 第三方回调code */\n authorizationCode: string;\n}\n\n/**\n * 获取仓库列表请求参数\n */\nexport interface GetRepoListRequest {\n /** 连接器名称 */\n name: ConnectorType;\n}\n\n/**\n * 撤销所有OAuth连接请求参数\n */\nexport interface RevokeAllRequest {\n /** 连接器名称 */\n name: ConnectorType;\n /** 安装ID列表(可选,如果不传则撤销所有) */\n installationIds?: string[];\n}\n\n/**\n * GitHub 仓库所有者信息\n */\nexport interface OauthGitHubRepoOwner {\n login: string;\n avatar_url: string;\n url: string;\n html_url: string;\n followers_url: string;\n starred_url: string;\n repos_url: string;\n events_url: string;\n}\n\n/**\n * GitHub 仓库信息\n */\nexport interface OauthGitHubRepo {\n name: string;\n full_name: string;\n private: boolean;\n html_url: string;\n url: string;\n teams_url: string;\n hooks_url: string;\n events_url: string;\n branches_url: string;\n git_commits_url: string;\n merges_url: string;\n pulls_url: string;\n git_url: string;\n clone_url: string;\n svn_url: string;\n downloads_url: string;\n description: string;\n owner: OauthGitHubRepoOwner;\n forks_count: number;\n}\n/**\n * 工蜂仓库信息\n */\nexport interface OauthGongfengRepo {\n id: number; // 仓库 ID\n description: string; // 仓库描述\n public: boolean; // 是否公开\n archived: boolean; // 是否归档\n visibility_level: number; // 可见性级别\n public_visibility: number; // 公开可见性级别\n namespace: {\n name: string,\n description: string,\n path: string,\n }; // 命名空间信息\n owner: {\n username: string,\n web_url: string,\n name: string,\n state: string,\n avatar_url: string\n }; // 所有者信息\n name: string; // 仓库名称\n name_with_namespace: string; // 带命名空间的仓库名称\n path: string; // 仓库路径\n path_with_namespace: string; // 带命名空间的仓库路径\n type: string; // 仓库类型(如 \"GIT\")\n default_branch: string; // 默认分支\n ssh_url_to_repo: string; // SSH 克隆地址\n http_url_to_repo: string; // HTTP 克隆地址\n https_url_to_repo: string; // HTTPS 克隆地址\n web_url: string; // 仓库网页地址\n avatar_url: string;\n}\n\n/**\n * CNB仓库信息\n */\nexport interface OauthCnbRepo {\n name: string;\n [key: string]: any;\n}\n\n/**\n * 获取仓库列表响应\n */\nexport interface GetRepoListResponse {\n github_repos?: {\n [key: string | number]: OauthGitHubRepo[]\n }\n gongfeng_repos?: OauthGongfengRepo[]\n cnb_repos?: OauthCnbRepo[]\n}\n\n/**\n * OAuth 用户信息\n */\nexport interface OauthUserInfo {\n /** 头像 URL */\n avatarUrl: string;\n /** 用户名 */\n name: string;\n}\n\n/**\n * 获取 OAuth 用户信息请求参数\n */\nexport interface GetOauthUserRequest {\n /** 连接器名称 */\n name: ConnectorType;\n}\n\n/**\n * 获取 OAuth 用户信息响应\n */\nexport interface GetOauthUserResponse {\n /** 用户信息 */\n user: OauthUserInfo;\n}\n\n/**\n * Figma 文件信息\n */\nexport interface FigmaFileInfo {\n /** 文件名 */\n name: string;\n /** 角色 */\n role: string;\n /** 最后修改时间 */\n lastModified: string;\n /** 编辑器类型 */\n editorType: string;\n /** 缩略图 URL */\n thumbnailUrl: string;\n /** 版本 */\n version: string;\n /** 主文件 Key */\n mainFileKey: string;\n}\n\n/**\n * 获取文件信息请求参数\n */\nexport interface GetFileRequest {\n /** 连接器名称 */\n name: ConnectorType;\n /** 文件 URL */\n url: string;\n}\n\n/**\n * 获取文件信息响应\n */\nexport interface GetFileResponse {\n /** Figma 文件信息 */\n figmaFileInfo: FigmaFileInfo | null;\n}\n\n// ============================================================================\n// Backend Provider 配置\n// ============================================================================\n\n/**\n * Backend Provider 配置选项\n */\nexport interface BackendProviderConfig {\n /** API 基础 URL (例如: https://api.example.com) */\n baseUrl: string;\n /** 认证 Token */\n authToken?: string;\n}\n\n// ============================================================================\n// Backend Provider 接口\n// ============================================================================\n\n/**\n * 企业用户用量信息\n */\nexport interface EnterpriseUsage {\n /** 已使用 credit */\n credit: number;\n /** 周期开始时间 */\n cycleStartTime: string;\n /** 周期结束时间 */\n cycleEndTime: string;\n /** 周期重置时间 */\n cycleResetTime: string;\n /** 限额数量 */\n limitNum: number;\n}\n\n/**\n * IBackendProvider 接口\n *\n * 定义与后端 API 交互的抽象接口\n *\n * 注意:getAgents 和 getModels 方法已废弃并移除,\n * 请使用 IAgentAdapter 中的对应方法\n */\nexport interface IBackendProvider {\n /**\n * 获取当前账号信息\n * @returns Promise<Account | null> 账号信息,未登录时返回 null\n */\n getAccount(): Promise<Account | null>;\n\n /**\n * 获取用户连接器列表\n * @returns Promise<ListUserConnectorResponse | null> 用户连接器列表,失败时返回 null\n */\n getUserConnector(): Promise<ListUserConnectorResponse>;\n\n /**\n * 修改用户连接器连接状态\n * @param request 请求参数\n * @returns Promise<void>\n */\n modifyUserConnectorConnectStatus(request: ModifyUserConnectorConnectStatusRequest): Promise<void>;\n\n /**\n * 修改用户连接器仓库\n * @param request 请求参数\n * @returns Promise<void>\n */\n modifyUserConnectorRepo(request: ModifyUserConnectorRepoRequest): Promise<void>;\n\n /**\n * 修改用户连接器激活状态\n * @param request 请求参数\n * @returns Promise<void>\n */\n modifyUserConnectorActiveStatus(request: ModifyUserConnectorActiveStatusRequest): Promise<void>;\n\n /**\n * 删除用户连接器\n * @param name 连接器名称\n * @returns Promise<void>\n */\n deleteUserConnector(name: ConnectorType): Promise<void>;\n\n /**\n * 添加任务\n * @param request 请求参数\n * @returns Promise<AddTaskResponse>\n */\n addConnectorTask(request: AddTaskRequest): Promise<AddTaskResponse>;\n\n /**\n * 获取任务连接器列表\n * @param taskId 任务ID\n * @returns Promise<ListTaskConnectorResponse>\n */\n getTaskConnector(taskId: string): Promise<ListTaskConnectorResponse>;\n\n /**\n * 修改任务连接器激活状态\n * @param request 请求参数\n * @returns Promise<ModifyTaskConnectorActiveStatusResponse>\n */\n modifyTaskConnectorActiveStatus(request: ModifyTaskConnectorActiveStatusRequest): Promise<ModifyTaskConnectorActiveStatusResponse>;\n\n /**\n * 修改任务连接器仓库\n * @param request 请求参数\n * @returns Promise<ModifyTaskConnectorRepoResponse>\n */\n modifyTaskConnectorRepo(request: ModifyTaskConnectorRepoRequest): Promise<ModifyTaskConnectorRepoResponse>;\n\n /**\n * oauth回调,后端用第三方的code换token的\n */\n saveOauthToken(request: SaveOauthTokenRequest): Promise<void>;\n\n /**\n * 获取OAuth连接器的仓库列表\n * @param request 请求参数\n * @returns Promise<GetRepoListResponse> 仓库列表响应\n */\n getRepoList(request: GetRepoListRequest): Promise<GetRepoListResponse>;\n\n /**\n * 撤销OAuth连接器的所有连接\n * @param request 请求参数\n * @returns Promise<void>\n */\n revokeAll(request: RevokeAllRequest): Promise<void>;\n\n /**\n * 获取 OAuth 用户信息\n * @param request 请求参数\n * @returns Promise<GetOauthUserResponse> 用户信息响应\n */\n getOauthUser(request: GetOauthUserRequest): Promise<GetOauthUserResponse>;\n\n /**\n * 获取文件信息\n * @param request 请求参数\n * @returns Promise<GetFileResponse> 文件信息响应\n */\n getFile(request: GetFileRequest): Promise<GetFileResponse>;\n\n /**\n * 触发登录流程\n * - Web 环境: 跳转到登录页面\n * - IDE 环境: 通过 IPC 通知 IDE 打开登录流程\n */\n login(): Promise<void>;\n\n /**\n * 登出账号\n */\n logout(): Promise<void>;\n\n /**\n * 重新加载窗口(可选,仅 IPC 环境支持)\n * @param params 可选参数,如 locale\n */\n reloadWindow?(params?: { locale?: string }): Promise<void>;\n\n /**\n * 监听事件(可选,用于 IPC 环境)\n * @param event 事件名称\n * @param callback 回调函数\n * @returns 取消订阅函数\n */\n on?(event: string, callback: (data?: unknown) => void): () => void;\n\n /** 获取企业用户用量信息\n * @param enterpriseId 企业 ID\n * @returns Promise<EnterpriseUsage | null> 企业用户用量信息\n */\n getEnterpriseUsage?(enterpriseId: string): Promise<EnterpriseUsage | null>;\n\n /**\n * 获取支持的插件列表(用于 Welcome 页面的快捷操作)\n * API 端点: GET /v2/as/support/plugins\n * @returns Promise<SupportPlugin[]> 支持的插件列表\n */\n getSupportPlugins?(): Promise<SupportPlugin[]>;\n}\n\n// ============================================================================\n// Support Plugins 相关类型\n// ============================================================================\n\n/**\n * 支持的插件信息\n * 用于 Welcome 页面的 QuickActions 快捷操作\n */\nexport interface SupportPlugin {\n /** 插件唯一标识 */\n id: string;\n /** 插件显示名称 */\n name: string;\n /** 插件对应的 prompt 内容 */\n prompt: string;\n /** 插件市场名称 */\n pluginMarketplaceName: string;\n /** 插件名称 */\n pluginName: string;\n}\n\n/**\n * 获取支持插件列表响应\n */\nexport interface GetSupportPluginsResponse {\n /** 插件列表 */\n plugins: SupportPlugin[];\n}\n\n// ============================================================================\n// Plugin Management 相关类型\n// ============================================================================\n\n/**\n * 插件作用域\n */\nexport type PluginScope = 'user' | 'project';\n\n/**\n * 插件操作类型\n */\nexport type PluginOperation = 'enable' | 'disable';\n\n/**\n * 批量插件操作项\n */\nexport interface BatchPluginOperationItem {\n /** 插件名称 */\n readonly pluginName: string;\n /** 市场名称 */\n readonly marketplaceName: string;\n /** 作用域 */\n readonly scope: PluginScope;\n /** 操作类型 */\n readonly operation: PluginOperation;\n}\n\n/**\n * 批量插件操作请求\n */\nexport interface BatchPluginOperationRequest {\n /** 操作项列表 */\n readonly items: BatchPluginOperationItem[];\n}\n\n/**\n * 批量插件操作失败项\n */\nexport interface BatchPluginOperationFailedItem extends BatchPluginOperationItem {\n /** 错误信息 */\n readonly error: string;\n}\n\n/**\n * 批量插件操作结果\n */\nexport interface BatchPluginOperationResult {\n /** 是否全部成功 */\n readonly success: boolean;\n /** 成功的插件列表 */\n readonly succeededPlugins: BatchPluginOperationItem[];\n /** 失败的插件列表 */\n readonly failedPlugins: BatchPluginOperationFailedItem[];\n}\n","/**\n * Backend Provider 实现\n *\n * 封装与后端 API 的 HTTP 通信\n */\nimport { CommodityCode, UserResource, AccountStatus } from './types';\n\nimport type { IBackendProvider, BackendProviderConfig, Account, AccountPlan, EditionDisplayType, EnterpriseUsage, ListUserConnectorResponse, ModifyUserConnectorConnectStatusRequest, ModifyUserConnectorRepoRequest, ModifyUserConnectorActiveStatusRequest, AddTaskRequest, AddTaskResponse, ListTaskConnectorResponse, ModifyTaskConnectorActiveStatusRequest, ModifyTaskConnectorActiveStatusResponse, ModifyTaskConnectorRepoRequest, ModifyTaskConnectorRepoResponse, SaveOauthTokenRequest, GetRepoListRequest, GetRepoListResponse, RevokeAllRequest, GetOauthUserRequest, GetOauthUserResponse, GetFileRequest, GetFileResponse, BatchPluginOperationRequest, SupportPlugin, BatchPluginOperationResult } from './types';\nimport { accountService } from '../account';\nimport { httpService } from '../http';\n\n/** 获取当前域名的登录页面 URL */\nconst getLoginUrl = () => `${window.location.origin}/login`;\n\n/** 获取当前域名的账号选择页面 URL */\nconst getSelectAccountUrl = () => `${window.location.origin}/login/select`;\n\n/** localStorage 中存储选中账号 ID 的 key */\nexport const SELECTED_ACCOUNT_KEY = 'CODEBUDDY_IDE_SELECTED_ACCOUNT_ID';\n\n/**\n * 套餐代码对应的 i18n key 映射\n */\nconst CommodityCodeText: Record<CommodityCode, string> = {\n [CommodityCode.free]: 'plan.codebuddyFreePlan',\n [CommodityCode.proMon]: 'plan.codebuddyProPlanMonthly',\n [CommodityCode.proMonPlus]: 'plan.codebuddyProPlanMonthly',\n [CommodityCode.gift]: 'plan.codebuddyProPlanTrial',\n [CommodityCode.activity]: 'plan.codebuddyGrowthPlan',\n [CommodityCode.proYear]: 'plan.codebuddyProPlanYearly',\n [CommodityCode.freeMon]: 'plan.codebuddyProPlanDaily',\n [CommodityCode.extra]: 'plan.codebuddyCreditPackage',\n};\n\n/**\n * 获取套餐名称的 i18n key\n */\nconst getPackageName = (packageCode: CommodityCode): string => CommodityCodeText[packageCode] || '';\n\n/**\n * Backend Provider 实现类\n *\n * 职责:\n * - 触发登录/登出流程\n * - 获取 account 后自动同步到 accountService\n * - 使用单例 httpService 进行所有 HTTP 请求\n *\n * 注意:getAgents 和 getModels 方法已废弃并移除,\n * 请使用 IAgentAdapter 中的对应方法\n */\nexport class BackendProvider implements IBackendProvider {\n constructor(config: BackendProviderConfig) {\n // 设置 httpService 配置\n httpService.setBaseURL(config.baseUrl);\n\n if (config.authToken) {\n httpService.setAuthToken(config.authToken);\n }\n\n // 注册 401 回调,执行完整的登出流程\n httpService.onUnauthorized(() => {\n console.log('[BackendProvider] User unauthorized (401/403), triggering logout');\n this.logout().catch(error => {\n console.error('[BackendProvider] Logout failed in 401 handler:', error);\n });\n });\n }\n\n /**\n * 获取当前账号信息\n * API 端点: GET /console/accounts (返回账号列表)\n *\n * 逻辑:\n * 1. 从 localStorage 读取 CODEBUDDY_IDE_SELECTED_ACCOUNT_ID\n * 2. 根据 CODEBUDDY_IDE_SELECTED_ACCOUNT_ID 找到对应账号\n * - personal 类型: 用 uid 匹配\n * - 其他类型: 用 enterpriseId 匹配\n * 3. 如果没有选中的账号,跳转到账号选择页面\n * 4. 获取套餐信息并合并到账号中\n * 5. 同步到 accountService\n */\n async getAccount(): Promise<Account | null> {\n try {\n const result = await httpService.get<{ data: { accounts: Account[] } }>(\n '/console/accounts'\n );\n\n const accounts: Account[] = result.data?.accounts || [];\n if (!accounts || accounts.length === 0) {\n accountService.setAccount(null);\n return null;\n }\n\n // 从 localStorage 读取选中的账号 ID\n const selectedAccountId = localStorage.getItem(SELECTED_ACCOUNT_KEY);\n let selectedAccount: Account | undefined;\n\n if (selectedAccountId) {\n // 查找选中的账号\n selectedAccount = accounts.find(account => {\n // personal 类型用 uid 匹配,其他类型用 enterpriseId 匹配\n if (account.type === 'personal') {\n return account.uid === selectedAccountId;\n }\n return account.enterpriseId === selectedAccountId;\n });\n\n if (selectedAccount) {\n const account = await this.enrichAccountWithUsage(selectedAccount);\n accountService.setAccount(account);\n return account;\n }\n }\n\n // 如果只有一个账号,自动选中它\n if (accounts.length === 1) {\n selectedAccount = accounts[0];\n // 保存选中的账号 ID 到 localStorage\n const accountId = selectedAccount.type === 'personal'\n ? selectedAccount.uid\n : selectedAccount.enterpriseId;\n if (accountId) {\n localStorage.setItem(SELECTED_ACCOUNT_KEY, accountId);\n }\n const account = await this.enrichAccountWithUsage(selectedAccount);\n console.log('account (auto-selected)', account);\n accountService.setAccount(account);\n return account;\n }\n\n // 多个账号但没有选中的,跳转到账号选择页面\n const redirectUrl = encodeURIComponent(window.location.href);\n window.location.href = `${getSelectAccountUrl()}?platform=website&state=0&redirect_uri=${redirectUrl}`;\n accountService.setAccount(null);\n return null;\n } catch (error) {\n console.error('[BackendProvider] getAccount failed:', error);\n accountService.setAccount(null);\n return null;\n }\n }\n\n /**\n * 获取用户连接器列表\n * API 端点: GET /console/as/connector/user/\n */\n async getUserConnector(): Promise<ListUserConnectorResponse> {\n const result = await httpService.get<any>('/console/as/connector/user/');\n\n if (result.code === 0) {\n const { connectors } = result.data;\n return {\n connectors: connectors.map((connector: any) => ({\n ...connector,\n connectStatus: connector.connect_status,\n activeStatus: connector.active_status,\n displayName: connector.display_name,\n oauthClientId: connector.oauth_client_id,\n oauthRedirectUrl: connector.oauth_redirect_url,\n }))\n };\n }\n\n throw result;\n }\n\n /**\n * 修改用户连接器连接状态\n * API 端点: PATCH /console/as/connector/user/:name/connect_status\n */\n async modifyUserConnectorConnectStatus(request: ModifyUserConnectorConnectStatusRequest): Promise<void> {\n // 构造请求体,将驼峰命名转换为下划线命名\n const body: Record<string, any> = {\n name: request.name,\n connect_status: request.connectStatus,\n };\n\n if (request.activeStatus !== undefined) {\n body.active_status = request.activeStatus;\n }\n\n if (request.repos !== undefined) {\n body.repos = request.repos;\n }\n\n const result = await httpService.patch<any>(\n `/console/as/connector/user/${request.name}/connect_status`,\n body\n );\n\n if (result.code === 0) {\n return;\n }\n\n throw result;\n }\n\n /**\n * 修改用户连接器仓库\n * API 端点: PATCH /console/as/connector/user/:name/repo/\n */\n async modifyUserConnectorRepo(request: ModifyUserConnectorRepoRequest): Promise<void> {\n const body: Record<string, any> = {\n name: request.name,\n };\n\n if (request.repo !== undefined) {\n body.repo = request.repo;\n }\n\n const result = await httpService.patch<any>(\n `/console/as/connector/user/${request.name}/repo`,\n body\n );\n\n if (result.code === 0) {\n return;\n }\n\n throw result;\n }\n\n /**\n * 修改用户连接器激活状态\n * API 端点: PATCH /console/as/connector/user/:name/active_status\n */\n async modifyUserConnectorActiveStatus(request: ModifyUserConnectorActiveStatusRequest): Promise<void> {\n const body = {\n name: request.name,\n active_status: request.activeStatus,\n };\n\n const result = await httpService.patch<any>(\n `/console/as/connector/user/${request.name}/active_status`,\n body\n );\n\n if (result.code === 0) {\n return;\n }\n\n throw result;\n }\n\n /**\n * 删除用户连接器\n * API 端点: DELETE /console/as/connector/user/:name/\n */\n async deleteUserConnector(name: 'github' | 'gongfeng'): Promise<void> {\n const result = await httpService.delete<any>(\n `/console/as/connector/user/${name}/`\n );\n\n if (result.code === 0) {\n return;\n }\n\n throw result;\n }\n\n /**\n * 添加任务\n * API 端点: POST /console/as/connector/task/\n */\n async addConnectorTask(request: AddTaskRequest): Promise<AddTaskResponse> {\n // 构造请求体,将驼峰命名转换为下划线命名\n const body: Record<string, any> = {\n task_id: request.taskId,\n };\n\n if (request.connectors !== undefined) {\n body.connectors = request.connectors.map(c => ({\n name: c.name,\n repos: c.repos,\n active_status: c.activeStatus,\n }));\n }\n\n const result = await httpService.post<any>(\n '/console/as/connector/task/',\n body\n );\n\n if (result.code === 0) {\n return {\n taskId: result.data?.task_id || request.taskId,\n };\n }\n\n throw result;\n }\n\n /**\n * 获取任务连接器列表\n * API 端点: GET /console/as/connector/task/:taskid\n */\n async getTaskConnector(taskId: string): Promise<ListTaskConnectorResponse> {\n const result = await httpService.get<any>(\n `/console/as/connector/task/${taskId}`\n );\n\n if (result.code === 0) {\n const { connectors } = result.data;\n return {\n connectors: (connectors || []).map((connector: any) => ({\n ...connector,\n activeStatus: connector.active_status,\n connectStatus: connector.connect_status,\n displayName: connector.display_name,\n oauthClientId: connector.oauth_client_id,\n oauthRedirectUrl: connector.oauth_redirect_url,\n }))\n };\n }\n\n throw result;\n }\n\n /**\n * 修改任务连接器激活状态\n * API 端点: PATCH /console/as/connector/task/:taskid/active_status\n */\n async modifyTaskConnectorActiveStatus(request: ModifyTaskConnectorActiveStatusRequest): Promise<ModifyTaskConnectorActiveStatusResponse> {\n const body = {\n task_id: request.taskId,\n name: request.name,\n active_status: request.activeStatus,\n };\n\n const result = await httpService.patch<any>(\n `/console/as/connector/task/${request.taskId}/active_status`,\n body\n );\n\n if (result.code === 0) {\n return {\n taskId: result.data?.task_id || request.taskId,\n };\n }\n\n throw result;\n }\n\n /**\n * 修改任务连接器仓库\n * API 端点: PATCH /console/as/connector/task/:taskid/repo\n */\n async modifyTaskConnectorRepo(request: ModifyTaskConnectorRepoRequest): Promise<ModifyTaskConnectorRepoResponse> {\n const body = {\n task_id: request.taskId,\n name: request.name,\n repo: request.repo,\n };\n\n const result = await httpService.patch<any>(\n `/console/as/connector/task/${request.taskId}/repo`,\n body\n );\n\n if (result.code === 0) {\n return {\n taskId: result.data?.task_id || request.taskId,\n };\n }\n\n throw result;\n }\n\n /**\n * 根据账号类型获取用量信息并合并到账号中\n * - 企业用户:调用 getEnterpriseUsage 获取月度限额\n * - 个人用户:调用 getCurrentPlan 获取套餐信息\n */\n private async enrichAccountWithUsage(selectedAccount: Account): Promise<Account> {\n const isEnterpriseUser = !!(selectedAccount.enterpriseId && selectedAccount.enterpriseId !== '');\n\n try {\n if (isEnterpriseUser) {\n // 企业用户:获取企业用量\n const enterpriseUsage = await this.getEnterpriseUsage(selectedAccount.enterpriseId!);\n const editionType = this.getEditionDisplayType(selectedAccount.type, false);\n\n if (enterpriseUsage) {\n const usageLeft = (enterpriseUsage.limitNum - enterpriseUsage.credit).toString();\n const usageTotal = enterpriseUsage.limitNum.toString();\n return {\n ...selectedAccount,\n editionType,\n usageLeft,\n usageTotal,\n refreshAt: enterpriseUsage.cycleResetTime ? new Date(enterpriseUsage.cycleResetTime).getTime() : undefined,\n };\n }\n return { ...selectedAccount, editionType };\n } else {\n // 个人用户:获取套餐信息\n const plan = await this.getCurrentPlan();\n const editionType = this.getEditionDisplayType(selectedAccount.type, plan.isPro);\n console.log('account', { ...selectedAccount, ...plan, editionType });\n return { ...selectedAccount, ...plan, editionType };\n }\n } catch (error) {\n console.error('[BackendProvider] enrichAccountWithUsage failed:', error);\n return { ...selectedAccount };\n }\n }\n\n /**\n * 获取当前套餐信息\n * 从计量计费接口获取用户的套餐信息\n * API: POST /billing/meter/get-user-resource\n */\n private async getCurrentPlan(): Promise<AccountPlan> {\n // 默认套餐信息\n const defaultPlan: AccountPlan = {\n isPro: false,\n expireAt: 0,\n renewFlag: 0,\n PackageCode: undefined,\n name: '',\n };\n\n try {\n // 构造请求参数\n const now = new Date();\n const futureDate = new Date(now.getTime() + 101 * 365 * 24 * 60 * 60 * 1000);\n const formatDate = (d: Date) => {\n const pad = (n: number) => n.toString().padStart(2, '0');\n return `${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())} ${pad(d.getHours())}:${pad(d.getMinutes())}:${pad(d.getSeconds())}`;\n };\n\n const body = {\n PageNumber: 1,\n // PageSize、用户购买包的数量、暂定 100 上限。\n PageSize: 100,\n ProductCode: 'p_tcaca',\n Status: [AccountStatus.valid, AccountStatus.usedUp], // 0-有效, 3-已用完\n PackageEndTimeRangeBegin: formatDate(now),\n PackageEndTimeRangeEnd: formatDate(futureDate),\n };\n\n const result = await httpService.post<any>(\n '/billing/meter/get-user-resource',\n body\n );\n\n // 响应格式: { Response: { Data: { Accounts: [...] } } }\n const resources = (result?.data?.Response?.Data?.Accounts || []) as UserResource[];\n\n if (!resources || resources.length === 0) {\n return defaultPlan;\n }\n\n // 解析时间字符串或时间戳为毫秒时间戳\n const parseTime = (time: string | number | undefined): number => {\n if (!time) { return 0; }\n return new Date(time).getTime();\n };\n\n // 每日刷新的套餐类型\n const dailyCredits = [CommodityCode.free, CommodityCode.freeMon];\n\n // 转换所有套餐为 PlanResource 列表\n const planResources = resources.map(r => {\n const isDaily = dailyCredits.includes(r.PackageCode);\n const endTime = isDaily ? r.CycleEndTime : r.DeductionEndTime;\n\n return {\n id: r.ResourceId,\n name: isDaily ? 'plan.addonCredits' : getPackageName(r.PackageCode),\n packageCode: r.PackageCode,\n isDaily,\n total: Number(r.CycleCapacitySizePrecise) || 0,\n used: Math.max(0, Number(r.CycleCapacitySizePrecise) - Number(r.CycleCapacityRemainPrecise)) || 0,\n left: Number(r.CycleCapacityRemainPrecise) || 0,\n expireAt: parseTime(endTime),\n refreshAt: isDaily ? undefined : parseTime(r.CycleEndTime),\n };\n }).sort((a, b) => {\n const getPriority = (code: CommodityCode) => {\n // 付费套餐优先\n if ([CommodityCode.proMon, CommodityCode.proMonPlus, CommodityCode.proYear, CommodityCode.extra].includes(code)) {\n return 1;\n }\n // 赠送/试用套餐次之\n if ([CommodityCode.gift, CommodityCode.activity].includes(code)) {\n return 2;\n }\n // 免费版最后\n if ([CommodityCode.free, CommodityCode.freeMon].includes(code)) {\n return 3;\n }\n return 4;\n };\n return getPriority(a.packageCode) - getPriority(b.packageCode);\n });\n\n // 查找 Pro 套餐(proYear 或 proMon)\n const proPlan = resources.find((r: any) =>\n r.PackageCode === CommodityCode.proYear || r.PackageCode === CommodityCode.proMon || r.PackageCode === CommodityCode.proMonPlus\n );\n\n // 查找试用套餐\n const trialPlan = resources.find((r: any) => r.PackageCode === CommodityCode.gift || r.PackageCode === CommodityCode.freeMon);\n\n const activePlan = proPlan || trialPlan;\n\n // 计算所有套餐的总量、已用和剩余\n const totalUsageLeft = planResources.reduce((sum, r) => sum + r.left, 0);\n const totalUsageTotal = planResources.reduce((sum, r) => sum + r.total, 0);\n const totalUsageUsed = planResources.reduce((sum, r) => sum + r.used, 0);\n\n if (activePlan) {\n return {\n isPro: !!proPlan,\n // 试用套餐\n isTria: trialPlan ? [AccountStatus.valid, AccountStatus.usedUp].includes(trialPlan.Status) : false,\n expireAt: parseTime(activePlan.DeductionEndTime || activePlan.ExpiredTime || activePlan.CycleEndTime),\n refreshAt: parseTime(activePlan.CycleEndTime),\n renewFlag: Number(activePlan.AutoRenewFlag) === 1 ? 1 : 0,\n PackageCode: activePlan.PackageCode,\n name: getPackageName(activePlan.PackageCode),\n usageTotal: String(totalUsageTotal),\n usageUsed: String(totalUsageUsed),\n usageLeft: String(totalUsageLeft),\n resources: planResources,\n };\n }\n\n return {\n ...defaultPlan,\n usageTotal: String(totalUsageTotal),\n usageUsed: String(totalUsageUsed),\n usageLeft: String(totalUsageLeft),\n resources: planResources\n };\n } catch (error) {\n console.error('[BackendProvider] getCurrentPlan error:', error);\n return defaultPlan;\n }\n }\n\n /**\n * 通过回调code,换token\n * @param request\n * @returns\n */\n async saveOauthToken(request: SaveOauthTokenRequest) {\n const body = {\n authorization_code: request.authorizationCode,\n };\n\n const result = await httpService.post<any>(\n `/console/as/connector/oauth/${request.name}/connect`,\n body\n );\n\n if (result.code === 0) {\n return;\n }\n\n throw result;\n }\n\n /**\n * 获取OAuth连接器的仓库列表\n */\n async getRepoList(request: GetRepoListRequest): Promise<GetRepoListResponse> {\n const result = await httpService.get<any>(\n `/console/as/connector/oauth/${request.name}/repos`\n );\n\n if (result.code === 0) {\n return result.data;\n }\n\n throw result;\n }\n\n /**\n * 撤销OAuth连接器的所有连接\n */\n async revokeAll(request: RevokeAllRequest): Promise<void> {\n const installationIds = request?.installationIds ?? [];\n\n const body: { name: string; installation_ids?: string[] } = {\n name: request.name,\n };\n\n if (installationIds) {\n body.installation_ids = installationIds;\n }\n\n const result = await httpService.post<any>(\n `/console/as/connector/oauth/${request.name}/revokeall`,\n body\n );\n\n if (result.code === 0) {\n return;\n }\n\n throw result;\n }\n\n /**\n * 获取 OAuth 用户信息\n * API 端点: GET /console/as/connector/oauth/:name/oauthuser\n */\n async getOauthUser(request: GetOauthUserRequest): Promise<GetOauthUserResponse> {\n const result = await httpService.get<any>(\n `/console/as/connector/oauth/${request.name}/oauthuser`\n );\n\n if (result.code === 0) {\n const data = result.data || {};\n return {\n user: {\n avatarUrl: data.user?.avatar_url || '',\n name: data.user?.name || '',\n }\n };\n }\n\n throw result;\n }\n\n /**\n * 获取文件信息\n * API 端点: GET /console/as/connector/oauth/:name/file\n */\n async getFile(request: GetFileRequest): Promise<GetFileResponse> {\n const body: { url: string; } = {\n url: request.url,\n };\n\n const result = await httpService.post<any>(\n `/console/as/connector/oauth/${request.name}/file`,\n body\n );\n\n if (result.code === 0) {\n const data = result.data || {};\n const figmaFileInfo = data.figma_file_info;\n return {\n figmaFileInfo: figmaFileInfo ? {\n name: figmaFileInfo.name || '',\n role: figmaFileInfo.role || '',\n lastModified: figmaFileInfo.last_modified || '',\n editorType: figmaFileInfo.editor_type || '',\n thumbnailUrl: figmaFileInfo.thumbnail_url || '',\n version: figmaFileInfo.version || '',\n mainFileKey: figmaFileInfo.main_file_key || '',\n } : null\n };\n }\n\n throw result;\n }\n\n\n /**\n * 根据账号类型和 Pro 状态计算版本展示类型\n * - personal + isPro = 'pro'\n * - personal + !isPro = 'free'\n * - ultimate = 'ultimate' (旗舰版/团队版)\n * - exclusive = 'exclusive' (专享版/企业版)\n */\n private getEditionDisplayType(type: string, isPro: boolean): EditionDisplayType {\n if (type === 'personal') {\n return isPro ? 'pro' : 'free';\n }\n if (type === 'ultimate') {\n return 'ultimate';\n }\n if (type === 'exclusive') {\n return 'exclusive';\n }\n // 默认返回 free\n return 'free';\n }\n\n /**\n * 触发登录流程\n * Web 环境: 跳转到登录页面\n */\n async login(): Promise<void> {\n // 获取当前页面 URL 作为回调地址\n // todo 支持弹窗和跳转、弹窗优化点 别让主页面 reload\n const redirectUrl = encodeURIComponent(window.location.href);\n window.location.href = `${getLoginUrl()}?platform=website&state=0&redirect_uri=${redirectUrl}`;\n }\n\n /**\n * 登出账号\n * Web 环境: 通过 iframe 访问登出 URL 清除 cookie\n */\n async logout(): Promise<void> {\n const baseURL = httpService.getAxiosInstance().defaults.baseURL;\n const url = `${baseURL}/console/logout`;\n\n try {\n await new Promise<void>(resolve => {\n const iframe = document.createElement('iframe');\n iframe.style.cssText = 'position:fixed;top:-9999px;left:-9999px;width:1px;height:1px;border:none;';\n iframe.src = url;\n\n const timeout = setTimeout(() => {\n cleanup();\n resolve();\n }, 5000);\n\n const cleanup = () => {\n clearTimeout(timeout);\n if (iframe.parentNode) {\n iframe.parentNode.removeChild(iframe);\n }\n };\n\n iframe.onerror = () => {\n cleanup();\n resolve();\n };\n\n document.body.appendChild(iframe);\n });\n } catch (error) {\n console.error('[BackendProvider] logout failed:', error);\n }\n\n // 清除 localStorage 中的选中账号\n localStorage.removeItem(SELECTED_ACCOUNT_KEY);\n\n // 清空 accountService,会触发订阅者更新,UI 会自动切换到未登录状态\n accountService.clearAccount();\n }\n\n /**\n * 批量切换插件状态\n * Web 环境不支持此功能\n */\n async batchTogglePlugins(request: BatchPluginOperationRequest): Promise<BatchPluginOperationResult> {\n console.warn('[BackendProvider] batchTogglePlugins is not supported in web environment');\n return {\n success: false,\n succeededPlugins: [],\n failedPlugins: request.items.map(item => ({\n ...item,\n error: 'Plugin batch toggle is not supported in web environment'\n }))\n };\n }\n\n /**\n * 获取企业用户用量信息\n * API: POST /billing/meter/get-enterprise-user-usage\n */\n async getEnterpriseUsage(enterpriseId: string): Promise<EnterpriseUsage | null> {\n try {\n const result = await httpService.post<any>(\n '/billing/meter/get-enterprise-user-usage',\n {},\n {\n headers: {\n 'X-Enterprise-Id': enterpriseId,\n },\n }\n );\n\n // 响应格式: { code, data, msg } 或直接是 EnterpriseUsage\n const usageData = result?.data?.data || result?.data || result;\n\n if (usageData && typeof usageData.limitNum === 'number') {\n return usageData as EnterpriseUsage;\n }\n\n return null;\n } catch (error) {\n console.error('[BackendProvider] getEnterpriseUsage error:', error);\n return null;\n }\n }\n\n /**\n * 获取支持的插件列表\n * API 端点: GET /console/as/support/plugins\n * 用于 Welcome 页面的 QuickActions 快捷操作\n */\n async getSupportPlugins(): Promise<SupportPlugin[]> {\n try {\n const result = await httpService.get<any>('/console/as/support/plugins');\n\n // 响应格式: { code: 0, data: { plugins: [...] } }\n if (result?.code === 0 && result?.data?.plugins) {\n return result.data.plugins;\n }\n\n return [];\n } catch (error) {\n console.error('[BackendProvider] getSupportPlugins error:', error);\n return [];\n }\n }\n}\n\n/**\n * 创建 BackendProvider 实例\n */\nexport function createBackendProvider(config: BackendProviderConfig): BackendProvider {\n return new BackendProvider(config);\n}\n","/**\n * IPC Backend Provider 实现\n *\n * 通过 IWidgetChannel 与后端通信\n * 使用统一的消息格式: { type: 'backend', requestId, params: { type, params } }\n */\n\nimport type { IBackendProvider, Account, ListUserConnectorResponse, ModifyUserConnectorConnectStatusRequest, ModifyUserConnectorRepoRequest, ModifyUserConnectorActiveStatusRequest, AddTaskRequest, AddTaskResponse, ListTaskConnectorResponse, ModifyTaskConnectorActiveStatusRequest, ModifyTaskConnectorActiveStatusResponse, ModifyTaskConnectorRepoRequest, ModifyTaskConnectorRepoResponse, GetOauthUserRequest, GetOauthUserResponse, SaveOauthTokenRequest, GetRepoListRequest, GetRepoListResponse, RevokeAllRequest, GetFileRequest, GetFileResponse, BatchPluginOperationRequest, BatchPluginOperationResult, SupportPlugin } from './types';\nimport { IWidgetChannel } from '../common';\nimport { accountService } from '../account';\n\n/**\n * IPC Backend Provider 配置\n */\nexport interface IPCBackendProviderConfig {\n /** Widget Channel 接口 */\n channel: IWidgetChannel;\n /** 是否启用调试日志 */\n debug?: boolean;\n /** 请求超时时间(毫秒,默认 30000) */\n timeoutMs?: number;\n}\n\n/**\n * Backend 请求类型常量\n */\nconst BACKEND_REQUEST_TYPES = {\n LOGIN: 'backend:login',\n LOGOUT: 'backend:logout',\n GET_ACCOUNT: 'backend:get-account',\n GET_USER_CONNECTOR: 'backend:get-user-connector',\n MODIFY_USER_CONNECTOR_CONNECT_STATUS: 'backend:modify-user-connector-connect-status',\n MODIFY_USER_CONNECTOR_REPO: 'backend:modify-user-connector-repo',\n MODIFY_USER_CONNECTOR_ACTIVE_STATUS: 'backend:modify-user-connector-active-status',\n DELETE_USER_CONNECTOR: 'backend:delete-user-connector',\n ADD_CONNECTOR_TASK: 'backend:add-connector-task',\n GET_TASK_CONNECTOR: 'backend:get-task-connector',\n MODIFY_TASK_CONNECTOR_ACTIVE_STATUS: 'backend:modify-task-connector-active-status',\n MODIFY_TASK_CONNECTOR_REPO: 'backend:modify-task-connector-repo',\n GET_OAUTH_USER: 'backend:get-oauth-user',\n SAVE_OAUTH_TOKEN: 'backend:save-oauth-token',\n GET_REPO_LIST: 'backend:get-repo-list',\n REVOKE_ALL: 'backend:revoke-all',\n GET_FILE: 'backend:get-file',\n RELOAD_WINDOW: 'backend:reload-window',\n BATCH_TOGGLE_PLUGINS: 'backend:batch-toggle-plugins',\n GET_SUPPORT_PLUGINS: 'backend:get-support-plugins',\n} as const;\n\n/**\n * 生成唯一请求 ID\n */\nfunction generateRequestId(): string {\n return `req-${Date.now()}-${Math.random().toString(36).slice(2)}`;\n}\n\n/**\n * IPC Backend Provider 实现类\n *\n * 通过 IWidgetChannel 与后端通信\n */\nexport class IPCBackendProvider implements IBackendProvider {\n private readonly channel: IWidgetChannel;\n private readonly debug: boolean;\n private readonly timeoutMs: number;\n\n constructor(config: IPCBackendProviderConfig) {\n this.channel = config.channel;\n this.debug = config.debug ?? false;\n this.timeoutMs = config.timeoutMs ?? 30000;\n\n this.log('Initialized with IWidgetChannel');\n }\n\n /**\n * 发送统一格式的后端请求\n * @param requestType 请求类型\n * @param params 请求参数\n * @returns 响应数据\n */\n private async sendBackendRequest<T>(requestType: string, params?: unknown): Promise<T> {\n const message = {\n type: 'backend',\n requestId: generateRequestId(),\n params: {\n type: requestType,\n params: params,\n },\n };\n\n this.log('Sending backend request:', message);\n\n const response = await this.channel.callMethod('__backend__', message, this.timeoutMs);\n\n this.log('Received response:', response);\n\n // 检查响应中是否有错误\n if (response?.error) {\n throw new Error(response.error);\n }\n\n // 从响应的 data 字段中提取实际数据\n return (response?.data !== undefined ? response.data : response) as T;\n }\n\n /**\n * 获取当前账号信息\n * IDE 环境: 通过 IPC 获取账号信息,并同步到 accountService\n */\n async getAccount(): Promise<Account | null> {\n this.log('Getting account via IPC');\n\n try {\n const account = await this.sendBackendRequest<Account | null>(\n BACKEND_REQUEST_TYPES.GET_ACCOUNT\n );\n // 同步到 accountService\n accountService.setAccount(account);\n return account;\n } catch (error) {\n this.log('Get account failed:', error);\n accountService.setAccount(null);\n return null;\n }\n }\n\n /**\n * 获取用户连接器列表\n * IDE 环境: 通过 IPC 获取用户连接器列表\n */\n async getUserConnector(): Promise<ListUserConnectorResponse> {\n this.log('Getting user connector via IPC');\n\n try {\n return await this.sendBackendRequest<ListUserConnectorResponse>(\n BACKEND_REQUEST_TYPES.GET_USER_CONNECTOR\n );\n } catch (error) {\n this.log('Get user connector failed:', error);\n throw error;\n }\n }\n\n /**\n * 修改用户连接器连接状态\n * IDE 环境: 通过 IPC 修改用户连接器连接状态\n */\n async modifyUserConnectorConnectStatus(request: ModifyUserConnectorConnectStatusRequest): Promise<void> {\n this.log('Modifying user connector connect status via IPC:', request);\n\n try {\n await this.sendBackendRequest<void>(\n BACKEND_REQUEST_TYPES.MODIFY_USER_CONNECTOR_CONNECT_STATUS,\n request\n );\n } catch (error) {\n this.log('Modify user connector connect status failed:', error);\n throw error;\n }\n }\n\n /**\n * 修改用户连接器仓库\n * IDE 环境: 通过 IPC 修改用户连接器仓库\n */\n async modifyUserConnectorRepo(request: ModifyUserConnectorRepoRequest): Promise<void> {\n this.log('Modifying user connector repo via IPC:', request);\n\n try {\n await this.sendBackendRequest<void>(\n BACKEND_REQUEST_TYPES.MODIFY_USER_CONNECTOR_REPO,\n request\n );\n } catch (error) {\n this.log('Modify user connector repo failed:', error);\n throw error;\n }\n }\n\n /**\n * 修改用户连接器激活状态\n * IDE 环境: 通过 IPC 修改用户连接器激活状态\n */\n async modifyUserConnectorActiveStatus(request: ModifyUserConnectorActiveStatusRequest): Promise<void> {\n this.log('Modifying user connector active status via IPC:', request);\n\n try {\n await this.sendBackendRequest<void>(\n BACKEND_REQUEST_TYPES.MODIFY_USER_CONNECTOR_ACTIVE_STATUS,\n request\n );\n } catch (error) {\n this.log('Modify user connector active status failed:', error);\n throw error;\n }\n }\n\n /**\n * 删除用户连接器\n * IDE 环境: 通过 IPC 删除用户连接器\n */\n async deleteUserConnector(name: 'github' | 'gongfeng'): Promise<void> {\n this.log('Deleting user connector via IPC:', name);\n\n try {\n await this.sendBackendRequest<void>(\n BACKEND_REQUEST_TYPES.DELETE_USER_CONNECTOR,\n { name }\n );\n } catch (error) {\n this.log('Delete user connector failed:', error);\n throw error;\n }\n }\n\n /**\n * 添加任务\n * IDE 环境: 通过 IPC 添加任务\n */\n async addConnectorTask(request: AddTaskRequest): Promise<AddTaskResponse> {\n this.log('Adding connector task via IPC:', request);\n\n try {\n return await this.sendBackendRequest<AddTaskResponse>(\n BACKEND_REQUEST_TYPES.ADD_CONNECTOR_TASK,\n request\n );\n } catch (error) {\n this.log('Add task failed:', error);\n throw error;\n }\n }\n\n /**\n * 获取任务连接器列表\n * IDE 环境: 通过 IPC 获取任务连接器列表\n */\n async getTaskConnector(taskId: string): Promise<ListTaskConnectorResponse> {\n this.log('Getting task connector via IPC:', taskId);\n\n try {\n return await this.sendBackendRequest<ListTaskConnectorResponse>(\n BACKEND_REQUEST_TYPES.GET_TASK_CONNECTOR,\n { taskId }\n );\n } catch (error) {\n this.log('Get task connector failed:', error);\n throw error;\n }\n }\n\n /**\n * 修改任务连接器激活状态\n * IDE 环境: 通过 IPC 修改任务连接器激活状态\n */\n async modifyTaskConnectorActiveStatus(request: ModifyTaskConnectorActiveStatusRequest): Promise<ModifyTaskConnectorActiveStatusResponse> {\n this.log('Modifying task connector active status via IPC:', request);\n\n try {\n return await this.sendBackendRequest<ModifyTaskConnectorActiveStatusResponse>(\n BACKEND_REQUEST_TYPES.MODIFY_TASK_CONNECTOR_ACTIVE_STATUS,\n request\n );\n } catch (error) {\n this.log('Modify task connector active status failed:', error);\n throw error;\n }\n }\n\n /**\n * 修改任务连接器仓库\n * IDE 环境: 通过 IPC 修改任务连接器仓库\n */\n async modifyTaskConnectorRepo(request: ModifyTaskConnectorRepoRequest): Promise<ModifyTaskConnectorRepoResponse> {\n this.log('Modifying task connector repo via IPC:', request);\n\n try {\n return await this.sendBackendRequest<ModifyTaskConnectorRepoResponse>(\n BACKEND_REQUEST_TYPES.MODIFY_TASK_CONNECTOR_REPO,\n request\n );\n } catch (error) {\n this.log('Modify task connector repo failed:', error);\n throw error;\n }\n }\n\n /**\n * 获取 OAuth 用户信息\n * IDE 环境: 通过 IPC 获取 OAuth 用户信息\n */\n async getOauthUser(request: GetOauthUserRequest): Promise<GetOauthUserResponse> {\n this.log('Getting OAuth user via IPC:', request);\n\n try {\n return await this.sendBackendRequest<GetOauthUserResponse>(\n BACKEND_REQUEST_TYPES.GET_OAUTH_USER,\n request\n );\n } catch (error) {\n this.log('Get OAuth user failed:', error);\n throw error;\n }\n }\n\n /**\n * 通过回调code,换token\n * IDE 环境: 通过 IPC 保存 OAuth Token\n */\n async saveOauthToken(request: SaveOauthTokenRequest): Promise<void> {\n this.log('Saving OAuth token via IPC:', request);\n\n try {\n await this.sendBackendRequest<void>(\n BACKEND_REQUEST_TYPES.SAVE_OAUTH_TOKEN,\n request\n );\n } catch (error) {\n this.log('Save OAuth token failed:', error);\n throw error;\n }\n }\n\n /**\n * 获取OAuth连接器的仓库列表\n * IDE 环境: 通过 IPC 获取仓库列表\n */\n async getRepoList(request: GetRepoListRequest): Promise<GetRepoListResponse> {\n this.log('Getting repo list via IPC:', request);\n\n try {\n return await this.sendBackendRequest<GetRepoListResponse>(\n BACKEND_REQUEST_TYPES.GET_REPO_LIST,\n request\n );\n } catch (error) {\n this.log('Get repo list failed:', error);\n throw error;\n }\n }\n\n /**\n * 撤销OAuth连接器的所有连接\n * IDE 环境: 通过 IPC 撤销所有连接\n */\n async revokeAll(request: RevokeAllRequest): Promise<void> {\n this.log('Revoking all connections via IPC:', request);\n\n try {\n await this.sendBackendRequest<void>(\n BACKEND_REQUEST_TYPES.REVOKE_ALL,\n request\n );\n } catch (error) {\n this.log('Revoke all connections failed:', error);\n throw error;\n }\n }\n\n /**\n * 获取文件信息\n * IDE 环境: 通过 IPC 获取文件信息\n */\n async getFile(request: GetFileRequest): Promise<GetFileResponse> {\n this.log('Getting file via IPC:', request);\n\n try {\n return await this.sendBackendRequest<GetFileResponse>(\n BACKEND_REQUEST_TYPES.GET_FILE,\n request\n );\n } catch (error) {\n this.log('Get file failed:', error);\n throw error;\n }\n }\n\n /**\n * 触发登录流程\n * IDE 环境: 通过 IPC 通知 IDE 打开登录流程\n */\n async login(): Promise<void> {\n this.log('Triggering login via IPC');\n\n try {\n await this.sendBackendRequest<void>(BACKEND_REQUEST_TYPES.LOGIN);\n } catch (error) {\n this.log('Login request failed:', error);\n throw error;\n }\n }\n\n /**\n * 登出账号\n * IDE 环境: 通过 IPC 通知 IDE 登出\n */\n async logout(): Promise<void> {\n this.log('Triggering logout via IPC');\n\n try {\n await this.sendBackendRequest<void>(BACKEND_REQUEST_TYPES.LOGOUT);\n // 清空 accountService\n accountService.clearAccount();\n } catch (error) {\n this.log('Logout request failed:', error);\n throw error;\n }\n }\n\n /**\n * 重新加载窗口\n * IDE 环境: 通过 IPC 通知 IDE 重新加载窗口(用于应用语言设置等)\n * @param params 可选参数,如 locale\n */\n async reloadWindow(params?: { locale?: string }): Promise<void> {\n this.log('Triggering reload window via IPC', params);\n\n try {\n await this.sendBackendRequest<void>(BACKEND_REQUEST_TYPES.RELOAD_WINDOW, params);\n } catch (error) {\n this.log('Reload window request failed:', error);\n throw error;\n }\n }\n\n /**\n * 批量切换插件状态\n * IDE 环境: 通过 IPC 调用 Extension Host 的 PluginService\n */\n async batchTogglePlugins(request: BatchPluginOperationRequest): Promise<BatchPluginOperationResult> {\n this.log('Batch toggling plugins via IPC:', request);\n\n try {\n return await this.sendBackendRequest<BatchPluginOperationResult>(\n BACKEND_REQUEST_TYPES.BATCH_TOGGLE_PLUGINS,\n request\n );\n } catch (error) {\n this.log('Batch toggle plugins failed:', error);\n throw error;\n }\n }\n\n /**\n * 获取支持的插件列表\n * IDE 环境: 通过 IPC 调用后端 API\n * 用于 Welcome 页面的 QuickActions 快捷操作\n *\n * 调用链:\n * 1. agent-ui: IPCBackendProvider.getSupportPlugins()\n * 2. Extension Host: BackendBridgeService.handleGetSupportPlugins()\n * 3. Backend API: GET /v2/as/support/plugins\n * 4. 返回 SupportPlugin[] 数据给 agent-ui\n */\n async getSupportPlugins(): Promise<SupportPlugin[]> {\n this.log('Getting support plugins via IPC');\n\n try {\n const result = await this.sendBackendRequest<{ plugins: SupportPlugin[] }>(\n BACKEND_REQUEST_TYPES.GET_SUPPORT_PLUGINS,\n {}\n );\n return result?.plugins || [];\n } catch (error) {\n this.log('Get support plugins failed:', error);\n return [];\n }\n }\n\n /**\n * 调试日志\n */\n private log(...args: unknown[]): void {\n if (this.debug) {\n console.log('[IPCBackendProvider]', ...args);\n }\n }\n}\n\n/**\n * 创建 IPCBackendProvider 实例\n */\nexport function createIPCBackendProvider(config: IPCBackendProviderConfig): IPCBackendProvider {\n return new IPCBackendProvider(config);\n}\n"],"x_google_ignoreList":[13,14,15,16,17,18,19,20,21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40,41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60,61],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"index.mjs","names":["utils","prototype","utils","PlatformFormData","AxiosError","encode","toFormData","utils","AxiosURLSearchParams","AxiosURLSearchParams","utils","platform","toFormData","platform","utils","utils","utils","transitionalDefaults","formDataToJSON","toFormData","AxiosError","platform","utils","utils","parseHeaders","defaults","AxiosHeaders","AxiosError","utils","AxiosError","speedometer","throttle","utils","platform","platform","AxiosHeaders","utils","AxiosHeaders","utils","platform","isURLSameOrigin","cookies","resolveConfig","AxiosHeaders","AxiosError","transitionalDefaults","utils","CanceledError","platform","AxiosError","CanceledError","utils","platform","utils","AxiosError","resolveConfig","composeSignals","AxiosHeaders","httpAdapter","xhrAdapter","fetchAdapter","utils","AxiosError","CanceledError","AxiosHeaders","adapters","defaults","validators","AxiosError","validator","InterceptorManager","utils","AxiosHeaders","CanceledError","utils","Axios","defaults","CanceledError","CancelToken","toFormData","AxiosError","AxiosHeaders","formDataToJSON","utils","adapters","HttpStatusCode","axios"],"sources":["../../agent-provider/src/common/_legacy/tool-schemas.ts","../../agent-provider/src/common/_legacy/MockAgentProvider.ts","../../agent-client-protocol/src/common/types.ts","../../agent-client-protocol/src/common/transport/streamable-http.ts","../../agent-client-protocol/src/common/client/artifacts.ts","../../agent-client-protocol/src/common/client/constants.ts","../../agent-client-protocol/src/common/client/errors.ts","../../agent-client-protocol/src/common/client/events.ts","../../agent-client-protocol/src/common/client/extensions.ts","../../agent-client-protocol/src/common/client/permissions.ts","../../agent-client-protocol/src/common/client/questions.ts","../../agent-client-protocol/src/common/client/client.ts","../../agent-provider/src/common/providers/cloud-agent-provider/cloud-connection.ts","../../agent-provider/node_modules/axios/lib/helpers/bind.js","../../agent-provider/node_modules/axios/lib/utils.js","../../agent-provider/node_modules/axios/lib/core/AxiosError.js","../../agent-provider/node_modules/axios/lib/helpers/null.js","../../agent-provider/node_modules/axios/lib/helpers/toFormData.js","../../agent-provider/node_modules/axios/lib/helpers/AxiosURLSearchParams.js","../../agent-provider/node_modules/axios/lib/helpers/buildURL.js","../../agent-provider/node_modules/axios/lib/core/InterceptorManager.js","../../agent-provider/node_modules/axios/lib/defaults/transitional.js","../../agent-provider/node_modules/axios/lib/platform/browser/classes/URLSearchParams.js","../../agent-provider/node_modules/axios/lib/platform/browser/classes/FormData.js","../../agent-provider/node_modules/axios/lib/platform/browser/classes/Blob.js","../../agent-provider/node_modules/axios/lib/platform/browser/index.js","../../agent-provider/node_modules/axios/lib/platform/common/utils.js","../../agent-provider/node_modules/axios/lib/platform/index.js","../../agent-provider/node_modules/axios/lib/helpers/toURLEncodedForm.js","../../agent-provider/node_modules/axios/lib/helpers/formDataToJSON.js","../../agent-provider/node_modules/axios/lib/defaults/index.js","../../agent-provider/node_modules/axios/lib/helpers/parseHeaders.js","../../agent-provider/node_modules/axios/lib/core/AxiosHeaders.js","../../agent-provider/node_modules/axios/lib/core/transformData.js","../../agent-provider/node_modules/axios/lib/cancel/isCancel.js","../../agent-provider/node_modules/axios/lib/cancel/CanceledError.js","../../agent-provider/node_modules/axios/lib/core/settle.js","../../agent-provider/node_modules/axios/lib/helpers/parseProtocol.js","../../agent-provider/node_modules/axios/lib/helpers/speedometer.js","../../agent-provider/node_modules/axios/lib/helpers/throttle.js","../../agent-provider/node_modules/axios/lib/helpers/progressEventReducer.js","../../agent-provider/node_modules/axios/lib/helpers/isURLSameOrigin.js","../../agent-provider/node_modules/axios/lib/helpers/cookies.js","../../agent-provider/node_modules/axios/lib/helpers/isAbsoluteURL.js","../../agent-provider/node_modules/axios/lib/helpers/combineURLs.js","../../agent-provider/node_modules/axios/lib/core/buildFullPath.js","../../agent-provider/node_modules/axios/lib/core/mergeConfig.js","../../agent-provider/node_modules/axios/lib/helpers/resolveConfig.js","../../agent-provider/node_modules/axios/lib/adapters/xhr.js","../../agent-provider/node_modules/axios/lib/helpers/composeSignals.js","../../agent-provider/node_modules/axios/lib/helpers/trackStream.js","../../agent-provider/node_modules/axios/lib/adapters/fetch.js","../../agent-provider/node_modules/axios/lib/adapters/adapters.js","../../agent-provider/node_modules/axios/lib/core/dispatchRequest.js","../../agent-provider/node_modules/axios/lib/env/data.js","../../agent-provider/node_modules/axios/lib/helpers/validator.js","../../agent-provider/node_modules/axios/lib/core/Axios.js","../../agent-provider/node_modules/axios/lib/cancel/CancelToken.js","../../agent-provider/node_modules/axios/lib/helpers/spread.js","../../agent-provider/node_modules/axios/lib/helpers/isAxiosError.js","../../agent-provider/node_modules/axios/lib/helpers/HttpStatusCode.js","../../agent-provider/node_modules/axios/lib/axios.js","../../agent-provider/src/http/http-service.ts","../../agent-provider/src/http/index.ts","../../agent-provider/src/account/account-service.ts","../../agent-provider/src/common/utils/concurrency.ts","../../agent-provider/src/common/providers/cloud-agent-provider/cos-upload-service.ts","../../agent-provider/src/common/providers/cloud-agent-provider/e2b-filesystem.ts","../../agent-provider/src/common/providers/cloud-agent-provider/cloud-provider.ts","../../agent-provider/src/common/client/session.ts","../../agent-provider/src/common/client/session-manager.ts","../../agent-provider/src/common/client/client.ts","../../agent-provider/src/common/client/types.ts","../../agent-provider/src/backend/types.ts","../../agent-provider/src/backend/backend-provider.ts","../../agent-provider/src/backend/ipc-backend-provider.ts"],"sourcesContent":["/**\n * ACP Tool Input/Output Schema 定义\n *\n * 用于约束 ACP 协议中 ToolCallUpdate 的 rawInput 和 rawOutput 字段\n * 基于 getCraftToolProvider 使用的工具定义\n */\n\nimport { z } from 'zod';\n\n// ============================================================================\n// 基础类型\n// ============================================================================\n\n/**\n * 工具类型标识\n */\nexport type ToolName =\n | 'list_dir'\n | 'search_file'\n | 'read_file'\n | 'read_lints'\n | 'rag_search'\n | 'read_rules'\n | 'mcp_get_tool_description'\n | 'mcp_call_tool'\n | 'fetch_mcp_resource'\n | 'create_rule'\n | 'update_memory'\n | 'search_content'\n | 'write_to_file'\n | 'replace_in_file'\n | 'delete_file'\n | 'execute_command'\n | 'preview_url'\n | 'ask_followup_question'\n | 'invoke_integration'\n | 'call_integration'\n | 'search_integration_tool'\n | 'supabase_get_logs'\n | 'supabase_execute_sql'\n | 'supabase_apply_migration'\n | 'supabase_list_migration'\n | 'supabase_list_tables'\n | 'cloud_studio_fetch_log'\n | 'cloud_studio_execute_command'\n | 'cloud_studio_deploy_sandbox'\n | 'component_get_prompt'\n | 'web_fetch'\n | 'use_skill'\n | 'web_search'\n | 'task'\n | 'codebase_search'\n | 'lsp'\n | 'spec_create'\n | 'spec_update';\n\n// ============================================================================\n// 工具输入类型定义\n// ============================================================================\n\nexport interface ListDirInput {\n target_directory: string;\n ignore_globs?: string;\n}\n\nexport interface SearchFileInput {\n target_directory: string;\n pattern: string;\n recursive: boolean;\n caseSensitive: boolean;\n}\n\nexport interface ReadFileInput {\n filePath: string;\n offset?: number;\n limit?: number;\n}\n\nexport interface ReadLintsInput {\n paths?: string;\n}\n\nexport interface RagSearchInput {\n queryString: string;\n knowledgeBaseNames: string;\n}\n\nexport interface ReadRulesInput {\n ruleNames: string;\n}\n\nexport interface McpGetToolDescriptionInput {\n toolRequests: string;\n}\n\nexport interface McpCallToolInput {\n serverName: string;\n toolName: string;\n arguments: string;\n maxOutputLength?: number;\n}\n\nexport interface FetchMcpResourceInput {\n server: string;\n uri: string;\n arguments?: Record<string, unknown>;\n downloadPath?: string;\n}\n\nexport interface CreateRuleInput {\n ruleScope: string;\n ruleName: string;\n ruleType: string;\n ruleContent: string;\n ruleDescription?: string;\n}\n\nexport interface UpdateMemoryInput {\n action?: 'create' | 'update' | 'delete';\n existing_knowledge_id?: string;\n knowledge_to_store?: string;\n title?: string;\n}\n\nexport interface SearchContentInput {\n pattern: string;\n directory: string;\n fileTypes?: string;\n contextBefore?: number;\n contextAfter?: number;\n contextAround?: number;\n outputMode?: string;\n caseSensitive?: boolean;\n}\n\nexport interface WriteToFileInput {\n filePath: string;\n content: string;\n}\n\nexport interface ReplaceInFileInput {\n filePath: string;\n old_str: string;\n new_str: string;\n}\n\nexport interface DeleteFileInput {\n target_file: string;\n explanation?: string;\n}\n\nexport interface ExecuteCommandInput {\n command: string;\n requires_approval: boolean;\n}\n\nexport interface PreviewUrlInput {\n url: string;\n}\n\nexport interface AskFollowupQuestionInput {\n questions: Array<{\n question: string;\n header: string;\n options: Array<{\n label: string;\n description: string;\n }>;\n multiSelect?: boolean;\n }>;\n}\n\nexport interface InvokeIntegrationInput {\n [key: string]: unknown;\n}\n\nexport interface CallIntegrationInput {\n [key: string]: unknown;\n}\n\nexport interface SearchIntegrationToolInput {\n [key: string]: unknown;\n}\n\nexport interface SupabaseGetLogsInput {\n [key: string]: unknown;\n}\n\nexport interface SupabaseExecuteSqlInput {\n [key: string]: unknown;\n}\n\nexport interface SupabaseApplyMigrationInput {\n [key: string]: unknown;\n}\n\nexport interface SupabaseListMigrationInput {\n [key: string]: unknown;\n}\n\nexport interface SupabaseListTablesInput {\n [key: string]: unknown;\n}\n\nexport interface CloudStudioFetchLogInput {\n [key: string]: unknown;\n}\n\nexport interface CloudStudioExecuteCommandInput {\n [key: string]: unknown;\n}\n\nexport interface CloudStudioDeploySandboxInput {\n [key: string]: unknown;\n}\n\nexport interface ComponentGetPromptInput {\n [key: string]: unknown;\n}\n\nexport interface WebFetchInput {\n url: string;\n fetchInfo: string;\n}\n\nexport interface UseSkillInput {\n command: string;\n}\n\nexport interface WebSearchInput {\n explanation: string;\n searchTerm: string;\n}\n\nexport interface TaskInput {\n subagent_name: string;\n description: string;\n prompt: string;\n subagent_path?: string;\n}\n\nexport interface CodebaseSearchInput {\n query: string;\n path: string;\n limit?: number;\n}\n\nexport interface LspInput {\n [key: string]: unknown;\n}\n\nexport interface SpecCreateInput {\n name: string;\n overview: string;\n relative_history: string;\n}\n\nexport interface SpecUpdateInput {\n status?: 'prepare' | 'ready' | 'building' | 'finished';\n}\n\n/**\n * 工具输入类型映射\n */\nexport interface ToolInputMap {\n list_dir: ListDirInput;\n search_file: SearchFileInput;\n read_file: ReadFileInput;\n read_lints: ReadLintsInput;\n rag_search: RagSearchInput;\n read_rules: ReadRulesInput;\n mcp_get_tool_description: McpGetToolDescriptionInput;\n mcp_call_tool: McpCallToolInput;\n fetch_mcp_resource: FetchMcpResourceInput;\n create_rule: CreateRuleInput;\n update_memory: UpdateMemoryInput;\n search_content: SearchContentInput;\n write_to_file: WriteToFileInput;\n replace_in_file: ReplaceInFileInput;\n delete_file: DeleteFileInput;\n execute_command: ExecuteCommandInput;\n preview_url: PreviewUrlInput;\n ask_followup_question: AskFollowupQuestionInput;\n invoke_integration: InvokeIntegrationInput;\n call_integration: CallIntegrationInput;\n search_integration_tool: SearchIntegrationToolInput;\n supabase_get_logs: SupabaseGetLogsInput;\n supabase_execute_sql: SupabaseExecuteSqlInput;\n supabase_apply_migration: SupabaseApplyMigrationInput;\n supabase_list_migration: SupabaseListMigrationInput;\n supabase_list_tables: SupabaseListTablesInput;\n cloud_studio_fetch_log: CloudStudioFetchLogInput;\n cloud_studio_execute_command: CloudStudioExecuteCommandInput;\n cloud_studio_deploy_sandbox: CloudStudioDeploySandboxInput;\n component_get_prompt: ComponentGetPromptInput;\n web_fetch: WebFetchInput;\n use_skill: UseSkillInput;\n web_search: WebSearchInput;\n task: TaskInput;\n codebase_search: CodebaseSearchInput;\n lsp: LspInput;\n spec_create: SpecCreateInput;\n spec_update: SpecUpdateInput;\n}\n\n// ============================================================================\n// 工具输出类型定义\n// ============================================================================\n\nexport interface ListFilesResult {\n type: 'list_files_result';\n files: Array<{\n filePath: string;\n size: string;\n modifyTime: string;\n }>;\n root: string;\n listing?: string;\n}\n\nexport interface SearchFileResult {\n type: 'search_file_result';\n path: string;\n pattern: string;\n recursive?: boolean;\n caseSensitive?: boolean;\n results: Array<{\n filePath: string;\n size: string;\n modifyTime: string;\n }>;\n}\n\nexport interface ReadFileResult {\n type: 'read_file_result';\n path: string;\n content: string;\n totalLineCount: number;\n hasMore: boolean;\n diagnostic?: string;\n hint?: string;\n image?: {\n data: string;\n mimeType: string;\n };\n}\n\nexport interface ReadLintsResult {\n type: 'read_lints_result';\n diagnostics: string[];\n totalCount?: number;\n hint?: string;\n isTruncated?: boolean;\n}\n\nexport interface KnowledgeSearchResult {\n type: 'knowledge_base_result';\n selectedKnowledgeBases: string;\n queryInput: string;\n}\n\nexport interface RuleMatchResult {\n type: 'rule_match_result';\n ruleDescription: string;\n filePaths: string[];\n}\n\nexport interface McpCallToolResult {\n type: 'mcp_call_tool_result';\n serverName: string;\n toolName: string;\n data: Array<{\n type: 'text' | 'image' | 'resource';\n text?: string;\n data?: string;\n mimeType?: string;\n resource?: {\n uri: string;\n mimeType?: string;\n text?: string;\n blob?: string;\n };\n }>;\n isError?: boolean;\n error?: unknown;\n hint?: string;\n}\n\nexport interface FetchMcpResourceToolResult {\n type: 'fetch_mcp_resource_result';\n server: string;\n uri: string;\n content: string;\n downloadPath?: string;\n}\n\nexport interface RuleCreateResult {\n type: 'rule_create_result';\n ruleName: string;\n createState: 'success' | 'invoke' | 'cancelled';\n hint?: string;\n filePath?: string;\n}\n\nexport interface UpdateMemoryResult {\n type: 'update_memory_result';\n success: boolean;\n message: string;\n action: 'create' | 'update' | 'delete';\n knowledge_id?: string;\n}\n\nexport interface SearchContentResult {\n type: 'search_content_result';\n directory: string;\n pattern: string;\n fileTypes: string;\n matches: Array<{\n filePath: string;\n content: string;\n startLine: number;\n endLine: number;\n size: string;\n modifyTime: string;\n }>;\n totalCount: number;\n hasMore: boolean;\n offset: number;\n limit: number;\n contextBefore: number;\n contextAfter: number;\n contextAround?: number;\n outputMode: string;\n caseSensitive: boolean;\n hint?: string;\n}\n\nexport interface WriteToFileResult {\n type: 'write_to_file_result';\n path: string;\n addLineCount: number;\n removedLines: number;\n addedChars?: number;\n removedChars?: number;\n bytesWritten: number;\n isNewFile: boolean;\n oldContent?: string;\n diagnostic?: string;\n}\n\nexport interface ReplaceInFileResult {\n type: 'replace_in_file_result';\n path: string;\n addLineCount?: number;\n removedLines?: number;\n addedChars?: number;\n removedChars?: number;\n matchCount?: number;\n hint?: string;\n diagnosticChange?: {\n added: string;\n removed: string;\n unchanged: string;\n };\n}\n\nexport interface DeleteFilesResult {\n type: 'delete_file_result';\n path: string;\n recursive: boolean;\n hint?: string;\n}\n\nexport interface ExecuteCommandResult {\n type: 'execute_command_result';\n stdout: string;\n stderr: string;\n exitCode: number;\n hint?: string;\n serviceInfo?: {\n isWatchCommand: boolean;\n isServiceOutput: boolean;\n serviceReady: boolean;\n message: string;\n };\n use_standalone_terminal?: boolean;\n}\n\nexport interface PreviewToolResult {\n type: 'preview_tool_result';\n url: string;\n message: string;\n}\n\nexport interface MultiQuestionResult {\n type: 'multi_question_result';\n questions: Array<{\n id: string;\n question: string;\n options: string[];\n multiSelect?: boolean;\n title?: string;\n }>;\n answers: Record<string, string | string[]>;\n message: string;\n}\n\nexport interface InvokeIntegrationToolResult {\n type: 'invoke_integration_tool_result';\n recommend: {\n id: string;\n type: string;\n status: 'connected' | 'disconnected';\n };\n message: string;\n}\n\nexport interface CallIntegrationToolResult {\n type: 'call_integration_tool_result';\n integrationId: string;\n toolName: string;\n data: {\n type: 'text';\n text: string;\n };\n isError?: boolean;\n error?: unknown;\n}\n\nexport interface SearchIntegrationToolResult {\n type: 'search_integration_tool_result';\n data: Array<{\n integrationId: string;\n integrationName: string;\n toolName: string;\n description: string;\n inputSchema: Record<string, unknown>;\n }>;\n hint?: string;\n}\n\nexport interface SupabaseToolsResult {\n type: 'supabase_get_logs_result' | 'supabase_execute_sql_result' | 'supabase_apply_migration_result' | 'supabase_list_migration_result' | 'supabase_list_tables_result';\n message: string;\n}\n\nexport interface CloudStudioFetchLogResult {\n type: 'cloud_studio_fetch_log_result';\n success: boolean;\n logs: Record<string, string>;\n}\n\nexport interface CloudStudioExecuteCommandResult {\n type: 'cloud_studio_execute_command_result';\n success: boolean;\n message: string;\n}\n\nexport interface CloudStudioDeployResult {\n type: 'cloud_studio_integration_result';\n previewUrl?: string;\n steps: Array<{\n status: 'idle' | 'success' | 'running' | 'error';\n name: 'createSandbox' | 'uploadProject' | 'installDependencies' | 'startService' | 'preview';\n error?: {\n code?: number;\n message?: string;\n };\n }>;\n}\n\nexport interface ComponentGetPromptResult {\n type: 'component_get_prompt_result';\n componentType: string;\n webFramework: string;\n data: {\n type: 'text';\n text: string;\n };\n}\n\nexport interface WebFetchToolResult {\n type: 'web_fetch_tool_result';\n message: string;\n data: string;\n loading?: string;\n title?: string;\n favicon?: string;\n}\n\nexport interface UseSkillToolResult {\n type: 'use_skill_tool_result';\n commandMessage: string;\n message: string;\n}\n\nexport interface WebSearchToolResult {\n type: 'web_search_tool_result';\n data: Array<{\n passage: string;\n uri: string;\n site: string;\n title: string;\n snippets?: string[];\n content?: string;\n }>;\n searchInput?: string;\n}\n\nexport interface TaskToolResult {\n type: 'task_tool_result';\n toolInfo?: Array<{\n name: string;\n info: string;\n needApprove?: boolean;\n toolCallId?: string;\n executeStatus?: 'ing' | 'completed' | 'cancel' | 'fail';\n }>;\n startCallTool?: boolean;\n finalResult?: string;\n toolCallBrief?: string;\n}\n\nexport interface CodebaseSearchResult {\n type: 'codebase_search_result';\n query: string;\n path: string;\n content: string;\n}\n\nexport interface LSPToolResult {\n type: 'lsp_tool_result';\n operation: string;\n result: string;\n resultCount?: number;\n fileCount?: number;\n character?: number;\n}\n\nexport interface PlanCreateToolResult {\n type: 'plan_create_tool_result';\n message: string;\n data: string;\n}\n\nexport interface PlanUpdateToolResult {\n type: 'plan_update_tool_result';\n status: 'prepare' | 'ready' | 'building' | 'finished';\n data: string;\n reminder: string;\n}\n\n/**\n * 工具输出类型映射\n */\nexport interface ToolOutputMap {\n list_dir: ListFilesResult;\n search_file: SearchFileResult;\n read_file: ReadFileResult;\n read_lints: ReadLintsResult;\n rag_search: KnowledgeSearchResult;\n read_rules: RuleMatchResult;\n mcp_get_tool_description: Record<string, unknown>;\n mcp_call_tool: McpCallToolResult;\n fetch_mcp_resource: FetchMcpResourceToolResult;\n create_rule: RuleCreateResult;\n update_memory: UpdateMemoryResult;\n search_content: SearchContentResult;\n write_to_file: WriteToFileResult;\n replace_in_file: ReplaceInFileResult;\n delete_file: DeleteFilesResult;\n execute_command: ExecuteCommandResult;\n preview_url: PreviewToolResult;\n ask_followup_question: MultiQuestionResult;\n invoke_integration: InvokeIntegrationToolResult;\n call_integration: CallIntegrationToolResult;\n search_integration_tool: SearchIntegrationToolResult;\n supabase_get_logs: SupabaseToolsResult;\n supabase_execute_sql: SupabaseToolsResult;\n supabase_apply_migration: SupabaseToolsResult;\n supabase_list_migration: SupabaseToolsResult;\n supabase_list_tables: SupabaseToolsResult;\n cloud_studio_fetch_log: CloudStudioFetchLogResult;\n cloud_studio_execute_command: CloudStudioExecuteCommandResult;\n cloud_studio_deploy_sandbox: CloudStudioDeployResult;\n component_get_prompt: ComponentGetPromptResult;\n web_fetch: WebFetchToolResult;\n use_skill: UseSkillToolResult;\n web_search: WebSearchToolResult;\n task: TaskToolResult;\n codebase_search: CodebaseSearchResult;\n lsp: LSPToolResult;\n spec_create: PlanCreateToolResult;\n spec_update: PlanUpdateToolResult;\n}\n\n// ============================================================================\n// Zod Schema 定义 (用于运行时验证)\n// ============================================================================\n\n/**\n * 工具输入 Schema 定义\n * 用于验证和约束 rawInput\n */\nexport const ToolInputSchemas = {\n list_dir: z.object({\n target_directory: z.string().describe('要列出内容的目录路径'),\n ignore_globs: z.string().optional().describe('可选的 glob 模式数组,用于忽略特定文件'),\n }),\n\n search_file: z.object({\n target_directory: z.string().describe('搜索的目录绝对路径'),\n pattern: z.string().describe('文件模式(如 \"*.js\"),支持通配符'),\n recursive: z.boolean().describe('是否递归搜索子目录'),\n caseSensitive: z.boolean().describe('是否区分大小写'),\n }),\n\n read_file: z.object({\n filePath: z.string().describe('要读取的文件的绝对路径'),\n offset: z.number().optional().describe('开始读取的行号'),\n limit: z.number().optional().describe('要读取的行数'),\n }),\n\n read_lints: z.object({\n paths: z.string().optional().describe('要读取 lint 错误的文件或目录路径'),\n }),\n\n rag_search: z.object({\n queryString: z.string().describe('用户的实际问题或搜索查询'),\n knowledgeBaseNames: z.string().describe('知识库名称,多个用逗号分隔'),\n }),\n\n read_rules: z.object({\n ruleNames: z.string().describe('要读取的规则关键词,用逗号分隔,格式:{ruleName}_{ruleId}'),\n }),\n\n mcp_get_tool_description: z.object({\n toolRequests: z.string().describe('JSON 字符串,二维数组格式:[[\"server1\", \"tool1\"], [\"server2\", \"tool2\"]]'),\n }),\n\n mcp_call_tool: z.object({\n serverName: z.string().describe('MCP 服务器名称'),\n toolName: z.string().describe('要调用的工具名称'),\n arguments: z.string().describe('目标 MCP 工具的参数,JSON 格式字符串'),\n maxOutputLength: z.number().optional().describe('控制工具输出的最大长度,默认 200000'),\n }),\n\n fetch_mcp_resource: z.object({\n server: z.string().describe('MCP 服务器标识符'),\n uri: z.string().describe('要读取的资源 URI'),\n arguments: z.record(z.unknown()).optional().describe('资源模板的参数'),\n downloadPath: z.string().optional().describe('可选的绝对路径,用于保存资源到磁盘'),\n }),\n\n create_rule: z.object({\n ruleScope: z.string().describe('规则范围,project rule 或 user rule'),\n ruleName: z.string().describe('规则文件名,不带扩展名'),\n ruleType: z.string().describe('规则类型,always、manual 或 requested'),\n ruleContent: z.string().describe('规则内容,使用 Markdown 格式'),\n ruleDescription: z.string().optional().describe('规则描述,使用 Markdown 格式'),\n }),\n\n update_memory: z.object({\n action: z.enum(['create', 'update', 'delete']).optional().describe('执行的操作'),\n existing_knowledge_id: z.string().optional().describe('更新或删除时必需,现有记忆的 ID'),\n knowledge_to_store: z.string().optional().describe('要存储的特定记忆'),\n title: z.string().optional().describe('记忆的标题'),\n }),\n\n search_content: z.object({\n pattern: z.string().describe('要搜索的关键字或正则表达式模式'),\n directory: z.string().describe('要搜索的目录的绝对路径'),\n fileTypes: z.string().optional().describe('可选的逗号分隔文件扩展名'),\n contextBefore: z.number().optional().describe('每个匹配前显示的行数'),\n contextAfter: z.number().optional().describe('每个匹配后显示的行数'),\n contextAround: z.number().optional().describe('每个匹配前后显示的行数'),\n outputMode: z.string().optional().describe('输出模式:content、files_with_matches 或 count'),\n caseSensitive: z.boolean().optional().describe('是否区分大小写'),\n }),\n\n write_to_file: z.object({\n filePath: z.string().describe('目标文件的绝对路径'),\n content: z.string().describe('要写入的内容'),\n }),\n\n replace_in_file: z.object({\n filePath: z.string().describe('要修改的文件的绝对路径'),\n old_str: z.string().describe('要替换的文本'),\n new_str: z.string().describe('替换后的文本'),\n }),\n\n delete_file: z.object({\n target_file: z.string().describe('要删除的文件的绝对路径'),\n explanation: z.string().optional().describe('为什么使用此工具的一句话解释'),\n }),\n\n execute_command: z.object({\n command: z.string().describe('要执行的 CLI 命令'),\n requires_approval: z.boolean().describe('命令是否需要用户批准'),\n }),\n\n preview_url: z.object({\n url: z.string().describe('要打开的完整、有效的 HTTP/HTTPS URL'),\n }),\n\n ask_followup_question: z.object({\n questions: z.array(z.object({\n question: z.string(),\n header: z.string().max(12),\n options: z.array(z.object({\n label: z.string().max(50),\n description: z.string(),\n })).min(2).max(4),\n multiSelect: z.boolean().optional(),\n })).min(1).max(4),\n }),\n\n invoke_integration: z.object({}).passthrough(),\n\n call_integration: z.object({}).passthrough(),\n\n search_integration_tool: z.object({}).passthrough(),\n\n supabase_get_logs: z.object({}).passthrough(),\n\n supabase_execute_sql: z.object({}).passthrough(),\n\n supabase_apply_migration: z.object({}).passthrough(),\n\n supabase_list_migration: z.object({}).passthrough(),\n\n supabase_list_tables: z.object({}).passthrough(),\n\n cloud_studio_fetch_log: z.object({}).passthrough(),\n\n cloud_studio_execute_command: z.object({}).passthrough(),\n\n cloud_studio_deploy_sandbox: z.object({}).passthrough(),\n\n component_get_prompt: z.object({}).passthrough(),\n\n web_fetch: z.object({\n url: z.string().describe('要获取内容的 URL'),\n fetchInfo: z.string().describe('用户想要获取的信息描述'),\n }),\n\n use_skill: z.object({\n command: z.string().describe('技能名称(不含参数),如 \"pdf\" 或 \"xlsx\"'),\n }),\n\n web_search: z.object({\n explanation: z.string().describe('为什么使用此工具的一句话解释'),\n searchTerm: z.string().describe('要在网络上搜索的搜索词'),\n }),\n\n task: z.object({\n subagent_name: z.string().describe('要调用的子代理名称'),\n description: z.string().describe('任务的简短描述(3-5 个词)'),\n prompt: z.string().describe('子代理要执行的任务'),\n subagent_path: z.string().optional().describe('子代理定义文件的路径'),\n }),\n\n codebase_search: z.object({\n query: z.string().describe('关于你想理解的内容的完整问题'),\n path: z.string().describe('限制搜索范围的目录路径前缀'),\n limit: z.number().max(100).optional().describe('返回的最大结果数,默认 10'),\n }),\n\n lsp: z.object({}).passthrough(),\n\n spec_create: z.object({\n name: z.string().describe('Plan 名称,用作稳定标识符/文件名'),\n overview: z.string().describe('用一两句话精确概括本次 plan 的主要内容'),\n relative_history: z.string().describe('准备阶段的上下文,包含用户需求、代码位置、额外上下文等'),\n }),\n\n spec_update: z.object({\n status: z.enum(['prepare', 'ready', 'building', 'finished']).optional().describe('Plan 状态'),\n }),\n} satisfies Record<ToolName, z.ZodTypeAny>;\n\n/**\n * 工具输出 Schema 定义\n * 用于验证和约束 rawOutput\n */\nexport const ToolOutputSchemas = {\n list_dir: z.object({\n type: z.literal('list_files_result'),\n files: z.array(z.object({\n filePath: z.string(),\n size: z.string(),\n modifyTime: z.string(),\n })),\n root: z.string(),\n listing: z.string().optional(),\n }),\n\n search_file: z.object({\n type: z.literal('search_file_result'),\n path: z.string(),\n pattern: z.string(),\n recursive: z.boolean().optional(),\n caseSensitive: z.boolean().optional(),\n results: z.array(z.object({\n filePath: z.string(),\n size: z.string(),\n modifyTime: z.string(),\n })),\n }),\n\n read_file: z.object({\n type: z.literal('read_file_result'),\n path: z.string(),\n content: z.string(),\n totalLineCount: z.number(),\n hasMore: z.boolean(),\n diagnostic: z.string().optional(),\n hint: z.string().optional(),\n image: z.object({\n data: z.string(),\n mimeType: z.string(),\n }).optional(),\n }),\n\n read_lints: z.object({\n type: z.literal('read_lints_result'),\n diagnostics: z.array(z.string()),\n totalCount: z.number().optional(),\n hint: z.string().optional(),\n isTruncated: z.boolean().optional(),\n }),\n\n rag_search: z.object({\n type: z.literal('knowledge_base_result'),\n selectedKnowledgeBases: z.string(),\n queryInput: z.string(),\n }),\n\n read_rules: z.object({\n type: z.literal('rule_match_result'),\n ruleDescription: z.string(),\n filePaths: z.array(z.string()),\n }),\n\n mcp_get_tool_description: z.object({}).passthrough(),\n\n mcp_call_tool: z.object({\n type: z.literal('mcp_call_tool_result'),\n serverName: z.string(),\n toolName: z.string(),\n data: z.array(z.union([\n z.object({\n type: z.literal('text'),\n text: z.string(),\n }),\n z.object({\n type: z.literal('image'),\n data: z.string(),\n mimeType: z.string(),\n }),\n z.object({\n type: z.literal('resource'),\n resource: z.object({\n uri: z.string(),\n mimeType: z.string().optional(),\n text: z.string().optional(),\n blob: z.string().optional(),\n }),\n }),\n ])),\n isError: z.boolean().optional(),\n error: z.unknown().optional(),\n hint: z.string().optional(),\n }),\n\n fetch_mcp_resource: z.object({\n type: z.literal('fetch_mcp_resource_result'),\n server: z.string(),\n uri: z.string(),\n content: z.string(),\n downloadPath: z.string().optional(),\n }),\n\n create_rule: z.object({\n type: z.literal('rule_create_result'),\n ruleName: z.string(),\n createState: z.enum(['success', 'invoke', 'cancelled']),\n hint: z.string().optional(),\n filePath: z.string().optional(),\n }),\n\n update_memory: z.object({\n type: z.literal('update_memory_result'),\n success: z.boolean(),\n message: z.string(),\n action: z.enum(['create', 'update', 'delete']),\n knowledge_id: z.string().optional(),\n }),\n\n search_content: z.object({\n type: z.literal('search_content_result'),\n directory: z.string(),\n pattern: z.string(),\n fileTypes: z.string(),\n matches: z.array(z.object({\n filePath: z.string(),\n content: z.string(),\n startLine: z.number(),\n endLine: z.number(),\n size: z.string(),\n modifyTime: z.string(),\n })),\n totalCount: z.number(),\n hasMore: z.boolean(),\n offset: z.number(),\n limit: z.number(),\n contextBefore: z.number(),\n contextAfter: z.number(),\n contextAround: z.number().optional(),\n outputMode: z.string(),\n caseSensitive: z.boolean(),\n hint: z.string().optional(),\n }),\n\n write_to_file: z.object({\n type: z.literal('write_to_file_result'),\n path: z.string(),\n addLineCount: z.number(),\n removedLines: z.number(),\n addedChars: z.number().optional(),\n removedChars: z.number().optional(),\n bytesWritten: z.number(),\n isNewFile: z.boolean(),\n oldContent: z.string().optional(),\n diagnostic: z.string().optional(),\n }),\n\n replace_in_file: z.object({\n type: z.literal('replace_in_file_result'),\n path: z.string(),\n addLineCount: z.number().optional(),\n removedLines: z.number().optional(),\n addedChars: z.number().optional(),\n removedChars: z.number().optional(),\n matchCount: z.number().optional(),\n hint: z.string().optional(),\n diagnosticChange: z.object({\n added: z.string(),\n removed: z.string(),\n unchanged: z.string(),\n }).optional(),\n }),\n\n delete_file: z.object({\n type: z.literal('delete_file_result'),\n path: z.string(),\n recursive: z.boolean(),\n hint: z.string().optional(),\n }),\n\n execute_command: z.object({\n type: z.literal('execute_command_result'),\n stdout: z.string(),\n stderr: z.string(),\n exitCode: z.number(),\n hint: z.string().optional(),\n serviceInfo: z.object({\n isWatchCommand: z.boolean(),\n isServiceOutput: z.boolean(),\n serviceReady: z.boolean(),\n message: z.string(),\n }).optional(),\n use_standalone_terminal: z.boolean().optional(),\n }),\n\n preview_url: z.object({\n type: z.literal('preview_tool_result'),\n url: z.string(),\n message: z.string(),\n }),\n\n ask_followup_question: z.object({\n type: z.literal('multi_question_result'),\n questions: z.array(z.object({\n id: z.string(),\n question: z.string(),\n options: z.array(z.string()),\n multiSelect: z.boolean().optional(),\n title: z.string().optional(),\n })),\n answers: z.record(z.union([z.string(), z.array(z.string())])),\n message: z.string(),\n }),\n\n invoke_integration: z.object({\n type: z.literal('invoke_integration_tool_result'),\n recommend: z.object({\n id: z.string(),\n type: z.string(),\n status: z.enum(['connected', 'disconnected']),\n }),\n message: z.string(),\n }),\n\n call_integration: z.object({\n type: z.literal('call_integration_tool_result'),\n integrationId: z.string(),\n toolName: z.string(),\n data: z.object({\n type: z.literal('text'),\n text: z.string(),\n }),\n isError: z.boolean().optional(),\n error: z.unknown().optional(),\n }),\n\n search_integration_tool: z.object({\n type: z.literal('search_integration_tool_result'),\n data: z.array(z.object({\n integrationId: z.string(),\n integrationName: z.string(),\n toolName: z.string(),\n description: z.string(),\n inputSchema: z.record(z.unknown()),\n })),\n hint: z.string().optional(),\n }),\n\n supabase_get_logs: z.object({\n type: z.enum([\n 'supabase_get_logs_result',\n 'supabase_execute_sql_result',\n 'supabase_apply_migration_result',\n 'supabase_list_migration_result',\n 'supabase_list_tables_result',\n ]),\n message: z.string(),\n }),\n\n supabase_execute_sql: z.object({\n type: z.enum([\n 'supabase_get_logs_result',\n 'supabase_execute_sql_result',\n 'supabase_apply_migration_result',\n 'supabase_list_migration_result',\n 'supabase_list_tables_result',\n ]),\n message: z.string(),\n }),\n\n supabase_apply_migration: z.object({\n type: z.enum([\n 'supabase_get_logs_result',\n 'supabase_execute_sql_result',\n 'supabase_apply_migration_result',\n 'supabase_list_migration_result',\n 'supabase_list_tables_result',\n ]),\n message: z.string(),\n }),\n\n supabase_list_migration: z.object({\n type: z.enum([\n 'supabase_get_logs_result',\n 'supabase_execute_sql_result',\n 'supabase_apply_migration_result',\n 'supabase_list_migration_result',\n 'supabase_list_tables_result',\n ]),\n message: z.string(),\n }),\n\n supabase_list_tables: z.object({\n type: z.enum([\n 'supabase_get_logs_result',\n 'supabase_execute_sql_result',\n 'supabase_apply_migration_result',\n 'supabase_list_migration_result',\n 'supabase_list_tables_result',\n ]),\n message: z.string(),\n }),\n\n cloud_studio_fetch_log: z.object({\n type: z.literal('cloud_studio_fetch_log_result'),\n success: z.boolean(),\n logs: z.record(z.string()),\n }),\n\n cloud_studio_execute_command: z.object({\n type: z.literal('cloud_studio_execute_command_result'),\n success: z.boolean(),\n message: z.string(),\n }),\n\n cloud_studio_deploy_sandbox: z.object({\n type: z.literal('cloud_studio_integration_result'),\n previewUrl: z.string().optional(),\n steps: z.array(z.object({\n status: z.enum(['idle', 'success', 'running', 'error']),\n name: z.enum(['createSandbox', 'uploadProject', 'installDependencies', 'startService', 'preview']),\n error: z.object({\n code: z.number().optional(),\n message: z.string().optional(),\n }).optional(),\n })),\n }),\n\n component_get_prompt: z.object({\n type: z.literal('component_get_prompt_result'),\n componentType: z.string(),\n webFramework: z.string(),\n data: z.object({\n type: z.literal('text'),\n text: z.string(),\n }),\n }),\n\n web_fetch: z.object({\n type: z.literal('web_fetch_tool_result'),\n message: z.string(),\n data: z.string(),\n loading: z.string().optional(),\n title: z.string().optional(),\n favicon: z.string().optional(),\n }),\n\n use_skill: z.object({\n type: z.literal('use_skill_tool_result'),\n commandMessage: z.string(),\n message: z.string(),\n }),\n\n web_search: z.object({\n type: z.literal('web_search_tool_result'),\n data: z.array(z.object({\n passage: z.string(),\n uri: z.string(),\n site: z.string(),\n title: z.string(),\n snippets: z.array(z.string()).optional(),\n content: z.string().optional(),\n })),\n searchInput: z.string().optional(),\n }),\n\n task: z.object({\n type: z.literal('task_tool_result'),\n toolInfo: z.array(z.object({\n name: z.string(),\n info: z.string(),\n needApprove: z.boolean().optional(),\n toolCallId: z.string().optional(),\n executeStatus: z.enum(['ing', 'completed', 'cancel', 'fail']).optional(),\n })).optional(),\n startCallTool: z.boolean().optional(),\n finalResult: z.string().optional(),\n toolCallBrief: z.string().optional(),\n }),\n\n codebase_search: z.object({\n type: z.literal('codebase_search_result'),\n query: z.string(),\n path: z.string(),\n content: z.string(),\n }),\n\n lsp: z.object({\n type: z.literal('lsp_tool_result'),\n operation: z.string(),\n result: z.string(),\n resultCount: z.number().optional(),\n fileCount: z.number().optional(),\n character: z.number().optional(),\n }),\n\n spec_create: z.object({\n type: z.literal('plan_create_tool_result'),\n message: z.string(),\n data: z.string(),\n }),\n\n spec_update: z.object({\n type: z.literal('plan_update_tool_result'),\n status: z.enum(['prepare', 'ready', 'building', 'finished']),\n data: z.string(),\n reminder: z.string(),\n }),\n} satisfies Record<ToolName, z.ZodTypeAny>;\n\n// ============================================================================\n// 工具 Schema 验证工具函数\n// ============================================================================\n\n/**\n * 验证工具输入\n * @param toolName 工具名称\n * @param input 输入数据\n * @returns 验证结果\n */\nexport function validateToolInput<TName extends ToolName>(\n toolName: TName,\n input: unknown\n): { success: true; data: ToolInputMap[TName] } | { success: false; error: z.ZodError } {\n const schema = ToolInputSchemas[toolName];\n const result = schema.safeParse(input);\n if (result.success) {\n return { success: true, data: result.data as ToolInputMap[TName] };\n }\n return { success: false, error: result.error };\n}\n\n/**\n * 验证工具输出\n * @param toolName 工具名称\n * @param output 输出数据\n * @returns 验证结果\n */\nexport function validateToolOutput<TName extends ToolName>(\n toolName: TName,\n output: unknown\n): { success: true; data: ToolOutputMap[TName] } | { success: false; error: z.ZodError } {\n const schema = ToolOutputSchemas[toolName];\n const result = schema.safeParse(output);\n if (result.success) {\n return { success: true, data: result.data as ToolOutputMap[TName] };\n }\n return { success: false, error: result.error };\n}\n","/**\n * Mock Agent Provider\n *\n * 职责:\n * 1. 专门用于 Mock 数据验证链路\n * 2. 返回符合 ACP 协议的模拟数据\n * 3. 用于前端开发和测试阶段\n *\n * Mock 数据包括:\n * - 流式文本消息 (agent_message_chunk)\n * - 嵌入资源内容\n * - 资源链接\n * - 工具调用流程 (tool_call + tool_call_update)\n */\n\nimport type {\n AuthenticateRequest,\n AuthenticateResponse,\n CancelNotification,\n InitializeRequest,\n InitializeResponse,\n LoadSessionRequest,\n LoadSessionResponse,\n NewSessionRequest,\n NewSessionResponse,\n PromptRequest,\n PromptResponse,\n SessionNotification,\n SetSessionModelRequest,\n SetSessionModelResponse,\n SetSessionModeRequest,\n SetSessionModeResponse\n} from '@agentclientprotocol/sdk';\n\nimport type {\n PermissionRequestParams,\n PermissionResponse,\n SessionUpdateParams} from './acp-client-protocol';\nimport {\n DeleteFilesResult,\n ReadFileResult,\n validateToolOutput} from './tool-schemas.js';\nimport type { IAgentProvider } from './types';\n\nexport interface MockAgentProviderConfig {\n useRealProvider?: boolean; // 是否使用真实 Provider(保留向后兼容)\n}\n\n/**\n * ACP 协议 Mock 数据 - 初始化响应\n */\nexport const mockInitializeResponse: InitializeResponse = {\n protocolVersion: 1,\n agentCapabilities: {\n loadSession: true,\n promptCapabilities: {\n image: true,\n audio: false,\n embeddedContext: true\n },\n mcpCapabilities: {\n http: true,\n sse: true\n }\n }\n};\n\n/**\n * ACP 协议 Mock 数据 - 会话创建响应\n */\nexport const mockNewSessionResponse = (sessionId: string): NewSessionResponse => ({\n sessionId\n});\n\n/**\n * Mock 会话数据存储\n * 每个会话都有对应的模拟历史消息\n */\nconst mockSessionHistories = new Map<string, Array<{\n type: 'user' | 'assistant';\n content: string;\n timestamp: number;\n}>>();\n\n// 初始化会话历史数据\nmockSessionHistories.set('1', [\n {\n type: 'user',\n content: '帮我开发一个五子棋游戏,需要包含以下功能:\\n1. 双人对战模式\\n2. 悔棋功能\\n3. 计时器',\n timestamp: Date.now() - 1800000 - 60000,\n },\n {\n type: 'assistant',\n content: '好的,我来帮你开发一个五子棋游戏。我会创建以下文件结构:\\n\\n- `index.html` - 主页面\\n- `game.js` - 游戏逻辑\\n- `style.css` - 样式文件\\n\\n让我开始创建这些文件...',\n timestamp: Date.now() - 1800000 - 50000,\n },\n {\n type: 'assistant',\n content: '我已经完成了所有文件的创建和修改。游戏支持双人对战、悔棋和计时功能。你可以直接在浏览器中打开 index.html 开始游戏。',\n timestamp: Date.now() - 1800000,\n },\n]);\n\nmockSessionHistories.set('2', [\n {\n type: 'user',\n content: '登录页面的样式有问题,按钮颜色不对齐,帮我修复一下',\n timestamp: Date.now() - 7200000 - 120000,\n },\n {\n type: 'assistant',\n content: '我来检查登录页面的样式。让我先看看相关的 CSS 文件...',\n timestamp: Date.now() - 7200000 - 100000,\n },\n {\n type: 'assistant',\n content: '样式问题已修复,请查看效果。主要修改了按钮的 flex 布局和颜色变量。',\n timestamp: Date.now() - 7200000,\n },\n]);\n\nmockSessionHistories.set('3', [\n {\n type: 'user',\n content: 'API 接口响应太慢了,需要添加缓存和错误重试机制',\n timestamp: Date.now() - 14400000 - 180000,\n },\n {\n type: 'assistant',\n content: '好的,我来优化 API 接口。我会实现:\\n1. Redis 缓存层\\n2. 指数退避重试策略\\n3. 请求去重',\n timestamp: Date.now() - 14400000 - 150000,\n },\n {\n type: 'assistant',\n content: '已添加缓存和错误重试机制。性能提升了约 60%。',\n timestamp: Date.now() - 14400000,\n },\n]);\n\nmockSessionHistories.set('4', [\n {\n type: 'user',\n content: '帮我为核心模块添加单元测试,覆盖率要达到 80% 以上',\n timestamp: Date.now() - 86400000 - 300000,\n },\n {\n type: 'assistant',\n content: '好的,我会使用 Jest 来编写单元测试。让我先看看核心模块的代码结构...',\n timestamp: Date.now() - 86400000 - 250000,\n },\n {\n type: 'assistant',\n content: '测试覆盖率已达到 85%,超过了目标。主要测试了:\\n- 用户认证逻辑\\n- 数据验证\\n- 错误处理',\n timestamp: Date.now() - 86400000,\n },\n]);\n\nexport class MockAgentProvider implements IAgentProvider {\n // ========== ACP 原生事件回调集合 ==========\n private sessionUpdateCallbacks: Set<(params: SessionUpdateParams) => void> = new Set();\n private permissionRequestResolvers: Map<string, (params: PermissionRequestParams) => Promise<PermissionResponse>> = new Map();\n private errorCallbacks: Set<(error: Error, sessionId?: string) => void> = new Set();\n\n // Minimal connection tracking to keep API compatibility with previous SSE provider\n private activeConnections: Map<string, unknown> = new Map();\n /**\n * Mock 模式下,等待用户权限决策的解析器集合\n * key: toolCallId -> resolver(outcome)\n */\n private pendingPermissionResolvers: Map<string, (outcome: { outcome: 'selected' | 'cancelled'; optionId?: string; timedOut?: boolean }) => void> = new Map();\n /**\n * Mock 模式下,已初始化的会话集合\n */\n private initializedSessions: Map<string, { cwd: string; mcpServers: any[]; createdAt: number }> = new Map();\n /**\n * 临时缓存当前正在发送的消息的 SessionUpdateParams\n * key: sessionId -> params[]\n * 在 sendMockMessage 开始时初始化,结束时保存到 mockSessionHistories\n */\n private currentSessionUpdates: Map<string, SessionUpdateParams[]> = new Map();\n\n constructor(config: MockAgentProviderConfig = {}) {\n console.log('[MockAgentProvider] Initialized');\n\n // 初始化一些默认的 Mock session 数据\n this.initializeMockSessions();\n }\n\n setSessionMode?(params: SetSessionModeRequest): Promise<SetSessionModeResponse | void> {\n throw new Error('Method not implemented.');\n }\n setSessionModel?(params: SetSessionModelRequest): Promise<SetSessionModelResponse | void> {\n throw new Error('Method not implemented.');\n }\n extMethod?(method: string, params: Record<string, unknown>): Promise<Record<string, unknown>> {\n throw new Error('Method not implemented.');\n }\n extNotification?(method: string, params: Record<string, unknown>): Promise<void> {\n throw new Error('Method not implemented.');\n }\n\n /**\n * 初始化默认的 Mock session 数据\n */\n private initializeMockSessions(): void {\n const mockSessions = [\n {\n sessionId: '1',\n cwd: '/Users/example/Project',\n mcpServers: [],\n createdAt: Date.now() - 3600000, // 1小时前\n },\n {\n sessionId: '2',\n cwd: '/Users/example/Project',\n mcpServers: [],\n createdAt: Date.now() - 7200000, // 2小时前\n },\n {\n sessionId: '3',\n cwd: '/Users/example/Project',\n mcpServers: [],\n createdAt: Date.now() - 86400000, // 1天前\n },\n ];\n\n mockSessions.forEach(session => {\n this.initializedSessions.set(session.sessionId, {\n cwd: session.cwd,\n mcpServers: session.mcpServers,\n createdAt: session.createdAt,\n });\n });\n\n console.log('[MockAgentProvider] Initialized mock sessions:', mockSessions.length);\n }\n\n /**\n * 从 PromptRequest 中提取文本内容\n * Provider 自己决定如何处理 ContentBlock 数组\n */\n private extractTextFromRequest(request: PromptRequest): string {\n if (request.prompt && Array.isArray(request.prompt)) {\n return request.prompt\n .filter(block => block.type === 'text')\n .map(block => ('text' in block ? block.text : ''))\n .join('\\n');\n }\n return '';\n }\n\n /**\n * 发送消息 - 直接使用 Mock 数据\n * 接收 PromptRequest,由 Provider 决定如何处理\n */\n async sendMessage(request: PromptRequest): Promise<void> {\n // 提取文本内容 - 这是 Provider 的职责\n const content = this.extractTextFromRequest(request);\n const sessionId = request.sessionId;\n\n console.log('[MockAgentProvider] Sending mock message:', { sessionId, content });\n\n try {\n // 直接使用 Mock 数据\n await this.sendMockMessage(sessionId, content);\n } catch (error) {\n console.error('[MockAgentProvider] Send message failed:', error);\n\n // 触发错误事件\n this.emitEvent({\n type: 'error',\n sessionId,\n timestamp: Date.now(),\n data: { error: error instanceof Error ? error.message : 'Unknown error' },\n });\n\n throw error;\n }\n }\n\n /**\n * 使用 Mock 数据模拟流式响应\n */\n private async sendMockMessage(sessionId: string, content: string): Promise<void> {\n console.log('[MockAgentProvider] Using mock data for:', content);\n\n // ✅ 初始化临时事件缓存\n this.currentSessionUpdates.set(sessionId, []);\n\n // 发送连接成功事件\n this.emitEvent({\n type: 'connected',\n sessionId,\n timestamp: Date.now(),\n });\n\n // 模拟延迟\n await this.delay(300);\n\n // 1. 发送文本回复消息\n const messageId1 = `m-text-${Date.now()}`;\n const textReply = `我收到了你发送的「${content}」信息,这是符合 ACP 协议的 mock 数据。\\n\\n下面演示不同类型的 ContentBlock:\\n\\n`;\n await this.streamMockTextContent(sessionId, messageId1, textReply);\n\n await this.delay(500);\n\n // 2. 发送包含嵌入资源的回复\n const messageId2 = `m-resource-${Date.now()}`;\n await this.sendMockEmbeddedResource(sessionId, messageId2);\n\n await this.delay(500);\n\n // 3. 发送包含资源链接的回复\n const messageId3 = `m-link-${Date.now()}`;\n await this.sendMockResourceLink(sessionId, messageId3);\n\n await this.delay(500);\n\n // 4. 模拟工具调用流程\n const toolMessageId = `m-tool-${Date.now()}`;\n await this.sendMockEditContentToolCallFlow(sessionId, toolMessageId);\n await this.delay(500);\n\n const diffToolMessageId = `m-tool-d-${Date.now()}`;\n await this.sendMockEditDiffToolCallFlow(sessionId, diffToolMessageId);\n await this.delay(500);\n\n const readToolMessageId = `m-tool-r-${Date.now()}`;\n await this.sendMockReadContentToolCallFlow(sessionId, readToolMessageId);\n await this.delay(300);\n\n // 添加 search_file 和 search_content 的 mock 数据\n const searchContentMessageId = `m-tool-sc-${Date.now()}`;\n await this.sendMockSearchContentToolCallFlow(sessionId, searchContentMessageId);\n\n await this.delay(500);\n\n const searchFileMessageId = `m-tool-sf-${Date.now()}`;\n await this.sendMockSearchFileToolCallFlow(sessionId, searchFileMessageId);\n\n await this.delay(500);\n\n // 6. 发送 list_files 工具调用\n const listFilesMessageId = `m-tool-lf-${Date.now()}`;\n await this.sendMockListFilesToolCallFlow(sessionId, listFilesMessageId);\n\n await this.delay(500);\n\n // 7. 发送 ask_followup_question 工具调用\n const askFollowupMessageId = `m-tool-afq-${Date.now()}`;\n await this.sendMockAskFollowupQuestionToolCallFlow(sessionId, askFollowupMessageId);\n await this.delay(500);\n\n // 添加删除文件工具调用的 mock 数据\n const deleteFileMessageId = `m-tool-df-${Date.now()}`;\n await this.sendMockDeleteFileToolCallFlow(sessionId, deleteFileMessageId);\n\n await this.delay(300);\n\n // 8. 发送 execute_command 工具调用\n const executeCommandMessageId = `m-tool-ec-${Date.now()}`;\n await this.sendMockExecuteCommandToolCallFlow(sessionId, executeCommandMessageId);\n\n await this.delay(300);\n\n // 8.1 发送 execute_command 工具调用(需要权限确认)\n const executeCommandApprovalMessageId = `m-tool-ec-approval-${Date.now()}`;\n await this.sendMockExecuteCommandWithApprovalToolCallFlow(sessionId, executeCommandApprovalMessageId);\n\n await this.delay(300);\n\n // 8.2 发送 write_to_file 工具调用(需要权限确认)\n const writeFileApprovalMessageId = `m-tool-wtf-approval-${Date.now()}`;\n await this.sendMockWriteFileWithApprovalToolCallFlow(sessionId, writeFileApprovalMessageId);\n\n await this.delay(300);\n\n // 9. 发送 preview_url 工具调用 (测试 unknown tool)\n const previewUrlMessageId = `m-tool-pu-${Date.now()}`;\n await this.sendMockPreviewUrlToolCallFlow(sessionId, previewUrlMessageId);\n\n await this.delay(300);\n\n // 10. 发送 list_code_definition_names 工具调用 (测试 unknown tool)\n const listCodeDefMessageId = `m-tool-lcd-${Date.now()}`;\n await this.sendMockListCodeDefinitionNamesToolCallFlow(sessionId, listCodeDefMessageId);\n\n await this.delay(300);\n\n // 添加 web_fetch 工具调用的 mock 数据\n const webFetchMessageId = `m-tool-wf-${Date.now()}`;\n await this.sendMockWebFetchToolCallFlow(sessionId, webFetchMessageId);\n\n await this.delay(300);\n\n // 11. 发送 read_lints 工具调用\n const readLintsMessageId = `m-tool-rl-${Date.now()}`;\n await this.sendMockReadLintsToolCallFlow(sessionId, readLintsMessageId);\n\n await this.delay(300);\n\n // 10. 发送 web_search 工具调用\n const webSearchMessageId = `m-tool-ws-${Date.now()}`;\n await this.sendMockWebSearchToolCallFlow(sessionId, webSearchMessageId);\n\n await this.delay(300);\n\n // 发送完成事件\n this.emitEvent({\n type: 'done',\n sessionId,\n timestamp: Date.now(),\n });\n }\n\n /**\n * 流式发送 Mock 文本消息内容 (符合 ACP 协议的 agent_message_chunk)\n */\n private async streamMockTextContent(sessionId: string, messageId: string, content: string): Promise<void> {\n const chunkSize = 10; // 每次发送 10 个字符\n\n for (let i = 0; i < content.length; i += chunkSize) {\n // 获取当前块的内容\n const chunk = content.substring(i, i + chunkSize);\n\n console.log(`Text Chunk: \"${chunk}\" | messageId: ${messageId} | sessionId: ${sessionId}`);\n\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: chunk,\n }\n }\n } as SessionNotification,\n });\n\n // 模拟网络延迟\n await this.delay(50);\n }\n }\n\n /**\n * 延迟函数\n */\n private delay(ms: number): Promise<void> {\n return new Promise(resolve => setTimeout(resolve, ms));\n }\n\n /**\n * 取消消息 (Mock 模式下直接完成)\n * @deprecated 使用 cancel 代替\n */\n async cancelMessage(sessionId: string): Promise<void> {\n console.log('[MockAgentProvider] Message cancelled:', sessionId);\n\n // 发送完成事件\n this.emitEvent({\n type: 'done',\n sessionId,\n timestamp: Date.now(),\n });\n }\n\n /**\n * ACP 协议: prompt - 发送用户消息\n *\n * Mock 实现: 复用 sendMessage\n */\n async prompt(request: PromptRequest): Promise<PromptResponse> {\n console.log('[MockAgentProvider] prompt called:', request);\n await this.sendMessage(request);\n // Mock 模式下,返回 end_turn 响应\n return {\n stopReason: 'end_turn',\n };\n }\n\n /**\n * ACP 协议: cancel - 取消消息\n *\n * Mock 实现: 复用 cancelMessage\n */\n async cancel(params: CancelNotification): Promise<void> {\n console.log('[MockAgentProvider] cancel called:', params);\n return this.cancelMessage(params.sessionId);\n }\n\n // ========== ACP 原生事件监听器 ==========\n\n /**\n * 监听会话更新事件(ACP 协议原生)\n */\n onSessionUpdate(callback: (params: SessionUpdateParams) => void): () => void {\n this.sessionUpdateCallbacks.add(callback);\n return () => {\n this.sessionUpdateCallbacks.delete(callback);\n };\n }\n\n /**\n * 监听权限请求事件(ACP 协议原生)\n */\n onRequestPermission(\n callback: (params: PermissionRequestParams) => Promise<PermissionResponse>\n ): () => void {\n this.permissionRequestResolvers.set('__global__', callback);\n return () => {\n this.permissionRequestResolvers.delete('__global__');\n };\n }\n\n /**\n * 监听错误事件(扩展事件,非 ACP 标准协议)\n */\n onError(callback: (error: Error, sessionId?: string) => void): () => void {\n this.errorCallbacks.add(callback);\n return () => {\n this.errorCallbacks.delete(callback);\n };\n }\n\n /**\n * 触发会话更新事件(内部辅助方法)\n */\n private emitSessionUpdate(params: SessionUpdateParams): void {\n console.log('[MockAgentProvider] Emitting session update:', params, 'to', this.sessionUpdateCallbacks.size, 'callbacks');\n\n this.sessionUpdateCallbacks.forEach(callback => {\n try {\n callback(params);\n } catch (error) {\n console.error('[MockAgentProvider] Session update callback error:', error);\n }\n });\n }\n\n /**\n * 触发错误事件(内部辅助方法)\n */\n private emitError(error: Error, sessionId?: string): void {\n console.log('[MockAgentProvider] Emitting error:', error, 'for session:', sessionId);\n\n this.errorCallbacks.forEach(callback => {\n try {\n callback(error, sessionId);\n } catch (err) {\n console.error('[MockAgentProvider] Error callback error:', err);\n }\n });\n }\n\n /**\n * 向后兼容的 emitEvent 方法\n * 将旧的事件格式桥接到新的 ACP 原生事件\n */\n private emitEvent(event: any): void {\n // 根据 event.type 路由到对应的新方法\n if (event.type === 'error') {\n const error = event.data?.error ? new Error(event.data.error) : new Error('Unknown error');\n this.emitError(error, event.sessionId);\n } else if (event.type === 'session_update' && event.notification) {\n // 转换为 SessionUpdateParams\n const params: SessionUpdateParams = {\n notification: event.notification,\n messageId: event.messageId,\n };\n this.emitSessionUpdate(params);\n } else if (event.type === 'connected' || event.type === 'done') {\n // 这些事件类型不再需要,忽略\n console.log('[MockAgentProvider] Ignoring deprecated event type:', event.type);\n } else {\n console.warn('[MockAgentProvider] Unknown event type:', event.type);\n }\n }\n\n /**\n * ACP 协议: initialize - 初始化连接\n *\n * Mock 实现: 返回预设的初始化响应\n */\n async initialize(request: InitializeRequest): Promise<InitializeResponse> {\n console.log('[MockAgentProvider] initialize', request);\n\n // 模拟网络延迟\n await this.delay(100);\n\n // 返回 Mock 初始化响应\n return mockInitializeResponse;\n }\n\n /**\n * ACP 协议: authenticate - 身份验证\n *\n * Mock 实现: 直接返回成功\n */\n async authenticate(request: AuthenticateRequest): Promise<AuthenticateResponse | void> {\n console.log('[MockAgentProvider] authenticate', request);\n // Mock 模式下不需要身份验证\n return;\n }\n\n /**\n * ACP 协议: newSession - 创建新会话\n *\n * Mock 实现: 生成唯一的 sessionId 并存储会话信息\n */\n async newSession(request: NewSessionRequest): Promise<NewSessionResponse> {\n console.log('[MockAgentProvider] session/new', request);\n\n // 模拟网络延迟\n await this.delay(100);\n\n // 生成唯一的 sessionId (格式: sess_<timestamp>_<random>)\n const sessionId = `sess_${Date.now()}_${Math.random().toString(36).substring(2, 9)}`;\n\n // 存储会话信息\n this.initializedSessions.set(sessionId, {\n cwd: request.cwd,\n mcpServers: request.mcpServers || [],\n createdAt: Date.now()\n });\n\n console.log('[MockAgentProvider] Created session:', sessionId, 'with config:', {\n cwd: request.cwd,\n mcpServers: request.mcpServers\n });\n\n // 返回 sessionId\n return mockNewSessionResponse(sessionId);\n }\n\n /**\n * ACP 协议: session/load - 加载已有会话\n *\n * Mock 实现: 流式回放会话历史(完整的事件回放)\n */\n async loadSession(request: LoadSessionRequest): Promise<LoadSessionResponse> {\n console.log('[MockAgentProvider] session/load', request);\n\n const { sessionId, cwd, mcpServers } = request;\n\n // 更新或添加会话配置\n this.initializedSessions.set(sessionId, {\n cwd,\n mcpServers: mcpServers || [],\n createdAt: Date.now()\n });\n\n // 从 mockSessionHistories 加载历史\n const history = mockSessionHistories.get(sessionId);\n if (history && history.length > 0) {\n console.log('[MockAgentProvider] Loading session from mockSessionHistories:', sessionId, 'with', history.length, 'entries');\n\n // 模拟流式回放会话历史\n for (let i = 0; i < history.length; i++) {\n const message = history[i];\n\n // 检查是否为事件数组格式(JSON 字符串)\n if (message.type === 'assistant' && message.content && message.content.startsWith('[')) {\n try {\n const events: any[] = JSON.parse(message.content);\n console.log('[MockAgentProvider] Replaying events from history:', {\n index: i,\n eventCount: events.length,\n });\n\n // 回放所有事件(通过 emitEvent 桥接)\n for (const event of events) {\n await this.delay(20); // 每个事件间隔 20ms\n this.emitEvent(event);\n }\n } catch (error) {\n console.error('[MockAgentProvider] Failed to parse events from history:', error);\n }\n } else {\n // 旧格式:普通文本消息\n const messageId = `load-${sessionId}-${i}-${Date.now()}`;\n const sessionUpdateType = message.type === 'user'\n ? 'user_message_chunk'\n : 'agent_message_chunk';\n\n await this.delay(50);\n\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: message.timestamp,\n notification: {\n sessionId,\n update: {\n sessionUpdate: sessionUpdateType,\n content: {\n type: 'text',\n text: message.content,\n }\n }\n } as SessionNotification,\n });\n }\n }\n\n console.log('[MockAgentProvider] Session loaded from mockSessionHistories:', sessionId);\n return {};\n }\n\n // 如果没有任何历史数据\n console.log('[MockAgentProvider] No history found for session:', sessionId);\n return {};\n }\n\n /**\n * ACP 协议: session/request_permission - 请求用户权限\n *\n * Agent 调用此方法来请求用户批准工具调用。\n * 此方法会:\n * 1. 发送 request_permission 事件给 UI\n * 2. 阻塞等待用户响应\n * 3. 返回用户的决策结果\n *\n * @param sessionId ACP 会话 ID\n * @param toolCallId 工具调用 ID\n * @param toolCall 工具调用信息\n * @param options 可用的权限选项\n * @returns 用户的决策结果\n */\n async requestPermission(\n sessionId: string,\n toolCallId: string,\n toolCall: any,\n options: any[]\n ): Promise<{ outcome: 'selected' | 'cancelled'; optionId?: string; timedOut?: boolean }> {\n console.log('[MockAgentProvider] requestPermission called:', { sessionId, toolCallId, toolCall, options });\n\n // 发送权限请求事件给 UI\n this.emitEvent({\n type: 'request_permission',\n sessionId,\n messageId: `permission-${toolCallId}`,\n timestamp: Date.now(),\n data: {\n sessionId,\n toolCall,\n options,\n },\n });\n\n // 阻塞等待用户决策(或超时)\n const permissionOutcome = await new Promise<{\n outcome: 'selected' | 'cancelled';\n optionId?: string;\n timedOut?: boolean;\n }>(resolve => {\n // 保存解析器,respondToPermissionRequest 会调用它\n this.pendingPermissionResolvers.set(toolCallId, resolve);\n\n // 超时降级:如果 60s 内没有用户响应,默认拒绝(避免无限挂起)\n setTimeout(() => {\n if (this.pendingPermissionResolvers.has(toolCallId)) {\n this.pendingPermissionResolvers.delete(toolCallId);\n console.log('[MockAgentProvider] Permission request timed out:', toolCallId);\n resolve({ outcome: 'cancelled', optionId: 'reject-once', timedOut: true });\n }\n }, 60000);\n });\n\n console.log('[MockAgentProvider] Permission outcome:', permissionOutcome);\n return permissionOutcome;\n }\n\n /**\n * 直接释放权限请求的 await(更直接的方法)\n *\n * 这个方法直接从 pendingPermissionResolvers 中获取 resolver 并调用,\n * 不需要经过 respondToPermissionRequest 的中转。\n *\n * @param toolCallId 工具调用 ID\n * @param outcome 用户的决策结果\n */\n releasePermissionRequest(toolCallId: string, outcome: { outcome: 'selected' | 'cancelled'; optionId?: string }): void {\n console.log('[MockAgentProvider] releasePermissionRequest called:', { toolCallId, outcome });\n\n const resolver = this.pendingPermissionResolvers.get(toolCallId);\n if (resolver) {\n // 直接调用 resolver 释放 await\n resolver({ ...outcome, timedOut: false });\n this.pendingPermissionResolvers.delete(toolCallId);\n console.log('[MockAgentProvider] Permission request released:', toolCallId);\n } else {\n console.warn('[MockAgentProvider] No pending permission request found for:', toolCallId);\n }\n }\n\n /**\n * 响应用户的权限决策(兼容旧接口)\n *\n * 这个方法内部调用 releasePermissionRequest。\n * 保留这个方法是为了向后兼容和符合 ACP 协议的命名规范。\n *\n * @param sessionId ACP 会话 ID(未使用,保留以符合接口)\n * @param payload 权限响应数据\n */\n async respondToPermissionRequest(sessionId: string, payload: {\n toolCallId: string;\n outcome: { outcome: 'selected' | 'cancelled'; optionId?: string }\n }): Promise<void> {\n console.log('[MockAgentProvider] respondToPermissionRequest called:', payload);\n\n // 直接调用 releasePermissionRequest\n this.releasePermissionRequest(payload.toolCallId, payload.outcome);\n }\n\n /**\n * 销毁 Provider\n */\n destroy(): void {\n // 清空所有回调解集合\n this.activeConnections.clear();\n this.pendingPermissionResolvers.clear();\n this.permissionRequestResolvers.clear();\n this.initializedSessions.clear();\n this.sessionUpdateCallbacks.clear();\n this.errorCallbacks.clear();\n // ✅ 清空临时事件缓存\n this.currentSessionUpdates.clear();\n console.log('[MockAgentProvider] Destroyed');\n }\n\n /**\n * 获取所有已初始化的会话列表(扩展方法,非 ACP 标准协议)\n * 返回 sessionId 及其配置信息和标题\n */\n getAllSessions(): Array<{ sessionId: string; title: string; cwd: string; mcpServers: any[]; createdAt: number }> {\n const mockTitles = {\n '1': '开发五子棋游戏',\n '2': '修复登录页面样式',\n '3': 'API 接口优化',\n };\n\n const sessions = Array.from(this.initializedSessions.entries()).map(([sessionId, config]) => ({\n sessionId,\n title: mockTitles[sessionId as keyof typeof mockTitles] || `Session ${sessionId}`,\n ...config,\n }));\n console.log('[MockAgentProvider] Getting all sessions:', sessions.length);\n return sessions;\n }\n\n /**\n * 发送嵌入资源消息 (符合 ACP 协议)\n */\n private async sendMockEmbeddedResource(sessionId: string, messageId: string): Promise<void> {\n console.log(`📎 Sending embedded resource: ${messageId}`);\n\n // 发送文本介绍部分\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-intro`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: '这是一个嵌入的 Python 文件示例:\\n\\n',\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(200);\n\n // 发送嵌入资源内容块\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-resource`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'resource',\n resource: {\n uri: 'file:///mock/example.py',\n mimeType: 'text/x-python',\n text: 'def hello_world():\\n print(\"Hello, World!\")\\n return \"success\"\\n\\nif __name__ == \"__main__\":\\n hello_world()'\n }\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(200);\n\n // 发送结尾文本\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-conclusion`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: '\\n\\n以上是直接嵌入的文件内容,无需额外请求即可显示。',\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(100);\n }\n\n /**\n * 发送资源链接消息 (符合 ACP 协议)\n */\n private async sendMockResourceLink(sessionId: string, messageId: string): Promise<void> {\n console.log(`🔗 Sending resource link: ${messageId}`);\n\n // 发送文本介绍部分\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-intro`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: '这是一个指向外部文档的资源链接:\\n\\n',\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(200);\n\n // 发送资源链接内容块\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-link`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'resource_link',\n uri: 'https://agentclientprotocol.com/protocol/content',\n name: 'ACP Content Protocol',\n mimeType: 'text/html',\n description: 'Agent Client Protocol - Content specification',\n size: BigInt(256000)\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(200);\n\n // 发送结尾文本\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-conclusion`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: '\\n\\n点击上面的链接可以访问完整的协议文档。',\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(100);\n }\n\n /**\n * 发送完整的工具调用流程 (符合 ACP 协议)\n */\n private async sendMockEditDiffToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`🔧 Starting tool call flow: ${messageId}`);\n\n // 2. 发送工具调用事件 - 使用 ACP 协议的 tool_call session/update\n const toolCallId = `tool-${Date.now()}`;\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Write File',\n kind: 'edit',\n status: 'pending',\n locations: [{ path: 'example.py', line: 0 }]\n }\n } as SessionNotification,\n });\n\n await this.delay(100);\n\n // 调用 requestPermission 接口请求用户权限\n // 这个方法会:\n // 1. 发送 request_permission 事件给 UI\n // 2. 阻塞等待用户响应\n // 3. 返回用户的决策结果\n // const permissionOutcome = await this.requestPermission(\n // sessionId,\n // toolCallId,\n // {\n // toolCallId,\n // kind: 'edit',\n // arguments: {\n // file_path: 'example.py',\n // content: 'def hello_world():\\n print(\"Hello from ACP tool call!\")...'\n // }\n // },\n // [\n // { optionId: 'allow-once', name: 'Allow once', kind: 'allow_once' },\n // { optionId: 'reject-once', name: 'Reject', kind: 'reject_once' }\n // ]\n // );\n\n // 根据用户决策推进后续 Mock 行为,区分同意与拒绝,发送不同的 tool_call_update 与消息\n // if (permissionOutcome.outcome === 'selected') {\n const content = 'def hello_world():\\n print(\"Hello from ACP tool call!\")\\n return \"success\"\\n\\nif __name__ == \"__main__\":\\n hello_world()';\n const chunkSize = 10; // 每次发送 10 个字符\n\n for (let i = 0; i < content.length; i += chunkSize) {\n // 获取当前块的内容\n const chunk = content.substring(i, i + chunkSize);\n\n console.log(`🐸 tool call Chunk: \"${chunk}\" | messageId: ${messageId} | sessionId: ${sessionId}`);\n\n // ✅ 5.1 开始执行 (in_progress)\n\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress', // ✅ 最终完成状态\n content: [\n {\n type: 'diff', // ✅ 使用 diff 类型显示文件修改\n path: 'example.py',\n oldText: '', // 新文件\n newText: chunk\n }\n ],\n locations: [{ path: 'example.py', line: 0 }]\n }\n } as SessionNotification,\n });\n // 模拟网络延迟\n await this.delay(50);\n }\n\n // 模拟工具执行时间\n await this.delay(1200);\n\n // ✅ 5.2 执行完成 (completed)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed', // ✅ 最终完成状态\n content: [\n {\n type: 'diff', // ✅ 使用 diff 类型显示文件修改\n path: 'example.py',\n oldText: 'def 213', // 新文件\n newText: 'def hello_world():\\n print(\"Hello from ACP tool call!\")\\n return \"success\"\\n\\nif __name__ == \"__main__\":\\n hello_world()'\n }\n ],\n locations: [{ path: 'example.py', line: 0 }]\n }\n } as SessionNotification,\n });\n\n // 发送总结消息\n await this.delay(300);\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-summary`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: '✅ 工具调用完成!我成功创建了一个名为 `example.py` 的文件。\\n\\nACP 协议工具调用流程:\\n1. `tool_call` (pending) - 创建工具调用\\n2. `tool_call_update` (in_progress) - 开始执行\\n3. `tool_call_update` (completed) - 执行完成\\n\\n这就是完整的工具调用生命周期!',\n }\n }\n } as SessionNotification,\n });\n // } else {\n // // ✅ 5.3 权限拒绝 (failed)\n // await this.delay(300);\n // this.emitEvent({\n // type: 'session_update',\n // sessionId,\n // messageId,\n // timestamp: Date.now(),\n // notification: {\n // sessionId,\n // update: {\n // sessionUpdate: 'tool_call_update',\n // toolCallId,\n // status: 'failed', // ✅ 失败状态\n // content: [\n // {\n // type: 'content',\n // content: {\n // type: 'text',\n // text: permissionOutcome.timedOut\n // ? 'Permission request timed out; tool execution cancelled (mock)'\n // : 'User rejected permission; tool execution cancelled (mock)'\n // }\n // }\n // ]\n // }\n // } as SessionNotification,\n // });\n\n // // 发送总结消息\n // await this.delay(300);\n // this.emitEvent({\n // type: 'session_update',\n // sessionId,\n // messageId: `${messageId}-summary`,\n // timestamp: Date.now(),\n // notification: {\n // sessionId,\n // update: {\n // sessionUpdate: 'agent_message_chunk',\n // content: {\n // type: 'text',\n // text: permissionOutcome.timedOut\n // ? '⏰ 权限请求超时,工具调用已取消(mock)。'\n // : '❌ 已拒绝执行写文件操作,工具调用已取消(mock)。',\n // }\n // }\n // } as SessionNotification,\n // });\n // }\n\n await this.delay(100);\n\n // 5. 发送工具调用完成后的总结消息\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-summary`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: '✅ 工具调用完成!我成功创建了一个名为 `example.py` 的文件。\\n\\n工具调用流程包括:\\n1. 发送工具调用请求 (tool_call)\\n2. 返回执行结果 (tool_call_update)\\n\\n这就是 ACP 协议中完整的工具调用生命周期。',\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(100);\n }\n\n private async sendMockEditContentToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`🔧 Starting tool call flow: ${messageId}`);\n\n // 2. 发送工具调用事件 - 使用 ACP 协议的 tool_call session/update\n const toolCallId = `tool-${Date.now()}`;\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'write_to_file',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Write File',\n kind: 'edit',\n status: 'pending',\n locations: [{ path: 'example-edit-content.py', line: 0 }]\n }\n } as SessionNotification,\n });\n\n await this.delay(100);\n\n const content = 'def hello_world():\\n print(\"Hello from ACP tool call!\")\\n return \"success\"\\n\\nif __name__ == \"__main__\":\\n hello_world()';\n const chunkSize = 10; // 每次发送 10 个字符\n\n let accumulatedInput = '';\n for (let i = 0; i < content.length; i += chunkSize) {\n // 获取当前块的内容\n const chunk = content.substring(i, i + chunkSize);\n accumulatedInput += chunk;\n\n console.log(`🐸 tool call Chunk: \"${chunk}\" | messageId: ${messageId} | sessionId: ${sessionId}`);\n\n // ✅ 5.1 开始执行 (in_progress)\n\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'write_to_file',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress', // ✅ 最终完成状态\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: accumulatedInput,\n\n }\n }\n ],\n rawInput: {\n 'filePath': 'example-edit-content.py',\n 'content': accumulatedInput,\n },\n locations: [{ path: 'example-edit-content.py', line: 0 }]\n }\n } as SessionNotification,\n });\n // 模拟网络延迟\n await this.delay(50);\n }\n\n // 模拟工具执行时间\n await this.delay(1200);\n\n // ✅ 5.2 执行完成 (completed)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'write_to_file',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed', // ✅ 最终完成状态\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: content\n }\n }\n ],\n rawOutput: {\n 'type': 'write_to_file_result',\n 'path': 'example-edit-content.py',\n 'addLineCount': 10,\n 'removedLines': 20,\n },\n locations: [{ path: 'example-edit-content.py', line: 0 }]\n }\n } as SessionNotification,\n });\n\n // 发送总结消息\n await this.delay(300);\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-summary`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'write_to_file',\n },\n },\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: '✅ 工具调用完成!我成功创建了一个名为 `example.py` 的文件。\\n\\nACP 协议工具调用流程:\\n1. `tool_call` (pending) - 创建工具调用\\n2. `tool_call_update` (in_progress) - 开始执行\\n3. `tool_call_update` (completed) - 执行完成\\n\\n这就是完整的工具调用生命周期!',\n }\n },\n rawInput: {\n 'filePath': 'example-edit-content.py',\n 'content': accumulatedInput,\n }\n } as SessionNotification,\n });\n\n await this.delay(100);\n\n // 5. 发送工具调用完成后的总结消息\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-summary`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: '✅ 工具调用完成!我成功创建了一个名为 `example.py` 的文件。\\n\\n工具调用流程包括:\\n1. 发送工具调用请求 (tool_call)\\n2. 返回执行结果 (tool_call_update)\\n\\n这就是 ACP 协议中完整的工具调用生命周期。',\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(100);\n }\n\n /**\n * 发送完整的工具调用流程 - kind 为 read (符合 ACP 协议)\n *\n * 模拟 read_file 工具调用流程,流式发送 rawInput 和 rawOutput\n */\n private async sendMockReadContentToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`📖 Starting read tool call flow: ${messageId}`);\n\n // 配置:模拟读取文件\n const filePath = '/Users/liumingyuan/Project/caseTest/subagent/test.js';\n const toolCallId = `tool-${Date.now()}`;\n\n // 1. 发送 tool_call (pending) - 不带 rawInput\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'read_file'\n }\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Read File',\n status: 'pending',\n locations: [{ path: filePath, line: 0 }]\n }\n } as SessionNotification\n });\n\n console.log('[MockAgentProvider] tool_call (pending) sent');\n await this.delay(100);\n\n // 2. 流式发送 rawInput (模拟参数逐个到达)\n // 将 filePath 分块流式发送\n const filePathChunks = [\n '/Users/liumingyuan/',\n 'Project/caseTest',\n '/subagent/',\n 'test.js'\n ];\n\n let accumulatedInput = '';\n for (const chunk of filePathChunks) {\n accumulatedInput += chunk;\n\n // 尝试解析累积的 input\n try {\n // const parsed = JSON.parse(accumulatedInput);\n // const rawInput: ReadFileInput = {\n // filePath: parsed.filePath\n // };\n\n // 验证 rawInput\n // const inputValidation = validateToolInput('read_file', rawInput);\n // if (inputValidation.success) {\n // 发送 tool_call_update with rawInput\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'read_file'\n }\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n rawInput: {\n filePath: accumulatedInput\n }\n }\n } as SessionNotification\n });\n await this.delay(100);\n console.log('[MockAgentProvider] tool_call_update with rawInput:', chunk);\n // }\n } catch {\n // JSON 还未完整,继续累积\n }\n\n }\n\n // Mock file content\n const fileContent = `/**\n * Test file for subagent functionality\n */\n\nfunction helloWorld() {\n console.log('Hello from test.js!');\n return 'success';\n}\n\nclass TestClass {\n constructor(name) {\n this.name = name;\n }\n\n greet() {\n return \\`Hello, \\${this.name}!\\`;\n }\n}\n\n// Export functions\nmodule.exports = {\n helloWorld,\n TestClass\n};\n`;\n\n const totalLineCount = 26;\n const chunkSize = 20;\n let accumulatedContent = '';\n\n // 3. 流式返回结果 with in_progress\n for (let i = 0; i < fileContent.length; i += chunkSize) {\n const chunk = fileContent.substring(i, i + chunkSize);\n accumulatedContent += chunk;\n const progress = Math.round(((i + chunkSize) / fileContent.length) * 100);\n const isLastChunk = (i + chunkSize) >= fileContent.length;\n\n const rawOutput: ReadFileResult = {\n type: 'read_file_result',\n path: filePath,\n content: accumulatedContent,\n totalLineCount,\n hasMore: !isLastChunk,\n hint: `Reading file... ${Math.min(progress, 100)}% complete`\n };\n\n // 验证 rawOutput\n const outputValidation = validateToolOutput('read_file', rawOutput);\n\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'read_file'\n }\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: accumulatedContent\n }\n }\n ],\n locations: [{ path: filePath, line: 0 }],\n rawOutput: (outputValidation.success ? outputValidation.data : rawOutput) as unknown as Record<string, unknown>\n }\n } as SessionNotification\n });\n\n console.log(`📖 Read chunk ${i / chunkSize + 1}: progress ${progress}%`);\n await this.delay(30);\n }\n\n // 4. 发送 completed 事件\n const finalRawOutput: ReadFileResult = {\n type: 'read_file_result',\n path: filePath,\n content: fileContent,\n totalLineCount,\n hasMore: false\n };\n\n const finalOutputValidation = validateToolOutput('read_file', finalRawOutput);\n if (!finalOutputValidation.success) {\n throw new Error(`Invalid rawOutput: ${JSON.stringify(finalOutputValidation.error.errors)}`);\n }\n\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: fileContent\n }\n }\n ],\n locations: [{ path: filePath, line: 0 }],\n rawOutput: finalOutputValidation.data as unknown as Record<string, unknown>\n }\n } as SessionNotification\n });\n\n console.log('[MockAgentProvider] tool_call_update (completed) sent');\n }\n\n /**\n * 发送 search_content 工具调用流程 (符合 ACP 协议)\n *\n * 用于模拟在文件内容中搜索特定文本的工具调用\n * 支持显示匹配的文件路径、行号和上下文内容\n */\n private async sendMockSearchContentToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`🔍 Starting search_content tool call flow: ${messageId}`);\n\n const pattern = '一行工具';\n const directory = '/Users/liumingyuan/Project/caseTest/subagent';\n\n // 1. 发送工具调用事件\n const toolCallId = `tool-sc-${Date.now()}`;\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'search_content',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Search Content',\n kind: 'search', // ✅ kind 为 search\n status: 'pending',\n locations: [{ path: directory, line: 0 }],\n rawInput: {\n pattern,\n directory,\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(100);\n\n // 2. 发送 in_progress 状态和搜索结果\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `Searching for \"${pattern}\" in ${directory}...`,\n }\n }\n ],\n locations: [{ path: directory, line: 0 }]\n }\n } as SessionNotification,\n });\n\n await this.delay(500);\n\n // 3. 发送完成状态和匹配结果\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: 'Found 1 matching result\\n\\nconvert_to_oneline.py\\n 56| print(\"文件内容转一行工具\")\\n',\n }\n }\n ],\n locations: [{ path: directory, line: 0 }],\n rawOutput: {\n pattern,\n directory,\n matches: [\n {\n filePath: 'convert_to_oneline.py',\n fileName: 'convert_to_oneline.py',\n content: ' print(\"文件内容转一行工具\")',\n startLine: 56,\n endLine: 56,\n }\n ],\n totalCount: 1,\n hasMore: false,\n }\n }\n } as SessionNotification,\n });\n\n // 4. 发送总结消息\n await this.delay(300);\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-summary`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: `✅ 内容搜索完成!在 \\`${directory}\\` 目录中找到 1 个匹配结果。\\n\\n**搜索结果:**\\n- \\`convert_to_oneline.py\\` 第 56 行: \\`print(\"文件内容转一行工具\")\\`\\n\\n**工具调用流程:**\\n1. \\`tool_call\\` (pending) - 创建搜索工具调用\\n2. \\`tool_call_update\\` (in_progress) - 执行搜索\\n3. \\`tool_call_update\\` (completed) - 返回匹配结果`,\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(100);\n }\n\n /**\n * 发送 search_file 工具调用流程 (符合 ACP 协议)\n *\n * 用于模拟按文件名模式搜索文件的工具调用\n * 支持递归搜索和文件大小显示\n */\n private async sendMockSearchFileToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`🔍 Starting search_file tool call flow: ${messageId}`);\n\n const pattern = '*.py';\n const directory = '/Users/liumingyuan/Project/caseTest/subagent';\n\n // 1. 发送工具调用事件\n const toolCallId = `tool-sf-${Date.now()}`;\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'search_file',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Search File',\n kind: 'search', // ✅ kind 为 search\n status: 'pending',\n locations: [{ path: directory, line: 0 }],\n rawInput: {\n target_directory: directory,\n pattern,\n recursive: true,\n caseSensitive: false,\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(100);\n\n // 2. 发送 in_progress 状态\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `Searching for files matching \"${pattern}\" in ${directory}...`,\n }\n }\n ],\n locations: [{ path: directory, line: 0 }]\n }\n } as SessionNotification,\n });\n\n await this.delay(500);\n\n // 3. 发送完成状态和搜索结果\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: 'Found 5 files\\n\\nbubble_sort.py (10.38 KB)\\nconvert_to_oneline.py (2.2 KB)\\ntls12_examples.py (4.58 KB)\\n.codebuddy/skills/analyzing-financial-statements/calculate_ratios.py (12.22 KB)\\n.codebuddy/skills/analyzing-financial-statements/interpret_ratios.py (15.83 KB)\\n',\n }\n }\n ],\n locations: [{ path: directory, line: 0 }],\n rawOutput: {\n path: directory,\n pattern,\n recursive: true,\n caseSensitive: false,\n results: [\n { filePath: 'bubble_sort.py', size: '10.38 KB' },\n { filePath: 'convert_to_oneline.py', size: '2.2 KB' },\n { filePath: 'tls12_examples.py', size: '4.58 KB' },\n { filePath: '.codebuddy/skills/analyzing-financial-statements/calculate_ratios.py', size: '12.22 KB' },\n { filePath: '.codebuddy/skills/analyzing-financial-statements/interpret_ratios.py', size: '15.83 KB' },\n ],\n }\n }\n } as SessionNotification,\n });\n\n // 4. 发送总结消息\n await this.delay(300);\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-summary`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: `✅ 文件搜索完成!在 \\`${directory}\\` 目录中找到 5 个匹配的 Python 文件。\\n\\n**搜索结果:**\\n- \\`bubble_sort.py\\` (10.38 KB)\\n- \\`convert_to_oneline.py\\` (2.2 KB)\\n- \\`tls12_examples.py\\` (4.58 KB)\\n- \\`.codebuddy/skills/analyzing-financial-statements/calculate_ratios.py\\` (12.22 KB)\\n- \\`.codebuddy/skills/analyzing-financial-statements/interpret_ratios.py\\` (15.83 KB)\\n\\n**工具调用流程:**\\n1. \\`tool_call\\` (pending) - 创建搜索工具调用\\n2. \\`tool_call_update\\` (in_progress) - 执行搜索\\n3. \\`tool_call_update\\` (completed) - 返回搜索结果`,\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(100);\n }\n\n /**\n * 发送删除文件工具调用流程 (符合 ACP 协议)\n *\n * 用于模拟删除文件的工具调用\n * 支持显示删除操作的进度和结果\n */\n private async sendMockDeleteFileToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`🗑️ Starting delete_file tool call flow: ${messageId}`);\n\n // 配置:模拟删除文件\n const targetFile = '/Users/liumingyuan/Project/caseTest/subagent/temp_file.txt';\n const toolCallId = `tool-df-${Date.now()}`;\n\n // 1. 发送 tool_call (pending) - 不带 rawInput\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'delete_file'\n }\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Delete File',\n status: 'pending',\n locations: [{ path: targetFile, line: 0 }]\n }\n } as SessionNotification\n });\n\n console.log('[MockAgentProvider] delete_file tool_call (pending) sent');\n await this.delay(200);\n\n // 2. 流式发送 rawInput (模拟参数逐个到达)\n // 将 target_file 分块流式发送\n const filePathChunks = [\n '/Users/liumingyuan/',\n 'Project/caseTest',\n '/subagent/',\n 'temp_file.txt'\n ];\n\n let accumulatedInput = '';\n for (const chunk of filePathChunks) {\n accumulatedInput += chunk;\n\n // 发送 tool_call_update with rawInput\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'delete_file'\n }\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n rawInput: {\n target_file: accumulatedInput,\n explanation: 'Deleting temporary test file'\n }\n }\n } as SessionNotification\n });\n await this.delay(100);\n console.log('[MockAgentProvider] delete_file tool_call_update with rawInput:', chunk);\n }\n\n await this.delay(300);\n\n // 3. 模拟删除操作进度\n const progressSteps = [\n { progress: 25, message: 'Checking file permissions...' },\n { progress: 50, message: 'Validating file path...' },\n { progress: 75, message: 'Removing file...' },\n { progress: 100, message: 'File deleted successfully' }\n ];\n\n for (const step of progressSteps) {\n const rawOutput: DeleteFilesResult = {\n type: 'delete_file_result',\n path: targetFile,\n recursive: false,\n hint: step.message\n };\n\n // 验证 rawOutput\n const outputValidation = validateToolOutput('delete_file', rawOutput);\n\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'delete_file'\n }\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: step.progress === 100 ? 'deleted' : 'in_progress',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `🗑️ ${step.message} (${step.progress}%)`\n }\n }\n ],\n locations: [{ path: targetFile, line: 0 }],\n rawOutput: (outputValidation.success ? outputValidation.data : rawOutput) as unknown as Record<string, unknown>\n }\n } as SessionNotification\n });\n\n console.log(`🗑️ Delete progress: ${step.progress}% - ${step.message}`);\n await this.delay(200);\n }\n\n // 4. 发送最终完成事件\n const finalRawOutput: DeleteFilesResult = {\n type: 'delete_file_result',\n path: targetFile,\n recursive: false,\n hint: 'File successfully deleted from filesystem'\n };\n\n const finalOutputValidation = validateToolOutput('delete_file', finalRawOutput);\n if (!finalOutputValidation.success) {\n throw new Error(`Invalid delete_file rawOutput: ${JSON.stringify(finalOutputValidation.error.errors)}`);\n }\n\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `✅ 文件删除完成!\\n\\n**删除的文件:** \\`${targetFile}\\`\\n\\n**操作详情:**\\n- 文件路径验证 ✓\\n- 权限检查 ✓\\n- 文件删除 ✓\\n\\n文件已从文件系统中成功移除。`\n }\n }\n ],\n locations: [{ path: targetFile, line: 0 }],\n rawOutput: finalOutputValidation.data as unknown as Record<string, unknown>\n }\n } as SessionNotification\n });\n\n console.log('[MockAgentProvider] delete_file tool_call_update (completed) sent');\n\n // 5. 发送总结消息\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-summary`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: '🗑️ **删除文件工具调用演示完成**\\n\\n**工具调用流程:**\\n1. `tool_call` (pending) - 创建删除文件工具调用\\n2. `tool_call_update` (in_progress) - 流式发送参数和执行删除操作\\n3. `tool_call_update` (completed) - 返回删除结果\\n\\n**rawInput 包含:**\\n- `target_file`: 要删除的文件路径\\n- `explanation`: 删除原因说明\\n\\n**rawOutput 包含:**\\n- `type`: \\'delete_file_result\\'\\n- `path`: 被删除的文件路径\\n- `recursive`: 是否递归删除\\n- `hint`: 操作提示信息',\n }\n }\n } as SessionNotification,\n });\n\n await this.delay(100);\n }\n\n /**\n * 发送 ask_followup_question 工具调用流程 (符合 ACP 协议)\n *\n * 用于模拟向用户提问并获取回答的工具调用\n * 数据结构与 genie-ide 保持一致:question (单个问题字符串) + options (选项数组)\n */\n private async sendMockAskFollowupQuestionToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`❓ Starting ask_followup_question tool call flow: ${messageId}`);\n\n const toolCallId = `tool-afq-${Date.now()}`;\n\n // 问题数据 - 与 genie-ide 保持一致的数据结构\n const question = '你希望创建什么类型的项目?';\n const options = [\n 'Web 应用(前端 + 后端)',\n '命令行工具(CLI)',\n '移动应用(React Native / Flutter)',\n '桌面应用(Electron)',\n '后端服务(API / 微服务)',\n ];\n\n // 1. 发送 tool_call (pending)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'ask_followup_question',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Ask Followup Question',\n kind: 'other',\n status: 'pending',\n rawInput: {\n question,\n options: JSON.stringify(options),\n },\n },\n } as SessionNotification,\n });\n\n await this.delay(100);\n\n // 2. 发送 tool_call_update (in_progress) - 显示问题\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'ask_followup_question',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n rawInput: {\n question,\n options: JSON.stringify(options),\n },\n },\n } as SessionNotification,\n });\n\n console.log('[MockAgentProvider] ask_followup_question tool displayed, waiting for user interaction...');\n // 工具显示完成,等待用户交互(暂未实现交互逻辑)\n }\n\n /**\n * 发送 list_dir 工具调用流程 (符合 ACP 协议)\n *\n * 用于模拟列出目录中的文件\n * 数据结构符合 tool-schemas.ts 中的 ListDirInput 和 ListFilesResult\n */\n private async sendMockListFilesToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`📁 Starting list_dir tool call flow: ${messageId}`);\n\n const target_directory = '/Users/test/project/src';\n const toolCallId = `tool-ld-${Date.now()}`;\n\n // Mock 文件列表数据 - 符合 ListFilesResult.files 格式\n const mockFiles = [\n { filePath: `${target_directory}/index.ts`, size: '2.5 KB', modifyTime: '2024-01-15 10:30:00' },\n { filePath: `${target_directory}/App.tsx`, size: '8.2 KB', modifyTime: '2024-01-15 09:45:00' },\n { filePath: `${target_directory}/utils.ts`, size: '1.8 KB', modifyTime: '2024-01-14 16:20:00' },\n { filePath: `${target_directory}/components/`, size: '-', modifyTime: '2024-01-15 11:00:00' },\n { filePath: `${target_directory}/hooks/`, size: '-', modifyTime: '2024-01-14 14:30:00' },\n { filePath: `${target_directory}/styles/`, size: '-', modifyTime: '2024-01-13 09:00:00' },\n { filePath: `${target_directory}/types.d.ts`, size: '3.1 KB', modifyTime: '2024-01-12 18:45:00' },\n { filePath: `${target_directory}/constants.ts`, size: '0.9 KB', modifyTime: '2024-01-10 12:00:00' },\n ];\n\n // 生成 listing 格式的文本输出\n const listing = mockFiles\n .map(f => {\n const name = f.filePath.split('/').pop() || '';\n const isDir = f.filePath.endsWith('/');\n return isDir ? `📁 ${name}` : `📄 ${name} (${f.size})`;\n })\n .join('\\n');\n\n // 1. 发送 tool_call (pending)\n // rawInput 符合 ListDirInput: { target_directory, ignore_globs? }\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'list_dir',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'List Directory',\n kind: 'search',\n status: 'pending',\n locations: [{ path: target_directory, line: 0 }],\n rawInput: {\n target_directory,\n // ignore_globs 是可选的,这里不传\n },\n },\n } as SessionNotification,\n });\n\n await this.delay(100);\n\n // 2. 发送 tool_call_update (in_progress)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'list_dir',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `Listing directory ${target_directory}...`,\n },\n },\n ],\n locations: [{ path: target_directory, line: 0 }],\n },\n } as SessionNotification,\n });\n\n await this.delay(300);\n\n // 3. 发送 tool_call_update (completed) 包含文件列表\n // rawOutput 符合 ListFilesResult: { type, files, root, listing? }\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'list_dir',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `Listed ${mockFiles.length} items in ${target_directory}`,\n },\n },\n ],\n locations: [{ path: target_directory, line: 0 }],\n rawOutput: {\n type: 'list_files_result',\n files: mockFiles,\n root: target_directory,\n listing: listing,\n },\n },\n } as SessionNotification,\n });\n\n console.log('[MockAgentProvider] list_dir tool call completed');\n }\n\n /**\n * 发送 preview_url 工具调用流程 (符合 ACP 协议)\n *\n * 用于测试 unknown-tool-renderer 的显示效果\n * preview_url 是一个未知工具,会使用 UnknownToolRenderer 渲染\n */\n private async sendMockPreviewUrlToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`🌐 Starting preview_url tool call flow: ${messageId}`);\n\n const toolCallId = `tool-pu-${Date.now()}`;\n const url = 'https://example.com/preview';\n\n // 1. 发送 tool_call (pending)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'preview_url',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Preview URL',\n kind: 'other',\n status: 'pending',\n rawInput: {\n url,\n },\n },\n } as SessionNotification,\n });\n\n await this.delay(200);\n\n // 2. 发送 tool_call_update (in_progress)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'preview_url',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `Loading preview for ${url}...`,\n },\n },\n ],\n },\n } as SessionNotification,\n });\n\n await this.delay(800);\n\n // 3. 发送 tool_call_update (completed)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'preview_url',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `Preview loaded successfully for ${url}`,\n },\n },\n ],\n rawOutput: {\n url,\n title: 'Example Domain',\n screenshot: 'base64-encoded-screenshot-data...',\n status: 200,\n },\n },\n } as SessionNotification,\n });\n\n console.log('[MockAgentProvider] preview_url tool call completed');\n }\n\n /**\n * 发送 list_code_definition_names 工具调用流程 (符合 ACP 协议)\n *\n * 用于测试 unknown-tool-renderer 的显示效果\n * list_code_definition_names 是一个未知工具,会使用 UnknownToolRenderer 渲染\n */\n private async sendMockListCodeDefinitionNamesToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`📝 Starting list_code_definition_names tool call flow: ${messageId}`);\n\n const toolCallId = `tool-lcd-${Date.now()}`;\n const filePath = '/Users/test/project/src/utils.ts';\n\n // 1. 发送 tool_call (pending)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'list_code_definition_names',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'List Code Definitions',\n kind: 'other',\n status: 'pending',\n rawInput: {\n filePath,\n },\n },\n } as SessionNotification,\n });\n\n await this.delay(200);\n\n // 2. 发送 tool_call_update (in_progress)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'list_code_definition_names',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `Analyzing code definitions in ${filePath}...`,\n },\n },\n ],\n },\n } as SessionNotification,\n });\n\n await this.delay(600);\n\n // 3. 发送 tool_call_update (completed)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'list_code_definition_names',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `Found 5 code definitions in ${filePath}`,\n },\n },\n ],\n rawOutput: {\n filePath,\n definitions: [\n { name: 'formatDate', type: 'function', line: 5 },\n { name: 'parseJSON', type: 'function', line: 15 },\n { name: 'debounce', type: 'function', line: 28 },\n { name: 'Config', type: 'interface', line: 42 },\n { name: 'DEFAULT_OPTIONS', type: 'const', line: 55 },\n ],\n },\n },\n } as SessionNotification,\n });\n\n console.log('[MockAgentProvider] list_code_definition_names tool call completed');\n }\n\n /**\n * 发送 web_fetch 工具调用流程 (符合 ACP 协议)\n *\n * 用于模拟从 URL 获取网页内容的工具调用\n */\n private async sendMockWebFetchToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`🌐 Starting web_fetch tool call flow: ${messageId}`);\n\n const toolCallId = `tool-wf-${Date.now()}`;\n const targetUrl = 'https://github.com/anthropics/claude-code';\n\n // 1. 发送 tool_call (pending)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'web_fetch',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Web Fetch',\n kind: 'other',\n status: 'pending',\n rawInput: {\n url: targetUrl,\n fetchInfo: 'Fetching page content...',\n },\n },\n } as SessionNotification,\n });\n\n await this.delay(200);\n\n // 2. 流式发送 rawInput(模拟 URL 逐步到达)\n const urlChunks = [\n 'https://github.com/',\n 'anthropics/',\n 'claude-code',\n ];\n\n let accumulatedUrl = '';\n for (const chunk of urlChunks) {\n accumulatedUrl += chunk;\n\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'web_fetch',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n rawInput: {\n url: accumulatedUrl,\n fetchInfo: 'Fetching page content...',\n },\n },\n } as SessionNotification,\n });\n await this.delay(100);\n console.log('[MockAgentProvider] web_fetch tool_call_update with rawInput:', chunk);\n }\n\n await this.delay(500);\n\n // 3. 模拟获取进度\n const progressSteps = [\n { loading: 'Connecting to server...' },\n { loading: 'Downloading content...' },\n { loading: 'Parsing HTML...' },\n ];\n\n for (const step of progressSteps) {\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'web_fetch',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n rawInput: {\n url: targetUrl,\n fetchInfo: step.loading,\n },\n rawOutput: {\n loading: step.loading,\n },\n },\n } as SessionNotification,\n });\n console.log(`🌐 Fetch progress: ${step.loading}`);\n await this.delay(300);\n }\n\n // 4. 发送完成事件,包含模拟的网页内容\n const mockWebContent = `# Claude Code\n\nClaude Code is an agentic coding tool that lives in your terminal, understands your codebase, and helps you code faster through natural conversation.\n\n## Features\n\n- **Agentic coding**: Claude Code can read, write, and edit files, run commands, and more.\n- **Context-aware**: Claude Code understands your entire codebase and can answer questions about it.\n- **Natural conversation**: Just describe what you want to do in plain English.\n\n## Installation\n\n\\`\\`\\`bash\nnpm install -g @anthropic-ai/claude-code\n\\`\\`\\`\n\n## Usage\n\n\\`\\`\\`bash\nclaude-code\n\\`\\`\\`\n\nStart a conversation with Claude Code and let it help you with your coding tasks!`;\n\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'web_fetch',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed',\n rawInput: {\n url: targetUrl,\n fetchInfo: 'Fetch web page content',\n },\n rawOutput: {\n title: 'Claude Code - GitHub',\n favicon: 'https://github.githubassets.com/favicons/favicon.svg',\n data: mockWebContent,\n },\n },\n } as SessionNotification,\n });\n\n console.log('[MockAgentProvider] web_fetch tool_call_update (completed) sent');\n\n // 5. 发送总结消息\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-summary`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: '🌐 **Web Fetch 工具调用演示完成**\\n\\n**工具调用流程:**\\n1. `tool_call` (pending) - 创建 web_fetch 工具调用\\n2. `tool_call_update` (in_progress) - 流式发送 URL 和获取进度\\n3. `tool_call_update` (completed) - 返回网页内容\\n\\n**rawInput 包含:**\\n- `url`: 目标 URL\\n- `fetchInfo`: 获取说明\\n\\n**rawOutput 包含:**\\n- `title`: 网页标题\\n- `favicon`: 网站图标\\n- `data`: 网页内容(Markdown 格式)',\n },\n },\n } as SessionNotification,\n });\n\n await this.delay(100);\n }\n\n /**\n * 发送 web_search 工具调用流程 (符合 ACP 协议)\n *\n * 用于模拟网络搜索的工具调用\n * 数据结构符合 tool-schemas.ts 中的 web_search 定义\n */\n private async sendMockWebSearchToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`🔍 Starting web_search tool call flow: ${messageId}`);\n\n const toolCallId = `tool-ws-${Date.now()}`;\n const searchTerm = 'TypeScript 5.0 new features';\n const explanation = 'Searching for the latest TypeScript 5.0 features to provide accurate and up-to-date information.';\n\n // 1. 发送 tool_call (pending)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'web_search',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Web Search',\n kind: 'search',\n status: 'pending',\n rawInput: {\n searchTerm,\n explanation,\n },\n },\n } as SessionNotification,\n });\n\n await this.delay(200);\n\n // 2. 发送 tool_call_update (in_progress) - 流式显示搜索中\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'web_search',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `🔍 Searching the web for \"${searchTerm}\"...`,\n },\n },\n ],\n rawInput: {\n searchTerm,\n explanation,\n },\n },\n } as SessionNotification,\n });\n\n await this.delay(800);\n\n // 3. 发送 tool_call_update (completed) - 包含搜索结果\n const searchResults = [\n {\n title: 'Announcing TypeScript 5.0 - TypeScript Blog',\n url: 'https://devblogs.microsoft.com/typescript/announcing-typescript-5-0/',\n snippet: 'TypeScript 5.0 brings many new features, including decorators, const type parameters, and improvements to enums. This release also includes significant performance improvements...',\n publishedDate: '2023-03-16',\n },\n {\n title: 'TypeScript 5.0: What\\'s New - LogRocket Blog',\n url: 'https://blog.logrocket.com/whats-new-typescript-5-0/',\n snippet: 'TypeScript 5.0 introduces several exciting features: ECMAScript decorators, const type parameters, multiple config extends, all enums are union enums, and more...',\n publishedDate: '2023-04-10',\n },\n {\n title: 'TypeScript 5.0 Release Notes - GitHub',\n url: 'https://github.com/microsoft/TypeScript/releases/tag/v5.0.0',\n snippet: 'Official release notes for TypeScript 5.0. Key highlights include: Decorators, const Type Parameters, Supporting Multiple Configuration Files in extends...',\n publishedDate: '2023-03-16',\n },\n {\n title: 'A Complete Guide to TypeScript 5.0 Features - Medium',\n url: 'https://medium.com/typescript-5-features-guide',\n snippet: 'Explore all the new features in TypeScript 5.0: decorators with metadata, const type parameters for better type inference, improved module resolution...',\n publishedDate: '2023-05-22',\n },\n {\n title: 'TypeScript 5.0 Performance Improvements',\n url: 'https://www.typescriptlang.org/docs/handbook/release-notes/typescript-5-0.html',\n snippet: 'TypeScript 5.0 includes significant performance improvements: faster type checking, reduced memory usage, and improved build times. Package size reduced by 50%...',\n publishedDate: '2023-03-16',\n },\n ];\n\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'web_search',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `Found ${searchResults.length} results for \"${searchTerm}\"`,\n },\n },\n ],\n rawInput: {\n searchTerm,\n explanation,\n },\n rawOutput: {\n query: searchTerm,\n totalResults: searchResults.length,\n results: searchResults,\n searchEngine: 'web',\n timestamp: new Date().toISOString(),\n },\n },\n } as SessionNotification,\n });\n\n // 4. 发送总结消息\n await this.delay(300);\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId: `${messageId}-summary`,\n timestamp: Date.now(),\n notification: {\n sessionId,\n update: {\n sessionUpdate: 'agent_message_chunk',\n content: {\n type: 'text',\n text: `🔍 **Web 搜索完成**\\n\\n**搜索词:** \\`${searchTerm}\\`\\n\\n**搜索结果:** 找到 ${searchResults.length} 个相关结果\\n\\n**主要发现:**\\n1. **TypeScript 5.0 新特性** - 包括装饰器、const 类型参数、枚举改进等\\n2. **性能提升** - 类型检查更快,内存使用减少,包大小减少 50%\\n3. **配置增强** - 支持多配置文件继承\\n\\n**工具调用流程:**\\n1. \\`tool_call\\` (pending) - 创建搜索工具调用\\n2. \\`tool_call_update\\` (in_progress) - 执行网络搜索\\n3. \\`tool_call_update\\` (completed) - 返回搜索结果\\n\\n**rawInput 包含:**\\n- \\`searchTerm\\`: 搜索关键词\\n- \\`explanation\\`: 搜索原因说明\\n\\n**rawOutput 包含:**\\n- \\`query\\`: 搜索查询\\n- \\`totalResults\\`: 结果总数\\n- \\`results\\`: 搜索结果数组(包含 title、url、snippet、publishedDate)`,\n },\n },\n } as SessionNotification,\n });\n\n console.log('[MockAgentProvider] web_search tool call completed');\n }\n\n /**\n * 发送 read_lints 工具调用流程 (符合 ACP 协议)\n *\n * 数据结构 (来自 tool-schemas.ts):\n * - Input: ReadLintsInput = { paths?: string }\n * - Output: ReadLintsResult = {\n * type: 'read_lints_result';\n * diagnostics: string[];\n * totalCount?: number;\n * hint?: string;\n * isTruncated?: boolean;\n * }\n */\n private async sendMockReadLintsToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`🔍 Starting read_lints tool call flow: ${messageId}`);\n\n const toolCallId = `tool-rl-${Date.now()}`;\n const targetPath = '/Users/test/project/src/components/Button.tsx';\n\n // 1. 发送 tool_call (pending)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'read_lints',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Read Lints',\n kind: 'read',\n status: 'pending',\n rawInput: {\n paths: targetPath,\n },\n },\n } as SessionNotification,\n });\n\n await this.delay(200);\n\n // 2. 发送 tool_call_update (in_progress)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'read_lints',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n rawInput: {\n paths: targetPath,\n },\n },\n } as SessionNotification,\n });\n\n await this.delay(300);\n\n // 3. 发送 tool_call_update (completed) - 包含 lint 错误\n const diagnosticsText = `Button.tsx - ERROR (3 issues)\n[ERROR] Line 15, Column 8: 'useState' is defined but never used. Consider removing this import.\n[ERROR] Line 23, Column 12: Type 'string' is not assignable to type 'number'. Expected a numeric value.\n[ERROR] Line 45, Column 4: Missing return statement in function that is expected to return 'ReactNode'.`;\n\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'read_lints',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed',\n rawInput: {\n paths: targetPath,\n },\n rawOutput: {\n type: 'read_lints_result',\n diagnostics: [diagnosticsText],\n totalCount: 3,\n hint: 'Found 3 lint errors in Button.tsx',\n isTruncated: false,\n },\n },\n } as SessionNotification,\n });\n\n console.log('[MockAgentProvider] read_lints tool call completed');\n }\n\n /**\n * 发送 execute_command 工具调用流程 (符合 ACP 协议)\n *\n * 数据结构 (来自 tool-schemas.ts):\n * - Input: ExecuteCommandInput = { command: string; requires_approval: boolean; }\n * - Output: ExecuteCommandResult = {\n * type: 'execute_command_result';\n * stdout: string;\n * stderr: string;\n * exitCode: number;\n * hint?: string;\n * serviceInfo?: { isWatchCommand, isServiceOutput, serviceReady, message };\n * use_standalone_terminal?: boolean;\n * }\n */\n private async sendMockExecuteCommandToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`💻 Starting execute_command tool call flow: ${messageId}`);\n const toolCallId = `tool-ec-${Date.now()}`;\n const command = 'npm run build';\n // 1. 发送 tool_call (pending)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'execute_command',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Execute Command',\n kind: 'execute',\n status: 'pending',\n rawInput: {\n command,\n requires_approval: false,\n },\n },\n } as SessionNotification,\n });\n await this.delay(200);\n // 2. 发送 tool_call_update (in_progress) - 模拟命令执行中\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'execute_command',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'in_progress',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: '> npm run build\\n\\nRunning build script...',\n },\n },\n ],\n },\n } as SessionNotification,\n });\n await this.delay(1000);\n // 3. 发送 tool_call_update (completed) - 命令执行完成\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'execute_command',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'completed',\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: '> npm run build\\n\\n> genie@1.0.0 build\\n> tsc -b\\n\\nBuild completed successfully.',\n },\n },\n ],\n rawOutput: {\n type: 'execute_command_result',\n stdout: '> genie@1.0.0 build\\n> tsc -b\\n\\nBuild completed successfully.',\n stderr: '',\n exitCode: 0,\n hint: 'Build completed in 2.5s',\n },\n },\n } as SessionNotification,\n });\n console.log('[MockAgentProvider] execute_command tool call completed');\n }\n /**\n * 发送 execute_command 工具调用流程(需要权限确认)(符合 ACP 协议)\n *\n * 这个方法模拟需要用户确认才能执行的命令\n * 用于测试权限确认菜单 (Run/Skip/Reject)\n *\n * 数据结构 (来自 tool-schemas.ts):\n * - Input: ExecuteCommandInput = { command: string; requires_approval: boolean; }\n *\n * 权限请求通过 rawInput.permissionRequest 字段传递:\n * - permissionRequest.options: 可选的权限选项数组\n */\n private async sendMockExecuteCommandWithApprovalToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`💻 Starting execute_command (with approval) tool call flow: ${messageId}`);\n const toolCallId = `tool-ec-approval-${Date.now()}`;\n const command = 'rm -rf node_modules && npm install';\n // 权限请求选项 - 与 ACP 协议的 PermissionOption 结构一致\n const permissionRequest = {\n options: [\n { optionId: 'allow-once', name: '运行', kind: 'allow_once' },\n { optionId: 'skip-once', name: '跳过', kind: 'skip_once' },\n { optionId: 'reject-once', name: '拒绝', kind: 'reject_once' },\n ],\n };\n // 1. 发送 tool_call (pending) - 需要权限确认\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'execute_command',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Execute Command',\n kind: 'execute',\n status: 'pending',\n rawInput: {\n command,\n requires_approval: true, // 保留向后兼容\n permissionRequest, // 新增:权限请求信息\n },\n },\n } as SessionNotification,\n });\n await this.delay(200);\n // 2. 发送 tool_call_update (in_progress) - 等待用户确认\n // 这时 UI 应该显示权限确认菜单 (Run/Skip/Reject)\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'execute_command',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'pending', // 保持 pending 状态等待用户审批\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: `$ ${command}\\n\\nWaiting for approval...`,\n },\n },\n ],\n rawInput: {\n command,\n requires_approval: true,\n permissionRequest, // 确保 rawInput 中也包含权限请求\n },\n },\n } as SessionNotification,\n });\n console.log('[MockAgentProvider] execute_command (with approval) tool call waiting for user approval');\n // 注意:这里不发送 completed,等待用户在 UI 上点击 Run/Skip/Reject\n }\n /**\n * 发送 write_to_file 工具调用流程(需要权限确认)(符合 ACP 协议)\n *\n * 这个方法模拟需要用户确认才能执行的文件写入\n * 用于测试权限确认菜单 (允许/跳过/拒绝)\n *\n * 权限请求通过 rawInput.permissionRequest 字段传递:\n * - permissionRequest.options: 可选的权限选项数组\n */\n private async sendMockWriteFileWithApprovalToolCallFlow(sessionId: string, messageId: string): Promise<void> {\n console.log(`📝 Starting write_to_file (with approval) tool call flow: ${messageId}`);\n const toolCallId = `tool-wtf-approval-${Date.now()}`;\n const filePath = '/Users/test/project/src/config.ts';\n const content = `export const config = {\n apiUrl: 'https://api.example.com',\n debug: true,\n timeout: 5000,\n};`;\n // 权限请求选项 - 与 ACP 协议的 PermissionOption 结构一致\n const permissionRequest = {\n options: [\n { optionId: 'allow-once', name: '允许', kind: 'allow_once' },\n { optionId: 'skip-once', name: '跳过', kind: 'skip_once' },\n { optionId: 'reject-once', name: '拒绝', kind: 'reject_once' },\n ],\n };\n // 1. 发送 tool_call (pending) - 需要权限确认\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'write_to_file',\n },\n },\n update: {\n sessionUpdate: 'tool_call',\n toolCallId,\n title: 'Write File',\n kind: 'edit',\n status: 'pending',\n locations: [{ path: filePath, line: 0 }],\n rawInput: {\n filePath,\n content,\n permissionRequest, // 权限请求信息\n },\n },\n } as SessionNotification,\n });\n await this.delay(200);\n // 2. 发送 tool_call_update (pending) - 等待用户确认\n // 这时 UI 应该显示权限确认菜单\n this.emitEvent({\n type: 'session_update',\n sessionId,\n messageId,\n timestamp: Date.now(),\n notification: {\n sessionId,\n _meta: {\n 'codebuddy.ai': {\n toolName: 'write_to_file',\n },\n },\n update: {\n sessionUpdate: 'tool_call_update',\n toolCallId,\n status: 'pending', // 保持 pending 状态等待用户审批\n content: [\n {\n type: 'content',\n content: {\n type: 'text',\n text: content,\n },\n },\n ],\n locations: [{ path: filePath, line: 0 }],\n rawInput: {\n filePath,\n content,\n permissionRequest, // 确保 rawInput 中也包含权限请求\n },\n },\n } as SessionNotification,\n });\n console.log('[MockAgentProvider] write_to_file (with approval) tool call waiting for user approval');\n // 注意:这里不发送 completed,等待用户在 UI 上点击 允许/跳过/拒绝\n }\n}\n","/**\n * Protocol Extension Types for Agent Client Protocol\n *\n * This file contains codebuddy.ai extension types that extend the base ACP protocol.\n * For SDK type re-exports and capability extensions, see ./sdk.ts\n */\n\n// ============================================\n// Extension Method Constants\n// ============================================\n\n/**\n * Extension method names as const for type safety\n */\nexport const ExtensionMethod = {\n ARTIFACT: '_codebuddy.ai/artifact',\n /**\n * Question/ToolInput extension method.\n * Used for tools that require user input (e.g., AskUserQuestion).\n * Request: ToolInputRequest, Response: ToolInputResponse\n */\n QUESTION: '_codebuddy.ai/question',\n CHECKPOINT: '_codebuddy.ai/checkpoint',\n /** Usage notification for token/cost tracking */\n USAGE: '_codebuddy.ai/usage',\n /** Command extension method for executing custom commands */\n COMMAND: '_codebuddy.ai/command',\n /** Auth URL notification for external authentication */\n AUTH_URL: '_codebuddy.ai/authUrl',\n} as const;\n\nexport type ExtensionMethodName = (typeof ExtensionMethod)[keyof typeof ExtensionMethod];\n\n/**\n * All known extension methods\n */\nexport const KNOWN_EXTENSIONS = [\n ExtensionMethod.ARTIFACT,\n ExtensionMethod.QUESTION,\n ExtensionMethod.CHECKPOINT,\n ExtensionMethod.USAGE,\n ExtensionMethod.COMMAND,\n ExtensionMethod.AUTH_URL,\n] as const;\n\n// ============================================\n// Extension Notification Types (Discriminated Union)\n// ============================================\n\n/**\n * Discriminated union for extension notifications\n * Use `method` field as discriminant for type narrowing\n */\nexport type ExtensionNotification = ArtifactExtNotification;\n\nexport interface ArtifactExtNotification {\n method: typeof ExtensionMethod.ARTIFACT;\n params: ArtifactNotificationParams;\n}\n\n// ============================================\n// Artifact Types\n// ============================================\n\n/**\n * Types of artifacts that can be created by the agent\n * Uses discriminant for TypeScript type narrowing\n */\nexport type ArtifactType = 'plan' | 'tasks' | 'media' | 'overview';\n\n/**\n * Artifact lifecycle events\n */\nexport type ArtifactEvent = 'created' | 'updated' | 'deleted';\n\n/**\n * Base Artifact interface\n * Inspired by MCP Resource: https://modelcontextprotocol.io/specification/2025-11-25/server/resources\n */\nexport interface BaseArtifact<T extends ArtifactType = ArtifactType> {\n /** Artifact type (discriminant for type narrowing) */\n type: T;\n /**\n * Unique identifier URI\n * - Cloud Agent: agent:///{path} e.g. agent:///artifacts/plan.md\n * - Local Agent: file:///{path} e.g. file:///Users/xxx/project/plan.md\n */\n uri: string;\n /** Resource name */\n name: string;\n /** Display title */\n title?: string;\n /** Description */\n description?: string;\n}\n\n// ============================================\n// Plan Artifact\n// ============================================\n\n/**\n * Plan Artifact - Markdown document\n * mimeType: text/markdown\n */\nexport interface PlanArtifact extends BaseArtifact<'plan'> {\n /** MIME type fixed as text/markdown */\n mimeType: 'text/markdown';\n /** Markdown text content */\n text: string;\n /** Version number (for diff) */\n version?: number;\n /** Previous version content (for diff display) */\n previousText?: string;\n /** Whether editing is enabled */\n enableEdit?: boolean;\n}\n\n// ============================================\n// Tasks Artifact\n// ============================================\n\n/**\n * Task item status\n */\nexport type TaskItemStatus = 'pending' | 'in_progress' | 'completed' | 'cancelled';\n\n/**\n * Single task item\n */\nexport interface TaskItem {\n id: string;\n content: string;\n status: TaskItemStatus;\n order?: number;\n}\n\n/**\n * Tasks Artifact - Task list\n * mimeType: application/json\n */\nexport interface TasksArtifact extends BaseArtifact<'tasks'> {\n /** MIME type fixed as application/json */\n mimeType: 'application/json';\n /** Task list */\n tasks: TaskItem[];\n /** Whether editing is enabled */\n enableEdit?: boolean;\n}\n\n// ============================================\n// Media Artifact\n// ============================================\n\n/**\n * Media content type (auxiliary classification)\n */\nexport type MediaContentType = 'image' | 'video' | 'audio' | 'document' | 'spreadsheet' | 'presentation' | 'diagram' | 'code';\n\n/**\n * Media Artifact - Media files\n * Supports images, videos, documents and other browser-renderable files\n *\n * URI schemes:\n * - data:image/png;base64,... (inline small files)\n * - agent:///artifacts/... (cloud agent resources)\n * - file:///path/to/file (local agent files)\n */\nexport interface MediaArtifact extends BaseArtifact<'media'> {\n /** MIME type, e.g. image/png, video/mp4, application/pdf */\n mimeType: string;\n /** File size in bytes */\n size?: number;\n /** Media content classification (auxiliary for frontend rendering) */\n contentType?: MediaContentType;\n /** Width (for images/videos) */\n width?: number;\n /** Height (for images/videos) */\n height?: number;\n}\n\n// ============================================\n// Overview Artifact\n// ============================================\n\n/**\n * Overview Artifact - 任务完成后的总结文档\n * mimeType: text/markdown\n */\nexport interface OverviewArtifact extends BaseArtifact<'overview'> {\n /** MIME type fixed as text/markdown */\n mimeType: 'text/markdown';\n /** Markdown text content */\n text?: string;\n}\n\n// ============================================\n// Artifact Union Type\n// ============================================\n\n/**\n * Artifact union type\n * Uses type field for discriminated union\n */\nexport type Artifact = PlanArtifact | TasksArtifact | MediaArtifact | OverviewArtifact;\n\n// ============================================\n// Artifact Notification Types (Discriminated Union by event)\n// ============================================\n\nexport interface ArtifactCreatedParams {\n sessionId: string;\n event: 'created';\n artifact: Artifact;\n}\n\nexport interface ArtifactUpdatedParams {\n sessionId: string;\n event: 'updated';\n artifact: Artifact;\n}\n\nexport interface ArtifactDeletedParams {\n sessionId: string;\n event: 'deleted';\n artifact: Pick<Artifact, 'type' | 'uri'>; // Only type and uri needed for deletion\n}\n\n/**\n * Artifact notification discriminated by event\n */\nexport type ArtifactNotificationParams =\n | ArtifactCreatedParams\n | ArtifactUpdatedParams\n | ArtifactDeletedParams;\n\n// ============================================\n// Question Types\n// ============================================\n\n/**\n * Question option structure\n */\nexport interface QuestionOption {\n /** Display text (1-5 words) */\n label: string;\n /** Option description */\n description: string;\n}\n\n/**\n * Single question structure\n */\nexport interface UserQuestion {\n /** Question ID */\n id: string;\n /** Question text */\n question: string;\n /** Short label (max 12 chars) */\n header?: string;\n /** Available options (2-4) */\n options: QuestionOption[];\n /** Allow multiple selections */\n multiSelect?: boolean;\n}\n\n/**\n * Question request (Server -> Client)\n * Sent via extMethod: _codebuddy.ai/question\n */\nexport interface QuestionRequest {\n /** Session ID */\n sessionId: string;\n /** Associated tool call ID (links extMethod to tool_call for UI) */\n toolCallId: string;\n /** Input type */\n inputType?: string;\n /** Schema containing questions */\n schema?: {\n /** Questions to ask (1-4) */\n questions: UserQuestion[];\n };\n /** Questions to ask (1-4) - legacy format */\n questions?: UserQuestion[];\n /** Request timeout in ms */\n timeout?: number;\n}\n\n/**\n * Question response (Client -> Server)\n */\nexport interface QuestionResponse {\n /** Response outcome */\n outcome: 'submitted' | 'cancelled';\n /** User's answers keyed by question ID (when submitted) */\n answers?: Record<string, string | string[]>;\n /** Cancellation reason (when cancelled) */\n reason?: string;\n}\n\n// ============================================\n// Usage Update Types\n// ============================================\n\n/**\n * Token usage information\n * Sent via extNotification: _codebuddy.ai/usage\n */\nexport interface UsageUpdate {\n sessionId: string;\n inputTokens?: number;\n outputTokens?: number;\n totalTokens?: number;\n cost?: number;\n model?: string;\n _meta?: Record<string, unknown>;\n}\n\n// ============================================\n// Command Types\n// ============================================\n\n/**\n * Command notification parameters\n * Sent via extNotification: _codebuddy.ai/command\n */\nexport interface CommandNotificationParams {\n /** Session ID */\n sessionId: string;\n /** Command action to execute */\n action: string;\n /** Command parameters */\n params?: Record<string, unknown>;\n}\n\n// ============================================\n// Checkpoint Types\n// ============================================\n\n/**\n * Checkpoint event types\n */\nexport type CheckpointEvent = 'created' | 'updated';\n\n/**\n * Checkpoint notification parameters\n * Sent via extNotification: _codebuddy.ai/checkpoint\n */\nexport interface CheckpointNotificationParams {\n /** Session ID */\n sessionId: string;\n /** Event type */\n event: CheckpointEvent;\n /** Checkpoint information */\n checkpoint: CheckpointInfo;\n}\n\n/**\n * Checkpoint information visible to clients\n */\nexport interface CheckpointInfo {\n /** Checkpoint ID */\n id: string;\n /** Label/description */\n label?: string;\n /** Creation timestamp */\n createdAt: number;\n /** File change summary */\n fileChanges: FileChangeSummary;\n}\n\n/**\n * File change summary for a checkpoint\n */\nexport interface FileChangeSummary {\n /** List of changed files */\n files: FileChangeInfo[];\n /** Total lines added */\n totalAdditions: number;\n /** Total lines deleted */\n totalDeletions: number;\n}\n\n/**\n * Change type for a file\n */\nexport type FileChangeType = 'created' | 'modified' | 'deleted';\n\n/**\n * Information about a single file change\n */\nexport interface FileChangeInfo {\n /** File URI (agent://files/{path}) */\n uri: string;\n /** Type of change */\n changeType: FileChangeType;\n /** Lines added */\n additions: number;\n /** Lines deleted */\n deletions: number;\n /** Unified diff format content\n * https://unifiedjs.com/explore/package/unified-diff/\n */\n diff?: string;\n /** File language */\n language?: string;\n}\n\n// ============================================\n// Auth URL Types\n// ============================================\n\n/**\n * Auth URL notification parameters\n * Sent via extNotification: _codebuddy.ai/authUrl\n *\n * Used when the Agent CLI attempts to open an external browser for OAuth authentication.\n * The client displays this URL so users can manually copy and open it if the browser\n * doesn't open automatically.\n */\nexport interface AuthUrlNotificationParams {\n /** The authentication URL to display */\n authUrl: string;\n /** Optional authentication provider identifier (e.g., 'oauth', 'external') */\n provider?: string;\n}\n\n// ============================================\n// Session Update Meta Types\n// ============================================\n\n/**\n * Mode for session update messages\n * - 'stream': Real-time updates during prompt execution (should be broadcast)\n * - 'history': Historical messages during session/load (should NOT be broadcast)\n */\nexport type SessionUpdateMode = 'stream' | 'history';\n\n/**\n * codebuddy.ai extension metadata for session updates\n */\nexport interface SessionUpdateCodebuddyMeta {\n /** Mode indicating the source of this update */\n mode?: SessionUpdateMode;\n}\n\n/**\n * _meta structure for session update notifications\n */\nexport interface SessionUpdateMeta {\n 'codebuddy.ai'?: SessionUpdateCodebuddyMeta;\n [key: string]: unknown;\n}\n","// Streamable HTTP Transport for ACP\n// Enables browser-based clients to connect to cloud-hosted ACP agents.\n//\n// Protocol flow:\n// 1. Client establishes GET SSE connection, receives Acp-Connection-Id\n// 2. Client sends POST requests with Acp-Connection-Id header\n// 3. Notifications arrive via GET SSE, responses via POST SSE\n\nimport type { Stream } from '@agentclientprotocol/sdk';\n\ntype StreamMessage = Stream extends { readable: ReadableStream<infer T> } ? T : never;\n\nexport interface StreamableHttpOptions {\n // ACP endpoint URL, e.g. 'https://cloud-agent.example.com/acp'\n endpoint: string;\n // Authorization token (sent as Bearer token)\n authToken?: string;\n // Custom headers to include in all requests\n headers?: Record<string, string>;\n // Reconnect options for SSE connections\n reconnect?: {\n enabled?: boolean; // default: true\n initialDelay?: number; // ms, default: 1000\n maxDelay?: number; // ms, default: 30000\n maxRetries?: number; // default: Infinity\n jitter?: boolean; // default: true, adds ±25% jitter to prevent thundering herd\n };\n // AbortSignal to cancel the connection\n signal?: AbortSignal;\n // Custom fetch implementation (for testing or non-browser environments)\n fetch?: typeof fetch;\n // Callback when connection is established\n onConnect?: (connectionId: string) => void;\n // Callback when connection is closed\n onDisconnect?: (connectionId: string) => void;\n // Callback when an error occurs\n onError?: (error: Error) => void;\n // Heartbeat timeout in ms (default: 60000). If no heartbeat received within this time, reconnect.\n // Set to 0 to disable heartbeat detection.\n heartbeatTimeout?: number;\n // POST request timeout in ms (default: 30000)\n postTimeout?: number;\n // Backpressure options for message queue\n backpressure?: {\n highWaterMark?: number; // default: 100, pause reading SSE when queue reaches this\n lowWaterMark?: number; // default: 50, resume reading SSE when queue drops to this\n pauseTimeout?: number; // default: 5000ms, timeout for backpressure pause to prevent deadlock\n };\n}\n\n/**\n * Extended Stream interface with connection management\n */\nexport interface StreamableHttpTransport extends Stream {\n /**\n * Current connection ID, undefined if not connected\n */\n readonly connectionId: string | undefined;\n /**\n * Promise that resolves when the SSE connection is established\n * and the connectionId is available. Callers should await this\n * before sending messages to ensure the transport is ready.\n */\n readonly ready: Promise<void>;\n /**\n * Close the connection gracefully (sends DELETE request)\n */\n close(): Promise<void>;\n}\n\ninterface SSEEvent {\n type: string;\n data: string;\n id?: string;\n}\n\nfunction parseSSELine(\n line: string,\n currentEvent: Partial<SSEEvent>\n): { event?: SSEEvent; reset: boolean; isComment: boolean } {\n if (line === '') {\n if (currentEvent.data) {\n return {\n event: {\n type: currentEvent.type || 'message',\n data: currentEvent.data,\n id: currentEvent.id,\n },\n reset: true,\n isComment: false,\n };\n }\n return { reset: true, isComment: false };\n }\n\n // SSE comments (including heartbeats) start with ':'\n if (line.startsWith(':')) {\n return { reset: false, isComment: true };\n }\n\n const colonIndex = line.indexOf(':');\n if (colonIndex === -1) {\n return { reset: false, isComment: false };\n }\n\n const field = line.slice(0, colonIndex);\n let value = line.slice(colonIndex + 1);\n if (value.startsWith(' ')) {\n value = value.slice(1);\n }\n\n switch (field) {\n case 'event':\n // Reset data when starting a new event type\n // This prevents data from previous events being concatenated\n if (currentEvent.type && currentEvent.type !== value) {\n currentEvent.data = undefined;\n }\n currentEvent.type = value;\n break;\n case 'data':\n currentEvent.data = (currentEvent.data || '') + value;\n break;\n case 'id':\n currentEvent.id = value;\n break;\n }\n\n return { reset: false, isComment: false };\n}\n\n// Create a Streamable HTTP transport.\n//\n// Example:\n// const transport = streamableHttp({\n// endpoint: 'https://agent.example.com/acp',\n// authToken: 'token123',\n// onConnect: (id) => console.log('Connected:', id),\n// onDisconnect: (id) => console.log('Disconnected:', id),\n// });\n//\n// // Later, close gracefully\n// await transport.close();\nexport function streamableHttp(options: StreamableHttpOptions): StreamableHttpTransport {\n const {\n endpoint,\n authToken,\n headers: customHeaders = {},\n reconnect = {},\n signal: externalSignal,\n fetch: customFetch = globalThis.fetch,\n onConnect,\n onDisconnect,\n onError,\n heartbeatTimeout = 60000,\n postTimeout = 30000,\n backpressure = {},\n } = options;\n\n const {\n enabled: reconnectEnabled = true,\n initialDelay = 1000,\n maxDelay = 30000,\n maxRetries = Infinity,\n jitter: jitterEnabled = true,\n } = reconnect;\n\n const {\n highWaterMark = 100,\n lowWaterMark = 50,\n pauseTimeout = 5000,\n } = backpressure;\n\n // Connection state\n let connectionId: string | undefined;\n let lastEventId: string | undefined;\n let reconnectAttempts = 0;\n let closed = false;\n let isClosing = false;\n\n // Promise that resolves when GET SSE is connected and we have connectionId\n let connectionReady: Promise<void>;\n let resolveConnection: () => void;\n let rejectConnection: (error: Error) => void;\n\n // Connection version to track reconnections\n let connectionVersion = 0;\n\n connectionReady = new Promise((resolve, reject) => {\n resolveConnection = resolve;\n rejectConnection = reject;\n });\n\n const abortController = new AbortController();\n\n // Combine signals - use manual approach for broader compatibility\n function isAborted(): boolean {\n return abortController.signal.aborted || (externalSignal?.aborted ?? false);\n }\n\n function getSignal(): AbortSignal {\n // AbortSignal.any is available in newer Node.js versions\n const anyFn = (AbortSignal as unknown as { any?: (signals: AbortSignal[]) => AbortSignal }).any;\n if (externalSignal && typeof anyFn === 'function') {\n return anyFn([externalSignal, abortController.signal]);\n }\n return abortController.signal;\n }\n\n const combinedSignal = getSignal();\n\n // Message queue for incoming messages with backpressure\n const messageQueue: StreamMessage[] = [];\n const messageResolvers: Array<(value: StreamMessage | null) => void> = [];\n let streamError: Error | null = null;\n let isPaused = false;\n let resumeReading: (() => void) | null = null;\n\n // Track background SSE processors to handle POST responses asynchronously\n const backgroundSSEProcessors: Set<Promise<void>> = new Set();\n\n // Heartbeat tracking\n let lastActivity = Date.now();\n let heartbeatCheckTimer: ReturnType<typeof setInterval> | undefined;\n\n function enqueueMessage(message: StreamMessage): boolean {\n if (messageResolvers.length > 0) {\n const resolver = messageResolvers.shift()!;\n resolver(message);\n return true;\n } else {\n messageQueue.push(message);\n // Check if we should pause reading due to backpressure\n if (messageQueue.length >= highWaterMark) {\n isPaused = true;\n return false;\n }\n return true;\n }\n }\n\n function dequeueMessage(): Promise<StreamMessage | null> {\n if (closed) {\n return Promise.resolve(null);\n }\n if (streamError) {\n return Promise.reject(streamError);\n }\n if (messageQueue.length > 0) {\n const message = messageQueue.shift()!;\n // Check if we should resume reading\n if (isPaused && messageQueue.length <= lowWaterMark) {\n isPaused = false;\n const resume = resumeReading;\n if (resume) {\n // Use queueMicrotask to avoid potential stack issues\n queueMicrotask(() => resume());\n }\n }\n return Promise.resolve(message);\n }\n return new Promise(resolve => {\n messageResolvers.push(resolve);\n });\n }\n\n function updateLastActivity(): void {\n lastActivity = Date.now();\n }\n\n function startHeartbeatCheck(triggerReconnect: () => void): void {\n if (heartbeatTimeout <= 0) {\n return;\n }\n // Check heartbeat every 10 seconds\n heartbeatCheckTimer = setInterval(() => {\n if (Date.now() - lastActivity > heartbeatTimeout) {\n console.warn('[StreamableHTTP] Heartbeat timeout, triggering reconnect');\n triggerReconnect();\n }\n }, 10000);\n }\n\n function stopHeartbeatCheck(): void {\n if (heartbeatCheckTimer) {\n clearInterval(heartbeatCheckTimer);\n heartbeatCheckTimer = undefined;\n }\n }\n\n /**\n * Calculate reconnect delay with optional jitter\n */\n function calculateDelay(attempt: number): number {\n const baseDelay = Math.min(initialDelay * Math.pow(2, attempt - 1), maxDelay);\n if (!jitterEnabled) {\n return baseDelay;\n }\n // Add ±25% jitter to prevent thundering herd\n const jitterFactor = 0.25 * (Math.random() * 2 - 1);\n return Math.round(baseDelay * (1 + jitterFactor));\n }\n\n function closeWithError(error: Error): void {\n streamError = error;\n closed = true;\n stopHeartbeatCheck();\n // Resume any paused reading\n if (resumeReading) {\n resumeReading();\n resumeReading = null;\n }\n rejectConnection(error);\n onError?.(error);\n while (messageResolvers.length > 0) {\n const resolver = messageResolvers.shift()!;\n resolver(null);\n }\n }\n\n function closeNormally(): void {\n closed = true;\n stopHeartbeatCheck();\n // Resume any paused reading\n if (resumeReading) {\n resumeReading();\n resumeReading = null;\n }\n // Clear background SSE processors (they will exit due to closed=true)\n backgroundSSEProcessors.clear();\n while (messageResolvers.length > 0) {\n const resolver = messageResolvers.shift()!;\n resolver(null);\n }\n }\n\n // Send DELETE request to close connection gracefully\n async function sendDelete(): Promise<void> {\n if (!connectionId || isClosing) {\n return;\n }\n\n isClosing = true;\n const currentConnectionId = connectionId;\n\n try {\n const headers = buildHeaders();\n headers['Acp-Connection-Id'] = currentConnectionId;\n\n await customFetch(endpoint, {\n method: 'DELETE',\n headers,\n signal: AbortSignal.timeout(5000), // 5s timeout for DELETE\n });\n } catch {\n // Ignore DELETE errors - connection may already be closed\n } finally {\n if (currentConnectionId) {\n onDisconnect?.(currentConnectionId);\n }\n isClosing = false;\n }\n }\n\n function buildHeaders(): Record<string, string> {\n const headers: Record<string, string> = {\n ...customHeaders,\n };\n\n if (authToken) {\n headers['Authorization'] = `Bearer ${authToken}`;\n }\n\n return headers;\n }\n\n async function processSSEStream(\n reader: ReadableStreamDefaultReader<Uint8Array>\n ): Promise<void> {\n const decoder = new TextDecoder();\n let buffer = '';\n let currentEvent: Partial<SSEEvent> = {};\n\n try {\n while (true) {\n // Check for backpressure - wait if paused with timeout protection\n if (isPaused) {\n await new Promise<void>(resolve => {\n let resolved = false;\n\n // Timeout protection to prevent permanent deadlock\n const timeoutId = setTimeout(() => {\n if (!resolved) {\n resolved = true;\n console.warn('[StreamableHTTP] Backpressure pause timeout, forcing resume');\n resolve();\n }\n }, pauseTimeout);\n\n resumeReading = () => {\n if (!resolved) {\n resolved = true;\n clearTimeout(timeoutId);\n resolve();\n }\n };\n });\n resumeReading = null;\n }\n\n const { value, done } = await reader.read();\n if (done) {break;}\n\n buffer += decoder.decode(value, { stream: true });\n const lines = buffer.split('\\n');\n buffer = lines.pop() || '';\n\n for (const line of lines) {\n const { event, reset, isComment } = parseSSELine(line, currentEvent);\n\n // SSE comments (including heartbeats) update last activity\n if (isComment) {\n updateLastActivity();\n continue;\n }\n\n if (event) {\n // Any event means we're receiving data - update activity\n updateLastActivity();\n\n if (event.id) {\n lastEventId = event.id;\n }\n\n // Skip non-message events (like \"connected\")\n if (event.type !== 'message') {\n continue;\n }\n\n try {\n const message = JSON.parse(event.data) as StreamMessage;\n // Only enqueue valid JSON-RPC messages\n if (message && typeof message === 'object' && 'jsonrpc' in message) {\n enqueueMessage(message);\n }\n } catch {\n console.error('[StreamableHTTP] Failed to parse SSE data:', event.data);\n }\n }\n\n if (reset) {\n currentEvent = {};\n }\n }\n }\n } finally {\n reader.releaseLock();\n }\n }\n\n /**\n * Process SSE stream in background without blocking the caller.\n * This prevents deadlock when POST responses return SSE streams.\n */\n function processSSEStreamBackground(\n reader: ReadableStreamDefaultReader<Uint8Array>\n ): void {\n const promise = processSSEStream(reader)\n .catch(error => {\n console.error('[StreamableHTTP] Background SSE processing error:', error);\n onError?.(error instanceof Error ? error : new Error(String(error)));\n })\n .finally(() => {\n backgroundSSEProcessors.delete(promise);\n });\n backgroundSSEProcessors.add(promise);\n }\n\n // Establish GET SSE connection and get connectionId\n async function startSSEConnection(): Promise<void> {\n // Track current reader for heartbeat-triggered reconnect\n let currentReader: ReadableStreamDefaultReader<Uint8Array> | null = null;\n\n // Function to trigger reconnect (called from heartbeat timeout)\n const triggerReconnect = (): void => {\n if (currentReader) {\n currentReader.cancel().catch(() => {/* ignore */});\n }\n };\n\n while (!closed && !isAborted()) {\n try {\n const headers = buildHeaders();\n headers['Accept'] = 'text/event-stream';\n\n if (lastEventId) {\n headers['Last-Event-ID'] = lastEventId;\n }\n\n const response = await customFetch(endpoint, {\n method: 'GET',\n headers,\n signal: combinedSignal,\n });\n\n if (!response.ok) {\n throw new Error(`HTTP ${response.status}`);\n }\n\n // Get connection ID from response - MUST have it\n const newConnectionId = response.headers.get('Acp-Connection-Id');\n if (!newConnectionId) {\n throw new Error('Server did not return Acp-Connection-Id header');\n }\n\n // Track previous connection for disconnect callback\n const previousConnectionId = connectionId;\n\n // Update connection state atomically\n connectionVersion++;\n connectionId = newConnectionId;\n resolveConnection();\n\n // Notify callbacks\n if (previousConnectionId && previousConnectionId !== newConnectionId) {\n onDisconnect?.(previousConnectionId);\n }\n onConnect?.(newConnectionId);\n\n reconnectAttempts = 0;\n\n // Reset heartbeat tracking and start heartbeat check\n updateLastActivity();\n startHeartbeatCheck(triggerReconnect);\n\n const reader = response.body?.getReader();\n if (reader) {\n currentReader = reader;\n await processSSEStream(reader);\n currentReader = null;\n }\n\n // Stop heartbeat check when connection ends\n stopHeartbeatCheck();\n\n // Connection ended\n const endedConnectionId = connectionId;\n connectionId = undefined;\n\n // Notify disconnect\n if (endedConnectionId) {\n onDisconnect?.(endedConnectionId);\n }\n\n if (!reconnectEnabled || closed) {\n break;\n }\n\n // Reconnect - create new connectionReady promise BEFORE clearing connectionId\n // so that sendMessage waits for new connection\n connectionReady = new Promise((resolve, reject) => {\n resolveConnection = resolve;\n rejectConnection = reject;\n });\n\n // Add delay before reconnecting to prevent tight loop\n // This gives sendMessage a chance to use the connection before it's replaced\n const reconnectDelay = calculateDelay(1); // Use minimal delay with jitter\n await new Promise(resolve => setTimeout(resolve, reconnectDelay));\n } catch (error) {\n // Stop heartbeat check on error\n stopHeartbeatCheck();\n currentReader = null;\n\n if (isAborted() || closed) {\n break;\n }\n\n reconnectAttempts++;\n\n if (reconnectAttempts > maxRetries) {\n closeWithError(new Error(`SSE reconnect failed after ${maxRetries} attempts`));\n break;\n }\n\n // Use calculateDelay with jitter\n const delay = calculateDelay(reconnectAttempts);\n\n console.warn(\n `[StreamableHTTP] SSE error, retrying in ${delay}ms (attempt ${reconnectAttempts}):`,\n error\n );\n\n await new Promise(resolve => setTimeout(resolve, delay));\n }\n }\n }\n\n // Send message via POST\n async function sendMessage(message: StreamMessage): Promise<void> {\n if (closed) {\n throw new Error('Connection is closed');\n }\n\n // Wait for stable connection with retry logic\n // This handles rapid reconnection scenarios\n const maxWaitAttempts = 5;\n let currentConnectionId: string | undefined;\n\n for (let attempt = 0; attempt < maxWaitAttempts; attempt++) {\n // Wait for GET SSE to establish and get connectionId\n const versionBeforeWait = connectionVersion;\n await connectionReady;\n\n // Check if reconnection happened while we were waiting\n if (versionBeforeWait !== connectionVersion && versionBeforeWait > 0) {\n // A reconnection happened - wait for new connection\n await connectionReady;\n }\n\n // Capture connectionId after waiting\n currentConnectionId = connectionId;\n\n if (currentConnectionId) {\n break; // Got a valid connection\n }\n\n // Connection became undefined (reconnecting), wait a bit and retry\n if (attempt < maxWaitAttempts - 1) {\n await new Promise(resolve => setTimeout(resolve, 100));\n }\n }\n\n if (!currentConnectionId) {\n throw new Error('No connection ID available after multiple attempts');\n }\n\n const headers = buildHeaders();\n headers['Content-Type'] = 'application/json';\n headers['Accept'] = 'application/json, text/event-stream';\n headers['Acp-Connection-Id'] = currentConnectionId;\n\n // Create timeout controller for POST request\n const postController = new AbortController();\n let timeoutId: ReturnType<typeof setTimeout> | undefined;\n\n // Combine with external signal if present\n const postSignal = postTimeout > 0\n ? postController.signal\n : combinedSignal;\n\n if (postTimeout > 0) {\n timeoutId = setTimeout(() => postController.abort(), postTimeout);\n // Also abort if external signal is aborted\n if (externalSignal) {\n externalSignal.addEventListener('abort', () => postController.abort(), { once: true });\n }\n abortController.signal.addEventListener('abort', () => postController.abort(), { once: true });\n }\n\n try {\n const response = await customFetch(endpoint, {\n method: 'POST',\n headers,\n body: JSON.stringify(message),\n signal: postSignal,\n });\n\n if (!response.ok) {\n const errorText = await response.text().catch(() => 'Unknown error');\n throw new Error(`HTTP ${response.status}: ${errorText}`);\n }\n\n // Handle response based on content type\n const contentType = response.headers.get('Content-Type') || '';\n\n if (contentType.includes('text/event-stream')) {\n const reader = response.body?.getReader();\n if (reader) {\n // Process SSE in background to avoid blocking sendMessage\n // This prevents deadlock when backpressure causes processSSEStream to pause\n processSSEStreamBackground(reader);\n }\n } else if (contentType.includes('application/json')) {\n const data = await response.json();\n if (data && typeof data === 'object' && 'jsonrpc' in data) {\n enqueueMessage(data as StreamMessage);\n }\n }\n // 202 responses have no body\n } finally {\n if (timeoutId) {\n clearTimeout(timeoutId);\n }\n }\n }\n\n // Start SSE connection immediately\n startSSEConnection().catch(error => {\n console.error('[StreamableHTTP] SSE connection error:', error);\n });\n\n const readable = new ReadableStream<StreamMessage>({\n async pull(controller) {\n const message = await dequeueMessage();\n if (message === null) {\n controller.close();\n } else {\n controller.enqueue(message);\n }\n },\n cancel() {\n closeNormally();\n abortController.abort();\n },\n });\n\n const writable = new WritableStream<StreamMessage>({\n async write(message) {\n await sendMessage(message);\n },\n close() {\n closeNormally();\n abortController.abort();\n },\n abort(reason) {\n closeWithError(reason instanceof Error ? reason : new Error(String(reason)));\n abortController.abort();\n },\n });\n\n // Close the connection gracefully\n async function close(): Promise<void> {\n if (closed) {\n return;\n }\n\n // Send DELETE to server first\n await sendDelete();\n\n // Then close locally\n closeNormally();\n abortController.abort();\n }\n\n return {\n readable,\n writable,\n get connectionId() {\n return connectionId;\n },\n get ready() {\n return connectionReady;\n },\n close,\n };\n}\n\nexport default streamableHttp;\n","/**\n * Artifact Manager for Streamable HTTP ACP Client\n * Handles artifact notification processing\n */\n\nimport type {\n Artifact,\n ArtifactEvent,\n ArtifactNotificationParams,\n} from '../types.js';\nimport type { Logger } from './types.js';\n\n/**\n * Configuration for ArtifactManager\n */\nexport interface ArtifactManagerConfig {\n /** Logger instance */\n logger?: Logger;\n}\n\n/**\n * Callback for artifact events\n */\nexport type ArtifactEventCallback = (artifact: Artifact, event: ArtifactEvent) => void;\n\n/**\n * Manages artifact notifications from the agent\n */\nexport class ArtifactManager {\n private artifacts = new Map<string, Artifact>();\n private logger?: Logger;\n private eventCallbacks: Set<ArtifactEventCallback> = new Set();\n\n constructor(config: ArtifactManagerConfig) {\n this.logger = config.logger;\n }\n\n /**\n * Register a callback for artifact events\n */\n onArtifactEvent(callback: ArtifactEventCallback): () => void {\n this.eventCallbacks.add(callback);\n return () => {\n this.eventCallbacks.delete(callback);\n };\n }\n\n /**\n * Handle an artifact notification from the agent\n */\n handleNotification(notification: ArtifactNotificationParams): void {\n const { event } = notification;\n\n if (event === 'deleted') {\n const { artifact } = notification;\n const existing = this.artifacts.get(artifact.uri);\n this.logger?.debug(`Artifact deleted: ${artifact.uri}`);\n this.artifacts.delete(artifact.uri);\n\n // Notify callbacks with existing artifact if available\n if (existing) {\n this.notifyCallbacks(existing, event);\n }\n } else {\n const { artifact } = notification;\n this.logger?.debug(`Artifact ${event}: ${artifact.uri} (${artifact.type})`);\n this.artifacts.set(artifact.uri, artifact);\n this.notifyCallbacks(artifact, event);\n }\n }\n\n private notifyCallbacks(artifact: Artifact, event: ArtifactEvent): void {\n for (const callback of this.eventCallbacks) {\n try {\n callback(artifact, event);\n } catch (err) {\n this.logger?.error('Error in artifact event callback:', err);\n }\n }\n }\n\n /**\n * Get an artifact by URI (used internally for deleted event handling)\n */\n get(uri: string): Artifact | undefined {\n return this.artifacts.get(uri);\n }\n\n /**\n * Clear all artifacts\n */\n clear(): void {\n this.artifacts.clear();\n this.logger?.debug('Cleared all artifacts');\n }\n}\n","/**\n * Protocol constants for Streamable HTTP ACP Client\n */\n\nimport type { ClientCapabilities } from '../sdk.js';\nimport { ExtensionMethod, KNOWN_EXTENSIONS } from '../types.js';\n\n// Re-export extension constants\nexport { ExtensionMethod, KNOWN_EXTENSIONS };\n\n// ============================================\n// Default Timeouts (in milliseconds)\n// ============================================\n\n/**\n * Default timeout for initialize operation\n */\nexport const DEFAULT_INITIALIZE_TIMEOUT = 30_000; // 30 seconds\n\n/**\n * Default timeout for prompt operation\n */\nexport const DEFAULT_PROMPT_TIMEOUT = 300_000; // 5 minutes\n\n/**\n * Default timeout for permission requests\n */\nexport const DEFAULT_PERMISSION_TIMEOUT = 300_000; // 5 minutes\n\n/**\n * Default timeout for question requests (ask_followup_question)\n */\nexport const DEFAULT_QUESTION_TIMEOUT = 300_000; // 5 minutes\n\n/**\n * Default timeout for tool input requests\n * @deprecated Use DEFAULT_QUESTION_TIMEOUT instead\n */\nexport const DEFAULT_TOOL_INPUT_TIMEOUT = DEFAULT_QUESTION_TIMEOUT;\n\n// ============================================\n// Default Reconnect Configuration\n// ============================================\n\n/**\n * Default reconnect options\n */\nexport const DEFAULT_RECONNECT_OPTIONS = {\n enabled: true,\n initialDelay: 1000, // 1 second\n maxDelay: 30_000, // 30 seconds\n maxRetries: Infinity\n} as const;\n\n// ============================================\n// Client Capabilities\n// ============================================\n\n/**\n * Default client capabilities for cloud-based clients\n * Cloud clients typically have no direct file system access\n */\nexport const CLOUD_CLIENT_CAPABILITIES: ClientCapabilities = {\n fs: {\n readTextFile: false,\n writeTextFile: false\n }\n} as const;\n\n/**\n * Default client capabilities for local (Node.js) clients\n * Local clients have file system access\n */\nexport const LOCAL_CLIENT_CAPABILITIES: ClientCapabilities = {\n fs: {\n readTextFile: true,\n writeTextFile: true\n }\n} as const;\n","/**\n * Custom error classes for Streamable HTTP ACP Client\n */\n\n/**\n * Base error class for all ACP client errors\n */\nexport class ACPClientError extends Error {\n public readonly code: string;\n public readonly cause?: Error;\n\n constructor(message: string, code: string, cause?: Error) {\n super(message);\n this.name = 'ACPClientError';\n this.code = code;\n this.cause = cause;\n\n // Maintain proper stack trace in V8 environments\n if (Error.captureStackTrace) {\n Error.captureStackTrace(this, this.constructor);\n }\n }\n}\n\n/**\n * Error thrown when connection fails\n */\nexport class ConnectionError extends ACPClientError {\n constructor(message: string, cause?: Error) {\n super(message, 'CONNECTION_ERROR', cause);\n this.name = 'ConnectionError';\n }\n}\n\n/**\n * Error thrown when initialization fails\n */\nexport class InitializationError extends ACPClientError {\n constructor(message: string, cause?: Error) {\n super(message, 'INITIALIZATION_ERROR', cause);\n this.name = 'InitializationError';\n }\n}\n\n/**\n * Error thrown for session-related failures\n */\nexport class SessionError extends ACPClientError {\n public readonly sessionId?: string;\n\n constructor(message: string, sessionId?: string, cause?: Error) {\n super(message, 'SESSION_ERROR', cause);\n this.name = 'SessionError';\n this.sessionId = sessionId;\n }\n}\n\n/**\n * Error thrown when prompt operation fails\n */\nexport class PromptError extends ACPClientError {\n public readonly sessionId?: string;\n\n constructor(message: string, sessionId?: string, cause?: Error) {\n super(message, 'PROMPT_ERROR', cause);\n this.name = 'PromptError';\n this.sessionId = sessionId;\n }\n}\n\n/**\n * Error thrown for permission-related failures\n */\nexport class PermissionError extends ACPClientError {\n public readonly requestId?: string;\n\n constructor(message: string, requestId?: string, cause?: Error) {\n super(message, 'PERMISSION_ERROR', cause);\n this.name = 'PermissionError';\n this.requestId = requestId;\n }\n}\n\n/**\n * Error thrown when an operation times out\n */\nexport class TimeoutError extends ACPClientError {\n public readonly operation: string;\n public readonly timeoutMs: number;\n\n constructor(operation: string, timeoutMs: number) {\n super(`Operation '${operation}' timed out after ${timeoutMs}ms`, 'TIMEOUT_ERROR');\n this.name = 'TimeoutError';\n this.operation = operation;\n this.timeoutMs = timeoutMs;\n }\n}\n\n/**\n * Error thrown when client is in invalid state for an operation\n */\nexport class InvalidStateError extends ACPClientError {\n public readonly currentState: string;\n public readonly expectedStates: string[];\n\n constructor(operation: string, currentState: string, expectedStates: string[]) {\n super(\n `Cannot perform '${operation}' in state '${currentState}'. Expected: ${expectedStates.join(' or ')}`,\n 'INVALID_STATE_ERROR'\n );\n this.name = 'InvalidStateError';\n this.currentState = currentState;\n this.expectedStates = expectedStates;\n }\n}\n\n","/**\n * Type-safe event emitter for Streamable HTTP ACP Client\n * Platform-agnostic implementation (no browser dependencies)\n */\n\n/**\n * Event listener function type\n */\nexport type EventListener<T> = (data: T) => void | Promise<void>;\n\n/**\n * Type-safe event emitter implementation\n */\nexport class EventEmitter<TEvents extends Record<string, unknown>> {\n private listeners: Map<keyof TEvents, Set<EventListener<unknown>>> = new Map();\n private onceListeners: Map<keyof TEvents, Set<EventListener<unknown>>> = new Map();\n\n /**\n * Add an event listener\n */\n on<K extends keyof TEvents>(event: K, listener: EventListener<TEvents[K]>): this {\n if (!this.listeners.has(event)) {\n this.listeners.set(event, new Set());\n }\n this.listeners.get(event)!.add(listener as EventListener<unknown>);\n return this;\n }\n\n /**\n * Remove an event listener\n */\n off<K extends keyof TEvents>(event: K, listener: EventListener<TEvents[K]>): this {\n const eventListeners = this.listeners.get(event);\n if (eventListeners) {\n eventListeners.delete(listener as EventListener<unknown>);\n }\n\n const onceEventListeners = this.onceListeners.get(event);\n if (onceEventListeners) {\n onceEventListeners.delete(listener as EventListener<unknown>);\n }\n\n return this;\n }\n\n /**\n * Add a one-time event listener\n */\n once<K extends keyof TEvents>(event: K, listener: EventListener<TEvents[K]>): this {\n if (!this.onceListeners.has(event)) {\n this.onceListeners.set(event, new Set());\n }\n this.onceListeners.get(event)!.add(listener as EventListener<unknown>);\n return this;\n }\n\n /**\n * Emit an event to all registered listeners\n * Returns true if any listeners were invoked\n */\n emit<K extends keyof TEvents>(event: K, data: TEvents[K]): boolean {\n const regularListeners = this.listeners.get(event);\n const onceEventListeners = this.onceListeners.get(event);\n\n let hasListeners = false;\n\n // Call regular listeners\n if (regularListeners && regularListeners.size > 0) {\n hasListeners = true;\n for (const listener of regularListeners) {\n try {\n const result = listener(data);\n // Handle async listeners\n if (result instanceof Promise) {\n result.catch(err => {\n console.error(`Error in async event listener for '${String(event)}':`, err);\n });\n }\n } catch (err) {\n console.error(`Error in event listener for '${String(event)}':`, err);\n }\n }\n }\n\n // Call once listeners and remove them\n if (onceEventListeners && onceEventListeners.size > 0) {\n hasListeners = true;\n const listenersToCall = Array.from(onceEventListeners);\n this.onceListeners.delete(event);\n\n for (const listener of listenersToCall) {\n try {\n const result = listener(data);\n if (result instanceof Promise) {\n result.catch(err => {\n console.error(\n `Error in async once event listener for '${String(event)}':`,\n err\n );\n });\n }\n } catch (err) {\n console.error(`Error in once event listener for '${String(event)}':`, err);\n }\n }\n }\n\n return hasListeners;\n }\n\n /**\n * Remove all listeners for an event, or all listeners if no event specified\n */\n removeAllListeners<K extends keyof TEvents>(event?: K): this {\n if (event !== undefined) {\n this.listeners.delete(event);\n this.onceListeners.delete(event);\n } else {\n this.listeners.clear();\n this.onceListeners.clear();\n }\n return this;\n }\n\n /**\n * Get the number of listeners for an event\n */\n listenerCount<K extends keyof TEvents>(event: K): number {\n const regular = this.listeners.get(event)?.size ?? 0;\n const once = this.onceListeners.get(event)?.size ?? 0;\n return regular + once;\n }\n\n /**\n * Get all event names that have listeners\n */\n eventNames(): Array<keyof TEvents> {\n const names = new Set<keyof TEvents>();\n for (const event of this.listeners.keys()) {\n names.add(event);\n }\n for (const event of this.onceListeners.keys()) {\n names.add(event);\n }\n return Array.from(names);\n }\n}\n","/**\n * Extension Method Handler for Streamable HTTP ACP Client\n * Routes and handles custom extension notifications\n */\n\nimport type { UsageUpdate } from '../types.js';\nimport { ExtensionMethod, KNOWN_EXTENSIONS } from './constants.js';\nimport type { Logger } from './types.js';\n\n/**\n * Configuration for ExtensionManager\n */\nexport interface ExtensionManagerConfig {\n /** Logger instance */\n logger?: Logger;\n}\n\n/**\n * Handler for extension notifications\n */\nexport type ExtensionNotificationHandler = (\n method: string,\n params: Record<string, unknown>\n) => void | Promise<void>;\n\n/**\n * Manages extension methods and notifications\n */\nexport class ExtensionManager {\n private config: ExtensionManagerConfig;\n private handlers = new Map<string, ExtensionNotificationHandler>();\n private fallbackHandler?: ExtensionNotificationHandler;\n\n constructor(config: ExtensionManagerConfig = {}) {\n this.config = config;\n }\n\n /**\n * Register a handler for a specific extension method\n */\n registerHandler(method: string, handler: ExtensionNotificationHandler): () => void {\n this.handlers.set(method, handler);\n return () => {\n this.handlers.delete(method);\n };\n }\n\n /**\n * Set a fallback handler for unknown extensions\n */\n setFallbackHandler(handler: ExtensionNotificationHandler): void {\n this.fallbackHandler = handler;\n }\n\n /**\n * Handle an extension notification\n */\n async handleNotification(method: string, params: Record<string, unknown>): Promise<void> {\n this.config.logger?.debug(`Extension notification: ${method}`);\n\n // Try specific handler first\n const handler = this.handlers.get(method);\n if (handler) {\n await handler(method, params);\n return;\n }\n\n // Try fallback handler\n if (this.fallbackHandler) {\n await this.fallbackHandler(method, params);\n return;\n }\n\n // Log unknown extension\n if (!this.isKnownExtension(method)) {\n this.config.logger?.warn(`Unknown extension notification: ${method}`);\n }\n }\n\n /**\n * Check if a method is a known extension\n */\n isKnownExtension(method: string): boolean {\n return KNOWN_EXTENSIONS.includes(method as typeof KNOWN_EXTENSIONS[number]);\n }\n\n /**\n * Check if method is the artifact extension\n */\n isArtifactExtension(method: string): boolean {\n return method === ExtensionMethod.ARTIFACT;\n }\n\n /**\n * Clear all handlers\n */\n clear(): void {\n this.handlers.clear();\n this.fallbackHandler = undefined;\n }\n}\n\n/**\n * Parse usage update from extension params\n */\nexport function parseUsageUpdate(params: Record<string, unknown>): UsageUpdate {\n return {\n sessionId: params.sessionId as string,\n inputTokens: params.inputTokens as number | undefined,\n outputTokens: params.outputTokens as number | undefined,\n totalTokens: params.totalTokens as number | undefined,\n cost: params.cost as number | undefined,\n model: params.model as string | undefined,\n _meta: params._meta as Record<string, unknown> | undefined\n };\n}\n","/**\n * Permission Manager for Streamable HTTP ACP Client\n * Handles permission requests with timeout support\n */\n\nimport type { RequestPermissionRequest, RequestPermissionResponse } from '@agentclientprotocol/sdk';\n\nimport { DEFAULT_PERMISSION_TIMEOUT } from './constants.js';\nimport { TimeoutError } from './errors.js';\nimport type { Logger, PermissionHandler } from './types.js';\n\n/**\n * Pending permission request state\n */\ninterface PendingPermission {\n params: RequestPermissionRequest;\n resolve: (response: RequestPermissionResponse) => void;\n reject: (error: Error) => void;\n timeoutId?: ReturnType<typeof setTimeout>;\n createdAt: number;\n}\n\n/**\n * Configuration for PermissionManager\n */\nexport interface PermissionManagerConfig {\n /** Default timeout for permission requests (ms) */\n timeout?: number;\n /** Auto-reject permissions on timeout */\n autoRejectOnTimeout?: boolean;\n /** Auto-approve all permissions (for testing) */\n autoApprove?: boolean;\n /** Custom permission handler */\n handler?: PermissionHandler;\n /** Logger instance */\n logger?: Logger;\n}\n\n/**\n * Event callbacks for permission events\n */\nexport interface PermissionEventCallbacks {\n onRequest?: (requestId: string, params: RequestPermissionRequest) => void;\n onResolved?: (requestId: string, optionId: string) => void;\n onRejected?: (requestId: string, reason?: string) => void;\n onTimeout?: (requestId: string) => void;\n}\n\n/**\n * Manages permission requests from the agent\n */\nexport class PermissionManager {\n private pending = new Map<string, PendingPermission>();\n private config: PermissionManagerConfig;\n private callbacks: PermissionEventCallbacks = {};\n\n constructor(config: PermissionManagerConfig = {}) {\n this.config = {\n timeout: DEFAULT_PERMISSION_TIMEOUT,\n autoRejectOnTimeout: true,\n autoApprove: false,\n ...config\n };\n }\n\n /**\n * Set event callbacks\n */\n setCallbacks(callbacks: PermissionEventCallbacks): void {\n this.callbacks = callbacks;\n }\n\n /**\n * Handle a permission request from the agent\n */\n async handleRequest(params: RequestPermissionRequest): Promise<RequestPermissionResponse> {\n const requestId = params.toolCall.toolCallId;\n\n this.config.logger?.debug(`Permission request received: ${requestId}`);\n\n // Auto-approve mode\n if (this.config.autoApprove) {\n const firstOption = params.options[0];\n this.config.logger?.debug(`Auto-approving permission: ${requestId}`);\n return {\n outcome: {\n outcome: 'selected',\n optionId: firstOption?.optionId ?? 'approve'\n }\n };\n }\n\n // Custom handler\n if (this.config.handler) {\n return this.config.handler(params);\n }\n\n // Create pending permission\n return new Promise<RequestPermissionResponse>((resolve, reject) => {\n const pending: PendingPermission = {\n params,\n resolve,\n reject,\n createdAt: Date.now()\n };\n\n // Set up timeout\n if (this.config.timeout && this.config.timeout > 0) {\n pending.timeoutId = setTimeout(() => {\n this.handleTimeout(requestId);\n }, this.config.timeout);\n }\n\n this.pending.set(requestId, pending);\n\n // Notify callback\n this.callbacks.onRequest?.(requestId, params);\n });\n }\n\n /**\n * Handle timeout for a permission request\n */\n private handleTimeout(requestId: string): void {\n const pending = this.pending.get(requestId);\n if (!pending) {return;}\n\n this.config.logger?.warn(`Permission request timed out: ${requestId}`);\n this.callbacks.onTimeout?.(requestId);\n\n if (this.config.autoRejectOnTimeout) {\n pending.resolve({\n outcome: { outcome: 'cancelled' }\n });\n } else {\n pending.reject(new TimeoutError('permission', this.config.timeout ?? DEFAULT_PERMISSION_TIMEOUT));\n }\n\n this.pending.delete(requestId);\n }\n\n /**\n * Resolve a permission request with a selected option\n * Returns true if the permission was found and resolved\n */\n resolve(requestId: string, optionId: string): boolean {\n const pending = this.pending.get(requestId);\n if (!pending) {\n this.config.logger?.warn(`Permission request not found: ${requestId}`);\n return false;\n }\n\n // Clear timeout\n if (pending.timeoutId) {\n clearTimeout(pending.timeoutId);\n }\n\n this.config.logger?.debug(`Permission resolved: ${requestId} -> ${optionId}`);\n\n pending.resolve({\n outcome: { outcome: 'selected', optionId }\n });\n\n this.pending.delete(requestId);\n this.callbacks.onResolved?.(requestId, optionId);\n\n return true;\n }\n\n /**\n * Reject (cancel) a permission request\n * Returns true if the permission was found and rejected\n */\n reject(requestId: string, reason?: string): boolean {\n const pending = this.pending.get(requestId);\n if (!pending) {\n this.config.logger?.warn(`Permission request not found: ${requestId}`);\n return false;\n }\n\n // Clear timeout\n if (pending.timeoutId) {\n clearTimeout(pending.timeoutId);\n }\n\n this.config.logger?.debug(`Permission rejected: ${requestId}${reason ? ` - ${reason}` : ''}`);\n\n pending.resolve({\n outcome: { outcome: 'cancelled' }\n });\n\n this.pending.delete(requestId);\n this.callbacks.onRejected?.(requestId, reason);\n\n return true;\n }\n\n /**\n * Get all pending permissions\n */\n getPending(): Map<string, { params: RequestPermissionRequest; createdAt: number }> {\n const result = new Map<string, { params: RequestPermissionRequest; createdAt: number }>();\n for (const [id, pending] of this.pending) {\n result.set(id, {\n params: pending.params,\n createdAt: pending.createdAt\n });\n }\n return result;\n }\n\n /**\n * Get a specific pending permission\n */\n getPendingById(requestId: string): { params: RequestPermissionRequest; createdAt: number } | undefined {\n const pending = this.pending.get(requestId);\n if (!pending) {return undefined;}\n return {\n params: pending.params,\n createdAt: pending.createdAt\n };\n }\n\n /**\n * Check if there are any pending permissions\n */\n hasPending(): boolean {\n return this.pending.size > 0;\n }\n\n /**\n * Get the count of pending permissions\n */\n get pendingCount(): number {\n return this.pending.size;\n }\n\n /**\n * Clear all pending permissions (reject all)\n */\n clear(): void {\n for (const [requestId, pending] of this.pending) {\n if (pending.timeoutId) {\n clearTimeout(pending.timeoutId);\n }\n pending.resolve({\n outcome: { outcome: 'cancelled' }\n });\n this.callbacks.onRejected?.(requestId, 'cleared');\n }\n this.pending.clear();\n this.config.logger?.debug('Cleared all pending permissions');\n }\n\n /**\n * Update configuration\n */\n updateConfig(config: Partial<PermissionManagerConfig>): void {\n this.config = { ...this.config, ...config };\n }\n}\n","/**\n * Question Manager for Streamable HTTP ACP Client\n * Handles ask_followup_question requests with timeout support\n */\n\nimport type {\n QuestionRequest,\n QuestionResponse,\n} from '../types.js';\nimport { DEFAULT_QUESTION_TIMEOUT } from './constants.js';\nimport { TimeoutError } from './errors.js';\nimport type { Logger } from './types.js';\n\n// Re-export for convenience\nexport type { QuestionRequest, QuestionResponse };\n\n/**\n * Pending question state\n */\ninterface PendingQuestion {\n request: QuestionRequest;\n resolve: (response: QuestionResponse) => void;\n reject: (error: Error) => void;\n timeoutId?: ReturnType<typeof setTimeout>;\n createdAt: number;\n}\n\n/**\n * Configuration for QuestionManager\n */\nexport interface QuestionManagerConfig {\n /** Default timeout for question requests (ms) */\n timeout?: number;\n /** Auto-cancel requests on timeout */\n autoCancelOnTimeout?: boolean;\n /** Logger instance */\n logger?: Logger;\n}\n\n/**\n * User's answers to questions\n */\nexport type QuestionAnswers = Record<string, string | string[]>;\n\n/**\n * Event callbacks for question events\n */\nexport interface QuestionEventCallbacks {\n onRequest?: (toolCallId: string, request: QuestionRequest) => void;\n onAnswered?: (toolCallId: string, answers: QuestionAnswers) => void;\n onCancelled?: (toolCallId: string, reason?: string) => void;\n onTimeout?: (toolCallId: string) => void;\n}\n\n/**\n * Manages question requests from the agent (ask_followup_question tool)\n */\nexport class QuestionManager {\n private pending = new Map<string, PendingQuestion>();\n private config: QuestionManagerConfig;\n private callbacks: QuestionEventCallbacks = {};\n\n constructor(config: QuestionManagerConfig = {}) {\n this.config = {\n timeout: DEFAULT_QUESTION_TIMEOUT,\n autoCancelOnTimeout: true,\n ...config\n };\n }\n\n /**\n * Set event callbacks\n */\n setCallbacks(callbacks: QuestionEventCallbacks): void {\n this.callbacks = callbacks;\n }\n\n /**\n * Handle a question request from the agent\n * Called when receiving _codebuddy.ai/question extMethod\n */\n async handleRequest(request: QuestionRequest): Promise<QuestionResponse> {\n const toolCallId = request.toolCallId;\n\n this.config.logger?.debug(`Question request received: ${toolCallId}`);\n\n return new Promise<QuestionResponse>((resolve, reject) => {\n const pending: PendingQuestion = {\n request,\n resolve,\n reject,\n createdAt: Date.now()\n };\n\n // Set up timeout\n const timeout = request.timeout ?? this.config.timeout;\n if (timeout && timeout > 0) {\n pending.timeoutId = setTimeout(() => {\n this.handleTimeout(toolCallId);\n }, timeout);\n }\n\n this.pending.set(toolCallId, pending);\n\n // Notify callback\n this.callbacks.onRequest?.(toolCallId, request);\n });\n }\n\n /**\n * Handle timeout for a question request\n */\n private handleTimeout(toolCallId: string): void {\n const pending = this.pending.get(toolCallId);\n if (!pending) {return;}\n\n this.config.logger?.warn(`Question request timed out: ${toolCallId}`);\n this.callbacks.onTimeout?.(toolCallId);\n\n if (this.config.autoCancelOnTimeout) {\n pending.resolve({\n outcome: 'cancelled',\n reason: 'timeout'\n });\n } else {\n pending.reject(new TimeoutError('question', this.config.timeout ?? DEFAULT_QUESTION_TIMEOUT));\n }\n\n this.pending.delete(toolCallId);\n }\n\n /**\n * Answer a question request with user's selections\n * Returns true if the request was found and answered\n */\n answer(toolCallId: string, answers: QuestionAnswers): boolean {\n const pending = this.pending.get(toolCallId);\n if (!pending) {\n this.config.logger?.warn(`Question request not found: ${toolCallId}`);\n return false;\n }\n\n // Clear timeout\n if (pending.timeoutId) {\n clearTimeout(pending.timeoutId);\n }\n\n this.config.logger?.debug(`Question answered: ${toolCallId}`);\n\n pending.resolve({\n outcome: 'submitted',\n answers\n });\n\n this.pending.delete(toolCallId);\n this.callbacks.onAnswered?.(toolCallId, answers);\n\n return true;\n }\n\n /**\n * Cancel a question request\n * Returns true if the request was found and cancelled\n */\n cancel(toolCallId: string, reason?: string): boolean {\n const pending = this.pending.get(toolCallId);\n if (!pending) {\n this.config.logger?.warn(`Question request not found: ${toolCallId}`);\n return false;\n }\n\n // Clear timeout\n if (pending.timeoutId) {\n clearTimeout(pending.timeoutId);\n }\n\n this.config.logger?.debug(`Question cancelled: ${toolCallId}${reason ? ` - ${reason}` : ''}`);\n\n pending.resolve({\n outcome: 'cancelled',\n reason\n });\n\n this.pending.delete(toolCallId);\n this.callbacks.onCancelled?.(toolCallId, reason);\n\n return true;\n }\n\n /**\n * Get all pending question requests\n */\n getPending(): Map<string, { request: QuestionRequest; createdAt: number }> {\n const result = new Map<string, { request: QuestionRequest; createdAt: number }>();\n for (const [id, pending] of this.pending) {\n result.set(id, {\n request: pending.request,\n createdAt: pending.createdAt\n });\n }\n return result;\n }\n\n /**\n * Get a specific pending question request\n */\n getPendingById(toolCallId: string): { request: QuestionRequest; createdAt: number } | undefined {\n const pending = this.pending.get(toolCallId);\n if (!pending) {return undefined;}\n return {\n request: pending.request,\n createdAt: pending.createdAt\n };\n }\n\n /**\n * Check if there are any pending question requests\n */\n hasPending(): boolean {\n return this.pending.size > 0;\n }\n\n /**\n * Get the count of pending question requests\n */\n get pendingCount(): number {\n return this.pending.size;\n }\n\n /**\n * Clear all pending question requests (cancel all)\n */\n clear(): void {\n for (const [toolCallId, pending] of this.pending) {\n if (pending.timeoutId) {\n clearTimeout(pending.timeoutId);\n }\n pending.resolve({\n outcome: 'cancelled',\n reason: 'cleared'\n });\n this.callbacks.onCancelled?.(toolCallId, 'cleared');\n }\n this.pending.clear();\n this.config.logger?.debug('Cleared all pending question requests');\n }\n\n /**\n * Update configuration\n */\n updateConfig(config: Partial<QuestionManagerConfig>): void {\n this.config = { ...this.config, ...config };\n }\n}\n","/**\n * Streamable HTTP ACP Client\n * Production-grade client for connecting to cloud-hosted ACP agents\n */\n\nimport {\n type Client,\n ClientSideConnection,\n ContentBlock,\n type InitializeResponse,\n type LoadSessionResponse,\n type NewSessionResponse,\n type PromptResponse,\n PROTOCOL_VERSION,\n type RequestPermissionRequest,\n type RequestPermissionResponse,\n type SessionNotification,\n type SetSessionModelRequest,\n type SetSessionModelResponse,\n type SetSessionModeRequest,\n type SetSessionModeResponse} from '@agentclientprotocol/sdk';\n\nimport { streamableHttp, type StreamableHttpTransport } from '../transport/streamable-http.js';\nimport type {\n ArtifactNotificationParams,\n CheckpointNotificationParams\n} from '../types.js';\nimport { ArtifactManager } from './artifacts.js';\nimport {\n CLOUD_CLIENT_CAPABILITIES,\n DEFAULT_INITIALIZE_TIMEOUT,\n ExtensionMethod\n} from './constants.js';\nimport {\n ConnectionError,\n InitializationError,\n InvalidStateError,\n SessionError\n} from './errors.js';\nimport { EventEmitter } from './events.js';\nimport { ExtensionManager, parseUsageUpdate } from './extensions.js';\nimport { PermissionManager } from './permissions.js';\nimport { type QuestionAnswers, QuestionManager, type QuestionRequest } from './questions.js';\nimport type {\n ClientEvents,\n ClientState,\n PromptOptions,\n StreamableHttpClientOptions,\n} from './types.js';\n\n/**\n * Production-grade Streamable HTTP ACP Client\n *\n * Features:\n * - Full ACP protocol support (initialize, session, prompt, cancel)\n * - Artifact notification handling\n * - Permission handling with timeout support\n * - Extension method support\n * - Type-safe event system\n * - Configurable logging\n */\nexport class StreamableHttpClient {\n private connection!: ClientSideConnection;\n private transport?: StreamableHttpTransport;\n private options: StreamableHttpClientOptions;\n private state: ClientState = 'disconnected';\n private initializeResponse?: InitializeResponse;\n\n // Managers\n private artifactManager: ArtifactManager;\n private permissionManager: PermissionManager;\n private questionManager: QuestionManager;\n private extensionManager: ExtensionManager;\n\n // Event emitter\n private emitter = new EventEmitter<ClientEvents>();\n\n constructor(options: StreamableHttpClientOptions) {\n this.options = options;\n\n // Initialize artifact manager\n this.artifactManager = new ArtifactManager({\n logger: options.logger\n });\n\n // Initialize permission manager\n this.permissionManager = new PermissionManager({\n timeout: options.permissionTimeout,\n autoRejectOnTimeout: options.permissionAutoRejectOnTimeout ?? true,\n autoApprove: options.autoApprove,\n handler: options.requestPermissionHandler,\n logger: options.logger\n });\n\n // Set up permission event callbacks\n this.permissionManager.setCallbacks({\n onRequest: (requestId, params) => {\n this.emitter.emit('permissionRequest', { requestId, params });\n },\n onResolved: (requestId, optionId) => {\n this.emitter.emit('permissionResolved', { requestId, optionId });\n },\n onRejected: (requestId, reason) => {\n this.emitter.emit('permissionRejected', { requestId, reason });\n },\n onTimeout: requestId => {\n this.emitter.emit('permissionTimeout', { requestId });\n }\n });\n\n // Initialize question manager\n this.questionManager = new QuestionManager({\n timeout: options.questionTimeout,\n autoCancelOnTimeout: options.questionAutoCancelOnTimeout ?? true,\n logger: options.logger\n });\n\n // Set up question event callbacks\n this.questionManager.setCallbacks({\n onRequest: (toolCallId: string, request: QuestionRequest) => {\n this.emitter.emit('questionRequest', { toolCallId, request });\n options.onQuestionRequest?.(toolCallId, request);\n },\n onAnswered: (toolCallId: string, answers: QuestionAnswers) => {\n this.emitter.emit('questionAnswered', { toolCallId, answers });\n },\n onCancelled: (toolCallId: string, reason?: string) => {\n this.emitter.emit('questionCancelled', { toolCallId, reason });\n },\n onTimeout: (toolCallId: string) => {\n this.emitter.emit('questionTimeout', { toolCallId });\n }\n });\n\n // Initialize extension manager\n this.extensionManager = new ExtensionManager({\n logger: options.logger\n });\n }\n\n // ============================================\n // State Management\n // ============================================\n\n /**\n * Get current client state\n */\n get currentState(): ClientState {\n return this.state;\n }\n\n /**\n * Check if client is initialized\n */\n get isInitialized(): boolean {\n return this.state === 'initialized';\n }\n\n /**\n * Check if client is connected (but maybe not initialized)\n */\n get isConnected(): boolean {\n return this.state === 'connected' || this.state === 'initialized';\n }\n\n /**\n * Get agent capabilities from initialization response\n */\n get agentCapabilities() {\n return this.initializeResponse?.agentCapabilities;\n }\n\n /**\n * Get full initialization response\n */\n get initializeResult() {\n return this.initializeResponse;\n }\n\n /**\n * Get current transport connection ID\n */\n get connectionId(): string | undefined {\n return this.transport?.connectionId;\n }\n\n private setState(newState: ClientState): void {\n const previous = this.state;\n this.state = newState;\n\n this.options.logger?.debug(`State change: ${previous} -> ${newState}`);\n this.emitter.emit('stateChange', { previous, current: newState });\n\n // Emit specific state events\n switch (newState) {\n case 'connecting':\n this.emitter.emit('connecting', undefined);\n break;\n case 'connected':\n this.emitter.emit('connected', undefined);\n break;\n case 'disconnected':\n this.emitter.emit('disconnected', undefined);\n break;\n case 'error':\n // Error event is emitted separately with the actual error\n break;\n }\n }\n\n // ============================================\n // Connection Management\n // ============================================\n\n /**\n * Connect and initialize the client\n */\n async connect(): Promise<InitializeResponse> {\n if (this.state !== 'disconnected') {\n await this.disconnect();\n }\n if (this.state === 'initialized') {\n return this.initializeResponse!;\n }\n\n if (this.state === 'connecting') {\n throw new ConnectionError('Connection already in progress');\n }\n\n this.setState('connecting');\n\n try {\n // Create transport\n this.transport = streamableHttp({\n endpoint: this.options.endpoint,\n authToken: this.options.authToken,\n headers: this.options.headers,\n reconnect: this.options.reconnect,\n fetch: this.options.fetch,\n onConnect: connectionId => {\n this.options.logger?.debug(`Transport connected: ${connectionId}`);\n },\n onDisconnect: connectionId => {\n this.options.logger?.debug(`Transport disconnected: ${connectionId}`);\n },\n onError: error => {\n this.options.logger?.error('Transport error:', error);\n this.emitter.emit('error', error);\n },\n });\n\n // Create connection\n this.connection = new ClientSideConnection(\n () => this.createClientHandler(),\n this.transport\n );\n\n this.setState('connected');\n\n // Initialize protocol\n // Merge client capabilities with provider defaults (provider takes priority)\n const timeout = this.options.initializeTimeout ?? DEFAULT_INITIALIZE_TIMEOUT;\n const mergedCapabilities = {\n ...this.options.clientCapabilities,\n ...CLOUD_CLIENT_CAPABILITIES,\n _meta: {\n ...this.options.clientCapabilities?._meta,\n ...CLOUD_CLIENT_CAPABILITIES._meta\n }\n };\n const initPromise = this.connection.initialize({\n protocolVersion: PROTOCOL_VERSION,\n clientCapabilities: mergedCapabilities\n });\n\n // Apply timeout\n const timeoutPromise = new Promise<never>((_, reject) => {\n setTimeout(() => {\n reject(new InitializationError(`Initialize timed out after ${timeout}ms`));\n }, timeout);\n });\n\n this.initializeResponse = await Promise.race([initPromise, timeoutPromise]);\n this.setState('initialized');\n\n this.options.logger?.info('Client initialized successfully');\n\n return this.initializeResponse;\n } catch (err) {\n this.setState('error');\n const error = err instanceof Error ? err : new Error(String(err));\n this.emitter.emit('error', error);\n\n if (err instanceof InitializationError || err instanceof ConnectionError) {\n throw err;\n }\n throw new ConnectionError('Failed to connect', error);\n }\n }\n\n /**\n * Disconnect the client gracefully\n * Sends DELETE request to server before closing local resources\n */\n async disconnect(): Promise<void> {\n if (this.state === 'disconnected') {\n return;\n }\n\n this.options.logger?.info('Disconnecting client');\n\n // Close transport gracefully (sends DELETE to server)\n if (this.transport) {\n try {\n await this.transport.close();\n } catch (err) {\n this.options.logger?.warn('Error closing transport:', err);\n }\n this.transport = undefined;\n }\n\n // Clear pending permissions\n this.permissionManager.clear();\n\n // Clear pending question requests\n this.questionManager.clear();\n\n // Clear artifacts\n this.artifactManager.clear();\n\n // Reset state\n this.initializeResponse = undefined;\n this.setState('disconnected');\n }\n\n /**\n * Create the client handler for the connection\n */\n private createClientHandler(): Client {\n return {\n sessionUpdate: async (params: SessionNotification) => {\n await this.handleSessionUpdate(params);\n },\n requestPermission: async (params: RequestPermissionRequest) => this.handleRequestPermission(params),\n extNotification: async (method: string, params: Record<string, unknown>) => {\n console.log('[ACP-Client] extNotification callback invoked:', { method, paramsKeys: Object.keys(params) });\n await this.handleExtNotification(method, params);\n },\n extMethod: async (method: string, params: Record<string, unknown>): Promise<Record<string, unknown>> =>\n this.handleExtMethod(method, params as unknown as QuestionRequest)\n };\n }\n\n // ============================================\n // Session Management\n // ============================================\n\n /**\n * Create a new session\n */\n async createSession(cwd: string): Promise<NewSessionResponse> {\n this.ensureInitialized('createSession');\n\n try {\n const response = await this.connection.newSession({\n cwd,\n mcpServers: []\n });\n\n this.options.logger?.info(`Session created: ${response.sessionId}`);\n return response;\n } catch (err) {\n throw new SessionError(\n `Failed to create session: ${err instanceof Error ? err.message : String(err)}`,\n undefined,\n err instanceof Error ? err : undefined\n );\n }\n }\n\n /**\n * Load an existing session\n * Requires agent to support loadSession capability\n */\n async loadSession(sessionId: string, cwd: string): Promise<LoadSessionResponse> {\n this.ensureInitialized('loadSession');\n\n if (!this.agentCapabilities?.loadSession) {\n throw new SessionError('Agent does not support session loading', sessionId);\n }\n\n try {\n const response = await this.connection.loadSession({\n sessionId,\n cwd,\n mcpServers: []\n });\n\n this.options.logger?.info(`Session loaded: ${sessionId}`);\n return response;\n } catch (err) {\n throw new SessionError(\n `Failed to load session: ${err instanceof Error ? err.message : String(err)}`,\n sessionId,\n err instanceof Error ? err : undefined\n );\n }\n }\n\n /**\n * Set the session mode\n */\n async setSessionMode(params: SetSessionModeRequest): Promise<SetSessionModeResponse> {\n this.ensureInitialized('setSessionMode');\n\n this.options.logger?.debug(`Setting session mode: ${params.sessionId} -> ${params.modeId}`);\n\n return this.connection.setSessionMode(params);\n }\n\n /**\n * Set the session model\n * @experimental This API is unstable and may change\n */\n async setSessionModel(params: SetSessionModelRequest): Promise<SetSessionModelResponse> {\n this.ensureInitialized('setSessionModel');\n\n this.options.logger?.debug(`Setting session model: ${params.sessionId} -> ${params.modelId}`);\n\n return this.connection.unstable_setSessionModel(params);\n }\n\n // ============================================\n // Prompt\n // ============================================\n\n /**\n * Send a prompt to the agent\n */\n async prompt(\n sessionId: string,\n prompt: ContentBlock[],\n options?: PromptOptions\n ): Promise<PromptResponse> {\n this.ensureInitialized('prompt');\n\n this.options.logger?.debug(`Sending prompt to session: ${sessionId}`);\n\n return this.connection.prompt({\n sessionId,\n prompt,\n _meta: options?.planMode ? { planMode: true, ...options._meta } : options?._meta\n });\n }\n\n /**\n * Cancel ongoing operations for a session\n */\n async cancel(sessionId: string): Promise<void> {\n this.ensureInitialized('cancel');\n\n this.options.logger?.debug(`Cancelling session: ${sessionId}`);\n\n await this.connection.cancel({ sessionId });\n }\n\n // ============================================\n // Permission Management\n // ============================================\n\n /**\n * Resolve a pending permission request\n */\n resolvePermission(requestId: string, optionId: string): boolean {\n return this.permissionManager.resolve(requestId, optionId);\n }\n\n /**\n * Reject a pending permission request\n */\n rejectPermission(requestId: string, reason?: string): boolean {\n return this.permissionManager.reject(requestId, reason);\n }\n\n /**\n * Get all pending permissions\n */\n getPendingPermissions() {\n return this.permissionManager.getPending();\n }\n\n /**\n * Check if there are pending permissions\n */\n hasPendingPermissions(): boolean {\n return this.permissionManager.hasPending();\n }\n\n // ============================================\n // Question Management (ask_followup_question)\n // ============================================\n\n /**\n * Answer a pending question request with user's selections\n */\n answerQuestion(toolCallId: string, answers: QuestionAnswers): boolean {\n return this.questionManager.answer(toolCallId, answers);\n }\n\n /**\n * Cancel a pending question request\n */\n cancelQuestion(toolCallId: string, reason?: string): boolean {\n return this.questionManager.cancel(toolCallId, reason);\n }\n\n /**\n * Get all pending question requests\n */\n getPendingQuestions() {\n return this.questionManager.getPending();\n }\n\n /**\n * Check if there are pending question requests\n */\n hasPendingQuestions(): boolean {\n return this.questionManager.hasPending();\n }\n\n // ============================================\n // Extension Methods\n // ============================================\n\n /**\n * Send an extension method request\n */\n async extMethod(method: string, params: Record<string, unknown>): Promise<Record<string, unknown>> {\n this.ensureInitialized('extMethod');\n return this.connection.extMethod(method, params);\n }\n\n /**\n * Send an extension notification\n */\n async extNotification(method: string, params: Record<string, unknown>): Promise<void> {\n this.ensureInitialized('extNotification');\n return this.connection.extNotification(method, params);\n }\n\n // ============================================\n // Event Emitter Implementation\n // ============================================\n\n on<K extends keyof ClientEvents>(\n event: K,\n listener: (data: ClientEvents[K]) => void | Promise<void>\n ): this {\n this.emitter.on(event, listener);\n return this;\n }\n\n off<K extends keyof ClientEvents>(\n event: K,\n listener: (data: ClientEvents[K]) => void | Promise<void>\n ): this {\n this.emitter.off(event, listener);\n return this;\n }\n\n once<K extends keyof ClientEvents>(\n event: K,\n listener: (data: ClientEvents[K]) => void | Promise<void>\n ): this {\n this.emitter.once(event, listener);\n return this;\n }\n\n emit<K extends keyof ClientEvents>(event: K, data: ClientEvents[K]): boolean {\n return this.emitter.emit(event, data);\n }\n\n removeAllListeners<K extends keyof ClientEvents>(event?: K): this {\n this.emitter.removeAllListeners(event);\n return this;\n }\n\n // ============================================\n // Internal Handlers\n // ============================================\n\n private async handleSessionUpdate(params: SessionNotification): Promise<void> {\n // Forward to callback\n await this.options.onSessionUpdate?.(params);\n\n // Emit event\n this.emitter.emit('sessionUpdate', params);\n }\n\n private async handleRequestPermission(\n params: RequestPermissionRequest\n ): Promise<RequestPermissionResponse> {\n return this.permissionManager.handleRequest(params);\n }\n\n private async handleExtNotification(\n method: string,\n params: Record<string, unknown>\n ): Promise<void> {\n // Handle artifact notifications\n if (method === ExtensionMethod.ARTIFACT) {\n const notification = params as unknown as ArtifactNotificationParams;\n const artifactData = notification.artifact as any;\n console.log('[ACP-Client] Received artifact notification:', {\n event: notification.event,\n artifactUri: artifactData?.uri,\n artifactType: artifactData?.type,\n });\n\n // For deleted events, get full artifact before deletion for callbacks\n if (notification.event === 'deleted') {\n const existing = this.artifactManager.get(notification.artifact.uri);\n this.artifactManager.handleNotification(notification);\n if (existing) {\n await this.options.onArtifact?.(existing, 'deleted');\n this.emitter.emit('artifactDeleted', existing);\n }\n } else {\n const { artifact, event } = notification;\n // Let artifactManager handle the notification\n this.artifactManager.handleNotification(notification);\n // Get the artifact from manager or use the original\n const storedArtifact = this.artifactManager.get(artifact.uri) || artifact;\n\n console.log('[ACP-Client] Stored artifact:', {\n event,\n artifactUri: storedArtifact.uri,\n artifactType: storedArtifact.type,\n hasText: storedArtifact.type === 'plan' ? !!(storedArtifact as any).text : undefined,\n });\n\n await this.options.onArtifact?.(storedArtifact, event);\n\n if (event === 'created') {\n console.log('[ACP-Client] Emitting artifactCreated event');\n this.emitter.emit('artifactCreated', storedArtifact);\n if (storedArtifact.type === 'plan') {\n await this.options.onPlanReady?.(storedArtifact);\n }\n } else {\n console.log('[ACP-Client] Emitting artifactUpdated event');\n this.emitter.emit('artifactUpdated', storedArtifact);\n }\n }\n return;\n }\n\n // Handle usage update notifications\n if (method === ExtensionMethod.USAGE) {\n const usage = parseUsageUpdate(params);\n await this.options.onUsageUpdate?.(usage);\n this.emitter.emit('usageUpdate', usage);\n return;\n }\n\n // Handle checkpoint notifications\n if (method === ExtensionMethod.CHECKPOINT) {\n const notification = params as unknown as CheckpointNotificationParams;\n if (notification.event === 'created') {\n this.emitter.emit('checkpointCreated', notification.checkpoint);\n } else if (notification.event === 'updated') {\n this.emitter.emit('checkpointUpdated', notification.checkpoint);\n }\n return;\n }\n\n // Forward other extensions to callback and manager\n await this.options.onExtNotification?.(method, params);\n await this.extensionManager.handleNotification(method, params);\n }\n\n private async handleExtMethod(\n method: string,\n params: QuestionRequest\n ): Promise<Record<string, unknown>> {\n // Handle question requests (ask_followup_question via question extMethod)\n if (method === ExtensionMethod.QUESTION) {\n const response = await this.questionManager.handleRequest(params);\n\n // Convert QuestionResponse to ToolInputResponse format expected by SDK\n if (response.outcome === 'submitted' && response.answers) {\n return {\n outcome: {\n outcome: 'submitted',\n data: {\n answers: response.answers\n }\n }\n };\n } else {\n return {\n outcome: {\n outcome: 'cancelled',\n reason: response.reason\n }\n };\n }\n }\n\n // Unknown extension method\n this.options.logger?.warn(`Unknown extension method: ${method}`);\n return {\n outcome: {\n outcome: 'cancelled',\n reason: 'unknown method'\n }\n };\n }\n\n // ============================================\n // Helpers\n // ============================================\n\n private ensureInitialized(operation: string): void {\n if (this.state !== 'initialized') {\n throw new InvalidStateError(operation, this.state, ['initialized']);\n }\n }\n}\n\nexport default StreamableHttpClient;\n","/**\n * Cloud Agent Connection\n * Wraps StreamableHttpClient to implement AgentConnection interface\n */\n\nimport type {\n ContentBlock,\n InitializeResponse,\n LoadSessionResponse,\n NewSessionResponse,\n PromptResponse,\n RequestPermissionRequest,\n SessionNotification,\n SetSessionModelResponse,\n SetSessionModeResponse\n} from '@agentclientprotocol/sdk';\nimport {\n StreamableHttpClient\n} from '@genie/agent-client-protocol';\n\nimport type { SessionConnectionInfo } from '../../client/types.js';\nimport type {\n AgentCapabilities,\n AgentConnection,\n AgentStatus,\n CloudConnectionConfig,\n ConnectionEventListener,\n ConnectionEvents,\n CreateSessionParams,\n LoadSessionParams,\n PromptParams\n} from '../../types.js';\n\n/**\n * Cloud Agent Connection implementation\n * Uses Streamable HTTP transport to connect to cloud-hosted ACP agents\n * Uses composition pattern - implements event emitter methods internally\n *\n * TODO: Connection Lifecycle Responsibilities\n * CloudAgentProvider caches connections by endpoint link. This class needs to:\n * - Implement connection health checks (detect and handle connection failures/reconnection)\n * - Handle token expiration (refresh or re-authentication when tokens expire)\n * - Emit 'disconnected' event when connection becomes unhealthy so provider can clean up cache\n */\nexport class CloudAgentConnection implements AgentConnection {\n private client: StreamableHttpClient;\n private listeners: Map<keyof ConnectionEvents, Set<ConnectionEventListener<unknown>>> = new Map();\n private onceListeners: Map<keyof ConnectionEvents, Set<ConnectionEventListener<unknown>>> = new Map();\n\n /**\n * Flag to suppress sessionUpdate event emission during streaming.\n * When true, onSessionUpdate callback won't emit to avoid duplicate messages with promptStream.\n */\n private _isStreaming = false;\n\n /**\n * Session connection information (sandboxId, link, token, etc.)\n * Set by CloudAgentProvider.connect() after fetching session data from backend.\n */\n private _sessionConnectionInfo?: SessionConnectionInfo;\n\n readonly agentId: string;\n readonly transport = 'cloud' as const;\n readonly cwd: string;\n\n constructor(agentId: string, config: CloudConnectionConfig, cwd: string = '/workspace') {\n this.agentId = agentId;\n this.cwd = cwd;\n\n // Create the underlying StreamableHttpClient\n this.client = new StreamableHttpClient({\n endpoint: config.endpoint,\n authToken: config.authToken,\n headers: config.headers,\n reconnect: config.reconnect,\n initializeTimeout: config.initializeTimeout,\n permissionTimeout: config.permissionTimeout,\n permissionAutoRejectOnTimeout: config.permissionAutoRejectOnTimeout,\n autoApprove: config.autoApprove,\n logger: config.logger,\n fetch: config.fetch,\n clientCapabilities: config.clientCapabilities,\n // Forward events to our emitter (suppressed during streaming to avoid duplicates)\n onSessionUpdate: update => {\n if (!this._isStreaming) {\n this.emit('sessionUpdate', update);\n }\n },\n onArtifact: (artifact, event) => {\n console.log('[CloudConnection] onArtifact callback:', {\n event,\n artifactUri: artifact.uri,\n artifactType: artifact.type,\n });\n if (event === 'created') {\n this.emit('artifactCreated', artifact);\n } else if (event === 'updated') {\n this.emit('artifactUpdated', artifact);\n } else if (event === 'deleted') {\n this.emit('artifactDeleted', artifact);\n }\n },\n onUsageUpdate: usage => {\n this.emit('usageUpdate', usage);\n }\n });\n\n // Forward client events\n this.setupEventForwarding();\n }\n\n private setupEventForwarding(): void {\n // Forward connection state events\n this.client.on('connecting', () => { this.emit('connecting', undefined); });\n this.client.on('connected', () => { this.emit('connected', undefined); });\n this.client.on('disconnected', () => { this.emit('disconnected', undefined); });\n this.client.on('error', error => { this.emit('error', error); });\n this.client.on('stateChange', change => { this.emit('stateChange', change); });\n\n // Forward permission events\n this.client.on('permissionRequest', data => { this.emit('permissionRequest', data); });\n this.client.on('permissionResolved', data => { this.emit('permissionResolved', data); });\n this.client.on('permissionRejected', data => { this.emit('permissionRejected', data); });\n this.client.on('permissionTimeout', data => { this.emit('permissionTimeout', data); });\n\n // Forward question events\n this.client.on('questionRequest', data => { this.emit('questionRequest', data); });\n this.client.on('questionAnswered', data => { this.emit('questionAnswered', data); });\n this.client.on('questionCancelled', data => { this.emit('questionCancelled', data); });\n this.client.on('questionTimeout', data => { this.emit('questionTimeout', data); });\n\n // Forward checkpoint events\n this.client.on('checkpointCreated', checkpoint => { this.emit('checkpointCreated', checkpoint); });\n this.client.on('checkpointUpdated', checkpoint => { this.emit('checkpointUpdated', checkpoint); });\n }\n\n // ============================================\n // Event Emitter Implementation\n // ============================================\n\n on<K extends keyof ConnectionEvents>(event: K, listener: ConnectionEventListener<ConnectionEvents[K]>): this {\n if (!this.listeners.has(event)) {\n this.listeners.set(event, new Set());\n }\n this.listeners.get(event)!.add(listener as ConnectionEventListener<unknown>);\n return this;\n }\n\n off<K extends keyof ConnectionEvents>(event: K, listener: ConnectionEventListener<ConnectionEvents[K]>): this {\n const eventListeners = this.listeners.get(event);\n if (eventListeners) {\n eventListeners.delete(listener as ConnectionEventListener<unknown>);\n }\n const onceEventListeners = this.onceListeners.get(event);\n if (onceEventListeners) {\n onceEventListeners.delete(listener as ConnectionEventListener<unknown>);\n }\n return this;\n }\n\n once<K extends keyof ConnectionEvents>(event: K, listener: ConnectionEventListener<ConnectionEvents[K]>): this {\n if (!this.onceListeners.has(event)) {\n this.onceListeners.set(event, new Set());\n }\n this.onceListeners.get(event)!.add(listener as ConnectionEventListener<unknown>);\n return this;\n }\n\n emit<K extends keyof ConnectionEvents>(event: K, data: ConnectionEvents[K]): boolean {\n const regularListeners = this.listeners.get(event);\n const onceEventListeners = this.onceListeners.get(event);\n\n let hasListeners = false;\n\n // Call regular listeners\n if (regularListeners && regularListeners.size > 0) {\n hasListeners = true;\n for (const listener of regularListeners) {\n try {\n const result = listener(data);\n if (result instanceof Promise) {\n result.catch(err => {\n console.error(`Error in async event listener for '${String(event)}':`, err);\n });\n }\n } catch (err) {\n console.error(`Error in event listener for '${String(event)}':`, err);\n }\n }\n }\n\n // Call once listeners and remove them\n if (onceEventListeners && onceEventListeners.size > 0) {\n hasListeners = true;\n const listenersToCall = Array.from(onceEventListeners);\n this.onceListeners.delete(event);\n\n for (const listener of listenersToCall) {\n try {\n const result = listener(data);\n if (result instanceof Promise) {\n result.catch(err => {\n console.error(`Error in async once event listener for '${String(event)}':`, err);\n });\n }\n } catch (err) {\n console.error(`Error in once event listener for '${String(event)}':`, err);\n }\n }\n }\n\n return hasListeners;\n }\n\n removeAllListeners<K extends keyof ConnectionEvents>(event?: K): this {\n if (event !== undefined) {\n this.listeners.delete(event);\n this.onceListeners.delete(event);\n } else {\n this.listeners.clear();\n this.onceListeners.clear();\n }\n return this;\n }\n\n // ============================================\n // State Properties\n // ============================================\n\n get state(): AgentStatus {\n return this.client.currentState as AgentStatus;\n }\n\n get isInitialized(): boolean {\n return this.client.isInitialized;\n }\n\n get capabilities(): AgentCapabilities | undefined {\n return this.client.agentCapabilities as AgentCapabilities | undefined;\n }\n\n get initializeResult(): InitializeResponse | undefined {\n return this.client.initializeResult;\n }\n\n // ============================================\n // Connection Lifecycle\n // ============================================\n\n async connect(): Promise<InitializeResponse> {\n return this.client.connect();\n }\n\n async disconnect(): Promise<void> {\n await this.client.disconnect();\n }\n\n // ============================================\n // Session Management\n // ============================================\n\n async createSession(params: CreateSessionParams): Promise<NewSessionResponse> {\n // Cloud Side does not support creating new sessions directly\n // Use this.cwd (provided by CloudAgentProvider) as the working directory\n // params.cwd is ignored because cloud agents already have a fixed cwd from backend\n const loadedSession = await this.client.loadSession(this.agentId, this.cwd);\n return { ...loadedSession, sessionId: this.agentId };\n }\n\n async loadSession(params: LoadSessionParams): Promise<LoadSessionResponse> {\n if (!params.sessionId) {\n throw new Error('sessionId is required for loadSession');\n }\n return this.client.loadSession(params.sessionId, this.cwd);\n }\n\n async setSessionMode(sessionId: string, modeId: string): Promise<SetSessionModeResponse> {\n return this.client.setSessionMode({ sessionId, modeId });\n }\n\n async setSessionModel(sessionId: string, modelId: string): Promise<SetSessionModelResponse> {\n return this.client.setSessionModel({ sessionId, modelId });\n }\n\n // ============================================\n // Prompt Operations\n // ============================================\n\n async prompt(sessionId: string, params: PromptParams): Promise<PromptResponse> {\n const blocks =\n typeof params.content === 'string'\n ? [{ type: 'text', text: params.content }] as ContentBlock[]\n : params.content as ContentBlock[];\n\n return this.client.prompt(sessionId, blocks, {\n planMode: params.planMode,\n _meta: params._meta\n });\n }\n\n async *promptStream(sessionId: string, params: PromptParams): AsyncIterable<SessionNotification> {\n // Suppress sessionUpdate event emission during streaming to avoid duplicates\n this._isStreaming = true;\n\n // For streaming, we need to collect updates via the event system\n const updates: SessionNotification[] = [];\n let resolveUpdate: ((value: SessionNotification | null) => void) | null = null;\n let done = false;\n\n const listener = (update: SessionNotification) => {\n if (resolveUpdate) {\n resolveUpdate(update);\n resolveUpdate = null;\n } else {\n updates.push(update);\n }\n };\n\n this.client.on('sessionUpdate', listener);\n\n try {\n // Start the prompt (non-blocking)\n const promptPromise = this.prompt(sessionId, params);\n\n // Yield updates as they come in\n while (!done) {\n const update = updates.shift();\n if (update) {\n yield update;\n } else {\n // Wait for next update or prompt completion\n const nextUpdate = await new Promise<SessionNotification | null>(resolve => {\n resolveUpdate = resolve;\n // Check if prompt completed while we were setting up\n promptPromise.then(() => {\n if (resolveUpdate === resolve) {\n resolveUpdate = null;\n resolve(null);\n }\n }).catch(() => {\n if (resolveUpdate === resolve) {\n resolveUpdate = null;\n resolve(null);\n }\n });\n });\n\n if (nextUpdate === null) {\n done = true;\n } else {\n yield nextUpdate;\n }\n }\n }\n } finally {\n this._isStreaming = false;\n this.client.off('sessionUpdate', listener);\n }\n }\n\n async cancel(sessionId: string): Promise<void> {\n return this.client.cancel(sessionId);\n }\n\n // ============================================\n // Permission Management\n // ============================================\n\n resolvePermission(requestId: string, optionId: string): boolean {\n return this.client.resolvePermission(requestId, optionId);\n }\n\n rejectPermission(requestId: string, reason?: string): boolean {\n return this.client.rejectPermission(requestId, reason);\n }\n\n getPendingPermissions(): Map<string, { params: RequestPermissionRequest; createdAt: number }> {\n return this.client.getPendingPermissions();\n }\n\n hasPendingPermissions(): boolean {\n return this.client.hasPendingPermissions();\n }\n\n // ============================================\n // Question Management (ask_followup_question)\n // ============================================\n\n answerQuestion(toolCallId: string, answers: import('@genie/agent-client-protocol').QuestionAnswers): boolean {\n return this.client.answerQuestion(toolCallId, answers);\n }\n\n cancelQuestion(toolCallId: string, reason?: string): boolean {\n return this.client.cancelQuestion(toolCallId, reason);\n }\n\n getPendingQuestions() {\n return this.client.getPendingQuestions();\n }\n\n hasPendingQuestions(): boolean {\n return this.client.hasPendingQuestions();\n }\n\n // ============================================\n // Tool Callback Management\n // ============================================\n\n async toolCallback(sessionId: string, toolCallId: string, toolName: string, action: 'skip' | 'cancel'): Promise<{ success: boolean; error?: string }> {\n // Cloud connection does not support toolCallback yet\n return { success: false, error: 'toolCallback not supported for cloud connections' };\n }\n\n // ============================================\n // Session Connection Info\n // ============================================\n\n /**\n * Set session connection information\n * Called by CloudAgentProvider.connect() after fetching session data from backend.\n */\n setSessionConnectionInfo(info: SessionConnectionInfo): void {\n this._sessionConnectionInfo = info;\n }\n\n /**\n * Get session connection information\n * Contains sandboxId, link, token, etc.\n */\n get sessionConnectionInfo(): SessionConnectionInfo | undefined {\n return this._sessionConnectionInfo;\n }\n\n // ============================================\n // Extension Methods\n // ============================================\n\n async extMethod(method: string, params: Record<string, unknown>): Promise<Record<string, unknown>> {\n return this.client.extMethod(method, params);\n }\n}\n\nexport default CloudAgentConnection;\n","'use strict';\n\nexport default function bind(fn, thisArg) {\n return function wrap() {\n return fn.apply(thisArg, arguments);\n };\n}\n","'use strict';\n\nimport bind from './helpers/bind.js';\n\n// utils is a library of generic helper functions non-specific to axios\n\nconst {toString} = Object.prototype;\nconst {getPrototypeOf} = Object;\nconst {iterator, toStringTag} = Symbol;\n\nconst kindOf = (cache => thing => {\n const str = toString.call(thing);\n return cache[str] || (cache[str] = str.slice(8, -1).toLowerCase());\n})(Object.create(null));\n\nconst kindOfTest = (type) => {\n type = type.toLowerCase();\n return (thing) => kindOf(thing) === type\n}\n\nconst typeOfTest = type => thing => typeof thing === type;\n\n/**\n * Determine if a value is an Array\n *\n * @param {Object} val The value to test\n *\n * @returns {boolean} True if value is an Array, otherwise false\n */\nconst {isArray} = Array;\n\n/**\n * Determine if a value is undefined\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if the value is undefined, otherwise false\n */\nconst isUndefined = typeOfTest('undefined');\n\n/**\n * Determine if a value is a Buffer\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is a Buffer, otherwise false\n */\nfunction isBuffer(val) {\n return val !== null && !isUndefined(val) && val.constructor !== null && !isUndefined(val.constructor)\n && isFunction(val.constructor.isBuffer) && val.constructor.isBuffer(val);\n}\n\n/**\n * Determine if a value is an ArrayBuffer\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is an ArrayBuffer, otherwise false\n */\nconst isArrayBuffer = kindOfTest('ArrayBuffer');\n\n\n/**\n * Determine if a value is a view on an ArrayBuffer\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is a view on an ArrayBuffer, otherwise false\n */\nfunction isArrayBufferView(val) {\n let result;\n if ((typeof ArrayBuffer !== 'undefined') && (ArrayBuffer.isView)) {\n result = ArrayBuffer.isView(val);\n } else {\n result = (val) && (val.buffer) && (isArrayBuffer(val.buffer));\n }\n return result;\n}\n\n/**\n * Determine if a value is a String\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is a String, otherwise false\n */\nconst isString = typeOfTest('string');\n\n/**\n * Determine if a value is a Function\n *\n * @param {*} val The value to test\n * @returns {boolean} True if value is a Function, otherwise false\n */\nconst isFunction = typeOfTest('function');\n\n/**\n * Determine if a value is a Number\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is a Number, otherwise false\n */\nconst isNumber = typeOfTest('number');\n\n/**\n * Determine if a value is an Object\n *\n * @param {*} thing The value to test\n *\n * @returns {boolean} True if value is an Object, otherwise false\n */\nconst isObject = (thing) => thing !== null && typeof thing === 'object';\n\n/**\n * Determine if a value is a Boolean\n *\n * @param {*} thing The value to test\n * @returns {boolean} True if value is a Boolean, otherwise false\n */\nconst isBoolean = thing => thing === true || thing === false;\n\n/**\n * Determine if a value is a plain Object\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is a plain Object, otherwise false\n */\nconst isPlainObject = (val) => {\n if (kindOf(val) !== 'object') {\n return false;\n }\n\n const prototype = getPrototypeOf(val);\n return (prototype === null || prototype === Object.prototype || Object.getPrototypeOf(prototype) === null) && !(toStringTag in val) && !(iterator in val);\n}\n\n/**\n * Determine if a value is a Date\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is a Date, otherwise false\n */\nconst isDate = kindOfTest('Date');\n\n/**\n * Determine if a value is a File\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is a File, otherwise false\n */\nconst isFile = kindOfTest('File');\n\n/**\n * Determine if a value is a Blob\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is a Blob, otherwise false\n */\nconst isBlob = kindOfTest('Blob');\n\n/**\n * Determine if a value is a FileList\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is a File, otherwise false\n */\nconst isFileList = kindOfTest('FileList');\n\n/**\n * Determine if a value is a Stream\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is a Stream, otherwise false\n */\nconst isStream = (val) => isObject(val) && isFunction(val.pipe);\n\n/**\n * Determine if a value is a FormData\n *\n * @param {*} thing The value to test\n *\n * @returns {boolean} True if value is an FormData, otherwise false\n */\nconst isFormData = (thing) => {\n let kind;\n return thing && (\n (typeof FormData === 'function' && thing instanceof FormData) || (\n isFunction(thing.append) && (\n (kind = kindOf(thing)) === 'formdata' ||\n // detect form-data instance\n (kind === 'object' && isFunction(thing.toString) && thing.toString() === '[object FormData]')\n )\n )\n )\n}\n\n/**\n * Determine if a value is a URLSearchParams object\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is a URLSearchParams object, otherwise false\n */\nconst isURLSearchParams = kindOfTest('URLSearchParams');\n\nconst [isReadableStream, isRequest, isResponse, isHeaders] = ['ReadableStream', 'Request', 'Response', 'Headers'].map(kindOfTest);\n\n/**\n * Trim excess whitespace off the beginning and end of a string\n *\n * @param {String} str The String to trim\n *\n * @returns {String} The String freed of excess whitespace\n */\nconst trim = (str) => str.trim ?\n str.trim() : str.replace(/^[\\s\\uFEFF\\xA0]+|[\\s\\uFEFF\\xA0]+$/g, '');\n\n/**\n * Iterate over an Array or an Object invoking a function for each item.\n *\n * If `obj` is an Array callback will be called passing\n * the value, index, and complete array for each item.\n *\n * If 'obj' is an Object callback will be called passing\n * the value, key, and complete object for each property.\n *\n * @param {Object|Array} obj The object to iterate\n * @param {Function} fn The callback to invoke for each item\n *\n * @param {Boolean} [allOwnKeys = false]\n * @returns {any}\n */\nfunction forEach(obj, fn, {allOwnKeys = false} = {}) {\n // Don't bother if no value provided\n if (obj === null || typeof obj === 'undefined') {\n return;\n }\n\n let i;\n let l;\n\n // Force an array if not already something iterable\n if (typeof obj !== 'object') {\n /*eslint no-param-reassign:0*/\n obj = [obj];\n }\n\n if (isArray(obj)) {\n // Iterate over array values\n for (i = 0, l = obj.length; i < l; i++) {\n fn.call(null, obj[i], i, obj);\n }\n } else {\n // Iterate over object keys\n const keys = allOwnKeys ? Object.getOwnPropertyNames(obj) : Object.keys(obj);\n const len = keys.length;\n let key;\n\n for (i = 0; i < len; i++) {\n key = keys[i];\n fn.call(null, obj[key], key, obj);\n }\n }\n}\n\nfunction findKey(obj, key) {\n key = key.toLowerCase();\n const keys = Object.keys(obj);\n let i = keys.length;\n let _key;\n while (i-- > 0) {\n _key = keys[i];\n if (key === _key.toLowerCase()) {\n return _key;\n }\n }\n return null;\n}\n\nconst _global = (() => {\n /*eslint no-undef:0*/\n if (typeof globalThis !== \"undefined\") return globalThis;\n return typeof self !== \"undefined\" ? self : (typeof window !== 'undefined' ? window : global)\n})();\n\nconst isContextDefined = (context) => !isUndefined(context) && context !== _global;\n\n/**\n * Accepts varargs expecting each argument to be an object, then\n * immutably merges the properties of each object and returns result.\n *\n * When multiple objects contain the same key the later object in\n * the arguments list will take precedence.\n *\n * Example:\n *\n * ```js\n * var result = merge({foo: 123}, {foo: 456});\n * console.log(result.foo); // outputs 456\n * ```\n *\n * @param {Object} obj1 Object to merge\n *\n * @returns {Object} Result of all merge properties\n */\nfunction merge(/* obj1, obj2, obj3, ... */) {\n const {caseless} = isContextDefined(this) && this || {};\n const result = {};\n const assignValue = (val, key) => {\n const targetKey = caseless && findKey(result, key) || key;\n if (isPlainObject(result[targetKey]) && isPlainObject(val)) {\n result[targetKey] = merge(result[targetKey], val);\n } else if (isPlainObject(val)) {\n result[targetKey] = merge({}, val);\n } else if (isArray(val)) {\n result[targetKey] = val.slice();\n } else {\n result[targetKey] = val;\n }\n }\n\n for (let i = 0, l = arguments.length; i < l; i++) {\n arguments[i] && forEach(arguments[i], assignValue);\n }\n return result;\n}\n\n/**\n * Extends object a by mutably adding to it the properties of object b.\n *\n * @param {Object} a The object to be extended\n * @param {Object} b The object to copy properties from\n * @param {Object} thisArg The object to bind function to\n *\n * @param {Boolean} [allOwnKeys]\n * @returns {Object} The resulting value of object a\n */\nconst extend = (a, b, thisArg, {allOwnKeys}= {}) => {\n forEach(b, (val, key) => {\n if (thisArg && isFunction(val)) {\n a[key] = bind(val, thisArg);\n } else {\n a[key] = val;\n }\n }, {allOwnKeys});\n return a;\n}\n\n/**\n * Remove byte order marker. This catches EF BB BF (the UTF-8 BOM)\n *\n * @param {string} content with BOM\n *\n * @returns {string} content value without BOM\n */\nconst stripBOM = (content) => {\n if (content.charCodeAt(0) === 0xFEFF) {\n content = content.slice(1);\n }\n return content;\n}\n\n/**\n * Inherit the prototype methods from one constructor into another\n * @param {function} constructor\n * @param {function} superConstructor\n * @param {object} [props]\n * @param {object} [descriptors]\n *\n * @returns {void}\n */\nconst inherits = (constructor, superConstructor, props, descriptors) => {\n constructor.prototype = Object.create(superConstructor.prototype, descriptors);\n constructor.prototype.constructor = constructor;\n Object.defineProperty(constructor, 'super', {\n value: superConstructor.prototype\n });\n props && Object.assign(constructor.prototype, props);\n}\n\n/**\n * Resolve object with deep prototype chain to a flat object\n * @param {Object} sourceObj source object\n * @param {Object} [destObj]\n * @param {Function|Boolean} [filter]\n * @param {Function} [propFilter]\n *\n * @returns {Object}\n */\nconst toFlatObject = (sourceObj, destObj, filter, propFilter) => {\n let props;\n let i;\n let prop;\n const merged = {};\n\n destObj = destObj || {};\n // eslint-disable-next-line no-eq-null,eqeqeq\n if (sourceObj == null) return destObj;\n\n do {\n props = Object.getOwnPropertyNames(sourceObj);\n i = props.length;\n while (i-- > 0) {\n prop = props[i];\n if ((!propFilter || propFilter(prop, sourceObj, destObj)) && !merged[prop]) {\n destObj[prop] = sourceObj[prop];\n merged[prop] = true;\n }\n }\n sourceObj = filter !== false && getPrototypeOf(sourceObj);\n } while (sourceObj && (!filter || filter(sourceObj, destObj)) && sourceObj !== Object.prototype);\n\n return destObj;\n}\n\n/**\n * Determines whether a string ends with the characters of a specified string\n *\n * @param {String} str\n * @param {String} searchString\n * @param {Number} [position= 0]\n *\n * @returns {boolean}\n */\nconst endsWith = (str, searchString, position) => {\n str = String(str);\n if (position === undefined || position > str.length) {\n position = str.length;\n }\n position -= searchString.length;\n const lastIndex = str.indexOf(searchString, position);\n return lastIndex !== -1 && lastIndex === position;\n}\n\n\n/**\n * Returns new array from array like object or null if failed\n *\n * @param {*} [thing]\n *\n * @returns {?Array}\n */\nconst toArray = (thing) => {\n if (!thing) return null;\n if (isArray(thing)) return thing;\n let i = thing.length;\n if (!isNumber(i)) return null;\n const arr = new Array(i);\n while (i-- > 0) {\n arr[i] = thing[i];\n }\n return arr;\n}\n\n/**\n * Checking if the Uint8Array exists and if it does, it returns a function that checks if the\n * thing passed in is an instance of Uint8Array\n *\n * @param {TypedArray}\n *\n * @returns {Array}\n */\n// eslint-disable-next-line func-names\nconst isTypedArray = (TypedArray => {\n // eslint-disable-next-line func-names\n return thing => {\n return TypedArray && thing instanceof TypedArray;\n };\n})(typeof Uint8Array !== 'undefined' && getPrototypeOf(Uint8Array));\n\n/**\n * For each entry in the object, call the function with the key and value.\n *\n * @param {Object<any, any>} obj - The object to iterate over.\n * @param {Function} fn - The function to call for each entry.\n *\n * @returns {void}\n */\nconst forEachEntry = (obj, fn) => {\n const generator = obj && obj[iterator];\n\n const _iterator = generator.call(obj);\n\n let result;\n\n while ((result = _iterator.next()) && !result.done) {\n const pair = result.value;\n fn.call(obj, pair[0], pair[1]);\n }\n}\n\n/**\n * It takes a regular expression and a string, and returns an array of all the matches\n *\n * @param {string} regExp - The regular expression to match against.\n * @param {string} str - The string to search.\n *\n * @returns {Array<boolean>}\n */\nconst matchAll = (regExp, str) => {\n let matches;\n const arr = [];\n\n while ((matches = regExp.exec(str)) !== null) {\n arr.push(matches);\n }\n\n return arr;\n}\n\n/* Checking if the kindOfTest function returns true when passed an HTMLFormElement. */\nconst isHTMLForm = kindOfTest('HTMLFormElement');\n\nconst toCamelCase = str => {\n return str.toLowerCase().replace(/[-_\\s]([a-z\\d])(\\w*)/g,\n function replacer(m, p1, p2) {\n return p1.toUpperCase() + p2;\n }\n );\n};\n\n/* Creating a function that will check if an object has a property. */\nconst hasOwnProperty = (({hasOwnProperty}) => (obj, prop) => hasOwnProperty.call(obj, prop))(Object.prototype);\n\n/**\n * Determine if a value is a RegExp object\n *\n * @param {*} val The value to test\n *\n * @returns {boolean} True if value is a RegExp object, otherwise false\n */\nconst isRegExp = kindOfTest('RegExp');\n\nconst reduceDescriptors = (obj, reducer) => {\n const descriptors = Object.getOwnPropertyDescriptors(obj);\n const reducedDescriptors = {};\n\n forEach(descriptors, (descriptor, name) => {\n let ret;\n if ((ret = reducer(descriptor, name, obj)) !== false) {\n reducedDescriptors[name] = ret || descriptor;\n }\n });\n\n Object.defineProperties(obj, reducedDescriptors);\n}\n\n/**\n * Makes all methods read-only\n * @param {Object} obj\n */\n\nconst freezeMethods = (obj) => {\n reduceDescriptors(obj, (descriptor, name) => {\n // skip restricted props in strict mode\n if (isFunction(obj) && ['arguments', 'caller', 'callee'].indexOf(name) !== -1) {\n return false;\n }\n\n const value = obj[name];\n\n if (!isFunction(value)) return;\n\n descriptor.enumerable = false;\n\n if ('writable' in descriptor) {\n descriptor.writable = false;\n return;\n }\n\n if (!descriptor.set) {\n descriptor.set = () => {\n throw Error('Can not rewrite read-only method \\'' + name + '\\'');\n };\n }\n });\n}\n\nconst toObjectSet = (arrayOrString, delimiter) => {\n const obj = {};\n\n const define = (arr) => {\n arr.forEach(value => {\n obj[value] = true;\n });\n }\n\n isArray(arrayOrString) ? define(arrayOrString) : define(String(arrayOrString).split(delimiter));\n\n return obj;\n}\n\nconst noop = () => {}\n\nconst toFiniteNumber = (value, defaultValue) => {\n return value != null && Number.isFinite(value = +value) ? value : defaultValue;\n}\n\n/**\n * If the thing is a FormData object, return true, otherwise return false.\n *\n * @param {unknown} thing - The thing to check.\n *\n * @returns {boolean}\n */\nfunction isSpecCompliantForm(thing) {\n return !!(thing && isFunction(thing.append) && thing[toStringTag] === 'FormData' && thing[iterator]);\n}\n\nconst toJSONObject = (obj) => {\n const stack = new Array(10);\n\n const visit = (source, i) => {\n\n if (isObject(source)) {\n if (stack.indexOf(source) >= 0) {\n return;\n }\n\n if(!('toJSON' in source)) {\n stack[i] = source;\n const target = isArray(source) ? [] : {};\n\n forEach(source, (value, key) => {\n const reducedValue = visit(value, i + 1);\n !isUndefined(reducedValue) && (target[key] = reducedValue);\n });\n\n stack[i] = undefined;\n\n return target;\n }\n }\n\n return source;\n }\n\n return visit(obj, 0);\n}\n\nconst isAsyncFn = kindOfTest('AsyncFunction');\n\nconst isThenable = (thing) =>\n thing && (isObject(thing) || isFunction(thing)) && isFunction(thing.then) && isFunction(thing.catch);\n\n// original code\n// https://github.com/DigitalBrainJS/AxiosPromise/blob/16deab13710ec09779922131f3fa5954320f83ab/lib/utils.js#L11-L34\n\nconst _setImmediate = ((setImmediateSupported, postMessageSupported) => {\n if (setImmediateSupported) {\n return setImmediate;\n }\n\n return postMessageSupported ? ((token, callbacks) => {\n _global.addEventListener(\"message\", ({source, data}) => {\n if (source === _global && data === token) {\n callbacks.length && callbacks.shift()();\n }\n }, false);\n\n return (cb) => {\n callbacks.push(cb);\n _global.postMessage(token, \"*\");\n }\n })(`axios@${Math.random()}`, []) : (cb) => setTimeout(cb);\n})(\n typeof setImmediate === 'function',\n isFunction(_global.postMessage)\n);\n\nconst asap = typeof queueMicrotask !== 'undefined' ?\n queueMicrotask.bind(_global) : ( typeof process !== 'undefined' && process.nextTick || _setImmediate);\n\n// *********************\n\n\nconst isIterable = (thing) => thing != null && isFunction(thing[iterator]);\n\n\nexport default {\n isArray,\n isArrayBuffer,\n isBuffer,\n isFormData,\n isArrayBufferView,\n isString,\n isNumber,\n isBoolean,\n isObject,\n isPlainObject,\n isReadableStream,\n isRequest,\n isResponse,\n isHeaders,\n isUndefined,\n isDate,\n isFile,\n isBlob,\n isRegExp,\n isFunction,\n isStream,\n isURLSearchParams,\n isTypedArray,\n isFileList,\n forEach,\n merge,\n extend,\n trim,\n stripBOM,\n inherits,\n toFlatObject,\n kindOf,\n kindOfTest,\n endsWith,\n toArray,\n forEachEntry,\n matchAll,\n isHTMLForm,\n hasOwnProperty,\n hasOwnProp: hasOwnProperty, // an alias to avoid ESLint no-prototype-builtins detection\n reduceDescriptors,\n freezeMethods,\n toObjectSet,\n toCamelCase,\n noop,\n toFiniteNumber,\n findKey,\n global: _global,\n isContextDefined,\n isSpecCompliantForm,\n toJSONObject,\n isAsyncFn,\n isThenable,\n setImmediate: _setImmediate,\n asap,\n isIterable\n};\n","'use strict';\n\nimport utils from '../utils.js';\n\n/**\n * Create an Error with the specified message, config, error code, request and response.\n *\n * @param {string} message The error message.\n * @param {string} [code] The error code (for example, 'ECONNABORTED').\n * @param {Object} [config] The config.\n * @param {Object} [request] The request.\n * @param {Object} [response] The response.\n *\n * @returns {Error} The created error.\n */\nfunction AxiosError(message, code, config, request, response) {\n Error.call(this);\n\n if (Error.captureStackTrace) {\n Error.captureStackTrace(this, this.constructor);\n } else {\n this.stack = (new Error()).stack;\n }\n\n this.message = message;\n this.name = 'AxiosError';\n code && (this.code = code);\n config && (this.config = config);\n request && (this.request = request);\n if (response) {\n this.response = response;\n this.status = response.status ? response.status : null;\n }\n}\n\nutils.inherits(AxiosError, Error, {\n toJSON: function toJSON() {\n return {\n // Standard\n message: this.message,\n name: this.name,\n // Microsoft\n description: this.description,\n number: this.number,\n // Mozilla\n fileName: this.fileName,\n lineNumber: this.lineNumber,\n columnNumber: this.columnNumber,\n stack: this.stack,\n // Axios\n config: utils.toJSONObject(this.config),\n code: this.code,\n status: this.status\n };\n }\n});\n\nconst prototype = AxiosError.prototype;\nconst descriptors = {};\n\n[\n 'ERR_BAD_OPTION_VALUE',\n 'ERR_BAD_OPTION',\n 'ECONNABORTED',\n 'ETIMEDOUT',\n 'ERR_NETWORK',\n 'ERR_FR_TOO_MANY_REDIRECTS',\n 'ERR_DEPRECATED',\n 'ERR_BAD_RESPONSE',\n 'ERR_BAD_REQUEST',\n 'ERR_CANCELED',\n 'ERR_NOT_SUPPORT',\n 'ERR_INVALID_URL'\n// eslint-disable-next-line func-names\n].forEach(code => {\n descriptors[code] = {value: code};\n});\n\nObject.defineProperties(AxiosError, descriptors);\nObject.defineProperty(prototype, 'isAxiosError', {value: true});\n\n// eslint-disable-next-line func-names\nAxiosError.from = (error, code, config, request, response, customProps) => {\n const axiosError = Object.create(prototype);\n\n utils.toFlatObject(error, axiosError, function filter(obj) {\n return obj !== Error.prototype;\n }, prop => {\n return prop !== 'isAxiosError';\n });\n\n AxiosError.call(axiosError, error.message, code, config, request, response);\n\n axiosError.cause = error;\n\n axiosError.name = error.name;\n\n customProps && Object.assign(axiosError, customProps);\n\n return axiosError;\n};\n\nexport default AxiosError;\n","// eslint-disable-next-line strict\nexport default null;\n","'use strict';\n\nimport utils from '../utils.js';\nimport AxiosError from '../core/AxiosError.js';\n// temporary hotfix to avoid circular references until AxiosURLSearchParams is refactored\nimport PlatformFormData from '../platform/node/classes/FormData.js';\n\n/**\n * Determines if the given thing is a array or js object.\n *\n * @param {string} thing - The object or array to be visited.\n *\n * @returns {boolean}\n */\nfunction isVisitable(thing) {\n return utils.isPlainObject(thing) || utils.isArray(thing);\n}\n\n/**\n * It removes the brackets from the end of a string\n *\n * @param {string} key - The key of the parameter.\n *\n * @returns {string} the key without the brackets.\n */\nfunction removeBrackets(key) {\n return utils.endsWith(key, '[]') ? key.slice(0, -2) : key;\n}\n\n/**\n * It takes a path, a key, and a boolean, and returns a string\n *\n * @param {string} path - The path to the current key.\n * @param {string} key - The key of the current object being iterated over.\n * @param {string} dots - If true, the key will be rendered with dots instead of brackets.\n *\n * @returns {string} The path to the current key.\n */\nfunction renderKey(path, key, dots) {\n if (!path) return key;\n return path.concat(key).map(function each(token, i) {\n // eslint-disable-next-line no-param-reassign\n token = removeBrackets(token);\n return !dots && i ? '[' + token + ']' : token;\n }).join(dots ? '.' : '');\n}\n\n/**\n * If the array is an array and none of its elements are visitable, then it's a flat array.\n *\n * @param {Array<any>} arr - The array to check\n *\n * @returns {boolean}\n */\nfunction isFlatArray(arr) {\n return utils.isArray(arr) && !arr.some(isVisitable);\n}\n\nconst predicates = utils.toFlatObject(utils, {}, null, function filter(prop) {\n return /^is[A-Z]/.test(prop);\n});\n\n/**\n * Convert a data object to FormData\n *\n * @param {Object} obj\n * @param {?Object} [formData]\n * @param {?Object} [options]\n * @param {Function} [options.visitor]\n * @param {Boolean} [options.metaTokens = true]\n * @param {Boolean} [options.dots = false]\n * @param {?Boolean} [options.indexes = false]\n *\n * @returns {Object}\n **/\n\n/**\n * It converts an object into a FormData object\n *\n * @param {Object<any, any>} obj - The object to convert to form data.\n * @param {string} formData - The FormData object to append to.\n * @param {Object<string, any>} options\n *\n * @returns\n */\nfunction toFormData(obj, formData, options) {\n if (!utils.isObject(obj)) {\n throw new TypeError('target must be an object');\n }\n\n // eslint-disable-next-line no-param-reassign\n formData = formData || new (PlatformFormData || FormData)();\n\n // eslint-disable-next-line no-param-reassign\n options = utils.toFlatObject(options, {\n metaTokens: true,\n dots: false,\n indexes: false\n }, false, function defined(option, source) {\n // eslint-disable-next-line no-eq-null,eqeqeq\n return !utils.isUndefined(source[option]);\n });\n\n const metaTokens = options.metaTokens;\n // eslint-disable-next-line no-use-before-define\n const visitor = options.visitor || defaultVisitor;\n const dots = options.dots;\n const indexes = options.indexes;\n const _Blob = options.Blob || typeof Blob !== 'undefined' && Blob;\n const useBlob = _Blob && utils.isSpecCompliantForm(formData);\n\n if (!utils.isFunction(visitor)) {\n throw new TypeError('visitor must be a function');\n }\n\n function convertValue(value) {\n if (value === null) return '';\n\n if (utils.isDate(value)) {\n return value.toISOString();\n }\n\n if (utils.isBoolean(value)) {\n return value.toString();\n }\n\n if (!useBlob && utils.isBlob(value)) {\n throw new AxiosError('Blob is not supported. Use a Buffer instead.');\n }\n\n if (utils.isArrayBuffer(value) || utils.isTypedArray(value)) {\n return useBlob && typeof Blob === 'function' ? new Blob([value]) : Buffer.from(value);\n }\n\n return value;\n }\n\n /**\n * Default visitor.\n *\n * @param {*} value\n * @param {String|Number} key\n * @param {Array<String|Number>} path\n * @this {FormData}\n *\n * @returns {boolean} return true to visit the each prop of the value recursively\n */\n function defaultVisitor(value, key, path) {\n let arr = value;\n\n if (value && !path && typeof value === 'object') {\n if (utils.endsWith(key, '{}')) {\n // eslint-disable-next-line no-param-reassign\n key = metaTokens ? key : key.slice(0, -2);\n // eslint-disable-next-line no-param-reassign\n value = JSON.stringify(value);\n } else if (\n (utils.isArray(value) && isFlatArray(value)) ||\n ((utils.isFileList(value) || utils.endsWith(key, '[]')) && (arr = utils.toArray(value))\n )) {\n // eslint-disable-next-line no-param-reassign\n key = removeBrackets(key);\n\n arr.forEach(function each(el, index) {\n !(utils.isUndefined(el) || el === null) && formData.append(\n // eslint-disable-next-line no-nested-ternary\n indexes === true ? renderKey([key], index, dots) : (indexes === null ? key : key + '[]'),\n convertValue(el)\n );\n });\n return false;\n }\n }\n\n if (isVisitable(value)) {\n return true;\n }\n\n formData.append(renderKey(path, key, dots), convertValue(value));\n\n return false;\n }\n\n const stack = [];\n\n const exposedHelpers = Object.assign(predicates, {\n defaultVisitor,\n convertValue,\n isVisitable\n });\n\n function build(value, path) {\n if (utils.isUndefined(value)) return;\n\n if (stack.indexOf(value) !== -1) {\n throw Error('Circular reference detected in ' + path.join('.'));\n }\n\n stack.push(value);\n\n utils.forEach(value, function each(el, key) {\n const result = !(utils.isUndefined(el) || el === null) && visitor.call(\n formData, el, utils.isString(key) ? key.trim() : key, path, exposedHelpers\n );\n\n if (result === true) {\n build(el, path ? path.concat(key) : [key]);\n }\n });\n\n stack.pop();\n }\n\n if (!utils.isObject(obj)) {\n throw new TypeError('data must be an object');\n }\n\n build(obj);\n\n return formData;\n}\n\nexport default toFormData;\n","'use strict';\n\nimport toFormData from './toFormData.js';\n\n/**\n * It encodes a string by replacing all characters that are not in the unreserved set with\n * their percent-encoded equivalents\n *\n * @param {string} str - The string to encode.\n *\n * @returns {string} The encoded string.\n */\nfunction encode(str) {\n const charMap = {\n '!': '%21',\n \"'\": '%27',\n '(': '%28',\n ')': '%29',\n '~': '%7E',\n '%20': '+',\n '%00': '\\x00'\n };\n return encodeURIComponent(str).replace(/[!'()~]|%20|%00/g, function replacer(match) {\n return charMap[match];\n });\n}\n\n/**\n * It takes a params object and converts it to a FormData object\n *\n * @param {Object<string, any>} params - The parameters to be converted to a FormData object.\n * @param {Object<string, any>} options - The options object passed to the Axios constructor.\n *\n * @returns {void}\n */\nfunction AxiosURLSearchParams(params, options) {\n this._pairs = [];\n\n params && toFormData(params, this, options);\n}\n\nconst prototype = AxiosURLSearchParams.prototype;\n\nprototype.append = function append(name, value) {\n this._pairs.push([name, value]);\n};\n\nprototype.toString = function toString(encoder) {\n const _encode = encoder ? function(value) {\n return encoder.call(this, value, encode);\n } : encode;\n\n return this._pairs.map(function each(pair) {\n return _encode(pair[0]) + '=' + _encode(pair[1]);\n }, '').join('&');\n};\n\nexport default AxiosURLSearchParams;\n","'use strict';\n\nimport utils from '../utils.js';\nimport AxiosURLSearchParams from '../helpers/AxiosURLSearchParams.js';\n\n/**\n * It replaces all instances of the characters `:`, `$`, `,`, `+`, `[`, and `]` with their\n * URI encoded counterparts\n *\n * @param {string} val The value to be encoded.\n *\n * @returns {string} The encoded value.\n */\nfunction encode(val) {\n return encodeURIComponent(val).\n replace(/%3A/gi, ':').\n replace(/%24/g, '$').\n replace(/%2C/gi, ',').\n replace(/%20/g, '+').\n replace(/%5B/gi, '[').\n replace(/%5D/gi, ']');\n}\n\n/**\n * Build a URL by appending params to the end\n *\n * @param {string} url The base of the url (e.g., http://www.google.com)\n * @param {object} [params] The params to be appended\n * @param {?(object|Function)} options\n *\n * @returns {string} The formatted url\n */\nexport default function buildURL(url, params, options) {\n /*eslint no-param-reassign:0*/\n if (!params) {\n return url;\n }\n \n const _encode = options && options.encode || encode;\n\n if (utils.isFunction(options)) {\n options = {\n serialize: options\n };\n } \n\n const serializeFn = options && options.serialize;\n\n let serializedParams;\n\n if (serializeFn) {\n serializedParams = serializeFn(params, options);\n } else {\n serializedParams = utils.isURLSearchParams(params) ?\n params.toString() :\n new AxiosURLSearchParams(params, options).toString(_encode);\n }\n\n if (serializedParams) {\n const hashmarkIndex = url.indexOf(\"#\");\n\n if (hashmarkIndex !== -1) {\n url = url.slice(0, hashmarkIndex);\n }\n url += (url.indexOf('?') === -1 ? '?' : '&') + serializedParams;\n }\n\n return url;\n}\n","'use strict';\n\nimport utils from './../utils.js';\n\nclass InterceptorManager {\n constructor() {\n this.handlers = [];\n }\n\n /**\n * Add a new interceptor to the stack\n *\n * @param {Function} fulfilled The function to handle `then` for a `Promise`\n * @param {Function} rejected The function to handle `reject` for a `Promise`\n *\n * @return {Number} An ID used to remove interceptor later\n */\n use(fulfilled, rejected, options) {\n this.handlers.push({\n fulfilled,\n rejected,\n synchronous: options ? options.synchronous : false,\n runWhen: options ? options.runWhen : null\n });\n return this.handlers.length - 1;\n }\n\n /**\n * Remove an interceptor from the stack\n *\n * @param {Number} id The ID that was returned by `use`\n *\n * @returns {Boolean} `true` if the interceptor was removed, `false` otherwise\n */\n eject(id) {\n if (this.handlers[id]) {\n this.handlers[id] = null;\n }\n }\n\n /**\n * Clear all interceptors from the stack\n *\n * @returns {void}\n */\n clear() {\n if (this.handlers) {\n this.handlers = [];\n }\n }\n\n /**\n * Iterate over all the registered interceptors\n *\n * This method is particularly useful for skipping over any\n * interceptors that may have become `null` calling `eject`.\n *\n * @param {Function} fn The function to call for each interceptor\n *\n * @returns {void}\n */\n forEach(fn) {\n utils.forEach(this.handlers, function forEachHandler(h) {\n if (h !== null) {\n fn(h);\n }\n });\n }\n}\n\nexport default InterceptorManager;\n","'use strict';\n\nexport default {\n silentJSONParsing: true,\n forcedJSONParsing: true,\n clarifyTimeoutError: false\n};\n","'use strict';\n\nimport AxiosURLSearchParams from '../../../helpers/AxiosURLSearchParams.js';\nexport default typeof URLSearchParams !== 'undefined' ? URLSearchParams : AxiosURLSearchParams;\n","'use strict';\n\nexport default typeof FormData !== 'undefined' ? FormData : null;\n","'use strict'\n\nexport default typeof Blob !== 'undefined' ? Blob : null\n","import URLSearchParams from './classes/URLSearchParams.js'\nimport FormData from './classes/FormData.js'\nimport Blob from './classes/Blob.js'\n\nexport default {\n isBrowser: true,\n classes: {\n URLSearchParams,\n FormData,\n Blob\n },\n protocols: ['http', 'https', 'file', 'blob', 'url', 'data']\n};\n","const hasBrowserEnv = typeof window !== 'undefined' && typeof document !== 'undefined';\n\nconst _navigator = typeof navigator === 'object' && navigator || undefined;\n\n/**\n * Determine if we're running in a standard browser environment\n *\n * This allows axios to run in a web worker, and react-native.\n * Both environments support XMLHttpRequest, but not fully standard globals.\n *\n * web workers:\n * typeof window -> undefined\n * typeof document -> undefined\n *\n * react-native:\n * navigator.product -> 'ReactNative'\n * nativescript\n * navigator.product -> 'NativeScript' or 'NS'\n *\n * @returns {boolean}\n */\nconst hasStandardBrowserEnv = hasBrowserEnv &&\n (!_navigator || ['ReactNative', 'NativeScript', 'NS'].indexOf(_navigator.product) < 0);\n\n/**\n * Determine if we're running in a standard browser webWorker environment\n *\n * Although the `isStandardBrowserEnv` method indicates that\n * `allows axios to run in a web worker`, the WebWorker will still be\n * filtered out due to its judgment standard\n * `typeof window !== 'undefined' && typeof document !== 'undefined'`.\n * This leads to a problem when axios post `FormData` in webWorker\n */\nconst hasStandardBrowserWebWorkerEnv = (() => {\n return (\n typeof WorkerGlobalScope !== 'undefined' &&\n // eslint-disable-next-line no-undef\n self instanceof WorkerGlobalScope &&\n typeof self.importScripts === 'function'\n );\n})();\n\nconst origin = hasBrowserEnv && window.location.href || 'http://localhost';\n\nexport {\n hasBrowserEnv,\n hasStandardBrowserWebWorkerEnv,\n hasStandardBrowserEnv,\n _navigator as navigator,\n origin\n}\n","import platform from './node/index.js';\nimport * as utils from './common/utils.js';\n\nexport default {\n ...utils,\n ...platform\n}\n","'use strict';\n\nimport utils from '../utils.js';\nimport toFormData from './toFormData.js';\nimport platform from '../platform/index.js';\n\nexport default function toURLEncodedForm(data, options) {\n return toFormData(data, new platform.classes.URLSearchParams(), Object.assign({\n visitor: function(value, key, path, helpers) {\n if (platform.isNode && utils.isBuffer(value)) {\n this.append(key, value.toString('base64'));\n return false;\n }\n\n return helpers.defaultVisitor.apply(this, arguments);\n }\n }, options));\n}\n","'use strict';\n\nimport utils from '../utils.js';\n\n/**\n * It takes a string like `foo[x][y][z]` and returns an array like `['foo', 'x', 'y', 'z']\n *\n * @param {string} name - The name of the property to get.\n *\n * @returns An array of strings.\n */\nfunction parsePropPath(name) {\n // foo[x][y][z]\n // foo.x.y.z\n // foo-x-y-z\n // foo x y z\n return utils.matchAll(/\\w+|\\[(\\w*)]/g, name).map(match => {\n return match[0] === '[]' ? '' : match[1] || match[0];\n });\n}\n\n/**\n * Convert an array to an object.\n *\n * @param {Array<any>} arr - The array to convert to an object.\n *\n * @returns An object with the same keys and values as the array.\n */\nfunction arrayToObject(arr) {\n const obj = {};\n const keys = Object.keys(arr);\n let i;\n const len = keys.length;\n let key;\n for (i = 0; i < len; i++) {\n key = keys[i];\n obj[key] = arr[key];\n }\n return obj;\n}\n\n/**\n * It takes a FormData object and returns a JavaScript object\n *\n * @param {string} formData The FormData object to convert to JSON.\n *\n * @returns {Object<string, any> | null} The converted object.\n */\nfunction formDataToJSON(formData) {\n function buildPath(path, value, target, index) {\n let name = path[index++];\n\n if (name === '__proto__') return true;\n\n const isNumericKey = Number.isFinite(+name);\n const isLast = index >= path.length;\n name = !name && utils.isArray(target) ? target.length : name;\n\n if (isLast) {\n if (utils.hasOwnProp(target, name)) {\n target[name] = [target[name], value];\n } else {\n target[name] = value;\n }\n\n return !isNumericKey;\n }\n\n if (!target[name] || !utils.isObject(target[name])) {\n target[name] = [];\n }\n\n const result = buildPath(path, value, target[name], index);\n\n if (result && utils.isArray(target[name])) {\n target[name] = arrayToObject(target[name]);\n }\n\n return !isNumericKey;\n }\n\n if (utils.isFormData(formData) && utils.isFunction(formData.entries)) {\n const obj = {};\n\n utils.forEachEntry(formData, (name, value) => {\n buildPath(parsePropPath(name), value, obj, 0);\n });\n\n return obj;\n }\n\n return null;\n}\n\nexport default formDataToJSON;\n","'use strict';\n\nimport utils from '../utils.js';\nimport AxiosError from '../core/AxiosError.js';\nimport transitionalDefaults from './transitional.js';\nimport toFormData from '../helpers/toFormData.js';\nimport toURLEncodedForm from '../helpers/toURLEncodedForm.js';\nimport platform from '../platform/index.js';\nimport formDataToJSON from '../helpers/formDataToJSON.js';\n\n/**\n * It takes a string, tries to parse it, and if it fails, it returns the stringified version\n * of the input\n *\n * @param {any} rawValue - The value to be stringified.\n * @param {Function} parser - A function that parses a string into a JavaScript object.\n * @param {Function} encoder - A function that takes a value and returns a string.\n *\n * @returns {string} A stringified version of the rawValue.\n */\nfunction stringifySafely(rawValue, parser, encoder) {\n if (utils.isString(rawValue)) {\n try {\n (parser || JSON.parse)(rawValue);\n return utils.trim(rawValue);\n } catch (e) {\n if (e.name !== 'SyntaxError') {\n throw e;\n }\n }\n }\n\n return (encoder || JSON.stringify)(rawValue);\n}\n\nconst defaults = {\n\n transitional: transitionalDefaults,\n\n adapter: ['xhr', 'http', 'fetch'],\n\n transformRequest: [function transformRequest(data, headers) {\n const contentType = headers.getContentType() || '';\n const hasJSONContentType = contentType.indexOf('application/json') > -1;\n const isObjectPayload = utils.isObject(data);\n\n if (isObjectPayload && utils.isHTMLForm(data)) {\n data = new FormData(data);\n }\n\n const isFormData = utils.isFormData(data);\n\n if (isFormData) {\n return hasJSONContentType ? JSON.stringify(formDataToJSON(data)) : data;\n }\n\n if (utils.isArrayBuffer(data) ||\n utils.isBuffer(data) ||\n utils.isStream(data) ||\n utils.isFile(data) ||\n utils.isBlob(data) ||\n utils.isReadableStream(data)\n ) {\n return data;\n }\n if (utils.isArrayBufferView(data)) {\n return data.buffer;\n }\n if (utils.isURLSearchParams(data)) {\n headers.setContentType('application/x-www-form-urlencoded;charset=utf-8', false);\n return data.toString();\n }\n\n let isFileList;\n\n if (isObjectPayload) {\n if (contentType.indexOf('application/x-www-form-urlencoded') > -1) {\n return toURLEncodedForm(data, this.formSerializer).toString();\n }\n\n if ((isFileList = utils.isFileList(data)) || contentType.indexOf('multipart/form-data') > -1) {\n const _FormData = this.env && this.env.FormData;\n\n return toFormData(\n isFileList ? {'files[]': data} : data,\n _FormData && new _FormData(),\n this.formSerializer\n );\n }\n }\n\n if (isObjectPayload || hasJSONContentType ) {\n headers.setContentType('application/json', false);\n return stringifySafely(data);\n }\n\n return data;\n }],\n\n transformResponse: [function transformResponse(data) {\n const transitional = this.transitional || defaults.transitional;\n const forcedJSONParsing = transitional && transitional.forcedJSONParsing;\n const JSONRequested = this.responseType === 'json';\n\n if (utils.isResponse(data) || utils.isReadableStream(data)) {\n return data;\n }\n\n if (data && utils.isString(data) && ((forcedJSONParsing && !this.responseType) || JSONRequested)) {\n const silentJSONParsing = transitional && transitional.silentJSONParsing;\n const strictJSONParsing = !silentJSONParsing && JSONRequested;\n\n try {\n return JSON.parse(data);\n } catch (e) {\n if (strictJSONParsing) {\n if (e.name === 'SyntaxError') {\n throw AxiosError.from(e, AxiosError.ERR_BAD_RESPONSE, this, null, this.response);\n }\n throw e;\n }\n }\n }\n\n return data;\n }],\n\n /**\n * A timeout in milliseconds to abort a request. If set to 0 (default) a\n * timeout is not created.\n */\n timeout: 0,\n\n xsrfCookieName: 'XSRF-TOKEN',\n xsrfHeaderName: 'X-XSRF-TOKEN',\n\n maxContentLength: -1,\n maxBodyLength: -1,\n\n env: {\n FormData: platform.classes.FormData,\n Blob: platform.classes.Blob\n },\n\n validateStatus: function validateStatus(status) {\n return status >= 200 && status < 300;\n },\n\n headers: {\n common: {\n 'Accept': 'application/json, text/plain, */*',\n 'Content-Type': undefined\n }\n }\n};\n\nutils.forEach(['delete', 'get', 'head', 'post', 'put', 'patch'], (method) => {\n defaults.headers[method] = {};\n});\n\nexport default defaults;\n","'use strict';\n\nimport utils from './../utils.js';\n\n// RawAxiosHeaders whose duplicates are ignored by node\n// c.f. https://nodejs.org/api/http.html#http_message_headers\nconst ignoreDuplicateOf = utils.toObjectSet([\n 'age', 'authorization', 'content-length', 'content-type', 'etag',\n 'expires', 'from', 'host', 'if-modified-since', 'if-unmodified-since',\n 'last-modified', 'location', 'max-forwards', 'proxy-authorization',\n 'referer', 'retry-after', 'user-agent'\n]);\n\n/**\n * Parse headers into an object\n *\n * ```\n * Date: Wed, 27 Aug 2014 08:58:49 GMT\n * Content-Type: application/json\n * Connection: keep-alive\n * Transfer-Encoding: chunked\n * ```\n *\n * @param {String} rawHeaders Headers needing to be parsed\n *\n * @returns {Object} Headers parsed into an object\n */\nexport default rawHeaders => {\n const parsed = {};\n let key;\n let val;\n let i;\n\n rawHeaders && rawHeaders.split('\\n').forEach(function parser(line) {\n i = line.indexOf(':');\n key = line.substring(0, i).trim().toLowerCase();\n val = line.substring(i + 1).trim();\n\n if (!key || (parsed[key] && ignoreDuplicateOf[key])) {\n return;\n }\n\n if (key === 'set-cookie') {\n if (parsed[key]) {\n parsed[key].push(val);\n } else {\n parsed[key] = [val];\n }\n } else {\n parsed[key] = parsed[key] ? parsed[key] + ', ' + val : val;\n }\n });\n\n return parsed;\n};\n","'use strict';\n\nimport utils from '../utils.js';\nimport parseHeaders from '../helpers/parseHeaders.js';\n\nconst $internals = Symbol('internals');\n\nfunction normalizeHeader(header) {\n return header && String(header).trim().toLowerCase();\n}\n\nfunction normalizeValue(value) {\n if (value === false || value == null) {\n return value;\n }\n\n return utils.isArray(value) ? value.map(normalizeValue) : String(value);\n}\n\nfunction parseTokens(str) {\n const tokens = Object.create(null);\n const tokensRE = /([^\\s,;=]+)\\s*(?:=\\s*([^,;]+))?/g;\n let match;\n\n while ((match = tokensRE.exec(str))) {\n tokens[match[1]] = match[2];\n }\n\n return tokens;\n}\n\nconst isValidHeaderName = (str) => /^[-_a-zA-Z0-9^`|~,!#$%&'*+.]+$/.test(str.trim());\n\nfunction matchHeaderValue(context, value, header, filter, isHeaderNameFilter) {\n if (utils.isFunction(filter)) {\n return filter.call(this, value, header);\n }\n\n if (isHeaderNameFilter) {\n value = header;\n }\n\n if (!utils.isString(value)) return;\n\n if (utils.isString(filter)) {\n return value.indexOf(filter) !== -1;\n }\n\n if (utils.isRegExp(filter)) {\n return filter.test(value);\n }\n}\n\nfunction formatHeader(header) {\n return header.trim()\n .toLowerCase().replace(/([a-z\\d])(\\w*)/g, (w, char, str) => {\n return char.toUpperCase() + str;\n });\n}\n\nfunction buildAccessors(obj, header) {\n const accessorName = utils.toCamelCase(' ' + header);\n\n ['get', 'set', 'has'].forEach(methodName => {\n Object.defineProperty(obj, methodName + accessorName, {\n value: function(arg1, arg2, arg3) {\n return this[methodName].call(this, header, arg1, arg2, arg3);\n },\n configurable: true\n });\n });\n}\n\nclass AxiosHeaders {\n constructor(headers) {\n headers && this.set(headers);\n }\n\n set(header, valueOrRewrite, rewrite) {\n const self = this;\n\n function setHeader(_value, _header, _rewrite) {\n const lHeader = normalizeHeader(_header);\n\n if (!lHeader) {\n throw new Error('header name must be a non-empty string');\n }\n\n const key = utils.findKey(self, lHeader);\n\n if(!key || self[key] === undefined || _rewrite === true || (_rewrite === undefined && self[key] !== false)) {\n self[key || _header] = normalizeValue(_value);\n }\n }\n\n const setHeaders = (headers, _rewrite) =>\n utils.forEach(headers, (_value, _header) => setHeader(_value, _header, _rewrite));\n\n if (utils.isPlainObject(header) || header instanceof this.constructor) {\n setHeaders(header, valueOrRewrite)\n } else if(utils.isString(header) && (header = header.trim()) && !isValidHeaderName(header)) {\n setHeaders(parseHeaders(header), valueOrRewrite);\n } else if (utils.isObject(header) && utils.isIterable(header)) {\n let obj = {}, dest, key;\n for (const entry of header) {\n if (!utils.isArray(entry)) {\n throw TypeError('Object iterator must return a key-value pair');\n }\n\n obj[key = entry[0]] = (dest = obj[key]) ?\n (utils.isArray(dest) ? [...dest, entry[1]] : [dest, entry[1]]) : entry[1];\n }\n\n setHeaders(obj, valueOrRewrite)\n } else {\n header != null && setHeader(valueOrRewrite, header, rewrite);\n }\n\n return this;\n }\n\n get(header, parser) {\n header = normalizeHeader(header);\n\n if (header) {\n const key = utils.findKey(this, header);\n\n if (key) {\n const value = this[key];\n\n if (!parser) {\n return value;\n }\n\n if (parser === true) {\n return parseTokens(value);\n }\n\n if (utils.isFunction(parser)) {\n return parser.call(this, value, key);\n }\n\n if (utils.isRegExp(parser)) {\n return parser.exec(value);\n }\n\n throw new TypeError('parser must be boolean|regexp|function');\n }\n }\n }\n\n has(header, matcher) {\n header = normalizeHeader(header);\n\n if (header) {\n const key = utils.findKey(this, header);\n\n return !!(key && this[key] !== undefined && (!matcher || matchHeaderValue(this, this[key], key, matcher)));\n }\n\n return false;\n }\n\n delete(header, matcher) {\n const self = this;\n let deleted = false;\n\n function deleteHeader(_header) {\n _header = normalizeHeader(_header);\n\n if (_header) {\n const key = utils.findKey(self, _header);\n\n if (key && (!matcher || matchHeaderValue(self, self[key], key, matcher))) {\n delete self[key];\n\n deleted = true;\n }\n }\n }\n\n if (utils.isArray(header)) {\n header.forEach(deleteHeader);\n } else {\n deleteHeader(header);\n }\n\n return deleted;\n }\n\n clear(matcher) {\n const keys = Object.keys(this);\n let i = keys.length;\n let deleted = false;\n\n while (i--) {\n const key = keys[i];\n if(!matcher || matchHeaderValue(this, this[key], key, matcher, true)) {\n delete this[key];\n deleted = true;\n }\n }\n\n return deleted;\n }\n\n normalize(format) {\n const self = this;\n const headers = {};\n\n utils.forEach(this, (value, header) => {\n const key = utils.findKey(headers, header);\n\n if (key) {\n self[key] = normalizeValue(value);\n delete self[header];\n return;\n }\n\n const normalized = format ? formatHeader(header) : String(header).trim();\n\n if (normalized !== header) {\n delete self[header];\n }\n\n self[normalized] = normalizeValue(value);\n\n headers[normalized] = true;\n });\n\n return this;\n }\n\n concat(...targets) {\n return this.constructor.concat(this, ...targets);\n }\n\n toJSON(asStrings) {\n const obj = Object.create(null);\n\n utils.forEach(this, (value, header) => {\n value != null && value !== false && (obj[header] = asStrings && utils.isArray(value) ? value.join(', ') : value);\n });\n\n return obj;\n }\n\n [Symbol.iterator]() {\n return Object.entries(this.toJSON())[Symbol.iterator]();\n }\n\n toString() {\n return Object.entries(this.toJSON()).map(([header, value]) => header + ': ' + value).join('\\n');\n }\n\n getSetCookie() {\n return this.get(\"set-cookie\") || [];\n }\n\n get [Symbol.toStringTag]() {\n return 'AxiosHeaders';\n }\n\n static from(thing) {\n return thing instanceof this ? thing : new this(thing);\n }\n\n static concat(first, ...targets) {\n const computed = new this(first);\n\n targets.forEach((target) => computed.set(target));\n\n return computed;\n }\n\n static accessor(header) {\n const internals = this[$internals] = (this[$internals] = {\n accessors: {}\n });\n\n const accessors = internals.accessors;\n const prototype = this.prototype;\n\n function defineAccessor(_header) {\n const lHeader = normalizeHeader(_header);\n\n if (!accessors[lHeader]) {\n buildAccessors(prototype, _header);\n accessors[lHeader] = true;\n }\n }\n\n utils.isArray(header) ? header.forEach(defineAccessor) : defineAccessor(header);\n\n return this;\n }\n}\n\nAxiosHeaders.accessor(['Content-Type', 'Content-Length', 'Accept', 'Accept-Encoding', 'User-Agent', 'Authorization']);\n\n// reserved names hotfix\nutils.reduceDescriptors(AxiosHeaders.prototype, ({value}, key) => {\n let mapped = key[0].toUpperCase() + key.slice(1); // map `set` => `Set`\n return {\n get: () => value,\n set(headerValue) {\n this[mapped] = headerValue;\n }\n }\n});\n\nutils.freezeMethods(AxiosHeaders);\n\nexport default AxiosHeaders;\n","'use strict';\n\nimport utils from './../utils.js';\nimport defaults from '../defaults/index.js';\nimport AxiosHeaders from '../core/AxiosHeaders.js';\n\n/**\n * Transform the data for a request or a response\n *\n * @param {Array|Function} fns A single function or Array of functions\n * @param {?Object} response The response object\n *\n * @returns {*} The resulting transformed data\n */\nexport default function transformData(fns, response) {\n const config = this || defaults;\n const context = response || config;\n const headers = AxiosHeaders.from(context.headers);\n let data = context.data;\n\n utils.forEach(fns, function transform(fn) {\n data = fn.call(config, data, headers.normalize(), response ? response.status : undefined);\n });\n\n headers.normalize();\n\n return data;\n}\n","'use strict';\n\nexport default function isCancel(value) {\n return !!(value && value.__CANCEL__);\n}\n","'use strict';\n\nimport AxiosError from '../core/AxiosError.js';\nimport utils from '../utils.js';\n\n/**\n * A `CanceledError` is an object that is thrown when an operation is canceled.\n *\n * @param {string=} message The message.\n * @param {Object=} config The config.\n * @param {Object=} request The request.\n *\n * @returns {CanceledError} The created error.\n */\nfunction CanceledError(message, config, request) {\n // eslint-disable-next-line no-eq-null,eqeqeq\n AxiosError.call(this, message == null ? 'canceled' : message, AxiosError.ERR_CANCELED, config, request);\n this.name = 'CanceledError';\n}\n\nutils.inherits(CanceledError, AxiosError, {\n __CANCEL__: true\n});\n\nexport default CanceledError;\n","'use strict';\n\nimport AxiosError from './AxiosError.js';\n\n/**\n * Resolve or reject a Promise based on response status.\n *\n * @param {Function} resolve A function that resolves the promise.\n * @param {Function} reject A function that rejects the promise.\n * @param {object} response The response.\n *\n * @returns {object} The response.\n */\nexport default function settle(resolve, reject, response) {\n const validateStatus = response.config.validateStatus;\n if (!response.status || !validateStatus || validateStatus(response.status)) {\n resolve(response);\n } else {\n reject(new AxiosError(\n 'Request failed with status code ' + response.status,\n [AxiosError.ERR_BAD_REQUEST, AxiosError.ERR_BAD_RESPONSE][Math.floor(response.status / 100) - 4],\n response.config,\n response.request,\n response\n ));\n }\n}\n","'use strict';\n\nexport default function parseProtocol(url) {\n const match = /^([-+\\w]{1,25})(:?\\/\\/|:)/.exec(url);\n return match && match[1] || '';\n}\n","'use strict';\n\n/**\n * Calculate data maxRate\n * @param {Number} [samplesCount= 10]\n * @param {Number} [min= 1000]\n * @returns {Function}\n */\nfunction speedometer(samplesCount, min) {\n samplesCount = samplesCount || 10;\n const bytes = new Array(samplesCount);\n const timestamps = new Array(samplesCount);\n let head = 0;\n let tail = 0;\n let firstSampleTS;\n\n min = min !== undefined ? min : 1000;\n\n return function push(chunkLength) {\n const now = Date.now();\n\n const startedAt = timestamps[tail];\n\n if (!firstSampleTS) {\n firstSampleTS = now;\n }\n\n bytes[head] = chunkLength;\n timestamps[head] = now;\n\n let i = tail;\n let bytesCount = 0;\n\n while (i !== head) {\n bytesCount += bytes[i++];\n i = i % samplesCount;\n }\n\n head = (head + 1) % samplesCount;\n\n if (head === tail) {\n tail = (tail + 1) % samplesCount;\n }\n\n if (now - firstSampleTS < min) {\n return;\n }\n\n const passed = startedAt && now - startedAt;\n\n return passed ? Math.round(bytesCount * 1000 / passed) : undefined;\n };\n}\n\nexport default speedometer;\n","/**\n * Throttle decorator\n * @param {Function} fn\n * @param {Number} freq\n * @return {Function}\n */\nfunction throttle(fn, freq) {\n let timestamp = 0;\n let threshold = 1000 / freq;\n let lastArgs;\n let timer;\n\n const invoke = (args, now = Date.now()) => {\n timestamp = now;\n lastArgs = null;\n if (timer) {\n clearTimeout(timer);\n timer = null;\n }\n fn.apply(null, args);\n }\n\n const throttled = (...args) => {\n const now = Date.now();\n const passed = now - timestamp;\n if ( passed >= threshold) {\n invoke(args, now);\n } else {\n lastArgs = args;\n if (!timer) {\n timer = setTimeout(() => {\n timer = null;\n invoke(lastArgs)\n }, threshold - passed);\n }\n }\n }\n\n const flush = () => lastArgs && invoke(lastArgs);\n\n return [throttled, flush];\n}\n\nexport default throttle;\n","import speedometer from \"./speedometer.js\";\nimport throttle from \"./throttle.js\";\nimport utils from \"../utils.js\";\n\nexport const progressEventReducer = (listener, isDownloadStream, freq = 3) => {\n let bytesNotified = 0;\n const _speedometer = speedometer(50, 250);\n\n return throttle(e => {\n const loaded = e.loaded;\n const total = e.lengthComputable ? e.total : undefined;\n const progressBytes = loaded - bytesNotified;\n const rate = _speedometer(progressBytes);\n const inRange = loaded <= total;\n\n bytesNotified = loaded;\n\n const data = {\n loaded,\n total,\n progress: total ? (loaded / total) : undefined,\n bytes: progressBytes,\n rate: rate ? rate : undefined,\n estimated: rate && total && inRange ? (total - loaded) / rate : undefined,\n event: e,\n lengthComputable: total != null,\n [isDownloadStream ? 'download' : 'upload']: true\n };\n\n listener(data);\n }, freq);\n}\n\nexport const progressEventDecorator = (total, throttled) => {\n const lengthComputable = total != null;\n\n return [(loaded) => throttled[0]({\n lengthComputable,\n total,\n loaded\n }), throttled[1]];\n}\n\nexport const asyncDecorator = (fn) => (...args) => utils.asap(() => fn(...args));\n","import platform from '../platform/index.js';\n\nexport default platform.hasStandardBrowserEnv ? ((origin, isMSIE) => (url) => {\n url = new URL(url, platform.origin);\n\n return (\n origin.protocol === url.protocol &&\n origin.host === url.host &&\n (isMSIE || origin.port === url.port)\n );\n})(\n new URL(platform.origin),\n platform.navigator && /(msie|trident)/i.test(platform.navigator.userAgent)\n) : () => true;\n","import utils from './../utils.js';\nimport platform from '../platform/index.js';\n\nexport default platform.hasStandardBrowserEnv ?\n\n // Standard browser envs support document.cookie\n {\n write(name, value, expires, path, domain, secure) {\n const cookie = [name + '=' + encodeURIComponent(value)];\n\n utils.isNumber(expires) && cookie.push('expires=' + new Date(expires).toGMTString());\n\n utils.isString(path) && cookie.push('path=' + path);\n\n utils.isString(domain) && cookie.push('domain=' + domain);\n\n secure === true && cookie.push('secure');\n\n document.cookie = cookie.join('; ');\n },\n\n read(name) {\n const match = document.cookie.match(new RegExp('(^|;\\\\s*)(' + name + ')=([^;]*)'));\n return (match ? decodeURIComponent(match[3]) : null);\n },\n\n remove(name) {\n this.write(name, '', Date.now() - 86400000);\n }\n }\n\n :\n\n // Non-standard browser env (web workers, react-native) lack needed support.\n {\n write() {},\n read() {\n return null;\n },\n remove() {}\n };\n\n","'use strict';\n\n/**\n * Determines whether the specified URL is absolute\n *\n * @param {string} url The URL to test\n *\n * @returns {boolean} True if the specified URL is absolute, otherwise false\n */\nexport default function isAbsoluteURL(url) {\n // A URL is considered absolute if it begins with \"<scheme>://\" or \"//\" (protocol-relative URL).\n // RFC 3986 defines scheme name as a sequence of characters beginning with a letter and followed\n // by any combination of letters, digits, plus, period, or hyphen.\n return /^([a-z][a-z\\d+\\-.]*:)?\\/\\//i.test(url);\n}\n","'use strict';\n\n/**\n * Creates a new URL by combining the specified URLs\n *\n * @param {string} baseURL The base URL\n * @param {string} relativeURL The relative URL\n *\n * @returns {string} The combined URL\n */\nexport default function combineURLs(baseURL, relativeURL) {\n return relativeURL\n ? baseURL.replace(/\\/?\\/$/, '') + '/' + relativeURL.replace(/^\\/+/, '')\n : baseURL;\n}\n","'use strict';\n\nimport isAbsoluteURL from '../helpers/isAbsoluteURL.js';\nimport combineURLs from '../helpers/combineURLs.js';\n\n/**\n * Creates a new URL by combining the baseURL with the requestedURL,\n * only when the requestedURL is not already an absolute URL.\n * If the requestURL is absolute, this function returns the requestedURL untouched.\n *\n * @param {string} baseURL The base URL\n * @param {string} requestedURL Absolute or relative URL to combine\n *\n * @returns {string} The combined full path\n */\nexport default function buildFullPath(baseURL, requestedURL, allowAbsoluteUrls) {\n let isRelativeUrl = !isAbsoluteURL(requestedURL);\n if (baseURL && (isRelativeUrl || allowAbsoluteUrls == false)) {\n return combineURLs(baseURL, requestedURL);\n }\n return requestedURL;\n}\n","'use strict';\n\nimport utils from '../utils.js';\nimport AxiosHeaders from \"./AxiosHeaders.js\";\n\nconst headersToObject = (thing) => thing instanceof AxiosHeaders ? { ...thing } : thing;\n\n/**\n * Config-specific merge-function which creates a new config-object\n * by merging two configuration objects together.\n *\n * @param {Object} config1\n * @param {Object} config2\n *\n * @returns {Object} New object resulting from merging config2 to config1\n */\nexport default function mergeConfig(config1, config2) {\n // eslint-disable-next-line no-param-reassign\n config2 = config2 || {};\n const config = {};\n\n function getMergedValue(target, source, prop, caseless) {\n if (utils.isPlainObject(target) && utils.isPlainObject(source)) {\n return utils.merge.call({caseless}, target, source);\n } else if (utils.isPlainObject(source)) {\n return utils.merge({}, source);\n } else if (utils.isArray(source)) {\n return source.slice();\n }\n return source;\n }\n\n // eslint-disable-next-line consistent-return\n function mergeDeepProperties(a, b, prop , caseless) {\n if (!utils.isUndefined(b)) {\n return getMergedValue(a, b, prop , caseless);\n } else if (!utils.isUndefined(a)) {\n return getMergedValue(undefined, a, prop , caseless);\n }\n }\n\n // eslint-disable-next-line consistent-return\n function valueFromConfig2(a, b) {\n if (!utils.isUndefined(b)) {\n return getMergedValue(undefined, b);\n }\n }\n\n // eslint-disable-next-line consistent-return\n function defaultToConfig2(a, b) {\n if (!utils.isUndefined(b)) {\n return getMergedValue(undefined, b);\n } else if (!utils.isUndefined(a)) {\n return getMergedValue(undefined, a);\n }\n }\n\n // eslint-disable-next-line consistent-return\n function mergeDirectKeys(a, b, prop) {\n if (prop in config2) {\n return getMergedValue(a, b);\n } else if (prop in config1) {\n return getMergedValue(undefined, a);\n }\n }\n\n const mergeMap = {\n url: valueFromConfig2,\n method: valueFromConfig2,\n data: valueFromConfig2,\n baseURL: defaultToConfig2,\n transformRequest: defaultToConfig2,\n transformResponse: defaultToConfig2,\n paramsSerializer: defaultToConfig2,\n timeout: defaultToConfig2,\n timeoutMessage: defaultToConfig2,\n withCredentials: defaultToConfig2,\n withXSRFToken: defaultToConfig2,\n adapter: defaultToConfig2,\n responseType: defaultToConfig2,\n xsrfCookieName: defaultToConfig2,\n xsrfHeaderName: defaultToConfig2,\n onUploadProgress: defaultToConfig2,\n onDownloadProgress: defaultToConfig2,\n decompress: defaultToConfig2,\n maxContentLength: defaultToConfig2,\n maxBodyLength: defaultToConfig2,\n beforeRedirect: defaultToConfig2,\n transport: defaultToConfig2,\n httpAgent: defaultToConfig2,\n httpsAgent: defaultToConfig2,\n cancelToken: defaultToConfig2,\n socketPath: defaultToConfig2,\n responseEncoding: defaultToConfig2,\n validateStatus: mergeDirectKeys,\n headers: (a, b , prop) => mergeDeepProperties(headersToObject(a), headersToObject(b),prop, true)\n };\n\n utils.forEach(Object.keys(Object.assign({}, config1, config2)), function computeConfigValue(prop) {\n const merge = mergeMap[prop] || mergeDeepProperties;\n const configValue = merge(config1[prop], config2[prop], prop);\n (utils.isUndefined(configValue) && merge !== mergeDirectKeys) || (config[prop] = configValue);\n });\n\n return config;\n}\n","import platform from \"../platform/index.js\";\nimport utils from \"../utils.js\";\nimport isURLSameOrigin from \"./isURLSameOrigin.js\";\nimport cookies from \"./cookies.js\";\nimport buildFullPath from \"../core/buildFullPath.js\";\nimport mergeConfig from \"../core/mergeConfig.js\";\nimport AxiosHeaders from \"../core/AxiosHeaders.js\";\nimport buildURL from \"./buildURL.js\";\n\nexport default (config) => {\n const newConfig = mergeConfig({}, config);\n\n let {data, withXSRFToken, xsrfHeaderName, xsrfCookieName, headers, auth} = newConfig;\n\n newConfig.headers = headers = AxiosHeaders.from(headers);\n\n newConfig.url = buildURL(buildFullPath(newConfig.baseURL, newConfig.url, newConfig.allowAbsoluteUrls), config.params, config.paramsSerializer);\n\n // HTTP basic authentication\n if (auth) {\n headers.set('Authorization', 'Basic ' +\n btoa((auth.username || '') + ':' + (auth.password ? unescape(encodeURIComponent(auth.password)) : ''))\n );\n }\n\n let contentType;\n\n if (utils.isFormData(data)) {\n if (platform.hasStandardBrowserEnv || platform.hasStandardBrowserWebWorkerEnv) {\n headers.setContentType(undefined); // Let the browser set it\n } else if ((contentType = headers.getContentType()) !== false) {\n // fix semicolon duplication issue for ReactNative FormData implementation\n const [type, ...tokens] = contentType ? contentType.split(';').map(token => token.trim()).filter(Boolean) : [];\n headers.setContentType([type || 'multipart/form-data', ...tokens].join('; '));\n }\n }\n\n // Add xsrf header\n // This is only done if running in a standard browser environment.\n // Specifically not if we're in a web worker, or react-native.\n\n if (platform.hasStandardBrowserEnv) {\n withXSRFToken && utils.isFunction(withXSRFToken) && (withXSRFToken = withXSRFToken(newConfig));\n\n if (withXSRFToken || (withXSRFToken !== false && isURLSameOrigin(newConfig.url))) {\n // Add xsrf header\n const xsrfValue = xsrfHeaderName && xsrfCookieName && cookies.read(xsrfCookieName);\n\n if (xsrfValue) {\n headers.set(xsrfHeaderName, xsrfValue);\n }\n }\n }\n\n return newConfig;\n}\n\n","import utils from './../utils.js';\nimport settle from './../core/settle.js';\nimport transitionalDefaults from '../defaults/transitional.js';\nimport AxiosError from '../core/AxiosError.js';\nimport CanceledError from '../cancel/CanceledError.js';\nimport parseProtocol from '../helpers/parseProtocol.js';\nimport platform from '../platform/index.js';\nimport AxiosHeaders from '../core/AxiosHeaders.js';\nimport {progressEventReducer} from '../helpers/progressEventReducer.js';\nimport resolveConfig from \"../helpers/resolveConfig.js\";\n\nconst isXHRAdapterSupported = typeof XMLHttpRequest !== 'undefined';\n\nexport default isXHRAdapterSupported && function (config) {\n return new Promise(function dispatchXhrRequest(resolve, reject) {\n const _config = resolveConfig(config);\n let requestData = _config.data;\n const requestHeaders = AxiosHeaders.from(_config.headers).normalize();\n let {responseType, onUploadProgress, onDownloadProgress} = _config;\n let onCanceled;\n let uploadThrottled, downloadThrottled;\n let flushUpload, flushDownload;\n\n function done() {\n flushUpload && flushUpload(); // flush events\n flushDownload && flushDownload(); // flush events\n\n _config.cancelToken && _config.cancelToken.unsubscribe(onCanceled);\n\n _config.signal && _config.signal.removeEventListener('abort', onCanceled);\n }\n\n let request = new XMLHttpRequest();\n\n request.open(_config.method.toUpperCase(), _config.url, true);\n\n // Set the request timeout in MS\n request.timeout = _config.timeout;\n\n function onloadend() {\n if (!request) {\n return;\n }\n // Prepare the response\n const responseHeaders = AxiosHeaders.from(\n 'getAllResponseHeaders' in request && request.getAllResponseHeaders()\n );\n const responseData = !responseType || responseType === 'text' || responseType === 'json' ?\n request.responseText : request.response;\n const response = {\n data: responseData,\n status: request.status,\n statusText: request.statusText,\n headers: responseHeaders,\n config,\n request\n };\n\n settle(function _resolve(value) {\n resolve(value);\n done();\n }, function _reject(err) {\n reject(err);\n done();\n }, response);\n\n // Clean up request\n request = null;\n }\n\n if ('onloadend' in request) {\n // Use onloadend if available\n request.onloadend = onloadend;\n } else {\n // Listen for ready state to emulate onloadend\n request.onreadystatechange = function handleLoad() {\n if (!request || request.readyState !== 4) {\n return;\n }\n\n // The request errored out and we didn't get a response, this will be\n // handled by onerror instead\n // With one exception: request that using file: protocol, most browsers\n // will return status as 0 even though it's a successful request\n if (request.status === 0 && !(request.responseURL && request.responseURL.indexOf('file:') === 0)) {\n return;\n }\n // readystate handler is calling before onerror or ontimeout handlers,\n // so we should call onloadend on the next 'tick'\n setTimeout(onloadend);\n };\n }\n\n // Handle browser request cancellation (as opposed to a manual cancellation)\n request.onabort = function handleAbort() {\n if (!request) {\n return;\n }\n\n reject(new AxiosError('Request aborted', AxiosError.ECONNABORTED, config, request));\n\n // Clean up request\n request = null;\n };\n\n // Handle low level network errors\n request.onerror = function handleError() {\n // Real errors are hidden from us by the browser\n // onerror should only fire if it's a network error\n reject(new AxiosError('Network Error', AxiosError.ERR_NETWORK, config, request));\n\n // Clean up request\n request = null;\n };\n\n // Handle timeout\n request.ontimeout = function handleTimeout() {\n let timeoutErrorMessage = _config.timeout ? 'timeout of ' + _config.timeout + 'ms exceeded' : 'timeout exceeded';\n const transitional = _config.transitional || transitionalDefaults;\n if (_config.timeoutErrorMessage) {\n timeoutErrorMessage = _config.timeoutErrorMessage;\n }\n reject(new AxiosError(\n timeoutErrorMessage,\n transitional.clarifyTimeoutError ? AxiosError.ETIMEDOUT : AxiosError.ECONNABORTED,\n config,\n request));\n\n // Clean up request\n request = null;\n };\n\n // Remove Content-Type if data is undefined\n requestData === undefined && requestHeaders.setContentType(null);\n\n // Add headers to the request\n if ('setRequestHeader' in request) {\n utils.forEach(requestHeaders.toJSON(), function setRequestHeader(val, key) {\n request.setRequestHeader(key, val);\n });\n }\n\n // Add withCredentials to request if needed\n if (!utils.isUndefined(_config.withCredentials)) {\n request.withCredentials = !!_config.withCredentials;\n }\n\n // Add responseType to request if needed\n if (responseType && responseType !== 'json') {\n request.responseType = _config.responseType;\n }\n\n // Handle progress if needed\n if (onDownloadProgress) {\n ([downloadThrottled, flushDownload] = progressEventReducer(onDownloadProgress, true));\n request.addEventListener('progress', downloadThrottled);\n }\n\n // Not all browsers support upload events\n if (onUploadProgress && request.upload) {\n ([uploadThrottled, flushUpload] = progressEventReducer(onUploadProgress));\n\n request.upload.addEventListener('progress', uploadThrottled);\n\n request.upload.addEventListener('loadend', flushUpload);\n }\n\n if (_config.cancelToken || _config.signal) {\n // Handle cancellation\n // eslint-disable-next-line func-names\n onCanceled = cancel => {\n if (!request) {\n return;\n }\n reject(!cancel || cancel.type ? new CanceledError(null, config, request) : cancel);\n request.abort();\n request = null;\n };\n\n _config.cancelToken && _config.cancelToken.subscribe(onCanceled);\n if (_config.signal) {\n _config.signal.aborted ? onCanceled() : _config.signal.addEventListener('abort', onCanceled);\n }\n }\n\n const protocol = parseProtocol(_config.url);\n\n if (protocol && platform.protocols.indexOf(protocol) === -1) {\n reject(new AxiosError('Unsupported protocol ' + protocol + ':', AxiosError.ERR_BAD_REQUEST, config));\n return;\n }\n\n\n // Send the request\n request.send(requestData || null);\n });\n}\n","import CanceledError from \"../cancel/CanceledError.js\";\nimport AxiosError from \"../core/AxiosError.js\";\nimport utils from '../utils.js';\n\nconst composeSignals = (signals, timeout) => {\n const {length} = (signals = signals ? signals.filter(Boolean) : []);\n\n if (timeout || length) {\n let controller = new AbortController();\n\n let aborted;\n\n const onabort = function (reason) {\n if (!aborted) {\n aborted = true;\n unsubscribe();\n const err = reason instanceof Error ? reason : this.reason;\n controller.abort(err instanceof AxiosError ? err : new CanceledError(err instanceof Error ? err.message : err));\n }\n }\n\n let timer = timeout && setTimeout(() => {\n timer = null;\n onabort(new AxiosError(`timeout ${timeout} of ms exceeded`, AxiosError.ETIMEDOUT))\n }, timeout)\n\n const unsubscribe = () => {\n if (signals) {\n timer && clearTimeout(timer);\n timer = null;\n signals.forEach(signal => {\n signal.unsubscribe ? signal.unsubscribe(onabort) : signal.removeEventListener('abort', onabort);\n });\n signals = null;\n }\n }\n\n signals.forEach((signal) => signal.addEventListener('abort', onabort));\n\n const {signal} = controller;\n\n signal.unsubscribe = () => utils.asap(unsubscribe);\n\n return signal;\n }\n}\n\nexport default composeSignals;\n","\nexport const streamChunk = function* (chunk, chunkSize) {\n let len = chunk.byteLength;\n\n if (!chunkSize || len < chunkSize) {\n yield chunk;\n return;\n }\n\n let pos = 0;\n let end;\n\n while (pos < len) {\n end = pos + chunkSize;\n yield chunk.slice(pos, end);\n pos = end;\n }\n}\n\nexport const readBytes = async function* (iterable, chunkSize) {\n for await (const chunk of readStream(iterable)) {\n yield* streamChunk(chunk, chunkSize);\n }\n}\n\nconst readStream = async function* (stream) {\n if (stream[Symbol.asyncIterator]) {\n yield* stream;\n return;\n }\n\n const reader = stream.getReader();\n try {\n for (;;) {\n const {done, value} = await reader.read();\n if (done) {\n break;\n }\n yield value;\n }\n } finally {\n await reader.cancel();\n }\n}\n\nexport const trackStream = (stream, chunkSize, onProgress, onFinish) => {\n const iterator = readBytes(stream, chunkSize);\n\n let bytes = 0;\n let done;\n let _onFinish = (e) => {\n if (!done) {\n done = true;\n onFinish && onFinish(e);\n }\n }\n\n return new ReadableStream({\n async pull(controller) {\n try {\n const {done, value} = await iterator.next();\n\n if (done) {\n _onFinish();\n controller.close();\n return;\n }\n\n let len = value.byteLength;\n if (onProgress) {\n let loadedBytes = bytes += len;\n onProgress(loadedBytes);\n }\n controller.enqueue(new Uint8Array(value));\n } catch (err) {\n _onFinish(err);\n throw err;\n }\n },\n cancel(reason) {\n _onFinish(reason);\n return iterator.return();\n }\n }, {\n highWaterMark: 2\n })\n}\n","import platform from \"../platform/index.js\";\nimport utils from \"../utils.js\";\nimport AxiosError from \"../core/AxiosError.js\";\nimport composeSignals from \"../helpers/composeSignals.js\";\nimport {trackStream} from \"../helpers/trackStream.js\";\nimport AxiosHeaders from \"../core/AxiosHeaders.js\";\nimport {progressEventReducer, progressEventDecorator, asyncDecorator} from \"../helpers/progressEventReducer.js\";\nimport resolveConfig from \"../helpers/resolveConfig.js\";\nimport settle from \"../core/settle.js\";\n\nconst isFetchSupported = typeof fetch === 'function' && typeof Request === 'function' && typeof Response === 'function';\nconst isReadableStreamSupported = isFetchSupported && typeof ReadableStream === 'function';\n\n// used only inside the fetch adapter\nconst encodeText = isFetchSupported && (typeof TextEncoder === 'function' ?\n ((encoder) => (str) => encoder.encode(str))(new TextEncoder()) :\n async (str) => new Uint8Array(await new Response(str).arrayBuffer())\n);\n\nconst test = (fn, ...args) => {\n try {\n return !!fn(...args);\n } catch (e) {\n return false\n }\n}\n\nconst supportsRequestStream = isReadableStreamSupported && test(() => {\n let duplexAccessed = false;\n\n const hasContentType = new Request(platform.origin, {\n body: new ReadableStream(),\n method: 'POST',\n get duplex() {\n duplexAccessed = true;\n return 'half';\n },\n }).headers.has('Content-Type');\n\n return duplexAccessed && !hasContentType;\n});\n\nconst DEFAULT_CHUNK_SIZE = 64 * 1024;\n\nconst supportsResponseStream = isReadableStreamSupported &&\n test(() => utils.isReadableStream(new Response('').body));\n\n\nconst resolvers = {\n stream: supportsResponseStream && ((res) => res.body)\n};\n\nisFetchSupported && (((res) => {\n ['text', 'arrayBuffer', 'blob', 'formData', 'stream'].forEach(type => {\n !resolvers[type] && (resolvers[type] = utils.isFunction(res[type]) ? (res) => res[type]() :\n (_, config) => {\n throw new AxiosError(`Response type '${type}' is not supported`, AxiosError.ERR_NOT_SUPPORT, config);\n })\n });\n})(new Response));\n\nconst getBodyLength = async (body) => {\n if (body == null) {\n return 0;\n }\n\n if(utils.isBlob(body)) {\n return body.size;\n }\n\n if(utils.isSpecCompliantForm(body)) {\n const _request = new Request(platform.origin, {\n method: 'POST',\n body,\n });\n return (await _request.arrayBuffer()).byteLength;\n }\n\n if(utils.isArrayBufferView(body) || utils.isArrayBuffer(body)) {\n return body.byteLength;\n }\n\n if(utils.isURLSearchParams(body)) {\n body = body + '';\n }\n\n if(utils.isString(body)) {\n return (await encodeText(body)).byteLength;\n }\n}\n\nconst resolveBodyLength = async (headers, body) => {\n const length = utils.toFiniteNumber(headers.getContentLength());\n\n return length == null ? getBodyLength(body) : length;\n}\n\nexport default isFetchSupported && (async (config) => {\n let {\n url,\n method,\n data,\n signal,\n cancelToken,\n timeout,\n onDownloadProgress,\n onUploadProgress,\n responseType,\n headers,\n withCredentials = 'same-origin',\n fetchOptions\n } = resolveConfig(config);\n\n responseType = responseType ? (responseType + '').toLowerCase() : 'text';\n\n let composedSignal = composeSignals([signal, cancelToken && cancelToken.toAbortSignal()], timeout);\n\n let request;\n\n const unsubscribe = composedSignal && composedSignal.unsubscribe && (() => {\n composedSignal.unsubscribe();\n });\n\n let requestContentLength;\n\n try {\n if (\n onUploadProgress && supportsRequestStream && method !== 'get' && method !== 'head' &&\n (requestContentLength = await resolveBodyLength(headers, data)) !== 0\n ) {\n let _request = new Request(url, {\n method: 'POST',\n body: data,\n duplex: \"half\"\n });\n\n let contentTypeHeader;\n\n if (utils.isFormData(data) && (contentTypeHeader = _request.headers.get('content-type'))) {\n headers.setContentType(contentTypeHeader)\n }\n\n if (_request.body) {\n const [onProgress, flush] = progressEventDecorator(\n requestContentLength,\n progressEventReducer(asyncDecorator(onUploadProgress))\n );\n\n data = trackStream(_request.body, DEFAULT_CHUNK_SIZE, onProgress, flush);\n }\n }\n\n if (!utils.isString(withCredentials)) {\n withCredentials = withCredentials ? 'include' : 'omit';\n }\n\n // Cloudflare Workers throws when credentials are defined\n // see https://github.com/cloudflare/workerd/issues/902\n const isCredentialsSupported = \"credentials\" in Request.prototype;\n request = new Request(url, {\n ...fetchOptions,\n signal: composedSignal,\n method: method.toUpperCase(),\n headers: headers.normalize().toJSON(),\n body: data,\n duplex: \"half\",\n credentials: isCredentialsSupported ? withCredentials : undefined\n });\n\n let response = await fetch(request, fetchOptions);\n\n const isStreamResponse = supportsResponseStream && (responseType === 'stream' || responseType === 'response');\n\n if (supportsResponseStream && (onDownloadProgress || (isStreamResponse && unsubscribe))) {\n const options = {};\n\n ['status', 'statusText', 'headers'].forEach(prop => {\n options[prop] = response[prop];\n });\n\n const responseContentLength = utils.toFiniteNumber(response.headers.get('content-length'));\n\n const [onProgress, flush] = onDownloadProgress && progressEventDecorator(\n responseContentLength,\n progressEventReducer(asyncDecorator(onDownloadProgress), true)\n ) || [];\n\n response = new Response(\n trackStream(response.body, DEFAULT_CHUNK_SIZE, onProgress, () => {\n flush && flush();\n unsubscribe && unsubscribe();\n }),\n options\n );\n }\n\n responseType = responseType || 'text';\n\n let responseData = await resolvers[utils.findKey(resolvers, responseType) || 'text'](response, config);\n\n !isStreamResponse && unsubscribe && unsubscribe();\n\n return await new Promise((resolve, reject) => {\n settle(resolve, reject, {\n data: responseData,\n headers: AxiosHeaders.from(response.headers),\n status: response.status,\n statusText: response.statusText,\n config,\n request\n })\n })\n } catch (err) {\n unsubscribe && unsubscribe();\n\n if (err && err.name === 'TypeError' && /Load failed|fetch/i.test(err.message)) {\n throw Object.assign(\n new AxiosError('Network Error', AxiosError.ERR_NETWORK, config, request),\n {\n cause: err.cause || err\n }\n )\n }\n\n throw AxiosError.from(err, err && err.code, config, request);\n }\n});\n\n\n","import utils from '../utils.js';\nimport httpAdapter from './http.js';\nimport xhrAdapter from './xhr.js';\nimport fetchAdapter from './fetch.js';\nimport AxiosError from \"../core/AxiosError.js\";\n\nconst knownAdapters = {\n http: httpAdapter,\n xhr: xhrAdapter,\n fetch: fetchAdapter\n}\n\nutils.forEach(knownAdapters, (fn, value) => {\n if (fn) {\n try {\n Object.defineProperty(fn, 'name', {value});\n } catch (e) {\n // eslint-disable-next-line no-empty\n }\n Object.defineProperty(fn, 'adapterName', {value});\n }\n});\n\nconst renderReason = (reason) => `- ${reason}`;\n\nconst isResolvedHandle = (adapter) => utils.isFunction(adapter) || adapter === null || adapter === false;\n\nexport default {\n getAdapter: (adapters) => {\n adapters = utils.isArray(adapters) ? adapters : [adapters];\n\n const {length} = adapters;\n let nameOrAdapter;\n let adapter;\n\n const rejectedReasons = {};\n\n for (let i = 0; i < length; i++) {\n nameOrAdapter = adapters[i];\n let id;\n\n adapter = nameOrAdapter;\n\n if (!isResolvedHandle(nameOrAdapter)) {\n adapter = knownAdapters[(id = String(nameOrAdapter)).toLowerCase()];\n\n if (adapter === undefined) {\n throw new AxiosError(`Unknown adapter '${id}'`);\n }\n }\n\n if (adapter) {\n break;\n }\n\n rejectedReasons[id || '#' + i] = adapter;\n }\n\n if (!adapter) {\n\n const reasons = Object.entries(rejectedReasons)\n .map(([id, state]) => `adapter ${id} ` +\n (state === false ? 'is not supported by the environment' : 'is not available in the build')\n );\n\n let s = length ?\n (reasons.length > 1 ? 'since :\\n' + reasons.map(renderReason).join('\\n') : ' ' + renderReason(reasons[0])) :\n 'as no adapter specified';\n\n throw new AxiosError(\n `There is no suitable adapter to dispatch the request ` + s,\n 'ERR_NOT_SUPPORT'\n );\n }\n\n return adapter;\n },\n adapters: knownAdapters\n}\n","'use strict';\n\nimport transformData from './transformData.js';\nimport isCancel from '../cancel/isCancel.js';\nimport defaults from '../defaults/index.js';\nimport CanceledError from '../cancel/CanceledError.js';\nimport AxiosHeaders from '../core/AxiosHeaders.js';\nimport adapters from \"../adapters/adapters.js\";\n\n/**\n * Throws a `CanceledError` if cancellation has been requested.\n *\n * @param {Object} config The config that is to be used for the request\n *\n * @returns {void}\n */\nfunction throwIfCancellationRequested(config) {\n if (config.cancelToken) {\n config.cancelToken.throwIfRequested();\n }\n\n if (config.signal && config.signal.aborted) {\n throw new CanceledError(null, config);\n }\n}\n\n/**\n * Dispatch a request to the server using the configured adapter.\n *\n * @param {object} config The config that is to be used for the request\n *\n * @returns {Promise} The Promise to be fulfilled\n */\nexport default function dispatchRequest(config) {\n throwIfCancellationRequested(config);\n\n config.headers = AxiosHeaders.from(config.headers);\n\n // Transform request data\n config.data = transformData.call(\n config,\n config.transformRequest\n );\n\n if (['post', 'put', 'patch'].indexOf(config.method) !== -1) {\n config.headers.setContentType('application/x-www-form-urlencoded', false);\n }\n\n const adapter = adapters.getAdapter(config.adapter || defaults.adapter);\n\n return adapter(config).then(function onAdapterResolution(response) {\n throwIfCancellationRequested(config);\n\n // Transform response data\n response.data = transformData.call(\n config,\n config.transformResponse,\n response\n );\n\n response.headers = AxiosHeaders.from(response.headers);\n\n return response;\n }, function onAdapterRejection(reason) {\n if (!isCancel(reason)) {\n throwIfCancellationRequested(config);\n\n // Transform response data\n if (reason && reason.response) {\n reason.response.data = transformData.call(\n config,\n config.transformResponse,\n reason.response\n );\n reason.response.headers = AxiosHeaders.from(reason.response.headers);\n }\n }\n\n return Promise.reject(reason);\n });\n}\n","export const VERSION = \"1.10.0\";","'use strict';\n\nimport {VERSION} from '../env/data.js';\nimport AxiosError from '../core/AxiosError.js';\n\nconst validators = {};\n\n// eslint-disable-next-line func-names\n['object', 'boolean', 'number', 'function', 'string', 'symbol'].forEach((type, i) => {\n validators[type] = function validator(thing) {\n return typeof thing === type || 'a' + (i < 1 ? 'n ' : ' ') + type;\n };\n});\n\nconst deprecatedWarnings = {};\n\n/**\n * Transitional option validator\n *\n * @param {function|boolean?} validator - set to false if the transitional option has been removed\n * @param {string?} version - deprecated version / removed since version\n * @param {string?} message - some message with additional info\n *\n * @returns {function}\n */\nvalidators.transitional = function transitional(validator, version, message) {\n function formatMessage(opt, desc) {\n return '[Axios v' + VERSION + '] Transitional option \\'' + opt + '\\'' + desc + (message ? '. ' + message : '');\n }\n\n // eslint-disable-next-line func-names\n return (value, opt, opts) => {\n if (validator === false) {\n throw new AxiosError(\n formatMessage(opt, ' has been removed' + (version ? ' in ' + version : '')),\n AxiosError.ERR_DEPRECATED\n );\n }\n\n if (version && !deprecatedWarnings[opt]) {\n deprecatedWarnings[opt] = true;\n // eslint-disable-next-line no-console\n console.warn(\n formatMessage(\n opt,\n ' has been deprecated since v' + version + ' and will be removed in the near future'\n )\n );\n }\n\n return validator ? validator(value, opt, opts) : true;\n };\n};\n\nvalidators.spelling = function spelling(correctSpelling) {\n return (value, opt) => {\n // eslint-disable-next-line no-console\n console.warn(`${opt} is likely a misspelling of ${correctSpelling}`);\n return true;\n }\n};\n\n/**\n * Assert object's properties type\n *\n * @param {object} options\n * @param {object} schema\n * @param {boolean?} allowUnknown\n *\n * @returns {object}\n */\n\nfunction assertOptions(options, schema, allowUnknown) {\n if (typeof options !== 'object') {\n throw new AxiosError('options must be an object', AxiosError.ERR_BAD_OPTION_VALUE);\n }\n const keys = Object.keys(options);\n let i = keys.length;\n while (i-- > 0) {\n const opt = keys[i];\n const validator = schema[opt];\n if (validator) {\n const value = options[opt];\n const result = value === undefined || validator(value, opt, options);\n if (result !== true) {\n throw new AxiosError('option ' + opt + ' must be ' + result, AxiosError.ERR_BAD_OPTION_VALUE);\n }\n continue;\n }\n if (allowUnknown !== true) {\n throw new AxiosError('Unknown option ' + opt, AxiosError.ERR_BAD_OPTION);\n }\n }\n}\n\nexport default {\n assertOptions,\n validators\n};\n","'use strict';\n\nimport utils from './../utils.js';\nimport buildURL from '../helpers/buildURL.js';\nimport InterceptorManager from './InterceptorManager.js';\nimport dispatchRequest from './dispatchRequest.js';\nimport mergeConfig from './mergeConfig.js';\nimport buildFullPath from './buildFullPath.js';\nimport validator from '../helpers/validator.js';\nimport AxiosHeaders from './AxiosHeaders.js';\n\nconst validators = validator.validators;\n\n/**\n * Create a new instance of Axios\n *\n * @param {Object} instanceConfig The default config for the instance\n *\n * @return {Axios} A new instance of Axios\n */\nclass Axios {\n constructor(instanceConfig) {\n this.defaults = instanceConfig || {};\n this.interceptors = {\n request: new InterceptorManager(),\n response: new InterceptorManager()\n };\n }\n\n /**\n * Dispatch a request\n *\n * @param {String|Object} configOrUrl The config specific for this request (merged with this.defaults)\n * @param {?Object} config\n *\n * @returns {Promise} The Promise to be fulfilled\n */\n async request(configOrUrl, config) {\n try {\n return await this._request(configOrUrl, config);\n } catch (err) {\n if (err instanceof Error) {\n let dummy = {};\n\n Error.captureStackTrace ? Error.captureStackTrace(dummy) : (dummy = new Error());\n\n // slice off the Error: ... line\n const stack = dummy.stack ? dummy.stack.replace(/^.+\\n/, '') : '';\n try {\n if (!err.stack) {\n err.stack = stack;\n // match without the 2 top stack lines\n } else if (stack && !String(err.stack).endsWith(stack.replace(/^.+\\n.+\\n/, ''))) {\n err.stack += '\\n' + stack\n }\n } catch (e) {\n // ignore the case where \"stack\" is an un-writable property\n }\n }\n\n throw err;\n }\n }\n\n _request(configOrUrl, config) {\n /*eslint no-param-reassign:0*/\n // Allow for axios('example/url'[, config]) a la fetch API\n if (typeof configOrUrl === 'string') {\n config = config || {};\n config.url = configOrUrl;\n } else {\n config = configOrUrl || {};\n }\n\n config = mergeConfig(this.defaults, config);\n\n const {transitional, paramsSerializer, headers} = config;\n\n if (transitional !== undefined) {\n validator.assertOptions(transitional, {\n silentJSONParsing: validators.transitional(validators.boolean),\n forcedJSONParsing: validators.transitional(validators.boolean),\n clarifyTimeoutError: validators.transitional(validators.boolean)\n }, false);\n }\n\n if (paramsSerializer != null) {\n if (utils.isFunction(paramsSerializer)) {\n config.paramsSerializer = {\n serialize: paramsSerializer\n }\n } else {\n validator.assertOptions(paramsSerializer, {\n encode: validators.function,\n serialize: validators.function\n }, true);\n }\n }\n\n // Set config.allowAbsoluteUrls\n if (config.allowAbsoluteUrls !== undefined) {\n // do nothing\n } else if (this.defaults.allowAbsoluteUrls !== undefined) {\n config.allowAbsoluteUrls = this.defaults.allowAbsoluteUrls;\n } else {\n config.allowAbsoluteUrls = true;\n }\n\n validator.assertOptions(config, {\n baseUrl: validators.spelling('baseURL'),\n withXsrfToken: validators.spelling('withXSRFToken')\n }, true);\n\n // Set config.method\n config.method = (config.method || this.defaults.method || 'get').toLowerCase();\n\n // Flatten headers\n let contextHeaders = headers && utils.merge(\n headers.common,\n headers[config.method]\n );\n\n headers && utils.forEach(\n ['delete', 'get', 'head', 'post', 'put', 'patch', 'common'],\n (method) => {\n delete headers[method];\n }\n );\n\n config.headers = AxiosHeaders.concat(contextHeaders, headers);\n\n // filter out skipped interceptors\n const requestInterceptorChain = [];\n let synchronousRequestInterceptors = true;\n this.interceptors.request.forEach(function unshiftRequestInterceptors(interceptor) {\n if (typeof interceptor.runWhen === 'function' && interceptor.runWhen(config) === false) {\n return;\n }\n\n synchronousRequestInterceptors = synchronousRequestInterceptors && interceptor.synchronous;\n\n requestInterceptorChain.unshift(interceptor.fulfilled, interceptor.rejected);\n });\n\n const responseInterceptorChain = [];\n this.interceptors.response.forEach(function pushResponseInterceptors(interceptor) {\n responseInterceptorChain.push(interceptor.fulfilled, interceptor.rejected);\n });\n\n let promise;\n let i = 0;\n let len;\n\n if (!synchronousRequestInterceptors) {\n const chain = [dispatchRequest.bind(this), undefined];\n chain.unshift.apply(chain, requestInterceptorChain);\n chain.push.apply(chain, responseInterceptorChain);\n len = chain.length;\n\n promise = Promise.resolve(config);\n\n while (i < len) {\n promise = promise.then(chain[i++], chain[i++]);\n }\n\n return promise;\n }\n\n len = requestInterceptorChain.length;\n\n let newConfig = config;\n\n i = 0;\n\n while (i < len) {\n const onFulfilled = requestInterceptorChain[i++];\n const onRejected = requestInterceptorChain[i++];\n try {\n newConfig = onFulfilled(newConfig);\n } catch (error) {\n onRejected.call(this, error);\n break;\n }\n }\n\n try {\n promise = dispatchRequest.call(this, newConfig);\n } catch (error) {\n return Promise.reject(error);\n }\n\n i = 0;\n len = responseInterceptorChain.length;\n\n while (i < len) {\n promise = promise.then(responseInterceptorChain[i++], responseInterceptorChain[i++]);\n }\n\n return promise;\n }\n\n getUri(config) {\n config = mergeConfig(this.defaults, config);\n const fullPath = buildFullPath(config.baseURL, config.url, config.allowAbsoluteUrls);\n return buildURL(fullPath, config.params, config.paramsSerializer);\n }\n}\n\n// Provide aliases for supported request methods\nutils.forEach(['delete', 'get', 'head', 'options'], function forEachMethodNoData(method) {\n /*eslint func-names:0*/\n Axios.prototype[method] = function(url, config) {\n return this.request(mergeConfig(config || {}, {\n method,\n url,\n data: (config || {}).data\n }));\n };\n});\n\nutils.forEach(['post', 'put', 'patch'], function forEachMethodWithData(method) {\n /*eslint func-names:0*/\n\n function generateHTTPMethod(isForm) {\n return function httpMethod(url, data, config) {\n return this.request(mergeConfig(config || {}, {\n method,\n headers: isForm ? {\n 'Content-Type': 'multipart/form-data'\n } : {},\n url,\n data\n }));\n };\n }\n\n Axios.prototype[method] = generateHTTPMethod();\n\n Axios.prototype[method + 'Form'] = generateHTTPMethod(true);\n});\n\nexport default Axios;\n","'use strict';\n\nimport CanceledError from './CanceledError.js';\n\n/**\n * A `CancelToken` is an object that can be used to request cancellation of an operation.\n *\n * @param {Function} executor The executor function.\n *\n * @returns {CancelToken}\n */\nclass CancelToken {\n constructor(executor) {\n if (typeof executor !== 'function') {\n throw new TypeError('executor must be a function.');\n }\n\n let resolvePromise;\n\n this.promise = new Promise(function promiseExecutor(resolve) {\n resolvePromise = resolve;\n });\n\n const token = this;\n\n // eslint-disable-next-line func-names\n this.promise.then(cancel => {\n if (!token._listeners) return;\n\n let i = token._listeners.length;\n\n while (i-- > 0) {\n token._listeners[i](cancel);\n }\n token._listeners = null;\n });\n\n // eslint-disable-next-line func-names\n this.promise.then = onfulfilled => {\n let _resolve;\n // eslint-disable-next-line func-names\n const promise = new Promise(resolve => {\n token.subscribe(resolve);\n _resolve = resolve;\n }).then(onfulfilled);\n\n promise.cancel = function reject() {\n token.unsubscribe(_resolve);\n };\n\n return promise;\n };\n\n executor(function cancel(message, config, request) {\n if (token.reason) {\n // Cancellation has already been requested\n return;\n }\n\n token.reason = new CanceledError(message, config, request);\n resolvePromise(token.reason);\n });\n }\n\n /**\n * Throws a `CanceledError` if cancellation has been requested.\n */\n throwIfRequested() {\n if (this.reason) {\n throw this.reason;\n }\n }\n\n /**\n * Subscribe to the cancel signal\n */\n\n subscribe(listener) {\n if (this.reason) {\n listener(this.reason);\n return;\n }\n\n if (this._listeners) {\n this._listeners.push(listener);\n } else {\n this._listeners = [listener];\n }\n }\n\n /**\n * Unsubscribe from the cancel signal\n */\n\n unsubscribe(listener) {\n if (!this._listeners) {\n return;\n }\n const index = this._listeners.indexOf(listener);\n if (index !== -1) {\n this._listeners.splice(index, 1);\n }\n }\n\n toAbortSignal() {\n const controller = new AbortController();\n\n const abort = (err) => {\n controller.abort(err);\n };\n\n this.subscribe(abort);\n\n controller.signal.unsubscribe = () => this.unsubscribe(abort);\n\n return controller.signal;\n }\n\n /**\n * Returns an object that contains a new `CancelToken` and a function that, when called,\n * cancels the `CancelToken`.\n */\n static source() {\n let cancel;\n const token = new CancelToken(function executor(c) {\n cancel = c;\n });\n return {\n token,\n cancel\n };\n }\n}\n\nexport default CancelToken;\n","'use strict';\n\n/**\n * Syntactic sugar for invoking a function and expanding an array for arguments.\n *\n * Common use case would be to use `Function.prototype.apply`.\n *\n * ```js\n * function f(x, y, z) {}\n * var args = [1, 2, 3];\n * f.apply(null, args);\n * ```\n *\n * With `spread` this example can be re-written.\n *\n * ```js\n * spread(function(x, y, z) {})([1, 2, 3]);\n * ```\n *\n * @param {Function} callback\n *\n * @returns {Function}\n */\nexport default function spread(callback) {\n return function wrap(arr) {\n return callback.apply(null, arr);\n };\n}\n","'use strict';\n\nimport utils from './../utils.js';\n\n/**\n * Determines whether the payload is an error thrown by Axios\n *\n * @param {*} payload The value to test\n *\n * @returns {boolean} True if the payload is an error thrown by Axios, otherwise false\n */\nexport default function isAxiosError(payload) {\n return utils.isObject(payload) && (payload.isAxiosError === true);\n}\n","const HttpStatusCode = {\n Continue: 100,\n SwitchingProtocols: 101,\n Processing: 102,\n EarlyHints: 103,\n Ok: 200,\n Created: 201,\n Accepted: 202,\n NonAuthoritativeInformation: 203,\n NoContent: 204,\n ResetContent: 205,\n PartialContent: 206,\n MultiStatus: 207,\n AlreadyReported: 208,\n ImUsed: 226,\n MultipleChoices: 300,\n MovedPermanently: 301,\n Found: 302,\n SeeOther: 303,\n NotModified: 304,\n UseProxy: 305,\n Unused: 306,\n TemporaryRedirect: 307,\n PermanentRedirect: 308,\n BadRequest: 400,\n Unauthorized: 401,\n PaymentRequired: 402,\n Forbidden: 403,\n NotFound: 404,\n MethodNotAllowed: 405,\n NotAcceptable: 406,\n ProxyAuthenticationRequired: 407,\n RequestTimeout: 408,\n Conflict: 409,\n Gone: 410,\n LengthRequired: 411,\n PreconditionFailed: 412,\n PayloadTooLarge: 413,\n UriTooLong: 414,\n UnsupportedMediaType: 415,\n RangeNotSatisfiable: 416,\n ExpectationFailed: 417,\n ImATeapot: 418,\n MisdirectedRequest: 421,\n UnprocessableEntity: 422,\n Locked: 423,\n FailedDependency: 424,\n TooEarly: 425,\n UpgradeRequired: 426,\n PreconditionRequired: 428,\n TooManyRequests: 429,\n RequestHeaderFieldsTooLarge: 431,\n UnavailableForLegalReasons: 451,\n InternalServerError: 500,\n NotImplemented: 501,\n BadGateway: 502,\n ServiceUnavailable: 503,\n GatewayTimeout: 504,\n HttpVersionNotSupported: 505,\n VariantAlsoNegotiates: 506,\n InsufficientStorage: 507,\n LoopDetected: 508,\n NotExtended: 510,\n NetworkAuthenticationRequired: 511,\n};\n\nObject.entries(HttpStatusCode).forEach(([key, value]) => {\n HttpStatusCode[value] = key;\n});\n\nexport default HttpStatusCode;\n","'use strict';\n\nimport utils from './utils.js';\nimport bind from './helpers/bind.js';\nimport Axios from './core/Axios.js';\nimport mergeConfig from './core/mergeConfig.js';\nimport defaults from './defaults/index.js';\nimport formDataToJSON from './helpers/formDataToJSON.js';\nimport CanceledError from './cancel/CanceledError.js';\nimport CancelToken from './cancel/CancelToken.js';\nimport isCancel from './cancel/isCancel.js';\nimport {VERSION} from './env/data.js';\nimport toFormData from './helpers/toFormData.js';\nimport AxiosError from './core/AxiosError.js';\nimport spread from './helpers/spread.js';\nimport isAxiosError from './helpers/isAxiosError.js';\nimport AxiosHeaders from \"./core/AxiosHeaders.js\";\nimport adapters from './adapters/adapters.js';\nimport HttpStatusCode from './helpers/HttpStatusCode.js';\n\n/**\n * Create an instance of Axios\n *\n * @param {Object} defaultConfig The default config for the instance\n *\n * @returns {Axios} A new instance of Axios\n */\nfunction createInstance(defaultConfig) {\n const context = new Axios(defaultConfig);\n const instance = bind(Axios.prototype.request, context);\n\n // Copy axios.prototype to instance\n utils.extend(instance, Axios.prototype, context, {allOwnKeys: true});\n\n // Copy context to instance\n utils.extend(instance, context, null, {allOwnKeys: true});\n\n // Factory for creating new instances\n instance.create = function create(instanceConfig) {\n return createInstance(mergeConfig(defaultConfig, instanceConfig));\n };\n\n return instance;\n}\n\n// Create the default instance to be exported\nconst axios = createInstance(defaults);\n\n// Expose Axios class to allow class inheritance\naxios.Axios = Axios;\n\n// Expose Cancel & CancelToken\naxios.CanceledError = CanceledError;\naxios.CancelToken = CancelToken;\naxios.isCancel = isCancel;\naxios.VERSION = VERSION;\naxios.toFormData = toFormData;\n\n// Expose AxiosError class\naxios.AxiosError = AxiosError;\n\n// alias for CanceledError for backward compatibility\naxios.Cancel = axios.CanceledError;\n\n// Expose all/spread\naxios.all = function all(promises) {\n return Promise.all(promises);\n};\n\naxios.spread = spread;\n\n// Expose isAxiosError\naxios.isAxiosError = isAxiosError;\n\n// Expose mergeConfig\naxios.mergeConfig = mergeConfig;\n\naxios.AxiosHeaders = AxiosHeaders;\n\naxios.formToJSON = thing => formDataToJSON(utils.isHTMLForm(thing) ? new FormData(thing) : thing);\n\naxios.getAdapter = adapters.getAdapter;\n\naxios.HttpStatusCode = HttpStatusCode;\n\naxios.default = axios;\n\n// this module should only have a default export\nexport default axios\n","import axios, { AxiosError,AxiosInstance, AxiosRequestConfig } from 'axios';\n\nimport {\n HttpServiceConfig,\n RequestErrorInterceptor,\n RequestInterceptor,\n ResponseErrorInterceptor,\n ResponseInterceptor,\n UnauthorizedCallback,\n} from './types';\n\n/**\n * HttpService - 单例 HTTP 客户端\n *\n * 特性:\n * - 单例模式,全局唯一实例,延迟初始化(首次使用时自动创建)\n * - 支持拦截器注册(其他模块可注入 header)\n * - 统一 401/403 处理\n * - 自动携带凭证(withCredentials)\n * - 类型安全\n *\n * @example\n * ```typescript\n * // 直接使用导出的 httpService(首次使用时自动初始化)\n * import { httpService } from '@genie/agent-provider';\n *\n * // 按需设置 baseURL\n * httpService.setBaseURL('https://api.example.com');\n *\n * // 按需设置 authToken\n * httpService.setAuthToken('your-token');\n *\n * // 注册请求拦截器(注入 header)\n * httpService.registerRequestInterceptor((config) => {\n * config.headers['X-Enterprise-Id'] = 'enterprise-123';\n * return config;\n * });\n *\n * // 注册 401 回调\n * httpService.onUnauthorized(() => {\n * console.log('User unauthorized');\n * window.location.href = '/login';\n * });\n *\n * // 发起请求\n * const data = await httpService.get<Account>('/console/accounts');\n * ```\n */\nexport class HttpService {\n private static instance: HttpService | null = null;\n private axiosInstance: AxiosInstance;\n private unauthorizedCallbacks: Set<UnauthorizedCallback> = new Set();\n private config: HttpServiceConfig;\n\n /**\n * 私有构造函数(单例模式)\n */\n private constructor(config: HttpServiceConfig = {}) {\n this.config = config;\n\n // 创建 axios 实例\n this.axiosInstance = axios.create({\n baseURL: config.baseURL?.replace(/\\/$/, '') || '',\n timeout: config.timeout || 60000,\n withCredentials: config.withCredentials !== false, // 默认 true\n headers: {\n 'Content-Type': 'application/json',\n 'Accept': 'application/json',\n ...config.headers,\n },\n });\n\n // 注册默认请求拦截器(添加 Authorization)\n this.registerDefaultRequestInterceptor();\n\n // 注册默认响应拦截器(处理 401/403)\n this.registerDefaultResponseInterceptor();\n }\n\n /**\n * 获取单例实例(延迟初始化)\n */\n public static getInstance(): HttpService {\n if (!HttpService.instance) {\n HttpService.instance = new HttpService();\n }\n return HttpService.instance;\n }\n\n /**\n * 重置单例实例(主要用于测试)\n */\n public static resetInstance(): void {\n HttpService.instance = null;\n }\n\n /**\n * 注册默认请求拦截器(添加 Authorization header)\n */\n private registerDefaultRequestInterceptor(): void {\n this.axiosInstance.interceptors.request.use(\n requestConfig => {\n // 添加 Authorization header\n if (this.config.authToken) {\n requestConfig.headers = requestConfig.headers || {};\n requestConfig.headers['Authorization'] = `Bearer ${this.config.authToken}`;\n }\n return requestConfig;\n },\n error => Promise.reject(error)\n );\n }\n\n /**\n * 注册默认响应拦截器(处理 401/403)\n */\n private registerDefaultResponseInterceptor(): void {\n this.axiosInstance.interceptors.response.use(\n response => response,\n (error: AxiosError) => {\n // 统一处理 401/403\n if (error.response?.status === 401 || error.response?.status === 403) {\n console.warn('[HttpService] Unauthorized (401/403), triggering callbacks');\n this.triggerUnauthorizedCallbacks();\n }\n return Promise.reject(error);\n }\n );\n }\n\n /**\n * 注册请求拦截器\n * @param onFulfilled 请求成功拦截器\n * @param onRejected 请求失败拦截器\n * @returns 拦截器 ID(用于移除)\n *\n * @example\n * ```typescript\n * const id = httpService.registerRequestInterceptor((config) => {\n * config.headers['X-Custom-Header'] = 'value';\n * return config;\n * });\n *\n * // 移除拦截器\n * httpService.ejectRequestInterceptor(id);\n * ```\n */\n public registerRequestInterceptor(\n onFulfilled: RequestInterceptor,\n onRejected?: RequestErrorInterceptor\n ): number {\n return this.axiosInstance.interceptors.request.use(onFulfilled, onRejected);\n }\n\n /**\n * 注册响应拦截器\n * @param onFulfilled 响应成功拦截器\n * @param onRejected 响应失败拦截器\n * @returns 拦截器 ID(用于移除)\n *\n * @example\n * ```typescript\n * const id = httpService.registerResponseInterceptor((response) => {\n * console.log('Response:', response.data);\n * return response;\n * });\n *\n * // 移除拦截器\n * httpService.ejectResponseInterceptor(id);\n * ```\n */\n public registerResponseInterceptor(\n onFulfilled: ResponseInterceptor,\n onRejected?: ResponseErrorInterceptor\n ): number {\n return this.axiosInstance.interceptors.response.use(onFulfilled, onRejected);\n }\n\n /**\n * 移除请求拦截器\n * @param id 拦截器 ID\n */\n public ejectRequestInterceptor(id: number): void {\n this.axiosInstance.interceptors.request.eject(id);\n }\n\n /**\n * 移除响应拦截器\n * @param id 拦截器 ID\n */\n public ejectResponseInterceptor(id: number): void {\n this.axiosInstance.interceptors.response.eject(id);\n }\n\n /**\n * 注册 401 未授权回调\n * @param callback 回调函数\n *\n * @example\n * ```typescript\n * httpService.onUnauthorized(() => {\n * console.log('User logged out');\n * window.location.href = '/login';\n * });\n * ```\n */\n public onUnauthorized(callback: UnauthorizedCallback): void {\n this.unauthorizedCallbacks.add(callback);\n }\n\n /**\n * 移除 401 未授权回调\n * @param callback 回调函数\n */\n public offUnauthorized(callback: UnauthorizedCallback): void {\n this.unauthorizedCallbacks.delete(callback);\n }\n\n /**\n * 触发所有 401 回调\n */\n private triggerUnauthorizedCallbacks(): void {\n this.unauthorizedCallbacks.forEach(callback => {\n try {\n callback();\n } catch (error) {\n console.error('[HttpService] Error in unauthorized callback:', error);\n }\n });\n }\n\n /**\n * 更新 authToken\n * @param token 新的 token\n */\n public setAuthToken(token: string | undefined): void {\n this.config.authToken = token;\n }\n\n /**\n * 更新 baseURL\n * @param baseURL 新的 baseURL\n */\n public setBaseURL(baseURL: string): void {\n this.config.baseURL = baseURL;\n this.axiosInstance.defaults.baseURL = baseURL.replace(/\\/$/, '');\n }\n\n /**\n * GET 请求\n * @param url 请求路径\n * @param config axios 配置\n * @returns 响应数据\n */\n public async get<T = any>(url: string, config?: AxiosRequestConfig): Promise<T> {\n const response = await this.axiosInstance.get<T>(url, config);\n return response.data;\n }\n\n /**\n * POST 请求\n * @param url 请求路径\n * @param data 请求体\n * @param config axios 配置\n * @returns 响应数据\n */\n public async post<T = any>(url: string, data?: any, config?: AxiosRequestConfig): Promise<T> {\n const response = await this.axiosInstance.post<T>(url, data, config);\n return response.data;\n }\n\n /**\n * PATCH 请求\n * @param url 请求路径\n * @param data 请求体\n * @param config axios 配置\n * @returns 响应数据\n */\n public async patch<T = any>(url: string, data?: any, config?: AxiosRequestConfig): Promise<T> {\n const response = await this.axiosInstance.patch<T>(url, data, config);\n return response.data;\n }\n\n /**\n * DELETE 请求\n * @param url 请求路径\n * @param config axios 配置\n * @returns 响应数据\n */\n public async delete<T = any>(url: string, config?: AxiosRequestConfig): Promise<T> {\n const response = await this.axiosInstance.delete<T>(url, config);\n return response.data;\n }\n\n /**\n * PUT 请求\n * @param url 请求路径\n * @param data 请求体\n * @param config axios 配置\n * @returns 响应数据\n */\n public async put<T = any>(url: string, data?: any, config?: AxiosRequestConfig): Promise<T> {\n const response = await this.axiosInstance.put<T>(url, data, config);\n return response.data;\n }\n\n /**\n * 获取原始 axios 实例(用于高级场景)\n */\n public getAxiosInstance(): AxiosInstance {\n return this.axiosInstance;\n }\n}\n","import { HttpService } from './http-service';\n\nexport * from './http-service';\nexport * from './types';\n\n/**\n * 导出 httpService 单例实例\n * 首次访问时自动初始化,无需手动调用初始化方法\n *\n * @example\n * ```typescript\n * import { httpService } from '@genie/agent-provider';\n *\n * // 按需设置 baseURL\n * httpService.setBaseURL('https://api.example.com');\n *\n * // 按需设置 authToken\n * httpService.setAuthToken('your-token');\n *\n * // 直接使用\n * const data = await httpService.get('/api/endpoint');\n * ```\n */\nexport const httpService = new Proxy({} as HttpService, {\n get(target, prop) {\n const instance = HttpService.getInstance();\n return (instance as any)[prop];\n },\n});\n\n","/**\n * AccountService - 账号状态管理单例\n *\n * 提供全局的账号状态管理,任何模块都可以:\n * - 获取当前账号: accountService.getAccount()\n * - 设置账号: accountService.setAccount(account)\n * - 订阅变化: accountService.subscribe(callback)\n * - 初始化时自动注册 HTTP 拦截器,注入账号相关 headers\n */\n\nimport type { Account } from '../backend/types';\nimport { httpService } from '../http';\n\n/**\n * 账号变化回调函数类型\n */\nexport type AccountChangeCallback = (account: Account | null) => void;\n\n/**\n * AccountService 类\n *\n * 单例模式,管理全局账号状态\n */\nclass AccountService {\n /** 当前账号 */\n private account: Account | null = null;\n\n /** 订阅者列表 */\n private listeners = new Set<AccountChangeCallback>();\n\n /** 是否已初始化(首次加载完成) */\n private initialized = false;\n\n /** 初始化等待队列 */\n private initPromise: Promise<Account | null> | null = null;\n private initResolve: ((account: Account | null) => void) | null = null;\n\n /** HTTP 请求拦截器 ID(未来可用于清理拦截器) */\n private requestInterceptorId: number | null = null;\n\n constructor() {\n // 创建初始化 Promise,允许其他模块等待首次加载完成\n this.initPromise = new Promise(resolve => {\n this.initResolve = resolve;\n });\n\n // 注册 HTTP 拦截器,注入账号相关 headers\n this.registerHttpInterceptor();\n }\n\n /**\n * 注册 HTTP 请求拦截器\n * 自动注入账号相关的 headers (X-User-Id, X-Enterprise-Id, X-Tenant-Id)\n */\n private registerHttpInterceptor(): void {\n try {\n this.requestInterceptorId = httpService.registerRequestInterceptor(config => {\n const account = this.getAccount();\n\n if (account?.uid) {\n config.headers = config.headers || {};\n config.headers['X-User-Id'] = account.uid;\n }\n\n if (account?.enterpriseId) {\n config.headers = config.headers || {};\n config.headers['X-Enterprise-Id'] = account.enterpriseId;\n config.headers['X-Tenant-Id'] = account.enterpriseId;\n }\n\n return config;\n });\n\n console.log('[AccountService] HTTP interceptor registered');\n } catch (error) {\n // HttpService 可能还未初始化,延迟注册\n console.warn('[AccountService] Failed to register HTTP interceptor, will retry on first request:', error);\n }\n }\n\n /**\n * 获取当前账号\n * @returns 当前账号,未登录或未加载时返回 null\n */\n getAccount(): Account | null {\n return this.account;\n }\n\n /**\n * 设置账号\n * @param account 账号信息,登出时传 null\n */\n setAccount(account: Account | null): void {\n const prev = this.account;\n const wasInitialized = this.initialized;\n this.account = account;\n\n // 首次设置时,标记为已初始化并 resolve 等待的 Promise\n if (!this.initialized) {\n this.initialized = true;\n this.initResolve?.(account);\n }\n\n // 首次初始化时总是通知订阅者,之后只有账号真正变化时才通知\n if (!wasInitialized || prev?.uid !== account?.uid) {\n this.notifyListeners();\n }\n }\n\n /**\n * 清除账号(登出)\n */\n clearAccount(): void {\n this.setAccount(null);\n }\n\n /**\n * 订阅账号变化\n * @param callback 变化时的回调函数\n * @returns 取消订阅函数\n */\n subscribe(callback: AccountChangeCallback): () => void {\n this.listeners.add(callback);\n return () => {\n this.listeners.delete(callback);\n };\n }\n\n /**\n * 等待首次账号加载完成\n * @returns Promise<Account | null>\n */\n waitForInit(): Promise<Account | null> {\n if (this.initialized) {\n return Promise.resolve(this.account);\n }\n return this.initPromise!;\n }\n\n /**\n * 是否已初始化\n */\n isInitialized(): boolean {\n return this.initialized;\n }\n\n /**\n * 是否已登录\n */\n isLoggedIn(): boolean {\n return this.account !== null;\n }\n\n /**\n * 通知所有订阅者\n */\n private notifyListeners(): void {\n this.listeners.forEach(callback => {\n try {\n callback(this.account);\n } catch (error) {\n console.error('[AccountService] Listener error:', error);\n }\n });\n }\n\n /**\n * 重置服务状态(仅用于测试)\n */\n _reset(): void {\n this.account = null;\n this.listeners.clear();\n this.initialized = false;\n this.initPromise = new Promise(resolve => {\n this.initResolve = resolve;\n });\n\n // 清理拦截器\n if (this.requestInterceptorId !== null) {\n try {\n httpService.ejectRequestInterceptor(this.requestInterceptorId);\n this.requestInterceptorId = null;\n } catch (error) {\n console.warn('[AccountService] Failed to eject interceptor during reset:', error);\n }\n }\n }\n}\n\n/**\n * 导出单例实例\n */\nexport const accountService = new AccountService();\n","/**\n * 并发控制工具\n *\n * 提供限制并发数量的工具函数,用于控制并行请求数量\n */\n\n/**\n * 并发执行任务,限制同时运行的任务数量\n *\n * @param tasks - 任务函数数组,每个函数返回 Promise\n * @param concurrency - 最大并发数,默认 5\n * @returns 所有任务的结果数组,顺序与输入一致\n *\n * @example\n * ```typescript\n * const urls = ['url1', 'url2', 'url3', 'url4', 'url5'];\n * const tasks = urls.map(url => () => fetch(url));\n * const results = await runWithConcurrency(tasks, 2); // 最多同时 2 个请求\n * ```\n */\nexport async function runWithConcurrency<T>(\n tasks: Array<() => Promise<T>>,\n concurrency: number = 5\n): Promise<T[]> {\n if (tasks.length === 0) {\n return [];\n }\n\n // 确保并发数至少为 1\n const limit = Math.max(1, concurrency);\n\n const results: T[] = new Array(tasks.length);\n let currentIndex = 0;\n\n async function runNext(): Promise<void> {\n while (currentIndex < tasks.length) {\n const index = currentIndex++;\n const task = tasks[index];\n results[index] = await task();\n }\n }\n\n // 启动 limit 个 worker\n const workers = Array(Math.min(limit, tasks.length))\n .fill(null)\n .map(() => runNext());\n\n await Promise.all(workers);\n\n return results;\n}\n\n/**\n * 并发执行任务,返回包含成功/失败状态的结果\n *\n * @param tasks - 任务函数数组\n * @param concurrency - 最大并发数,默认 5\n * @returns 所有任务的结果数组,包含状态信息\n *\n * @example\n * ```typescript\n * const results = await runWithConcurrencySettled(tasks, 3);\n * const successes = results.filter(r => r.status === 'fulfilled');\n * const failures = results.filter(r => r.status === 'rejected');\n * ```\n */\nexport async function runWithConcurrencySettled<T>(\n tasks: Array<() => Promise<T>>,\n concurrency: number = 5\n): Promise<PromiseSettledResult<T>[]> {\n if (tasks.length === 0) {\n return [];\n }\n\n const limit = Math.max(1, concurrency);\n const results: PromiseSettledResult<T>[] = new Array(tasks.length);\n let currentIndex = 0;\n\n async function runNext(): Promise<void> {\n while (currentIndex < tasks.length) {\n const index = currentIndex++;\n const task = tasks[index];\n try {\n const value = await task();\n results[index] = { status: 'fulfilled', value };\n } catch (reason) {\n results[index] = { status: 'rejected', reason };\n }\n }\n }\n\n const workers = Array(Math.min(limit, tasks.length))\n .fill(null)\n .map(() => runNext());\n\n await Promise.all(workers);\n\n return results;\n}\n\n/**\n * 创建一个并发限制器,可复用于多次调用\n *\n * @param concurrency - 最大并发数\n * @returns 并发限制器实例\n *\n * @example\n * ```typescript\n * const limiter = createConcurrencyLimiter(3);\n *\n * // 多个地方可以共用同一个限制器\n * const result1 = await limiter.run(() => fetch(url1));\n * const result2 = await limiter.run(() => fetch(url2));\n *\n * // 或者批量执行\n * const results = await limiter.runAll([\n * () => fetch(url1),\n * () => fetch(url2),\n * ]);\n * ```\n */\nexport function createConcurrencyLimiter(concurrency: number = 5) {\n const limit = Math.max(1, concurrency);\n let running = 0;\n const queue: Array<{\n task: () => Promise<unknown>;\n resolve: (value: unknown) => void;\n reject: (reason: unknown) => void;\n }> = [];\n\n async function processQueue(): Promise<void> {\n if (running >= limit || queue.length === 0) {\n return;\n }\n\n running++;\n const item = queue.shift()!;\n\n try {\n const result = await item.task();\n item.resolve(result);\n } catch (error) {\n item.reject(error);\n } finally {\n running--;\n processQueue();\n }\n }\n\n return {\n /**\n * 执行单个任务,受并发限制\n */\n run<T>(task: () => Promise<T>): Promise<T> {\n return new Promise<T>((resolve, reject) => {\n queue.push({\n task: task as () => Promise<unknown>,\n resolve: resolve as (value: unknown) => void,\n reject,\n });\n processQueue();\n });\n },\n\n /**\n * 批量执行任务,受并发限制\n */\n async runAll<T>(tasks: Array<() => Promise<T>>): Promise<T[]> {\n return Promise.all(tasks.map(task => this.run(task)));\n },\n\n /**\n * 获取当前正在运行的任务数\n */\n get runningCount(): number {\n return running;\n },\n\n /**\n * 获取队列中等待的任务数\n */\n get pendingCount(): number {\n return queue.length;\n },\n };\n}\n","/**\n * COS Upload Service\n *\n * 负责通过预签名 URL 上传文件到腾讯云 COS\n *\n * 上传流程:\n * 1. 生成 objectKey (uploads/{timestamp}-{random}-{filename})\n * 2. 批量请求后端获取预签名 URL (同时包含 upload_url 和 download_url)\n * 3. 使用 upload_url 并发上传文件到 COS\n * 4. 返回 download_url 作为访问地址\n *\n * 注意:文件级缓存(避免重复上传相同文件)由上层 agent-ui 的 upload-cache 处理\n */\n\nimport { httpService } from '../../../http/index.js';\nimport type { Logger } from '../../client/types.js';\nimport { runWithConcurrencySettled } from '../../utils/index.js';\n\n/**\n * 预签名 URL 请求参数\n */\nexport interface PresignedURLRequest {\n /** 对象路径数组 */\n object_keys: string[];\n}\n\n/**\n * 预签名 URL 响应项\n */\nexport interface PresignedURLItem {\n /** 对象路径 */\n object_key: string;\n /** 上传 URL (PUT) */\n upload_url: string;\n /** 下载 URL (GET) */\n download_url: string;\n}\n\n/**\n * 预签名 URL 响应\n */\nexport interface PresignedURLResponse {\n /** 过期时间 (秒) */\n expire: number;\n /** 预签名 URL 列表 */\n items: PresignedURLItem[];\n}\n\n/**\n * API 响应包装类型\n */\ninterface ApiResponse<T> {\n code: number;\n message: string;\n data?: T;\n}\n\n/**\n * 上传结果\n */\nexport interface UploadResult {\n /** 是否成功 */\n success: boolean;\n /** 访问 URL (成功时返回) */\n url?: string;\n /** 对象路径 */\n objectKey?: string;\n /** 错误信息 (失败时返回) */\n error?: string;\n}\n\n/**\n * COS Upload Service 配置\n */\nexport interface CosUploadServiceOptions {\n /** Logger 实例 */\n logger?: Logger;\n /** 上传并发数,默认 3 */\n uploadConcurrency?: number;\n}\n\n/**\n * COS 上传服务\n *\n * @example\n * ```typescript\n * const service = new CosUploadService({\n * logger: console,\n * uploadConcurrency: 3,\n * });\n *\n * const result = await service.uploadFile(file);\n * if (result.success) {\n * console.log('File URL:', result.url);\n * }\n * ```\n */\nexport class CosUploadService {\n private logger?: Logger;\n private uploadConcurrency: number;\n\n constructor(options: CosUploadServiceOptions = {}) {\n this.logger = options.logger;\n this.uploadConcurrency = options.uploadConcurrency ?? 3;\n }\n\n /**\n * 生成唯一的 objectKey\n *\n * 格式: uploads/{timestamp}-{randomId}-{encodedFilename}\n */\n private generateObjectKey(filename: string): string {\n const timestamp = Date.now();\n const randomId = Math.random().toString(36).substring(2, 10);\n const encodedFilename = encodeURIComponent(filename);\n return `uploads/${timestamp}-${randomId}-${encodedFilename}`;\n }\n\n /**\n * 批量获取预签名 URL\n *\n * POST /console/as/support/presigned_url\n */\n private async getPresignedUrls(objectKeys: string[]): Promise<PresignedURLResponse> {\n const apiResponse = await httpService.post<ApiResponse<PresignedURLResponse>>(\n '/console/as/support/presigned_url',\n { object_keys: objectKeys }\n );\n\n if (!apiResponse.data) {\n throw new Error('No data in presigned URL response');\n }\n\n return apiResponse.data;\n }\n\n /**\n * 上传单个文件到 COS\n *\n * @param file - 要上传的文件\n * @returns 上传结果,包含访问 URL 或错误信息\n */\n async uploadFile(file: File): Promise<UploadResult> {\n const filename = file.name;\n this.logger?.info(`[CosUploadService] Uploading file: ${filename}`);\n\n try {\n // 1. 生成 objectKey\n const objectKey = this.generateObjectKey(filename);\n this.logger?.debug(`[CosUploadService] Generated objectKey: ${objectKey}`);\n\n // 2. 获取预签名 URL\n const presignedResponse = await this.getPresignedUrls([objectKey]);\n const presignedItem = presignedResponse.items[0];\n if (!presignedItem) {\n throw new Error('No presigned URL item returned');\n }\n\n // 3. 上传文件到 COS(使用原生 fetch,因为这是上传到外部 COS)\n const uploadResponse = await fetch(presignedItem.upload_url, {\n method: 'PUT',\n body: file,\n headers: {\n 'Content-Type': file.type || 'application/octet-stream',\n },\n });\n\n if (!uploadResponse.ok) {\n const errorText = await uploadResponse.text().catch(() => uploadResponse.statusText);\n throw new Error(`COS upload failed: ${uploadResponse.status} ${errorText}`);\n }\n this.logger?.debug('[CosUploadService] File uploaded to COS');\n\n this.logger?.info(`[CosUploadService] Upload success: ${filename}`);\n return {\n success: true,\n url: presignedItem.download_url,\n objectKey,\n };\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : 'Unknown error';\n this.logger?.error(`[CosUploadService] Upload failed: ${filename}`, error);\n return {\n success: false,\n error: errorMessage,\n };\n }\n }\n\n /**\n * 批量上传文件到 COS\n *\n * 使用并发控制,限制同时上传的文件数量\n *\n * @param files - 要上传的文件数组\n * @returns 所有文件的上传结果\n */\n async uploadFiles(files: File[]): Promise<{\n success: boolean;\n urls?: string[];\n expireSeconds?: number;\n error?: string;\n results: UploadResult[];\n }> {\n if (files.length === 0) {\n return { success: true, urls: [], results: [] };\n }\n\n this.logger?.info(`[CosUploadService] Uploading ${files.length} file(s) with concurrency ${this.uploadConcurrency}`);\n\n try {\n // 1. 为所有文件生成 objectKey\n const fileInfos = files.map(file => ({\n file,\n objectKey: this.generateObjectKey(file.name),\n }));\n\n // 2. 批量获取预签名 URL\n const objectKeys = fileInfos.map(info => info.objectKey);\n const presignedResponse = await this.getPresignedUrls(objectKeys);\n this.logger?.debug(`[CosUploadService] Got ${presignedResponse.items.length} presigned URLs`);\n\n // 3. 验证返回的数量\n if (presignedResponse.items.length !== fileInfos.length) {\n throw new Error(`Expected ${fileInfos.length} presigned URLs, got ${presignedResponse.items.length}`);\n }\n\n // 4. 创建上传任务(使用后端返回的顺序来匹配)\n // 后端会在 objectKey 前加上 userID,所以我们使用数组索引来匹配\n const uploadTasks = fileInfos.map(({ file }, index) => async (): Promise<UploadResult> => {\n const presignedItem = presignedResponse.items[index];\n if (!presignedItem) {\n return {\n success: false,\n error: `No presigned URL for ${file.name}`,\n objectKey: fileInfos[index].objectKey,\n };\n }\n\n try {\n // 使用原生 fetch 上传到 COS\n const uploadResponse = await fetch(presignedItem.upload_url, {\n method: 'PUT',\n body: file,\n headers: {\n 'Content-Type': file.type || 'application/octet-stream',\n },\n });\n\n if (!uploadResponse.ok) {\n const errorText = await uploadResponse.text().catch(() => uploadResponse.statusText);\n return {\n success: false,\n error: `COS upload failed: ${uploadResponse.status} ${errorText}`,\n objectKey: presignedItem.object_key,\n };\n }\n\n this.logger?.debug(`[CosUploadService] Uploaded: ${file.name}`);\n return {\n success: true,\n url: presignedItem.download_url,\n objectKey: presignedItem.object_key,\n };\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : 'Unknown error';\n return {\n success: false,\n error: errorMessage,\n objectKey: presignedItem.object_key,\n };\n }\n });\n\n // 5. 并发执行上传任务\n const settledResults = await runWithConcurrencySettled(uploadTasks, this.uploadConcurrency);\n\n // 6. 处理结果\n const results: UploadResult[] = settledResults.map((result, index) => {\n if (result.status === 'fulfilled') {\n return result.value;\n }\n // rejected 的情况\n return {\n success: false,\n error: result.reason instanceof Error ? result.reason.message : 'Unknown error',\n objectKey: fileInfos[index].objectKey,\n };\n });\n\n const urls = results.filter(r => r.success && r.url).map(r => r.url!);\n const failedResults = results.filter(r => !r.success);\n\n if (failedResults.length > 0) {\n const failedErrors = failedResults.map(r => r.error).join('; ');\n return {\n success: false,\n error: `${failedResults.length} file(s) failed: ${failedErrors}`,\n expireSeconds: presignedResponse.expire,\n results,\n };\n }\n\n this.logger?.info(`[CosUploadService] All ${files.length} file(s) uploaded successfully`);\n return {\n success: true,\n urls,\n expireSeconds: presignedResponse.expire,\n results,\n };\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : 'Unknown error';\n this.logger?.error('[CosUploadService] Batch upload failed', error);\n return {\n success: false,\n error: errorMessage,\n results: files.map(() => ({ success: false, error: errorMessage })),\n };\n }\n }\n}\n\nexport default CosUploadService;\n","/**\n * E2B Filesystem Implementation\n *\n * Provides FilesResource implementation using E2B Sandbox SDK.\n * Directly uses e2b SDK types.\n *\n * @see https://e2b.dev/docs/filesystem/read-write\n * @see https://e2b.dev/docs/filesystem/watch\n */\n\nimport { type EntryInfo, Sandbox, type WatchHandle,type WriteInfo } from 'e2b';\n\nimport type {\n E2BSandboxConnectionInfo,\n FilesResource,\n FilesystemEvent,\n FilesystemListOpts,\n FilesystemRequestOpts,\n WatchOpts,\n WriteEntry\n} from '../../types.js';\n\n/**\n * E2B Filesystem Implementation\n *\n * Wraps E2B Sandbox SDK's filesystem operations to implement FilesResource interface.\n *\n * @example\n * ```typescript\n * const fs = await E2BFilesystem.connect({\n * sandboxId: 'sandbox-123',\n * apiKey: 'e2b_xxx'\n * });\n *\n * // Read/write files\n * await fs.write('/test.txt', 'Hello World');\n * const content = await fs.read('/test.txt');\n *\n * // Watch for changes\n * const handle = await fs.watchDir('/workspace', (event) => {\n * console.log('File changed:', event);\n * });\n * ```\n */\nexport class E2BFilesystem implements FilesResource {\n private sandbox: Sandbox;\n\n constructor(sandbox: Sandbox) {\n this.sandbox = sandbox;\n }\n\n /**\n * Connect to an E2B Sandbox and create filesystem instance\n */\n static async connect(info: E2BSandboxConnectionInfo): Promise<E2BFilesystem> {\n const sandbox = await Sandbox.connect(info.sandboxId, {\n domain: info.domain,\n apiUrl: info.apiUrl,\n requestTimeoutMs: info.requestTimeoutMs,\n debug: info.debug,\n headers: info.headers\n });\n return new E2BFilesystem(sandbox);\n }\n\n /**\n * Get the underlying E2B Sandbox instance\n */\n getSandbox(): Sandbox {\n return this.sandbox;\n }\n\n // ============================================\n // Read 方法重载实现\n // ============================================\n\n read(path: string, opts?: FilesystemRequestOpts & { format?: 'text' }): Promise<string>;\n read(path: string, opts: FilesystemRequestOpts & { format: 'bytes' }): Promise<Uint8Array>;\n read(path: string, opts: FilesystemRequestOpts & { format: 'blob' }): Promise<Blob>;\n read(path: string, opts: FilesystemRequestOpts & { format: 'stream' }): Promise<ReadableStream<Uint8Array>>;\n read(path: string, opts?: FilesystemRequestOpts & { format?: string }): Promise<string | Uint8Array | Blob | ReadableStream<Uint8Array>> {\n return this.sandbox.files.read(path, opts as any);\n }\n\n // ============================================\n // Write 方法重载实现\n // ============================================\n\n write(path: string, data: string | ArrayBuffer | Blob | ReadableStream, opts?: FilesystemRequestOpts): Promise<WriteInfo>;\n write(files: WriteEntry[], opts?: FilesystemRequestOpts): Promise<WriteInfo[]>;\n write(pathOrFiles: string | WriteEntry[], dataOrOpts?: string | ArrayBuffer | Blob | ReadableStream | FilesystemRequestOpts, opts?: FilesystemRequestOpts): Promise<WriteInfo | WriteInfo[]> {\n if (Array.isArray(pathOrFiles)) {\n // Batch write: write(files: WriteEntry[], opts?: FilesystemRequestOpts)\n return this.sandbox.files.write(pathOrFiles, dataOrOpts as FilesystemRequestOpts);\n }\n // Single file write: write(path: string, data: ..., opts?: FilesystemRequestOpts)\n return this.sandbox.files.write(pathOrFiles, dataOrOpts as string | ArrayBuffer | Blob | ReadableStream, opts);\n }\n\n // ============================================\n // 其他方法实现\n // ============================================\n\n async list(path: string, opts?: FilesystemListOpts): Promise<EntryInfo[]> {\n return this.sandbox.files.list(path, opts);\n }\n\n async exists(path: string, opts?: FilesystemRequestOpts): Promise<boolean> {\n return this.sandbox.files.exists(path, opts);\n }\n\n async makeDir(path: string, opts?: FilesystemRequestOpts): Promise<boolean> {\n return this.sandbox.files.makeDir(path, opts);\n }\n\n async remove(path: string, opts?: FilesystemRequestOpts): Promise<void> {\n return this.sandbox.files.remove(path, opts);\n }\n\n async rename(oldPath: string, newPath: string, opts?: FilesystemRequestOpts): Promise<EntryInfo> {\n return this.sandbox.files.rename(oldPath, newPath, opts);\n }\n\n async getInfo(path: string, opts?: FilesystemRequestOpts): Promise<EntryInfo> {\n return this.sandbox.files.getInfo(path, opts);\n }\n\n async watchDir(\n path: string,\n onEvent: (event: FilesystemEvent) => void | Promise<void>,\n opts?: WatchOpts & { onExit?: (err?: Error) => void | Promise<void> }\n ): Promise<WatchHandle> {\n return this.sandbox.files.watchDir(path, onEvent, opts);\n }\n}\n\nexport default E2BFilesystem;\n","/**\n * Cloud Agent Provider\n *\n * REST API based agent provider that manages cloud-hosted agents.\n * Implements the AgentProvider interface for the unified agent API.\n */\n\nimport { accountService } from '../../../account/index.js';\nimport type { SupportScene } from '../../../backend/types.js';\nimport { httpService } from '../../../http/index.js';\nimport type {\n AgentProvider,\n CloudAgentState,\n FilesResource,\n FilesystemProvider,\n ListAgentOptions,\n ListAgentResult,\n Logger,\n PaginationInfo,\n PickFileParams,\n PickFileResponse,\n UploadFileParams,\n UploadFileResponse,\n} from '../../client/types.js';\nimport type {\n AgentStatus,\n ClientCapabilities,\n E2BSandboxConnectionInfo,\n ModelInfo\n} from '../../types.js';\nimport { ApiResponse, ArchiveConversationResponse, Conversation as ConversationDto, CreateConversationRequest, CreateConversationResponse, DeleteConversationRequest, GetConversationResponse, GetConversationSessionResponse, ListConversationRequest, ListConversationResponse, PatchConversationRequest, PatchConversationResponse, ProductConfigResponse } from './api-types.js';\nimport { CloudAgentConnection } from './cloud-connection.js';\nimport { CosUploadService } from './cos-upload-service.js';\nimport { E2BFilesystem } from './e2b-filesystem.js';\n\n/**\n * Agent data stored in cloud backend\n */\n// @ts-expect-error 没对齐\ninterface CloudAgentData {\n id: string;\n name?: string;\n description?: string;\n endpoint: string;\n authToken?: string;\n headers?: Record<string, string>;\n metadata?: Record<string, unknown>;\n createdAt?: string;\n updatedAt?: string;\n}\n\n/**\n * Configuration for CloudAgentProvider\n */\nexport interface CloudAgentProviderOptions {\n /** Base endpoint URL for agent management API (e.g., 'https://api.example.com') */\n endpoint: string;\n /** Authorization token */\n authToken?: string;\n /** Custom headers */\n headers?: Record<string, string>;\n /** Logger instance */\n logger?: Logger;\n /** Client capabilities (sent during agent initialization) */\n clientCapabilities?: ClientCapabilities;\n}\n\n/**\n * Normalize a path by resolving `.` and `..` segments\n * This is a simplified version that works in browser environment\n */\nfunction normalizePath(path: string): string {\n const segments = path.split('/');\n const result: string[] = [];\n\n for (const segment of segments) {\n if (segment === '..') {\n // Don't pop if we're at root level\n if (result.length > 0 && result[result.length - 1] !== '') {\n result.pop();\n }\n } else if (segment !== '.' && segment !== '') {\n result.push(segment);\n }\n }\n\n // Preserve leading slash for absolute paths\n return (path.startsWith('/') ? '/' : '') + result.join('/');\n}\n\n/**\n * Resolve agent:/// URI to filesystem path\n *\n * 只处理 agent:// 协议的 URI,raw path 直接透传不做任何处理\n *\n * Supported formats:\n * - `agent:///workspace/{path}` → `/workspace/{path}`\n * - `agent:///plans/{path}` → `/root/.codebuddy/plans/{path}`\n * - `agent:///{path}` → `/{path}`\n * - Raw paths (e.g. `/foo/bar`) → 直接透传,不处理\n *\n * Security: Path traversal attacks are prevented by normalizing paths\n * and verifying they stay within expected boundaries.\n */\nfunction resolveAgentUri(input: string): string {\n // 只处理 agent:// 协议,raw path 直接返回\n if (!input.startsWith('agent://')) {\n return input;\n }\n\n const path = input.slice('agent://'.length);\n if (!path.startsWith('/')) {\n return input;\n }\n\n // Normalize path to prevent traversal attacks\n const normalizedPath = normalizePath(path);\n\n // 路径映射: /plans/{path} → /root/.codebuddy/plans/{path}\n if (normalizedPath.startsWith('/plans/') || normalizedPath === '/plans') {\n const mappedPath = '/root/.codebuddy' + normalizedPath;\n // Verify the mapped path stays within /root/.codebuddy/plans\n const finalPath = normalizePath(mappedPath);\n if (!finalPath.startsWith('/root/.codebuddy/plans')) {\n throw new Error(`Invalid path: path traversal detected in ${input}`);\n }\n return finalPath;\n }\n\n return normalizedPath;\n}\n\n/**\n * Create a FilesResource wrapper that resolves agent:/// URIs\n */\nfunction createAgentFilesystem(fs: E2BFilesystem): FilesResource {\n return {\n read: (path: string, opts?: any) => fs.read(resolveAgentUri(path), opts),\n write: (pathOrFiles: any, dataOrOpts?: any, opts?: any) => {\n if (Array.isArray(pathOrFiles)) {\n const resolved = pathOrFiles.map(f => ({ ...f, path: resolveAgentUri(f.path) }));\n return fs.write(resolved, dataOrOpts);\n }\n return fs.write(resolveAgentUri(pathOrFiles), dataOrOpts, opts);\n },\n list: (path: string, opts?: any) => fs.list(resolveAgentUri(path), opts),\n exists: (path: string, opts?: any) => fs.exists(resolveAgentUri(path), opts),\n makeDir: (path: string, opts?: any) => fs.makeDir(resolveAgentUri(path), opts),\n remove: (path: string, opts?: any) => fs.remove(resolveAgentUri(path), opts),\n rename: (oldPath: string, newPath: string, opts?: any) =>\n fs.rename(resolveAgentUri(oldPath), resolveAgentUri(newPath), opts),\n getInfo: (path: string, opts?: any) => fs.getInfo(resolveAgentUri(path), opts),\n watchDir: (path: string, onEvent: any, opts?: any) =>\n fs.watchDir(resolveAgentUri(path), onEvent, opts),\n } as FilesResource;\n}\n\n/**\n * CloudAgentProvider - Manages cloud-hosted agents via REST API\n *\n * API Endpoints (base path: /console/as):\n * - POST {endpoint}/console/as/conversations - Create new conversation\n * - GET {endpoint}/console/as/conversations/{id} - Get conversation details\n * - GET {endpoint}/console/as/conversations - List all conversations\n * - POST {endpoint}/console/as/conversations/{id}/delete - Delete conversation\n * - POST {endpoint}/console/as/conversations/{id}/archive - Archive conversation\n * - POST {endpoint}/console/as/conversations/{id} - Update conversation (rename)\n * - GET {endpoint}/console/as/conversations/{id}/session - Get conversation session (includes sandboxId)\n * - GET {endpoint}/console/enterprises/{id}/models - Get available models\n *\n * The provider stores agent endpoint configurations in the cloud backend.\n * When connect() is called, it creates a CloudAgentConnection to the agent's\n * endpoint and returns an Agent instance.\n *\n * @example\n * ```typescript\n * const provider = new CloudAgentProvider({\n * endpoint: 'https://staging-copilot.tencent.com',\n * authToken: 'token'\n * });\n *\n * // List all agents (uses default pagination and sorting)\n * const allAgents = await provider.list();\n *\n * // List agents with custom pagination\n * const page2 = await provider.list({\n * page: 2,\n * size: 50\n * });\n *\n * // List agents with filtering\n * const runningAgents = await provider.list({\n * filters: [\n * { field: 'status', value: 'running' }\n * ]\n * });\n *\n * // List agents with custom sorting\n * const sortedAgents = await provider.list({\n * sort: {\n * orderBy: 'createdAt',\n * order: 'desc'\n * }\n * });\n *\n * // List agents created in last 14 days with multiple filters\n * const recentAgents = await provider.list({\n * dayRange: 14,\n * filters: [\n * { field: 'status', value: 'running,stopped' }\n * ],\n * page: 1,\n * size: 20\n * });\n *\n * // Get agent state\n * const state = await provider.get('agent-id');\n *\n * // Connect to agent\n * const agent = await provider.connect('agent-id');\n *\n * // Use agent\n * const session = await agent.sessions.create({ cwd: '/workspace' });\n *\n * // Get available models\n * const models = await provider.getModels('my-repo');\n * ```\n */\nexport class CloudAgentProvider implements AgentProvider<CloudAgentConnection>, FilesystemProvider {\n private options: CloudAgentProviderOptions;\n private logger?: Logger;\n private requestInterceptorId: number | null = null;\n\n /** Cache for filesystem instances (keyed by agentId) */\n private filesystemCache: Map<string, FilesResource> = new Map();\n\n /** Cache for agent connections (keyed by endpoint link) */\n private connectionCache: Map<string, CloudAgentConnection> = new Map();\n\n /** COS upload service instance */\n private cosUploadService: CosUploadService;\n\n /** Event listeners for provider-level events */\n private eventListeners: Map<string, Set<(...args: any[]) => void>> = new Map();\n\n constructor(options: CloudAgentProviderOptions) {\n this.options = options;\n this.logger = options.logger;\n\n // 设置 baseURL(如果提供)\n if (options.endpoint) {\n httpService.setBaseURL(options.endpoint);\n }\n\n // 设置 authToken(如果提供)\n if (options.authToken) {\n httpService.setAuthToken(options.authToken);\n }\n\n // 注册请求拦截器,只注入 options 中的 headers\n if (options.headers && Object.keys(options.headers).length > 0) {\n this.requestInterceptorId = httpService.registerRequestInterceptor(config => {\n config.headers = config.headers || {};\n // 注入 options 中的自定义 headers\n Object.entries(options.headers!).forEach(([key, value]) => {\n config.headers[key] = value;\n });\n return config;\n });\n }\n\n // Initialize COS upload service\n this.cosUploadService = new CosUploadService({\n logger: this.logger,\n });\n }\n\n /**\n * Dispose the provider and clean up resources\n */\n dispose(): void {\n this.filesystemCache.clear();\n this.connectionCache.clear();\n\n // 移除注册的拦截器\n if (this.requestInterceptorId !== null) {\n httpService.ejectRequestInterceptor(this.requestInterceptorId);\n }\n }\n\n // ============================================\n // FilesystemProvider Implementation\n // ============================================\n\n /**\n * Get the filesystem provider (returns self)\n */\n get filesystem(): FilesystemProvider {\n return this;\n }\n\n /**\n * Get filesystem resource for an agent\n *\n * Creates or returns cached filesystem instance for the agent's sandbox.\n * The filesystem supports both `agent:///` URIs and raw paths.\n *\n * @param agentId - Agent ID to get filesystem for\n * @returns FilesResource instance for the agent's sandbox (with URI support)\n *\n * @example\n * ```typescript\n * const fs = await provider.getFilesystem(agentId);\n *\n * // Use agent:/// URIs\n * const content = await fs.read('agent:///files/src/app.ts');\n * await fs.write('agent:///artifacts/output.txt', 'Hello');\n *\n * // Raw paths still work (backward compatible)\n * const content2 = await fs.read('/src/app.ts');\n * ```\n */\n async getFilesystem(agentId: string): Promise<FilesResource> {\n // Check cache first\n const cached = this.filesystemCache.get(agentId);\n if (cached) {\n return cached;\n }\n\n // Get sandbox info from backend\n const info = await this.getSandboxInfo(agentId);\n\n // Create E2BFilesystem and wrap with URI support\n const e2bFilesystem = await E2BFilesystem.connect(info);\n const filesystem = createAgentFilesystem(e2bFilesystem);\n\n this.filesystemCache.set(agentId, filesystem);\n\n this.logger?.debug(`Created filesystem for agent: ${agentId}`);\n return filesystem;\n }\n\n /**\n * Get sandbox information from backend\n *\n * Uses GET {endpoint}/console/as/conversations/{agentId}/session\n * to retrieve sandbox information. Extracts sandboxId from the session response\n * and constructs the apiUrl for E2B proxy.\n *\n * @param agentId - Agent ID\n * @returns E2B Sandbox connection information with sandboxId and apiUrl\n */\n private async getSandboxInfo(agentId: string): Promise<E2BSandboxConnectionInfo> {\n const apiResponse = await httpService.get<ApiResponse<GetConversationSessionResponse>>(\n `/console/as/conversations/${agentId}/session`\n );\n\n if (!apiResponse.data) {\n throw new Error('No data in API response');\n }\n\n // Build E2BSandboxConnectionInfo from session response\n // apiUrl is from response endpoint with /v2 replaced by /console\n const apiUrl = apiResponse.data.e2bEndpoint.replace(/^http:\\/\\//, 'https://').replace('/v2', '/console');\n const currentEnterpriseId = localStorage.getItem('currentEnterpriseId');\n\n return {\n sandboxId: apiResponse.data.sandboxId,\n apiUrl,\n accessToken: apiResponse.data.token,\n headers: {\n // 只有当 enterpriseId 存在时才添加该 header\n ...(currentEnterpriseId && { 'X-Enterprise-Id': currentEnterpriseId }),\n },\n };\n }\n\n /**\n * Get agent state by ID\n */\n async get(agentId: string): Promise<CloudAgentState | undefined> {\n try {\n const apiResponse = await httpService.get<ApiResponse<GetConversationResponse>>(\n `/console/as/conversations/${agentId}`\n );\n\n if (!apiResponse.data) {\n throw new Error('No data in API response');\n }\n return this.toAgentState(apiResponse.data);\n } catch (error: any) {\n // axios will throw error on 404\n if (error.response?.status === 404) {\n return undefined;\n }\n this.logger?.error(`Failed to get agent ${agentId}:`, error);\n throw error;\n }\n }\n\n /**\n * List all agent states with pagination information\n *\n * @param options - Optional query parameters for filtering, sorting, and pagination\n * @returns Object containing agents array and pagination info\n */\n async list(options?: ListAgentOptions): Promise<ListAgentResult<CloudAgentState>> {\n try {\n // Build request parameters with defaults and user overrides\n const params: ListConversationRequest = {\n // Default values\n page: 1,\n size: 30,\n sort: {\n order: 'desc',\n orderBy: 'status'\n },\n // User overrides\n ...options && {\n ...(options.dayRange !== undefined && { dayRange: options.dayRange }),\n ...(options.page !== undefined && { page: options.page }),\n ...(options.size !== undefined && { size: options.size }),\n ...(options.sort !== undefined && { sort: options.sort }),\n ...(options.filters !== undefined && { filters: options.filters }),\n ...(options.title !== undefined && { title: options.title }),\n }\n };\n\n const url = this.buildGetUrl('/console/as/conversations/', params);\n const apiResponse = await httpService.get<ApiResponse<ListConversationResponse>>(url);\n\n if (!apiResponse.data) {\n throw new Error('No data in API response');\n }\n\n const agents = apiResponse.data.conversations.map(a => this.toAgentState(a));\n const pagination: PaginationInfo = apiResponse.data.pagination;\n\n console.log('[CloudAgentProvider] API response:', {\n agentsCount: agents.length,\n pagination\n });\n\n return { agents, pagination };\n } catch (error) {\n this.logger?.error('Failed to list agents:', error);\n throw error;\n }\n }\n\n /**\n * Create a new conversation\n * POST {endpoint}/console/as/conversations\n */\n async create(): Promise<string> {\n try {\n // All fields are optional in CreateConversationRequest\n const createPayload: CreateConversationRequest = {\n prompt: '',\n model: 'deepseek-r1',\n };\n const apiResponse = await httpService.post<ApiResponse<CreateConversationResponse>>(\n '/console/as/conversations/',\n createPayload\n );\n\n if (!apiResponse.data) {\n throw new Error('No data in API response');\n }\n\n this.logger?.info(`Created conversation: ${apiResponse.data.id}`);\n return apiResponse.data.id;\n } catch (error) {\n this.logger?.error('Failed to create conversation:', error);\n throw error;\n }\n }\n\n /**\n * Connect to an agent and return the connection\n *\n * This method:\n * 1. Fetches the agent configuration from the backend\n * 2. Checks if an existing connection can be reused (based on endpoint link)\n * 3. Creates a CloudAgentConnection to the agent's endpoint if not cached\n * 4. Saves session connection info to the connection\n * 5. Connects and initializes the connection\n * 6. Returns the connected CloudAgentConnection\n *\n * Connection caching:\n * - Connections are cached by endpoint link to enable reuse\n * - CloudAgentConnection is responsible for handling connection health checks\n * and token expiration internally\n */\n async connect(agentId: string): Promise<CloudAgentConnection> {\n // Fetch conversation session data from backend\n let sessionApiResponse: ApiResponse<GetConversationSessionResponse>;\n try {\n sessionApiResponse = await httpService.get<ApiResponse<GetConversationSessionResponse>>(\n `/console/as/conversations/${agentId}/session`\n );\n\n if (!sessionApiResponse.data) {\n throw new Error('No data in API response');\n }\n } catch (error: any) {\n // axios will throw error on 404\n if (error.response?.status === 404) {\n throw new Error(`Agent not found: ${agentId}`);\n }\n throw new Error(`Failed to get agent for connection: ${error.message}`);\n }\n\n const sessionData = sessionApiResponse.data;\n const endpoint = sessionData.link.replace(/^http:\\/\\//, 'https://');\n const cwd = sessionApiResponse.data.cwd || '/workspace';\n\n // Fetch conversation details to get name, createdAt and status\n let agentName: string | undefined;\n let agentCreatedAt: Date | undefined;\n let agentStatus: string | undefined;\n\n try {\n const agentApiResponse = await httpService.get<ApiResponse<GetConversationResponse>>(\n `/console/as/conversations/${agentId}`\n );\n if (agentApiResponse.data) {\n agentName = agentApiResponse.data.name;\n agentCreatedAt = agentApiResponse.data.createdAt ? new Date(agentApiResponse.data.createdAt) : undefined;\n // 优先使用 sessionStatus(会话状态),fallback 到 status(Agent 状态)\n agentStatus = agentApiResponse.data.sessionStatus || agentApiResponse.data.status;\n }\n } catch (error) {\n // Ignore errors when fetching conversation details (non-critical)\n this.logger?.debug(`Failed to fetch conversation details for ${agentId}:`, error);\n }\n\n // TEMP FIX: Disable connection cache to ensure GET long-polling reconnects\n // TODO: Remove this after backend implements history message push on every session/load\n const existingConnection = this.connectionCache.get(endpoint);\n if (existingConnection) {\n this.connectionCache.delete(endpoint);\n }\n\n // Prepare client capabilities with cwd in _meta (consistent with LocalAgentProvider)\n const clientCapabilities = {\n ...this.options.clientCapabilities,\n _meta: {\n ...this.options.clientCapabilities?._meta,\n 'codebuddy.ai': {\n ...this.options.clientCapabilities?._meta?.['codebuddy.ai'],\n cwd,\n }\n }\n };\n\n // Create connection to agent's endpoint\n // Note: CloudAgentConnection is responsible for:\n // - Connection health checks (reconnection on disconnect)\n // - Token expiration handling (refresh or re-authentication)\n const connection = new CloudAgentConnection(sessionData.sessionId || agentId, {\n endpoint,\n authToken: sessionData.token,\n // Backend does not provide custom headers currently\n // headers: data.headers,\n logger: this.logger,\n clientCapabilities\n }, cwd);\n\n // Save session connection info to the connection\n connection.setSessionConnectionInfo({\n sessionId: sessionData.sessionId,\n agentId: sessionData.id,\n link: sessionData.link,\n token: sessionData.token,\n sandboxId: sessionData.sandboxId,\n expireAt: sessionData.expireAt,\n cwd\n });\n\n // Connect and initialize\n try {\n await connection.connect();\n } catch (error) {\n this.logger?.error(`Failed to connect to agent ${agentId}:`, error);\n throw error;\n }\n\n // Cache the connection\n this.connectionCache.set(endpoint, connection);\n\n // Clean up cache when connection is disconnected\n connection.once('disconnected', () => {\n this.connectionCache.delete(endpoint);\n this.logger?.debug(`Connection removed from cache: ${endpoint}`);\n });\n\n this.logger?.info(`Connected to agent: ${agentId}`);\n\n // Emit sessionCreated event for agent-new-adapter to create conversation\n // SessionInfo format: Use agentId as both id and agentId (1:1 mapping per SessionManager)\n this.emitEvent('sessionCreated', {\n id: agentId, // Use agentId as sessionId (1:1 mapping in current design)\n agentId: agentId,\n name: agentName,\n status: agentStatus as AgentStatus,\n cwd: cwd,\n createdAt: agentCreatedAt\n });\n\n return connection;\n }\n\n /**\n * Delete a conversation by ID\n * POST {endpoint}/console/as/conversations/{agentId}/delete\n */\n async delete(agentId: string): Promise<boolean> {\n try {\n const requestBody: DeleteConversationRequest = { id: agentId };\n await httpService.post(\n `/console/as/conversations/${agentId}/delete`,\n requestBody\n );\n\n return true;\n } catch (error: any) {\n // axios will throw error on 404\n if (error.response?.status === 404) {\n return false;\n }\n this.logger?.error(`Failed to delete conversation ${agentId}:`, error);\n throw error;\n }\n }\n\n /**\n * Archive a conversation by ID\n * POST {endpoint}/console/as/conversations/{agentId}/archive\n *\n * @param agentId - Conversation ID to archive\n * @returns ArchiveConversationResponse containing the archived conversation ID\n *\n * @example\n * ```typescript\n * const result = await provider.archive('agent-123');\n * console.log('Archived conversation:', result.id);\n * ```\n */\n async archive(agentId: string): Promise<ArchiveConversationResponse> {\n try {\n const apiResponse = await httpService.post<ApiResponse<ArchiveConversationResponse>>(\n `/console/as/conversations/${agentId}/archive`\n );\n\n if (!apiResponse.data) {\n // 如果后端没有返回 data,使用传入的 agentId\n this.logger?.info(`Archived conversation: ${agentId}`);\n return { id: agentId };\n }\n\n this.logger?.info(`Archived conversation: ${apiResponse.data.id}`);\n return apiResponse.data;\n } catch (error) {\n this.logger?.error(`Failed to archive conversation ${agentId}:`, error);\n throw error;\n }\n }\n\n /**\n * Rename a conversation by ID\n * POST {endpoint}/console/as/conversations/{agentId}\n *\n * @param agentId - Conversation ID to rename\n * @param title - New title for the conversation\n * @returns PatchConversationResponse containing the renamed conversation ID\n *\n * @example\n * ```typescript\n * const result = await provider.rename('agent-123', 'New Title');\n * console.log('Renamed conversation:', result.id);\n * ```\n */\n async rename(agentId: string, title: string): Promise<PatchConversationResponse> {\n try {\n const body: PatchConversationRequest = { title };\n const apiResponse = await httpService.post<ApiResponse<PatchConversationResponse>>(\n `/console/as/conversations/${agentId}`,\n body\n );\n\n if (!apiResponse.data) {\n // 如果后端没有返回 data,使用传入的 agentId\n this.logger?.info(`Renamed conversation: ${agentId} to \"${title}\"`);\n return { id: agentId };\n }\n\n this.logger?.info(`Renamed conversation: ${apiResponse.data.id} to \"${title}\"`);\n return apiResponse.data;\n } catch (error) {\n this.logger?.error(`Failed to rename conversation ${agentId}:`, error);\n throw error;\n }\n }\n\n /**\n * Get available models from product configuration\n *\n * GET {endpoint}/console/enterprises/{personal|enterpriseId}/models?repos[]={repo}\n *\n * This method fetches the models from /console/enterprises API\n * and extracts the models array from the response.\n *\n * @param repo - Optional repository URL for context-specific config\n * @returns Array of ModelInfo with full model details\n */\n async getModels(repo?: string): Promise<ModelInfo[]> {\n try {\n // Get user info from accountService to determine personal vs enterprise account\n const account = accountService.getAccount();\n\n // Check if user is logged in\n if (!account) {\n this.logger?.warn('[CloudAgentProvider] No account info available, cannot get models');\n return [];\n }\n\n // Determine account type: personal or enterprise\n const accountIdentifier = account.enterpriseId || 'personal';\n\n // Build URL with new console API format\n let url = `/console/enterprises/${accountIdentifier}/models`;\n if (repo) {\n // Keep same parameter format: ?repos[]=encoded_repo_url\n url += `?repos[]=${encodeURIComponent(repo)}`;\n }\n\n // Build custom headers specific to this request\n // Note: Authorization, X-User-Id, X-Enterprise-Id, X-Tenant-Id, X-Product, X-User-Agent\n // are already added by the request interceptor\n const headers: Record<string, string> = {\n 'Accept': 'application/json, text/plain, */*',\n 'X-Requested-With': 'XMLHttpRequest',\n 'X-Request-ID': this.generateRequestId(),\n };\n\n this.logger?.debug(`[CloudAgentProvider] GET ${url}`);\n\n // Parse response - expecting { data: ProductConfiguration }\n const apiResponse = await httpService.get<ApiResponse<ProductConfigResponse>>(url, {\n headers\n });\n if (!apiResponse.data) {\n this.logger?.warn('[CloudAgentProvider] No data in config response, returning empty models');\n return [];\n }\n\n // Extract models from product configuration\n const productConfig = apiResponse.data;\n const models = productConfig.models ?? [];\n\n this.logger?.info(`[CloudAgentProvider] Retrieved ${models.length} models from /console/enterprises API`);\n\n // Convert LanguageModel to ModelInfo\n return models.map((model): ModelInfo => ({\n id: model.id,\n name: model.name ?? model.id,\n description: model.description,\n credits: model.credits,\n configurable: model.configurable,\n configured: model.configured,\n isDefault: model.isDefault,\n supportsImages: model.supportsImages,\n supportsReasoning: model.supportsReasoning,\n onlyReasoning: model.onlyReasoning,\n disabledMultimodal: model.disabledMultimodal,\n disabled: model.disabled,\n disabledReason: model.disabledReason,\n disabledAction: model.disabledAction,\n }));\n } catch (error) {\n this.logger?.error('[CloudAgentProvider] Failed to get models:', error);\n throw error;\n }\n }\n\n /**\n * Generate a unique request ID\n */\n private generateRequestId(): string {\n // Generate UUID without dashes\n return 'xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx'.replace(/x/g, () =>\n Math.floor(Math.random() * 16).toString(16)\n );\n }\n\n // ============================================\n // File Picker (Browser Environment)\n // ============================================\n\n /**\n * Common image MIME types for filtering\n */\n private static readonly IMAGE_MIME_TYPES = [\n 'image/png',\n 'image/jpeg',\n 'image/jpg',\n 'image/gif',\n 'image/webp',\n 'image/svg+xml',\n 'image/bmp',\n ];\n\n /**\n * Pick files using browser's native file input\n *\n * @param params - File picker parameters\n * @returns Response with selected file paths (filenames in browser)\n */\n async pickFile(params?: PickFileParams): Promise<PickFileResponse> {\n return new Promise(resolve => {\n const input = document.createElement('input');\n input.type = 'file';\n input.style.display = 'none';\n\n // Set accept attribute\n if (params?.filters && params.filters.length > 0) {\n const acceptTypes: string[] = [];\n for (const filter of params.filters) {\n for (const ext of filter.extensions) {\n const mimeType = this.extensionToMimeType(ext);\n acceptTypes.push(mimeType || `.${ext}`);\n }\n }\n input.accept = acceptTypes.join(',');\n } else {\n input.accept = CloudAgentProvider.IMAGE_MIME_TYPES.join(',');\n }\n\n input.multiple = params?.canSelectMany ?? false;\n\n input.onchange = () => {\n const files = input.files;\n if (!files || files.length === 0) {\n resolve({ files: [], canceled: true });\n } else {\n const fileArray = Array.from(files);\n this.logger?.info(`Picked ${fileArray.length} file(s)`);\n resolve({ files: fileArray, canceled: false });\n }\n document.body.removeChild(input);\n };\n\n input.oncancel = () => {\n resolve({ files: [], canceled: true });\n document.body.removeChild(input);\n };\n\n document.body.appendChild(input);\n input.click();\n });\n }\n\n /**\n * Convert file extension to MIME type\n */\n private extensionToMimeType(ext: string): string | null {\n const extLower = ext.toLowerCase().replace(/^\\./, '');\n const mimeMap: Record<string, string> = {\n 'png': 'image/png',\n 'jpg': 'image/jpeg',\n 'jpeg': 'image/jpeg',\n 'gif': 'image/gif',\n 'webp': 'image/webp',\n 'svg': 'image/svg+xml',\n 'bmp': 'image/bmp',\n 'ico': 'image/x-icon',\n };\n return mimeMap[extLower] || null;\n }\n\n // ============================================\n // File Upload (Browser Environment)\n // ============================================\n\n /**\n * Upload files to cloud storage via COS presigned URL\n *\n * @param params - files array (File objects in browser)\n * @returns Response with corresponding cloud URLs\n */\n async uploadFile(params: UploadFileParams): Promise<UploadFileResponse> {\n this.logger?.info(`[CloudAgentProvider] uploadFile called for ${params.files.length} file(s)`);\n\n // Filter out string paths (only File objects are supported in browser)\n const files = params.files.filter((f): f is File => typeof f !== 'string');\n if (files.length === 0) {\n return {\n success: false,\n error: 'No valid File objects provided',\n };\n }\n\n // Use CosUploadService for actual upload\n const result = await this.cosUploadService.uploadFiles(files);\n\n return {\n success: result.success,\n urls: result.urls,\n expireSeconds: result.expireSeconds,\n error: result.error,\n };\n }\n\n // ============================================\n // Event Emitter Implementation\n // ============================================\n\n /**\n * Register event listener\n * @param event - Event name\n * @param handler - Event handler function\n */\n on(event: string, handler: (...args: any[]) => void): () => void {\n if (!this.eventListeners.has(event)) {\n this.eventListeners.set(event, new Set());\n }\n this.eventListeners.get(event)!.add(handler);\n\n // Return unsubscribe function\n return () => {\n this.off(event, handler);\n };\n }\n\n /**\n * Unregister event listener\n * @param event - Event name\n * @param handler - Event handler function\n */\n off(event: string, handler: (...args: any[]) => void): void {\n const listeners = this.eventListeners.get(event);\n if (listeners) {\n listeners.delete(handler);\n }\n }\n\n /**\n * Emit event to all registered listeners\n * @param event - Event name\n * @param args - Event arguments\n */\n private emitEvent(event: string, ...args: any[]): void {\n const listeners = this.eventListeners.get(event);\n if (listeners && listeners.size > 0) {\n this.logger?.debug(`Emitting event: ${event}`, args);\n for (const handler of listeners) {\n try {\n handler(...args);\n } catch (error) {\n this.logger?.error(`Error in event handler for ${event}:`, error);\n }\n }\n }\n }\n\n // ============================================\n // Support Scenes\n // ============================================\n\n /**\n * 获取支持的场景列表\n * API 端点: GET /console/as/support/scenes\n * 用于 Welcome 页面的 QuickActions 快捷操作\n *\n * @returns Promise<SupportScene[]> 支持的场景列表\n */\n async getSupportScenes(): Promise<SupportScene[]> {\n try {\n const apiResponse = await httpService.get<ApiResponse<{ scenes: SupportScene[] }>>(\n '/console/as/support/scenes'\n );\n\n if (!apiResponse.data) {\n this.logger?.warn('[CloudAgentProvider] No data in support scenes response');\n return [];\n }\n\n const scenes = apiResponse.data.scenes || [];\n this.logger?.info(`[CloudAgentProvider] Retrieved ${scenes.length} support scenes`);\n return scenes;\n } catch (error) {\n this.logger?.error('[CloudAgentProvider] Failed to get support scenes:', error);\n return [];\n }\n }\n\n // ============================================\n // Helpers\n // ============================================\n\n private toAgentState(data: ConversationDto): CloudAgentState {\n // 优先使用 sessionStatus(会话状态),fallback 到 status(Agent 状态)\n const status = data.sessionStatus || data.status;\n return {\n id: data.id,\n name: data.name,\n description: data.summary,\n type: 'cloud',\n status: status as AgentStatus,\n createdAt: data.createdAt ? new Date(data.createdAt) : undefined,\n capabilities: this.options.clientCapabilities,\n };\n }\n\n /**\n * Helper: 将 GET 请求的 body 转换为 URL 查询参数\n */\n private buildGetUrl(path: string, params?: unknown): string {\n if (!params) {\n return path;\n }\n\n const searchParams = new URLSearchParams();\n for (const [key, value] of Object.entries(params as Record<string, unknown>)) {\n if (value !== undefined && value !== null) {\n const stringValue = typeof value === 'object'\n ? JSON.stringify(value)\n : String(value);\n searchParams.append(key, stringValue);\n }\n }\n\n const queryString = searchParams.toString();\n return queryString ? `${path}?${queryString}` : path;\n }\n}\n\nexport default CloudAgentProvider;\n","/**\n * ActiveSessionImpl - Implements the ActiveSession interface\n *\n * Represents an active session with its resources and operations.\n * Session is the primary API surface for client interactions.\n */\n\nimport type { PromptResponse as SdkPromptResponse,SessionNotification } from '@agentclientprotocol/sdk';\n\nimport type { AgentStatus,FilesResource, PromptParams } from '../types.js';\nimport type {\n ActiveSession,\n AgentCapabilities,\n AgentConnection,\n AgentState,\n Artifact,\n ArtifactsResource,\n ArtifactType,\n AvailableCommand,\n Logger,\n ModelInfo,\n PromptResponse,\n PromptsResource,\n QuestionAnswers,\n SessionAgentOperations,\n SessionConnectionInfo,\n SessionEventHandler,\n SessionEvents,\n SessionMode} from './types.js';\n\n/**\n * Event listener type\n */\ntype EventListener<T> = (data: T) => void | Promise<void>;\n\n/**\n * Filesystem getter function type\n * Returns a FilesResource instance for file operations\n */\nexport type FilesystemGetter = () => Promise<FilesResource>;\n\n/**\n * Options for creating an ActiveSessionImpl instance\n */\nexport interface ActiveSessionImplOptions {\n /** Logger instance */\n logger?: Logger;\n /** Getter function for filesystem resource (provided by SessionManager) */\n getFilesystem?: FilesystemGetter;\n /** Session connection information (for cloud sessions) */\n connectionInfo?: SessionConnectionInfo;\n}\n\n/**\n * ActiveSessionImpl - Implements the ActiveSession interface\n *\n * This class wraps an AgentConnection and provides the session-centric API.\n * It is created by SessionManager when creating or loading sessions.\n *\n * @example\n * ```typescript\n * // Created by client.sessions.new() or client.sessions.load()\n * const session = await client.sessions.new({ cwd: '/workspace' });\n *\n * // Access agent state\n * console.log(session.agentState.status);\n *\n * // Send prompt\n * const response = await session.prompts.send({ content: 'Hello!' });\n *\n * // Cleanup\n * session.disconnect();\n * ```\n */\nexport class ActiveSessionImpl implements ActiveSession {\n private _id: string;\n private _agentId: string;\n private _availableModes?: SessionMode[];\n private _currentMode?: string;\n private _availableModels?: ModelInfo[];\n private _currentModelId?: string;\n private _availableCommands: AvailableCommand[] = [];\n private logger?: Logger;\n private connection: AgentConnection;\n private _getFilesystem?: FilesystemGetter;\n private _connectionInfo?: SessionConnectionInfo;\n\n // Event emitter storage\n private listeners: Map<keyof SessionEvents, Set<EventListener<unknown>>> = new Map();\n private onceListeners: Map<keyof SessionEvents, Set<EventListener<unknown>>> = new Map();\n\n /**\n * Agent operations namespace\n */\n readonly agent: SessionAgentOperations;\n\n /**\n * Prompts resource namespace\n */\n readonly prompts: PromptsResource;\n\n /**\n * Artifacts resource namespace\n */\n readonly artifacts: ArtifactsResource;\n\n /**\n * Files resource namespace (lazily loaded via getter)\n */\n readonly files: FilesResource;\n\n /**\n * Create an ActiveSessionImpl instance\n *\n * @param sessionId - Session ID\n * @param agentId - Agent ID\n * @param connection - Already connected AgentConnection\n * @param options - Additional options\n */\n constructor(\n sessionId: string,\n agentId: string,\n connection: AgentConnection,\n options: ActiveSessionImplOptions = {}\n ) {\n this._id = sessionId;\n this._agentId = agentId;\n this.connection = connection;\n this.logger = options.logger;\n this._getFilesystem = options.getFilesystem;\n this._connectionInfo = options.connectionInfo;\n\n // Set up event forwarding from connection\n this.setupConnectionEvents(connection);\n\n // Initialize agent operations namespace\n this.agent = this.createAgentOperations();\n\n // Initialize resource namespaces\n this.prompts = this.createPromptsResource();\n this.artifacts = this.createArtifactsResource();\n this.files = this.createFilesResource();\n }\n\n // ============================================\n // Properties\n // ============================================\n\n /**\n * Session ID\n */\n get id(): string {\n return this._id;\n }\n\n /**\n * Agent ID\n */\n get agentId(): string {\n return this._agentId;\n }\n\n /**\n * Agent state (live connection state)\n * Returns LocalAgentState or CloudAgentState based on transport type\n */\n get agentState(): AgentState {\n return {\n id: this._agentId,\n status: this.connection.state as AgentStatus,\n capabilities: this.connection.capabilities,\n type: this.connection.transport,\n cwd: this.connection.cwd || ''\n };\n }\n\n /**\n * Get agent capabilities (available after connection)\n */\n get capabilities(): AgentCapabilities | undefined {\n return this.connection.capabilities;\n }\n\n /**\n * Available session modes\n */\n get availableModes(): SessionMode[] | undefined {\n return this._availableModes;\n }\n\n /**\n * Current session mode\n */\n get currentMode(): string | undefined {\n return this._currentMode;\n }\n\n /**\n * Available models for this session\n */\n get availableModels(): ModelInfo[] | undefined {\n return this._availableModels;\n }\n\n /**\n * Current model ID\n */\n get currentModelId(): string | undefined {\n return this._currentModelId;\n }\n\n /**\n * Available slash commands\n *\n * When Agent sends available_commands_update, this list is automatically updated.\n * Commands can be accessed directly without waiting for events.\n */\n get availableCommands(): AvailableCommand[] {\n return this._availableCommands;\n }\n\n /**\n * Set available commands (called when available_commands_update is received)\n */\n setAvailableCommands(commands: AvailableCommand[]): void {\n this._availableCommands = commands;\n this.logger?.info(`Session ${this._id}: Available commands updated, count: ${commands.length}`);\n }\n\n /**\n * Check if the session is active\n */\n get isActive(): boolean {\n return this.connection.isInitialized;\n }\n\n /**\n * Session connection information (only available for cloud sessions)\n * 会话连接信息,包括sandboxId、link、token等\n */\n get connectionInfo(): SessionConnectionInfo | undefined {\n return this._connectionInfo;\n }\n\n /**\n * Set session modes (called after create/load)\n */\n setModes(availableModes?: SessionMode[], currentMode?: string): void {\n this._availableModes = availableModes;\n this._currentMode = currentMode;\n }\n\n /**\n * Set available models (called after create/load)\n */\n setModels(availableModels?: ModelInfo[], currentModelId?: string): void {\n this._availableModels = availableModels;\n this._currentModelId = currentModelId;\n }\n\n // ============================================\n // Agent Operations Namespace\n // ============================================\n\n private createAgentOperations(): SessionAgentOperations {\n // eslint-disable-next-line @typescript-eslint/no-this-alias\n const self = this;\n return {\n get id(): string {\n return self._agentId;\n },\n get state(): AgentState {\n return self.agentState;\n },\n get isConnected(): boolean {\n return self.connection.isInitialized;\n },\n get capabilities(): AgentCapabilities | undefined {\n return self.connection.capabilities;\n }\n };\n }\n\n // ============================================\n // Prompts Resource\n // ============================================\n\n private createPromptsResource(): PromptsResource {\n return {\n send: async (params: PromptParams): Promise<PromptResponse> => {\n const connection = this.getConnectionOrThrow();\n const response = await connection.prompt(this._id, params);\n return this.mapPromptResponse(response);\n },\n\n stream: (params: PromptParams): AsyncIterable<SessionNotification> => {\n const connection = this.getConnectionOrThrow();\n return connection.promptStream(this._id, params);\n },\n\n cancel: async (): Promise<void> => {\n const connection = this.getConnectionOrThrow();\n await connection.cancel(this._id);\n }\n };\n }\n\n // ============================================\n // Artifacts Resource\n // ============================================\n\n private createArtifactsResource(): ArtifactsResource {\n // Artifact management has been simplified - these methods are no longer supported\n const notSupported = () => {\n throw new Error('Artifact management is no longer supported through this API');\n };\n\n return {\n list: async (_params?: { type?: ArtifactType }): Promise<Artifact[]> => {\n notSupported();\n return [];\n },\n\n retrieve: async (_artifactId: string): Promise<Artifact> => {\n notSupported();\n return undefined as unknown as Artifact;\n },\n\n content: async (_artifactId: string): Promise<string> => {\n notSupported();\n return '';\n }\n };\n }\n\n // ============================================\n // Files Resource\n // ============================================\n\n /**\n * Create files resource with lazy-loaded filesystem\n *\n * The filesystem is lazily loaded on first use to avoid unnecessary\n * connections to the sandbox. The actual filesystem instance is obtained\n * via the getter function provided by SessionManager.\n */\n private createFilesResource(): FilesResource {\n // eslint-disable-next-line @typescript-eslint/no-this-alias\n const self = this;\n let filesPromise: Promise<FilesResource> | null = null;\n\n /**\n * Get or create the filesystem instance\n */\n const getFs = async (): Promise<FilesResource> => {\n if (!self._getFilesystem) {\n throw new Error('Filesystem not available: provider does not support filesystem operations');\n }\n if (!filesPromise) {\n filesPromise = self._getFilesystem();\n }\n return filesPromise;\n };\n\n return {\n // Read operations - 支持多种格式重载\n read: (async (path: string, opts?: any) => (await getFs()).read(path, opts)) as FilesResource['read'],\n\n // Write operations - 支持单文件和批量写入重载\n write: (async (pathOrFiles: any, dataOrOpts?: any, opts?: any) => {\n const fs = await getFs();\n if (Array.isArray(pathOrFiles)) {\n // Batch write\n return fs.write(pathOrFiles, dataOrOpts);\n }\n // Single file write\n return fs.write(pathOrFiles, dataOrOpts, opts);\n }) as FilesResource['write'],\n\n // List with depth support\n list: async (path, opts) => (await getFs()).list(path, opts),\n exists: async (path, opts) => (await getFs()).exists(path, opts),\n makeDir: async (path, opts) => (await getFs()).makeDir(path, opts),\n remove: async (path, opts) => (await getFs()).remove(path, opts),\n rename: async (oldPath, newPath, opts) => (await getFs()).rename(oldPath, newPath, opts),\n\n // 新增 getInfo 方法\n getInfo: async (path, opts) => (await getFs()).getInfo(path, opts),\n\n // Watch operations - 支持扩展的 opts\n watchDir: async (path, onEvent, opts) => (await getFs()).watchDir(path, onEvent, opts)\n };\n }\n\n // ============================================\n // Permission Management\n // ============================================\n\n /**\n * Resolve a permission request\n */\n resolvePermission(requestId: string, optionId: string): boolean {\n return this.connection.resolvePermission(requestId, optionId);\n }\n\n /**\n * Reject a permission request\n */\n rejectPermission(requestId: string, reason?: string): boolean {\n return this.connection.rejectPermission(requestId, reason);\n }\n\n // ============================================\n // Question Management (ask_followup_question)\n // ============================================\n\n /**\n * Answer a question request with user's selections\n */\n answerQuestion(toolCallId: string, answers: QuestionAnswers): boolean {\n return this.connection.answerQuestion(toolCallId, answers);\n }\n\n /**\n * Cancel a question request\n */\n cancelQuestion(toolCallId: string, reason?: string): boolean {\n return this.connection.cancelQuestion(toolCallId, reason);\n }\n\n // ============================================\n // Tool Callback Management\n // ============================================\n\n /**\n * Callback for tool operations (skip or cancel)\n * @param toolCallId Tool call ID\n * @param toolName Tool name\n * @param action Action to perform ('skip' or 'cancel')\n */\n async toolCallback(toolCallId: string, toolName: string, action: 'skip' | 'cancel'): Promise<{ success: boolean; error?: string }> {\n const connection = this.getConnectionOrThrow();\n return await connection.toolCallback(this._id, toolCallId, toolName, action);\n }\n\n // ============================================\n // Session Mode Management\n // ============================================\n\n /**\n * Set the current session mode\n *\n * @param modeId - The mode ID to switch to (must be in availableModes)\n * @throws Error if modeId is not in availableModes or connection fails\n *\n * @example\n * ```typescript\n * // Switch to 'code' mode\n * await session.setMode('code');\n *\n * // Switch to 'architect' mode\n * await session.setMode('architect');\n * ```\n */\n async setMode(modeId: string): Promise<void> {\n // Validate modeId if availableModes is set\n if (this._availableModes) {\n const modeExists = this._availableModes.some(m => m.id === modeId);\n if (!modeExists) {\n const availableIds = this._availableModes.map(m => m.id).join(', ');\n throw new Error(`Invalid modeId: \"${modeId}\". Available modes: ${availableIds}`);\n }\n }\n\n const connection = this.getConnectionOrThrow();\n await connection.setSessionMode(this._id, modeId);\n\n // Update internal state\n this._currentMode = modeId;\n }\n\n /**\n * Set the current session model\n *\n * @param modelId - The model ID to switch to\n * @example\n * ```typescript\n * // Switch to Claude Sonnet 4\n * await session.setSessionModel('claude-sonnet-4-20250514');\n *\n * // Switch to GPT-4o\n * await session.setSessionModel('gpt-4o');\n * ```\n */\n async setSessionModel(modelId: string): Promise<void> {\n const connection = this.getConnectionOrThrow();\n await connection.setSessionModel(this._id, modelId);\n }\n\n // ============================================\n // Event Subscription\n // ============================================\n\n /**\n * Subscribe to session events\n */\n on<K extends keyof SessionEvents>(event: K, handler: SessionEventHandler<K>): this {\n if (!this.listeners.has(event)) {\n this.listeners.set(event, new Set());\n }\n this.listeners.get(event)!.add(handler as EventListener<unknown>);\n return this;\n }\n\n /**\n * Unsubscribe from session events\n */\n off<K extends keyof SessionEvents>(event: K, handler: SessionEventHandler<K>): this {\n const eventListeners = this.listeners.get(event);\n if (eventListeners) {\n eventListeners.delete(handler as EventListener<unknown>);\n }\n const onceEventListeners = this.onceListeners.get(event);\n if (onceEventListeners) {\n onceEventListeners.delete(handler as EventListener<unknown>);\n }\n return this;\n }\n\n /**\n * Subscribe to a session event once\n */\n once<K extends keyof SessionEvents>(event: K, handler: SessionEventHandler<K>): this {\n if (!this.onceListeners.has(event)) {\n this.onceListeners.set(event, new Set());\n }\n this.onceListeners.get(event)!.add(handler as EventListener<unknown>);\n return this;\n }\n\n /**\n * Emit an event to all registered listeners\n */\n private emit<K extends keyof SessionEvents>(event: K, data: SessionEvents[K]): boolean {\n const regularListeners = this.listeners.get(event);\n const onceEventListeners = this.onceListeners.get(event);\n\n let hasListeners = false;\n\n // Call regular listeners\n if (regularListeners && regularListeners.size > 0) {\n hasListeners = true;\n for (const listener of regularListeners) {\n try {\n const result = listener(data);\n if (result instanceof Promise) {\n result.catch(err => {\n console.error(`Error in async event listener for '${String(event)}':`, err);\n });\n }\n } catch (err) {\n console.error(`Error in event listener for '${String(event)}':`, err);\n }\n }\n }\n\n // Call once listeners and remove them\n if (onceEventListeners && onceEventListeners.size > 0) {\n hasListeners = true;\n const listenersToCall = Array.from(onceEventListeners);\n this.onceListeners.delete(event);\n\n for (const listener of listenersToCall) {\n try {\n const result = listener(data);\n if (result instanceof Promise) {\n result.catch(err => {\n console.error(`Error in async once event listener for '${String(event)}':`, err);\n });\n }\n } catch (err) {\n console.error(`Error in once event listener for '${String(event)}':`, err);\n }\n }\n }\n\n return hasListeners;\n }\n\n /**\n * Remove all listeners for an event\n */\n private removeAllListeners<K extends keyof SessionEvents>(event?: K): void {\n if (event !== undefined) {\n this.listeners.delete(event);\n this.onceListeners.delete(event);\n } else {\n this.listeners.clear();\n this.onceListeners.clear();\n }\n }\n\n // ============================================\n // Lifecycle\n // ============================================\n\n /**\n * Disconnect from the session/agent\n */\n disconnect(): void {\n this.connection.disconnect();\n this.removeAllListeners();\n this.logger?.info(`Session ${this._id}: Disconnected`);\n }\n\n /**\n * Symbol.dispose for 'using' keyword support\n * Automatically disconnects and cleans up when session goes out of scope\n *\n * @example\n * ```typescript\n * {\n * using session = await client.sessions.new({ cwd: '/workspace' });\n * // ... use session\n * } // session automatically disposed\n * ```\n */\n [Symbol.dispose](): void {\n this.disconnect();\n }\n\n // ============================================\n // Helpers\n // ============================================\n\n private getConnectionOrThrow(): AgentConnection {\n if (!this.connection.isInitialized) {\n throw new Error(`Session ${this._id}: Connection not initialized.`);\n }\n return this.connection;\n }\n\n private setupConnectionEvents(connection: AgentConnection): void {\n // Connection lifecycle events\n connection.on('connected', () => {\n this.emit('connected', undefined as never);\n });\n\n connection.on('disconnected', () => {\n this.emit('disconnected', undefined as never);\n });\n\n connection.on('error', error => {\n this.emit('error', error);\n });\n\n // Session updates\n connection.on('sessionUpdate', update => {\n this.emit('sessionUpdate', update);\n });\n\n // Artifact events\n connection.on('artifactCreated', artifact => {\n console.log('[Session] Forwarding artifactCreated:', {\n artifactUri: artifact.uri,\n artifactType: artifact.type,\n });\n this.emit('artifactCreated', artifact);\n });\n\n connection.on('artifactUpdated', artifact => {\n console.log('[Session] Forwarding artifactUpdated:', {\n artifactUri: artifact.uri,\n artifactType: artifact.type,\n });\n this.emit('artifactUpdated', artifact);\n });\n\n connection.on('artifactDeleted', artifact => {\n console.log('[Session] Forwarding artifactDeleted:', { artifactUri: artifact.uri });\n this.emit('artifactDeleted', artifact);\n });\n\n // Permission requests\n connection.on('permissionRequest', request => {\n this.emit('permissionRequest', request);\n });\n\n // Question requests\n connection.on('questionRequest', request => {\n this.emit('questionRequest', request);\n });\n\n connection.on('questionCancelled', () => {\n this.prompts.cancel();\n });\n\n // Usage updates\n connection.on('usageUpdate', usage => {\n this.emit('usageUpdate', usage);\n });\n\n // Checkpoint events\n connection.on('checkpointCreated', checkpoint => {\n this.emit('checkpointCreated', checkpoint);\n });\n\n connection.on('checkpointUpdated', checkpoint => {\n this.emit('checkpointUpdated', checkpoint);\n });\n\n // Command events\n connection.on('command', command => {\n console.log('[Session] Forwarding command:', {\n action: command.action,\n paramsKeys: command.params ? Object.keys(command.params) : []\n });\n this.emit('command', command);\n });\n }\n\n private mapPromptResponse(response: SdkPromptResponse): PromptResponse {\n return {\n stopReason: response.stopReason as PromptResponse['stopReason'],\n _meta: response._meta ?? undefined\n };\n }\n}\n","/**\n * SessionManager - Manages session lifecycle and connections\n *\n * Provides the core implementation for session-centric API operations:\n * - list() - Lists sessions (mapped from agents)\n * - createSession() - Creates new session (auto-creates agent)\n * - loadSession() - Loads existing session (finds agent by sessionId)\n */\n\nimport { ActiveSessionImpl } from './session.js';\nimport type {\n ActiveSession,\n AgentProvider,\n CreateSessionParams,\n ListAgentOptions,\n ListAgentResult,\n LoadSessionParams,\n Logger,\n ModelInfo,\n SessionInfo} from './types.js';\n\n/**\n * Options for creating a SessionManager instance\n */\nexport interface SessionManagerOptions {\n /** Agent provider (required) */\n provider: AgentProvider;\n /** Logger instance */\n logger?: Logger;\n}\n\n/**\n * SessionManager - Session lifecycle management\n *\n * This class manages the relationship between sessions and agents.\n * Since the backend is agent-centric, SessionManager handles the mapping:\n * - Sessions are views over agents\n * - sessionId may equal agentId in simple cases\n *\n * Features:\n * - Session caching: reuses existing ActiveSession instances\n * - Automatic cleanup on session disconnect\n *\n * @example\n * ```typescript\n * const manager = new SessionManager({ provider, logger });\n *\n * // List sessions\n * const sessions = await manager.listSessions();\n *\n * // Create new session\n * const session = await manager.createSession({ cwd: '/workspace' });\n *\n * // Load existing session (returns cached instance if available)\n * const loaded = await manager.loadSession({ sessionId: 'xxx', cwd: '/workspace' });\n * ```\n */\nexport class SessionManager {\n private provider: AgentProvider;\n private logger?: Logger;\n\n constructor(options: SessionManagerOptions) {\n this.provider = options.provider;\n this.logger = options.logger;\n }\n\n /**\n * List all sessions with pagination info (mapped from agents)\n *\n * Each agent maps to a session. The sessionId is derived from the agent.\n * Cloud: Returns server-side filtered/sorted/paginated results\n * Local: Returns client-side filtered/sorted results (synthetic pagination)\n *\n * @param options - Optional query parameters for filtering, sorting, and pagination\n */\n async listSessions(options?: ListAgentOptions): Promise<ListAgentResult<SessionInfo>> {\n const result = await this.provider.list(options);\n const sessions = result.agents.map(agent => ({\n // Use agentId as sessionId (since sessions are 1:1 with agents in current design)\n id: agent.id,\n agentId: agent.id,\n name: agent.name,\n status: agent.status,\n createdAt: agent.createdAt,\n lastActivityAt: agent.updatedAt,\n // cwd is only available for local agents\n cwd: agent.type === 'local' ? agent.cwd : undefined,\n isPlayground: agent.isPlayground\n }));\n\n console.log('[SessionManager] Returning sessions:', { count: sessions.length, pagination: result.pagination });\n return {\n agents: sessions,\n pagination: result.pagination\n };\n }\n\n /**\n * Create a new session\n *\n * Steps:\n * 1. Create new agent (if provider supports it) or use existing\n * 2. Connect to agent\n * 3. Call ACP newSession\n * 4. Register session mapping (for LocalAgentProvider)\n * 5. Return ActiveSession instance\n */\n async createSession(params: CreateSessionParams): Promise<ActiveSession> {\n this.logger?.info('Creating new session');\n\n // Step 1: Create new agent if provider supports it\n let agentId: string;\n\n if (this.provider.create) {\n // Pass params to create() - LocalAgentProvider uses cwd as agentId\n agentId = await this.provider.create(params);\n this.logger?.debug(`Created new agent: ${agentId}`);\n } else {\n // If provider doesn't support create, we need to throw\n // The provider must support creating agents for sessions.new()\n throw new Error('Provider does not support creating agents. Use sessions.load() with an existing sessionId.');\n }\n\n // Step 2: Connect to agent\n const connection = await this.provider.connect(agentId);\n this.logger?.debug(`Connected to agent: ${agentId}`);\n\n // Step 3: Create session via ACP\n const response = await connection.createSession({\n _meta: params._meta,\n cwd: params.cwd,\n mcpServers: params.mcpServers\n });\n\n // Step 4: Register session mapping (for LocalAgentProvider to support loadSession)\n if (this.provider.registerSession) {\n this.provider.registerSession(response.sessionId, agentId);\n this.logger?.debug(`Registered session mapping: ${response.sessionId} → ${agentId}`);\n }\n\n // Step 5: Create and return ActiveSession\n // Get connectionInfo from CloudAgentConnection if available\n const connectionInfo = (connection as any).sessionConnectionInfo;\n\n const session = new ActiveSessionImpl(\n response.sessionId,\n agentId,\n connection,\n {\n logger: this.logger,\n // Bind sessionId (not agentId) to create a getter function for filesystem\n // Note: conversation-channel-router binds sessionId to windowId,\n // so we must use sessionId for filesystem routing\n getFilesystem: this.provider.filesystem\n ? () => this.provider.filesystem!.getFilesystem(response.sessionId)\n : undefined,\n // Pass connectionInfo from CloudAgentConnection (if available)\n connectionInfo\n }\n );\n\n // Set modes from response\n session.setModes(\n response.modes?.availableModes,\n response.modes?.currentModeId\n );\n\n // Set models from response (convert ACP ModelInfo to local ModelInfo)\n if (response.models?.availableModels) {\n const localModels: ModelInfo[] = response.models.availableModels.map(m => ({\n id: m.modelId,\n name: m.name,\n description: m.description ?? undefined\n }));\n session.setModels(localModels, response.models?.currentModelId);\n }\n\n this.logger?.info(`Session created: ${response.sessionId}`);\n return session;\n }\n\n /**\n * Load an existing session\n *\n * Steps:\n * 1. Check cache for existing session\n * 2. Find agent by sessionId (sessionId === agentId in current design)\n * 3. Connect to agent\n * 4. Create ActiveSession instance\n * 5. Execute onSessionCreated callback (if provided) to allow early setup (e.g., event listeners)\n * 6. Call ACP loadSession\n * 7. Return ActiveSession instance (cached)\n */\n async loadSession(params: LoadSessionParams): Promise<ActiveSession> {\n this.logger?.info(`Loading session: ${params.sessionId}`);\n\n // Step 2: Find agent by sessionId\n // In current design, sessionId === agentId\n const agentId = params.sessionId;\n\n // Verify agent exists\n const agentState = await this.provider.get(agentId);\n if (!agentState) {\n throw new Error(`Session not found: ${params.sessionId}`);\n }\n\n // Step 3: Connect to agent\n const connection = await this.provider.connect(agentId);\n this.logger?.debug(`Connected to agent: ${agentId}`);\n\n // Step 4: Create ActiveSession instance (BEFORE loadSession call)\n // Get connectionInfo from CloudAgentConnection if available\n const connectionInfo = (connection as any).sessionConnectionInfo;\n\n const session = new ActiveSessionImpl(\n params.sessionId,\n agentId,\n connection,\n {\n logger: this.logger,\n // Bind sessionId (not agentId) to create a getter function for filesystem\n // Note: conversation-channel-router binds sessionId to windowId,\n // so we must use sessionId for filesystem routing\n getFilesystem: this.provider.filesystem\n ? () => this.provider.filesystem!.getFilesystem(params.sessionId)\n : undefined,\n // Pass connectionInfo from CloudAgentConnection (if available)\n connectionInfo\n }\n );\n\n // Step 5: Execute onSessionCreated callback (BEFORE loadSession)\n // This allows callers to set up event listeners before historical messages arrive\n if (params.onSessionCreated) {\n await params.onSessionCreated(session);\n }\n\n // Step 6: Load session via ACP (may trigger historical message push)\n const response = await connection.loadSession({\n sessionId: params.sessionId,\n cwd: agentState.type === 'local' ? agentState.cwd : params.cwd,\n mcpServers: params.mcpServers\n });\n\n // Step 7: Set modes and models from response\n session.setModes(\n response.modes?.availableModes,\n response.modes?.currentModeId\n );\n\n // Set models from response (convert ACP ModelInfo to local ModelInfo)\n if (response.models?.availableModels) {\n const localModels: ModelInfo[] = response.models.availableModels.map(m => ({\n id: m.modelId,\n name: m.name,\n description: m.description ?? undefined\n }));\n session.setModels(localModels, response.models?.currentModelId);\n }\n\n this.logger?.info(`Session loaded: ${params.sessionId}`);\n return session;\n }\n}\n","/**\n * AgentClient - Session-centric API\n *\n * Provides a unified entry point for managing sessions.\n * Sessions are the primary API surface; agents are internal implementation.\n */\n\nimport type { ModelInfo } from '../types.js';\nimport { SessionManager } from './session-manager.js';\nimport type {\n AgentClientOptions,\n AgentProvider,\n BatchPluginOperationRequest,\n BatchPluginOperationResult,\n ClientSessionsResource,\n InitializeWorkspaceParams,\n InitializeWorkspaceResponse,\n ListAgentOptions,\n Logger,\n ModelsResource,\n PickFileParams,\n PickFileResponse,\n PickFolderParams,\n PickFolderResponse,\n SearchFileParams,\n SearchFileResponse,\n SessionsResourceEventHandler,\n SessionsResourceEvents,\n UploadFileParams,\n UploadFileResponse,\n WorkspaceInfo} from './types.js';\n\n// ============================================\n// AgentClient - Session-Centric API\n// ============================================\n\n/**\n * AgentClient - Session-centric client\n *\n * Provides a session-centric API that internally manages agents.\n * Users interact with sessions; the agent lifecycle is handled internally.\n *\n * @example\n * ```typescript\n * // Create client with a provider\n * const provider = new CloudAgentProvider({\n * endpoint: 'https://api.example.com',\n * authToken: 'token'\n * });\n *\n * const client = new AgentClient({\n * provider,\n * logger: console\n * });\n *\n * // List all sessions\n * const sessions = await client.sessions.list();\n *\n * // Create new session (auto-creates agent and connects)\n * const session = await client.sessions.create({ cwd: '/workspace' });\n * console.log(session.agentState.status); // agent status\n * console.log(session.agentState.id); // agent ID\n *\n * // Send prompt\n * await session.prompts.send({ content: 'Hello' });\n *\n * // Get available models\n * const models = await client.sessions.models.list('my-repo');\n *\n * // Use 'using' keyword for automatic cleanup\n * {\n * using session = await client.sessions.create({ cwd: '/workspace' });\n * // ... use session\n * } // session automatically disposed\n *\n * // Or manually disconnect\n * session.disconnect();\n *\n * // Load existing session\n * const loadedSession = await client.sessions.load({\n * sessionId: 'xxx',\n * cwd: '/workspace'\n * });\n * ```\n */\nexport class AgentClient {\n private logger?: Logger;\n private provider: AgentProvider;\n private sessionManager: SessionManager;\n\n /**\n * Sessions resource namespace (primary API entry point)\n */\n readonly sessions: ClientSessionsResource;\n\n /**\n * 运行环境类型\n * - 'local': IDE 本地环境\n * - 'cloud': 云端环境\n */\n readonly environmentType: 'local' | 'cloud';\n\n constructor(options: AgentClientOptions) {\n this.logger = options.logger;\n this.provider = options.provider;\n this.environmentType = options.environmentType ?? 'cloud';\n\n // Initialize session manager\n this.sessionManager = new SessionManager({\n provider: this.provider,\n logger: this.logger\n });\n\n // Initialize sessions resource\n this.sessions = this.createSessionsResource();\n }\n\n // ============================================\n // Sessions Resource\n // ============================================\n\n private createSessionsResource(): ClientSessionsResource {\n return {\n list: async (options?: ListAgentOptions) => this.sessionManager.listSessions(options),\n\n create: async params => this.sessionManager.createSession(params),\n\n load: async params => {\n console.log('[AgentClient] sessions.load called:', params.sessionId);\n return this.sessionManager.loadSession(params);\n },\n\n archive: async (sessionId: string): Promise<{ id: string }> => {\n this.logger?.debug('AgentClient.sessions.archive called', { sessionId });\n\n try {\n // Check if provider supports archive\n if (this.provider.archive) {\n const result = await this.provider.archive(sessionId);\n this.logger?.info('Session archived successfully', { sessionId });\n return result;\n }\n\n // If provider does not support archive, throw error\n throw new Error('Provider does not support archive method');\n } catch (error) {\n this.logger?.error('Failed to archive session', error);\n throw error;\n }\n },\n\n rename: async (sessionId: string, title: string): Promise<{ id: string }> => {\n this.logger?.debug('AgentClient.sessions.rename called', { sessionId, title });\n\n try {\n // Check if provider supports rename\n if (this.provider.rename) {\n const result = await this.provider.rename(sessionId, title);\n this.logger?.info('Session renamed successfully', { sessionId, title });\n return result;\n }\n\n // If provider does not support rename, throw error\n throw new Error('Provider does not support rename method');\n } catch (error) {\n this.logger?.error('Failed to rename session', error);\n throw error;\n }\n },\n\n move: async (sessionId: string): Promise<{ id: string }> => {\n this.logger?.debug('AgentClient.sessions.move called', { sessionId });\n\n try {\n // Check if provider supports move\n if (this.provider.move) {\n const result = await this.provider.move(sessionId);\n this.logger?.info('Session moved successfully', { sessionId });\n return result;\n }\n\n // If provider does not support move, throw error\n throw new Error('Provider does not support move method');\n } catch (error) {\n this.logger?.error('Failed to move session', error);\n throw error;\n }\n },\n\n // Initialize workspace for future sessions\n initializeWorkspace: async (params: InitializeWorkspaceParams): Promise<InitializeWorkspaceResponse> => {\n this.logger?.debug('AgentClient.sessions.initializeWorkspace called', params);\n\n try {\n // openWorkspace 是 LocalAgentProvider 特有的能力\n if (this.provider.openWorkspace) {\n const result = await this.provider.openWorkspace(params);\n this.logger?.info('Workspace opened successfully', { cwd: params.cwd });\n return result;\n }\n\n // 如果 provider 不支持 openWorkspace,返回成功(向后兼容)\n this.logger?.warn('Provider does not support openWorkspace');\n return { success: true };\n\n } catch (error) {\n this.logger?.error('Failed to initialize workspace', error);\n return {\n success: false,\n error: error instanceof Error ? error.message : 'Unknown error'\n };\n }\n },\n\n // Get current workspaces list\n getCurrentWorkspaces: async (filter?: { activeOnly?: boolean }): Promise<WorkspaceInfo[]> => {\n this.logger?.debug('AgentClient.sessions.getCurrentWorkspaces called', filter);\n\n try {\n // getCurrentWorkspaces 是 LocalAgentProvider 特有的能力\n if ('getCurrentWorkspaces' in this.provider && typeof this.provider.getCurrentWorkspaces === 'function') {\n const result = await this.provider.getCurrentWorkspaces(filter);\n this.logger?.info('Current workspaces retrieved', { count: result.length });\n return result;\n }\n\n // 如果 provider 不支持 getCurrentWorkspaces,返回空数组(向后兼容)\n this.logger?.warn('Provider does not support getCurrentWorkspaces');\n return [];\n\n } catch (error) {\n this.logger?.error('Failed to get current workspaces', error);\n return [];\n }\n },\n\n // Event methods - forward to provider if supported\n on: <K extends keyof SessionsResourceEvents>(\n event: K,\n handler: SessionsResourceEventHandler<K>\n ): void => {\n if (this.provider.on) {\n this.provider.on(event as string, handler as (...args: any[]) => void);\n } else {\n this.logger?.warn(`Provider does not support event registration: ${String(event)}`);\n }\n },\n\n off: <K extends keyof SessionsResourceEvents>(\n event: K,\n handler: SessionsResourceEventHandler<K>\n ): void => {\n if (this.provider.off) {\n this.provider.off(event as string, handler as (...args: any[]) => void);\n } else {\n this.logger?.warn(`Provider does not support event unregistration: ${String(event)}`);\n }\n },\n\n openWorkspace: async (params: InitializeWorkspaceParams): Promise<InitializeWorkspaceResponse> => {\n try {\n if (this.provider && this.provider.openWorkspace) {\n const result = await this.provider.openWorkspace(params);\n this.logger?.info('Workspace opened successfully', { cwd: params.cwd });\n return result;\n }\n return { success: false, error: 'Provider does not support openWorkspace' };\n } catch (error) {\n this.logger?.error('Failed to open workspace', error);\n return {\n success: false,\n error: error instanceof Error ? error.message : 'Unknown error'\n };\n }\n },\n\n pickFile: async (params?: PickFileParams): Promise<PickFileResponse> => {\n try {\n if (this.provider && this.provider.pickFile) {\n const result = await this.provider.pickFile(params);\n this.logger?.info('File picker completed', { fileCount: result.files.length, canceled: result.canceled });\n return result;\n }\n return { files: [], canceled: true, error: 'Provider does not support pickFile' };\n } catch (error) {\n this.logger?.error('Failed to pick file', error);\n return {\n files: [],\n canceled: true,\n error: error instanceof Error ? error.message : 'Unknown error'\n };\n }\n },\n\n pickFolder: async (params?: PickFolderParams): Promise<PickFolderResponse> => {\n try {\n if (this.provider && this.provider.pickFolder) {\n const result = await this.provider.pickFolder(params);\n this.logger?.info('Folder picker completed', { folderPaths: result.folderPaths, canceled: result.canceled });\n return result;\n }\n return { folderPaths: [], canceled: true, error: 'Provider does not support pickFolder' };\n } catch (error) {\n this.logger?.error('Failed to pick folder', error);\n return {\n folderPaths: [],\n canceled: true,\n error: error instanceof Error ? error.message : 'Unknown error'\n };\n }\n },\n\n uploadFile: async (params: UploadFileParams): Promise<UploadFileResponse> => {\n try {\n if (this.provider && this.provider.uploadFile) {\n const result = await this.provider.uploadFile(params);\n this.logger?.info('File upload completed', { count: params.files.length, success: result.success });\n return result;\n }\n return { success: false, error: 'Provider does not support uploadFile' };\n } catch (error) {\n this.logger?.error('Failed to upload file', error);\n return {\n success: false,\n error: error instanceof Error ? error.message : 'Unknown error'\n };\n }\n },\n\n searchFile: async (params: SearchFileParams): Promise<SearchFileResponse> => {\n try {\n if (this.provider && this.provider.searchFile) {\n const result = await this.provider.searchFile(params);\n this.logger?.info('File search completed', { resultCount: result.results.length, hasError: !!result.error });\n return result;\n }\n return { results: [], error: 'Provider does not support searchFile' };\n } catch (error) {\n this.logger?.error('Failed to search file', error);\n return {\n results: [],\n error: error instanceof Error ? error.message : 'Unknown error'\n };\n }\n },\n\n batchTogglePlugins: async (request: BatchPluginOperationRequest): Promise<BatchPluginOperationResult> => {\n try {\n if (this.provider && this.provider.batchTogglePlugins) {\n const result = await this.provider.batchTogglePlugins(request);\n this.logger?.info('Batch toggle plugins completed', {\n succeededCount: result.succeededPlugins.length,\n failedCount: result.failedPlugins.length\n });\n return result;\n }\n return {\n success: false,\n succeededPlugins: [],\n failedPlugins: request.items.map(item => ({\n ...item,\n error: 'Provider does not support batchTogglePlugins'\n }))\n };\n } catch (error) {\n this.logger?.error('Failed to batch toggle plugins', error);\n return {\n success: false,\n succeededPlugins: [],\n failedPlugins: request.items.map(item => ({\n ...item,\n error: error instanceof Error ? error.message : 'Unknown error'\n }))\n };\n }\n },\n\n getInstalledPlugins: async (forceRefresh?: boolean) => {\n try {\n if (this.provider && 'getInstalledPlugins' in this.provider && typeof this.provider.getInstalledPlugins === 'function') {\n const result = await this.provider.getInstalledPlugins(forceRefresh);\n this.logger?.info('Got installed plugins', { count: result?.length ?? 0 });\n return result;\n }\n this.logger?.warn('Provider does not support getInstalledPlugins');\n return [];\n } catch (error) {\n this.logger?.error('Failed to get installed plugins', error);\n return [];\n }\n },\n\n installPlugins: async (pluginNames, marketplaceName, installScope) => {\n try {\n if (this.provider && 'installPlugins' in this.provider && typeof this.provider.installPlugins === 'function') {\n const result = await this.provider.installPlugins(pluginNames, marketplaceName, installScope);\n this.logger?.info('Install plugins', { pluginNames, marketplaceName, success: result.success });\n return result;\n }\n this.logger?.warn('Provider does not support installPlugins');\n return { success: false, error: 'Provider does not support installPlugins' };\n } catch (error) {\n this.logger?.error('Failed to install plugins', error);\n return {\n success: false,\n error: error instanceof Error ? error.message : 'Unknown error'\n };\n }\n },\n\n getSupportScenes: async () => {\n try {\n if (this.provider && 'getSupportScenes' in this.provider && typeof this.provider.getSupportScenes === 'function') {\n const result = await this.provider.getSupportScenes();\n this.logger?.info('Got support scenes', { count: result?.length ?? 0 });\n return result;\n }\n this.logger?.warn('Provider does not support getSupportScenes');\n return [];\n } catch (error) {\n this.logger?.error('Failed to get support scenes', error);\n return [];\n }\n },\n\n // Models resource - delegates to provider's getModels method\n models: this.createModelsResource(),\n };\n }\n\n private createModelsResource(): ModelsResource {\n return {\n list: async (repo: string): Promise<ModelInfo[]> => {\n // Check if provider supports getModels\n if (this.provider.getModels) {\n return this.provider.getModels(repo);\n }\n throw new Error('Provider does not support getModels method');\n }\n };\n }\n\n // ============================================\n // Lifecycle\n // ============================================\n\n /**\n * Dispose the client\n *\n * Note: Active sessions are not automatically disposed.\n * The caller is responsible for disconnecting sessions they created.\n */\n dispose(): void {\n this.logger?.info('AgentClient disposed');\n }\n}\n\nexport default AgentClient;\n","/**\n * API type definitions for AgentClient\n * Session-centric API design\n */\n\nimport type { RequestPermissionRequest,SessionNotification } from '@agentclientprotocol/sdk';\nimport type {\n Artifact,\n ArtifactEvent,\n ArtifactType,\n CheckpointInfo,\n ClientEvents,\n EventListener,\n QuestionAnswers,\n QuestionRequest,\n UsageUpdate} from '@genie/agent-client-protocol';\n\nimport type { BatchPluginOperationRequest, BatchPluginOperationResult } from '../../backend/types.js';\nimport type { AvailableCommand } from '../providers/local-agent-provider/acp/types.js';\nimport type {\n Agent as AgentInfo,\n AgentCapabilities,\n AgentConnection,\n AgentStatus,\n ArtifactsConfig,\n ArtifactTypeConfig,\n ClientCapabilities,\n CodebuddyAgentMeta,\n CodebuddyClientMeta,\n E2BSandboxConnectionInfo,\n // e2b SDK types (re-exported from types.js)\n EntryInfo,\n FilesResource,\n Filesystem,\n FilesystemProvider,\n McpServerConfig,\n // Model types\n ModelInfo,\n PromptContentBlock,\n PromptParams,\n Session,\n SessionMode} from '../types.js';\n\n// Re-export commonly used types\nexport type {\n AgentInfo,\n AgentConnection,\n AgentStatus,\n Session,\n SessionMode,\n PromptParams,\n PromptContentBlock,\n AgentCapabilities,\n ClientCapabilities,\n // codebuddy.ai extension types\n ArtifactTypeConfig,\n ArtifactsConfig,\n CodebuddyClientMeta,\n CodebuddyAgentMeta,\n // Other types\n Artifact,\n ArtifactType,\n ArtifactEvent,\n UsageUpdate,\n ClientEvents,\n EventListener,\n SessionNotification,\n RequestPermissionRequest,\n QuestionRequest,\n QuestionAnswers,\n // Filesystem types\n FilesResource,\n FilesystemProvider,\n McpServerConfig,\n E2BSandboxConnectionInfo,\n // e2b SDK types (re-exported from types.js)\n EntryInfo,\n Filesystem,\n // Model types\n ModelInfo,\n // ACP types\n AvailableCommand,\n // Plugin types\n BatchPluginOperationRequest,\n BatchPluginOperationResult\n};\n\n// ============================================\n// Session Connection Info (for cloud sessions)\n// ============================================\n\n/**\n * Session connection information\n * 包含连接到Agent会话所需的所有信息,包括sandbox连接凭证\n */\nexport interface SessionConnectionInfo {\n /** Session ID */\n sessionId: string;\n /** Agent ID */\n agentId: string;\n /** Session endpoint URL (Agent的WebSocket/HTTP端点) */\n link: string;\n /** Session token (JWT格式的认证令牌) */\n token: string;\n /** Sandbox ID (E2B沙箱的唯一标识) */\n sandboxId: string;\n /** Session expiration timestamp (unix timestamp) */\n expireAt: number;\n /** Current working directory (optional) */\n cwd?: string;\n}\n\n// ============================================\n// Agent State (Unified agent state object)\n// ============================================\n\n/**\n * Agent 来源类型\n */\nexport type AgentStateType = 'local' | 'cloud';\n\n/**\n * 云端 Agent 可见性\n */\nexport type CloudAgentVisibility = 'PRIVATE' | 'PUBLIC' | 'TEAM';\n\n/**\n * 云端 Agent 来源信息\n */\nexport interface CloudAgentSourceInfo {\n /** 提供商: github, gitlab 等 */\n provider: string;\n /** 分支/引用 */\n ref: string;\n /** 仓库路径 */\n repository: string;\n}\n\n/**\n * 云端 Agent 目标信息\n */\nexport interface CloudAgentTarget {\n /** 是否自动创建 PR */\n autoCreatePr: boolean;\n /** 分支名称 */\n branchName?: string;\n /** PR URL */\n prUrl?: string;\n /** Agent URL */\n url?: string;\n}\n\n/**\n * AgentState 基础接口\n * 所有类型的 AgentState 都必须实现此接口\n */\nexport interface BaseAgentState {\n /** Unique agent ID */\n id: string;\n /** Display name */\n name?: string;\n /** Description */\n description?: string;\n /** Agent type */\n type: AgentStateType;\n /** Current connection status */\n status: AgentStatus;\n /** Agent capabilities (available after connection) */\n capabilities?: AgentCapabilities;\n /** When the agent was created */\n createdAt?: Date;\n /** When the agent was last updated */\n updatedAt?: Date;\n /** 是否为 playground */\n isPlayground?: boolean;\n}\n\n/**\n * LocalAgentState - 本地 Agent 状态\n * 来自本地 IPC 通信的 Agent\n */\nexport interface LocalAgentState extends BaseAgentState {\n type: 'local';\n /** 工作目录 */\n cwd: string;\n}\n\n/**\n * CloudAgentState - 云端 Agent 状态\n * 来自远程 API 的云端实例\n */\nexport interface CloudAgentState extends BaseAgentState {\n type: 'cloud';\n}\n\n/**\n * AgentState - Unified agent state object exposed to client users\n *\n * This is the primary way clients access agent information.\n * Uses discriminated union pattern to distinguish between local and cloud agents.\n */\nexport type AgentState = LocalAgentState | CloudAgentState;\n\n/**\n * 类型守卫:判断是否为 LocalAgentState\n */\nexport function isLocalAgentState(state: AgentState): state is LocalAgentState {\n return state.type === 'local';\n}\n\n/**\n * 类型守卫:判断是否为 CloudAgentState\n */\nexport function isCloudAgentState(state: AgentState): state is CloudAgentState {\n return state.type === 'cloud';\n}\n\n/**\n * Logger interface\n */\nexport interface Logger {\n debug(message: string, ...args: unknown[]): void;\n info(message: string, ...args: unknown[]): void;\n warn(message: string, ...args: unknown[]): void;\n error(message: string, ...args: unknown[]): void;\n}\n\n// ============================================\n// Agent List Query Options\n// ============================================\n\n/**\n * Filter condition for listing agents\n */\nexport interface ListAgentFilter {\n /** Filter field name (e.g., 'status', 'name') */\n field: string;\n /** Filter value (comma-separated for multiple values) */\n value: string;\n}\n\n/**\n * Sort options for listing agents\n */\nexport interface ListAgentSort {\n /** Sort field (e.g., 'createdAt', 'status') */\n orderBy: string;\n /** Sort direction */\n order?: 'asc' | 'desc';\n}\n\n/**\n * Query options for listing agents\n *\n * These options are supported by both CloudAgentProvider and LocalAgentProvider.\n * Cloud: Server-side filtering, sorting, and pagination\n * Local: Client-side filtering and sorting, no pagination (returns all)\n */\nexport interface ListAgentOptions {\n /**\n * Page number (starts from 1)\n * Cloud: Used for API pagination\n * Local: Ignored (returns all sessions)\n */\n page?: number;\n\n /**\n * Page size\n * Cloud: Number of items per page (default 20, max 100)\n * Local: Ignored (returns all sessions)\n */\n size?: number;\n\n /**\n * Sort options\n * Cloud: Sorts results by specified field and order\n * Local: Sorts results by specified field and order\n */\n sort?: ListAgentSort;\n\n /**\n * Filter conditions\n * Cloud: Filters results by specified field values\n * Local: Filters results by specified field values\n */\n filters?: ListAgentFilter[];\n\n /**\n * Day range filter (e.g., agents created in last N days)\n * Cloud: Filters by creation date\n * Local: Filters by creation date\n */\n dayRange?: number;\n\n /**\n * Title search keyword (matches agent title)\n * Cloud: Server-side search\n * Local: Client-side search\n */\n title?: string;\n}\n\n/**\n * Pagination metadata returned from list operations\n */\nexport interface PaginationInfo {\n /** Current page number (starts from 1) */\n page: number;\n /** Page size */\n size: number;\n /** Total number of items */\n total: number;\n /** Total number of pages */\n totalPages: number;\n /** Whether there is a next page */\n hasNext: boolean;\n /** Whether there is a previous page */\n hasPrev: boolean;\n}\n\n/**\n * Response from list operations that includes pagination\n */\nexport interface ListAgentResult<T = AgentState> {\n /** List of agent states or session info */\n agents: T[];\n /** Pagination information */\n pagination: PaginationInfo;\n}\n\n// ============================================\n// Session-Centric Types\n// ============================================\n\n/**\n * Session information (returned by list, mapped from Agent)\n */\nexport interface SessionInfo {\n /** Session ID (from agent.session) */\n id: string;\n /** Associated agent ID */\n agentId: string;\n /** Agent name */\n name?: string;\n /** Agent status */\n status: AgentStatus;\n /** When the session/agent was created */\n createdAt?: Date;\n /** Last activity timestamp */\n lastActivityAt?: Date;\n /** Working directory (for local agents) */\n cwd?: string;\n /** Whether the session is a playground */\n isPlayground?: boolean;\n}\n\n/**\n * Parameters for creating a new session\n */\nexport interface CreateSessionParams {\n /** Working directory */\n cwd: string;\n /** MCP server configurations */\n mcpServers?: McpServerConfig[];\n _meta?: Record<string, unknown>;\n}\n\n/**\n * Parameters for loading an existing session\n */\nexport interface LoadSessionParams {\n /** Session ID to load (required) */\n sessionId: string;\n /** Working directory */\n cwd: string;\n /** MCP server configurations */\n mcpServers?: McpServerConfig[];\n /** Callback executed right after session instance is created, before connection.loadSession() is called */\n onSessionCreated?: (session: ActiveSession) => Promise<void> | void;\n}\n\n/**\n * Parameters for initializing a workspace\n */\nexport interface InitializeWorkspaceParams {\n /** Working directory */\n cwd: string;\n /** MCP server configurations */\n mcpServers?: McpServerConfig[];\n /** Whether to activate the workspace window to foreground (default true) */\n needActivated?: boolean;\n}\n\n/**\n * Response for workspace initialization\n */\nexport interface InitializeWorkspaceResponse {\n /** Whether initialization was successful */\n success: boolean;\n /** Error message (if failed) */\n error?: string;\n}\n\n// ============================================\n// Resource Interfaces\n// ============================================\n\n/**\n * Prompts resource interface (ACP verbs)\n * Operations use the current session automatically\n */\nexport interface PromptsResource {\n /** Send a prompt and wait for completion */\n send(params: PromptParams): Promise<PromptResponse>;\n\n /** Stream a prompt (yields session updates) */\n stream(params: PromptParams): AsyncIterable<SessionNotification>;\n\n /** Cancel an ongoing prompt */\n cancel(): Promise<void>;\n}\n\n/**\n * Artifacts resource interface\n */\nexport interface ArtifactsResource {\n /** List all artifacts */\n list(params?: { type?: ArtifactType }): Promise<Artifact[]>;\n\n /** Get a single artifact */\n retrieve(artifactId: string): Promise<Artifact>;\n\n /** Get artifact content */\n content(artifactId: string): Promise<string>;\n}\n\n/**\n * Models resource interface\n */\nexport interface ModelsResource {\n /** Get available models for a repository */\n list(repo?: string): Promise<ModelInfo[]>;\n}\n\n// ============================================\n// Response Types\n// ============================================\n\n/**\n * Prompt response\n */\nexport interface PromptResponse {\n /** Stop reason */\n stopReason: 'end_turn' | 'max_tokens' | 'tool_use' | 'cancelled' | 'error';\n /** Response metadata */\n _meta?: Record<string, unknown>;\n}\n\n// ============================================\n// Sessions Resource Events (for ClientSessionsResource)\n// ============================================\n\n/**\n * Sessions resource events for monitoring session list changes\n */\nexport interface SessionsResourceEvents {\n /** Emitted when the sessions list changes (create, delete, update) */\n sessionsChanged: SessionInfo[];\n /** Emitted when a new session is created */\n sessionCreated: SessionInfo;\n /** Emitted when a session is deleted */\n sessionDeleted: { sessionId: string };\n /** Emitted when a session is updated (status change, etc.) */\n sessionUpdated: SessionInfo;\n}\n\n/**\n * Event handler type for sessions resource events\n */\nexport type SessionsResourceEventHandler<K extends keyof SessionsResourceEvents> = (\n data: SessionsResourceEvents[K]\n) => void | Promise<void>;\n\n// ============================================\n// Session Events (for ActiveSession)\n// ============================================\n\n/**\n * Session events for event subscription\n */\nexport interface SessionEvents {\n /** Emitted when session updates occur */\n sessionUpdate: SessionNotification;\n /** Emitted when an artifact is created */\n artifactCreated: Artifact;\n /** Emitted when an artifact is updated */\n artifactUpdated: Artifact;\n /** Emitted when an artifact is deleted */\n artifactDeleted: Artifact;\n /** Emitted when a permission request is received */\n permissionRequest: { requestId: string; params: RequestPermissionRequest };\n /** Emitted when a question request is received (ask_followup_question) */\n questionRequest: { toolCallId: string; request: QuestionRequest };\n /** Emitted when usage data is updated */\n usageUpdate: UsageUpdate;\n /** Emitted when a checkpoint is created */\n checkpointCreated: CheckpointInfo;\n /** Emitted when a checkpoint is updated */\n checkpointUpdated: CheckpointInfo;\n /** Emitted when a command is received */\n command: { action: string; params?: Record<string, unknown> };\n /** Emitted when connected to agent */\n connected: void;\n /** Emitted when disconnected from agent */\n disconnected: void;\n /** Emitted when an error occurs */\n error: Error;\n}\n\n/**\n * Event handler type for session events\n */\nexport type SessionEventHandler<K extends keyof SessionEvents> = (data: SessionEvents[K]) => void | Promise<void>;\n\n// ============================================\n// Active Session Interface\n// ============================================\n\n/**\n * Agent operations (accessed via session.agent)\n */\nexport interface SessionAgentOperations {\n /** Agent ID */\n readonly id: string;\n /** Agent state */\n readonly state: AgentState;\n /** Whether the agent is connected */\n readonly isConnected: boolean;\n /** Agent capabilities */\n readonly capabilities?: AgentCapabilities;\n}\n\n/**\n * Active Session interface\n * Represents an active session with its resources and operations\n *\n * Key design:\n * - Session is the primary API surface\n * - agentState provides direct access to underlying agent state\n * - disconnect() is called directly on session (not session.agent)\n */\nexport interface ActiveSession {\n /** Session ID */\n readonly id: string;\n /** Agent ID */\n readonly agentId: string;\n /** Agent state (direct access to underlying agent state) */\n readonly agentState: AgentState;\n /** Agent capabilities (available after connection) */\n readonly capabilities?: AgentCapabilities;\n /** Available session modes */\n readonly availableModes?: SessionMode[];\n /** Current session mode */\n readonly currentMode?: string;\n /** Available models for this session */\n readonly availableModels?: ModelInfo[];\n /** Current model ID */\n readonly currentModelId?: string;\n /** Available slash commands (updated via available_commands_update) */\n readonly availableCommands: AvailableCommand[];\n /** Whether the session is active */\n readonly isActive: boolean;\n /**\n * Session connection information (only available for cloud sessions)\n * 会话连接信息,包括sandboxId、link、token等\n */\n readonly connectionInfo?: SessionConnectionInfo;\n\n // Agent operations namespace (optional agent-level access)\n /** Agent operations */\n readonly agent: SessionAgentOperations;\n\n // ACP Resources\n /** Prompts resource */\n readonly prompts: PromptsResource;\n /** Artifacts resource */\n readonly artifacts: ArtifactsResource;\n /** Files resource */\n readonly files: FilesResource;\n\n // Permission management\n /** Resolve a permission request */\n resolvePermission(requestId: string, optionId: string): boolean;\n /** Reject a permission request */\n rejectPermission(requestId: string, reason?: string): boolean;\n\n // Question management (ask_followup_question)\n /** Answer a question request with user's selections */\n answerQuestion(toolCallId: string, answers: QuestionAnswers): boolean;\n /** Cancel a question request */\n cancelQuestion(toolCallId: string, reason?: string): boolean;\n\n // Tool callback management\n /** Callback for tool operations (skip or cancel) */\n toolCallback(toolCallId: string, toolName: string, action: 'skip' | 'cancel'): Promise<{ success: boolean; error?: string }>;\n\n // Session mode management\n /** Set the current session mode */\n setMode(modeId: string): Promise<void>;\n /** Set the current session model */\n setSessionModel(modelId: string): Promise<void>;\n /** Set available commands (called when available_commands_update is received) */\n setAvailableCommands(commands: AvailableCommand[]): void;\n /** Set available modes (called after create/load) */\n setModes(availableModes?: SessionMode[], currentMode?: string): void;\n /** Set available models (called after create/load) */\n setModels(availableModels?: ModelInfo[], currentModelId?: string): void;\n\n // Event subscription\n /** Subscribe to an event */\n on<K extends keyof SessionEvents>(event: K, handler: SessionEventHandler<K>): this;\n /** Unsubscribe from an event */\n off<K extends keyof SessionEvents>(event: K, handler: SessionEventHandler<K>): this;\n /** Subscribe to an event once */\n once<K extends keyof SessionEvents>(event: K, handler: SessionEventHandler<K>): this;\n\n // Lifecycle - disconnect directly on session\n /** Disconnect from the session/agent */\n disconnect(): void;\n\n /** Symbol.dispose for 'using' keyword support */\n [Symbol.dispose](): void;\n}\n\n// ============================================\n// Provider & Client Types\n// ============================================\n\n/**\n * 环境类型\n */\nexport type EnvironmentType = 'local' | 'cloud';\n\n/**\n * Agent provider interface\n *\n * Responsible for:\n * - Managing agent state/configuration storage\n * - Creating connections to agents\n * - Abstracting away transport details (cloud/local)\n *\n * The provider.connect() method returns an AgentConnection.\n * The client wraps the connection in an ActiveSession instance.\n *\n * @typeParam C - Connection type used by this provider (e.g., CloudAgentConnection, LocalAgentConnection)\n */\nexport interface AgentProvider<C extends AgentConnection = AgentConnection> {\n /**\n * Create a new agent and return its ID\n *\n * @param params - Optional session params (used by LocalAgentProvider to get cwd)\n * @returns Agent ID (Cloud: UUID, Local: cwd)\n */\n create?(params?: CreateSessionParams): Promise<string>;\n /** Get agent state by ID */\n get(agentId: string): Promise<AgentState | undefined>;\n\n /**\n * List all agent states with pagination information\n *\n * @param options - Optional query parameters for filtering, sorting, and pagination\n * Cloud providers use these for API queries and return server pagination\n * Local providers apply client-side filtering and return synthetic pagination\n * @returns Object containing agents array and pagination info\n */\n list(options?: ListAgentOptions): Promise<ListAgentResult<AgentState>>;\n\n /** Connect to an agent and return the connection */\n connect(agentId: string): Promise<C>;\n /** Delete an agent by ID */\n delete(agentId: string): Promise<boolean>;\n\n /**\n * Archive an agent by ID (optional)\n * Used by CloudAgentProvider for archiving agents\n *\n * @param agentId - Agent ID to archive\n * @returns Object containing the archived agent ID\n */\n archive?(agentId: string): Promise<{ id: string }>;\n\n /**\n * Rename an agent by ID (optional)\n * Used by CloudAgentProvider and LocalAgentProvider for renaming agents\n *\n * @param agentId - Agent ID to rename\n * @param title - New title for the agent\n * @returns Object containing the renamed agent ID\n */\n rename?(agentId: string, title: string): Promise<{ id: string }>;\n\n /**\n * Move an agent by ID (optional)\n * Used by LocalAgentProvider for moving Playground sessions to Workspace\n *\n * @param agentId - Agent ID to move\n * @returns Object containing the moved agent ID\n */\n move?(agentId: string): Promise<{ id: string }>;\n\n /** Filesystem provider (optional - some providers may not support filesystem operations) */\n readonly filesystem?: FilesystemProvider;\n\n /**\n * Get available models for a repository (optional)\n * Implementation varies by provider type\n * @param repo - Repository identifier\n * @returns Array of model information\n */\n getModels?(repo?: string): Promise<ModelInfo[]>;\n\n /**\n * Register sessionId → agentId mapping (optional, used by LocalAgentProvider)\n * Called after session creation to maintain the mapping for loadSession\n *\n * @param sessionId - Session ID returned by connection.createSession()\n * @param agentId - Agent ID (cwd for Local)\n */\n registerSession?(sessionId: string, agentId: string): void;\n\n /**\n * Open a workspace window (optional, used by LocalAgentProvider)\n *\n * @param params - Workspace params including cwd\n * @returns Response with success status\n */\n openWorkspace?(params: InitializeWorkspaceParams): Promise<InitializeWorkspaceResponse>;\n\n /**\n * Pick files from file dialog (optional, used by LocalAgentProvider)\n *\n * @param params - File picker params including filters\n * @returns Response with file paths and cancel status\n */\n pickFile?(params?: PickFileParams): Promise<PickFileResponse>;\n\n /**\n * Pick folders from folder dialog (optional, used by LocalAgentProvider)\n *\n * @param params - Folder picker params\n * @returns Response with folder paths and cancel status\n */\n pickFolder?(params?: PickFolderParams): Promise<PickFolderResponse>;\n\n /**\n * Upload a file to cloud storage (optional)\n *\n * @param params - Upload parameters including file content\n * @returns Response with cloud URL after successful upload\n */\n uploadFile?(params: UploadFileParams): Promise<UploadFileResponse>;\n\n /**\n * Search for files in the workspace (optional, used by LocalAgentProvider)\n *\n * @param params - Search parameters including options\n * @returns Response with search results\n */\n searchFile?(params: SearchFileParams): Promise<SearchFileResponse>;\n\n /**\n * Batch toggle plugins (optional, used by LocalAgentProvider)\n *\n * @param request - Batch plugin operation request\n * @returns Batch operation result\n */\n batchTogglePlugins?(request: BatchPluginOperationRequest): Promise<BatchPluginOperationResult>;\n\n /**\n * Get installed plugins (optional, used by LocalAgentProvider)\n *\n * @param forceRefresh - Whether to force refresh the cache\n * @returns Array of installed plugins\n */\n getInstalledPlugins?(forceRefresh?: boolean): Promise<Array<{\n name: string;\n marketplaceName: string;\n status: string;\n description?: string;\n version?: string;\n installScope?: 'user' | 'project';\n }>>;\n\n /**\n * Install plugins (optional, used by LocalAgentProvider)\n *\n * @param pluginNames - Array of plugin names to install\n * @param marketplaceName - Marketplace name\n * @param installScope - Install scope ('user' | 'project')\n * @returns Result of the operation\n */\n installPlugins?(\n pluginNames: string[],\n marketplaceName: string,\n installScope?: 'user' | 'project'\n ): Promise<{ success: boolean; error?: string }>;\n\n /**\n * Register an event listener\n * Provider implementations should forward events to the underlying transport\n *\n * @param event - Event name\n * @param handler - Event handler function\n */\n on?(event: string, handler: (...args: any[]) => void): void;\n\n /**\n * Unregister an event listener\n *\n * @param event - Event name\n * @param handler - Event handler function to remove\n */\n off?(event: string, handler: (...args: any[]) => void): void;\n}\n\n/**\n * AgentClient initialization options\n */\nexport interface AgentClientOptions {\n /** Agent provider (required) */\n provider: AgentProvider;\n /** Logger instance */\n logger?: Logger;\n /** Client capabilities (sent during initialization) */\n clientCapabilities?: ClientCapabilities;\n /**\n * 运行环境类型\n * - 'local': IDE 本地环境\n * - 'cloud': 云端环境\n */\n environmentType?: EnvironmentType;\n}\n\n/**\n * Client sessions resource interface\n * Top-level API for session management\n *\n * Key design:\n * - list() returns sessions with pagination info (mapped from agents)\n * - create() creates a new session (auto-creates agent and connects)\n * - load() loads an existing session (finds agent by sessionId and connects)\n * - archive() archives a session/agent\n * - initializeWorkspace() initializes a workspace for future sessions\n */\nexport interface ClientSessionsResource {\n /**\n * List all sessions with pagination info\n * Cloud: Returns server-side filtered/sorted/paginated results\n * Local: Returns client-side filtered/sorted results (synthetic pagination)\n */\n list(options?: ListAgentOptions): Promise<ListAgentResult<SessionInfo>>;\n\n /** Create a new session (auto-creates agent and connects) */\n create(params: CreateSessionParams): Promise<ActiveSession>;\n\n /** Load an existing session (finds agent by sessionId and connects) */\n load(params: LoadSessionParams): Promise<ActiveSession>;\n\n /**\n * Archive a session/agent\n * @param sessionId - Session ID to archive\n * @returns Object containing the archived session ID\n */\n archive(sessionId: string): Promise<{ id: string }>;\n\n /**\n * Rename a session/agent\n * @param sessionId - Session ID to rename\n * @param title - New title for the session\n * @returns Object containing the renamed session ID\n */\n rename(sessionId: string, title: string): Promise<{ id: string }>;\n\n /**\n * Move a session (Playground → Workspace)\n * @param sessionId - Session ID to move\n * @returns Object containing the moved session ID\n */\n move(sessionId: string): Promise<{ id: string }>;\n\n /** Initialize a workspace for future sessions */\n initializeWorkspace(params: InitializeWorkspaceParams): Promise<InitializeWorkspaceResponse>;\n\n /** Models resource for getting available models */\n readonly models: ModelsResource;\n\n /** Get current workspaces list */\n getCurrentWorkspaces(filter?: { activeOnly?: boolean }): Promise<WorkspaceInfo[]>;\n\n // Event subscription for sessions list changes\n /** Subscribe to sessions resource events */\n on<K extends keyof SessionsResourceEvents>(\n event: K,\n handler: SessionsResourceEventHandler<K>\n ): void;\n\n /** Unsubscribe from sessions resource events */\n off<K extends keyof SessionsResourceEvents>(\n event: K,\n handler: SessionsResourceEventHandler<K>\n ): void;\n\n /** Open a workspace (for LocalAgentProvider) */\n openWorkspace(params: InitializeWorkspaceParams): Promise<InitializeWorkspaceResponse>;\n\n /** Pick files from file dialog (for LocalAgentProvider) */\n pickFile(params?: PickFileParams): Promise<PickFileResponse>;\n\n /** Pick folders from folder dialog (for LocalAgentProvider) */\n pickFolder(params?: PickFolderParams): Promise<PickFolderResponse>;\n\n /** Upload a file to cloud storage */\n uploadFile(params: UploadFileParams): Promise<UploadFileResponse>;\n\n /** Search for files in the workspace (for LocalAgentProvider) */\n searchFile(params: SearchFileParams): Promise<SearchFileResponse>;\n\n /** Batch toggle plugins (for LocalAgentProvider) */\n batchTogglePlugins(request: BatchPluginOperationRequest): Promise<BatchPluginOperationResult>;\n\n /** Get installed plugins (for LocalAgentProvider) */\n getInstalledPlugins?(forceRefresh?: boolean): Promise<Array<{\n name: string;\n marketplaceName: string;\n status: string;\n description?: string;\n version?: string;\n installScope?: 'user' | 'project';\n }>>;\n\n /** Install plugins (for LocalAgentProvider) */\n installPlugins(\n pluginNames: string[],\n marketplaceName: string,\n installScope?: 'user' | 'project'\n ): Promise<{ success: boolean; error?: string }>;\n\n /**\n * Get support scenes from backend API (for LocalAgentProvider)\n * API 端点: GET /v2/as/support/scenes (Local) 或 GET /console/as/support/scenes (Web)\n */\n getSupportScenes(): Promise<Array<{\n id: number;\n name: string;\n plugins: Array<{\n id: number;\n name: string;\n marketplaceName: string;\n }>;\n prompts: string[];\n }>>;\n\n}\n\n// ============================================\n// Workspace Types\n// ============================================\n\n/**\n * Workspace information (aligned with FolderSelectResult)\n */\nexport interface WorkspaceInfo {\n /** Folder path */\n path: string;\n /** Folder display name */\n label: string;\n}\n\n// ============================================\n// File Picker Types\n// ============================================\n\n/**\n * File filter for pickFile dialog\n */\nexport interface FileFilter {\n /** Display name for the filter */\n readonly name: string;\n /** File extensions (without dot) */\n readonly extensions: string[];\n}\n\n/**\n * Parameters for picking files\n */\nexport interface PickFileParams {\n /** Default path for the dialog */\n readonly defaultPath?: string;\n /** File type filters */\n readonly filters?: FileFilter[];\n /** Whether to allow multiple selection (default false) */\n readonly canSelectMany?: boolean;\n}\n\n/**\n * Response from picking files\n */\nexport interface PickFileResponse {\n /** Selected files - File objects in browser, absolute path strings in IDE */\n readonly files: Array<File | string>;\n /** Whether user cancelled the dialog */\n readonly canceled: boolean;\n /** Error message (if failed) */\n readonly error?: string;\n}\n\n// ============================================\n// Folder Picker Types\n// ============================================\n\n/**\n * Parameters for picking folders\n */\nexport interface PickFolderParams {\n /** Default path for the dialog */\n readonly defaultPath?: string;\n}\n\n/**\n * Response from picking folders\n */\nexport interface PickFolderResponse {\n /** Selected folder paths */\n readonly folderPaths: string[];\n /** Whether user cancelled the dialog */\n readonly canceled: boolean;\n /** Error message (if failed) */\n readonly error?: string;\n}\n\n// ============================================\n// File Upload Types\n// ============================================\n\n/**\n * Parameters for uploading files\n */\nexport interface UploadFileParams {\n /** Files to upload - File objects in browser, absolute path strings in IDE */\n readonly files: Array<File | string>;\n}\n\n/**\n * Response from uploading files\n */\nexport interface UploadFileResponse {\n /** Whether upload was successful */\n readonly success: boolean;\n /** Cloud URLs corresponding to each uploaded file (same order as input files) */\n readonly urls?: string[];\n /** URL expiration time in seconds (from backend) */\n readonly expireSeconds?: number;\n /** Error message (if upload failed) */\n readonly error?: string;\n}\n\n// ============================================\n// File Search Types\n// ============================================\n\n/**\n * Mention type for file/folder\n */\nexport enum MentionType {\n file = 'file',\n folder = 'folder'\n}\n\n/**\n * Search options for file search\n */\nexport interface SearchOptions {\n /** Search keyword */\n readonly search?: string;\n /** Number of results to return */\n readonly resultNum: number;\n}\n\n/**\n * File search result\n */\nexport interface SearchFileResult {\n /** Full path */\n path: string;\n /** Relative path */\n relativePath: string;\n /** File name */\n fileName?: string;\n /** Folder name */\n folderName?: string;\n /** Type (file or folder) */\n type: MentionType.file | MentionType.folder;\n}\n\n/**\n * Parameters for searching files\n */\nexport interface SearchFileParams {\n /** Search options */\n readonly options: SearchOptions;\n /** Search path*/\n readonly cwd?: string;\n}\n\n/**\n * Response from searching files\n */\nexport interface SearchFileResponse {\n /** Search results */\n readonly results: SearchFileResult[];\n /** Error message (if failed) */\n readonly error?: string;\n}\n","/**\n * Backend Provider 类型定义\n *\n * 定义 IBackendProvider 接口和配置\n */\n\n// ============================================================================\n// Account 相关类型\n// ============================================================================\n\n/**\n * 账号版本类型\n */\nexport type Edition = 'pro' | 'personal' | 'ultimate' | 'exclusive';\n\n/**\n * 版本展示类型(用于 UI 展示)\n * - free: 免费版(个人版未订阅 Pro)\n * - pro: Pro 版(个人版已订阅 Pro)\n * - ultimate: 旗舰版(团队版)\n * - exclusive: 专享版(企业版)\n */\nexport type EditionDisplayType = 'free' | 'pro' | 'ultimate' | 'exclusive';\n\n/**\n * 部署状态\n */\nexport interface DeployStatus {\n statusCode: number;\n statusMsg: string;\n detailMsg: string;\n}\n\n/**\n * 套餐代码\n */\n\n/**\n * TCACA_code_001_PqouKr6QWV CodeBuddy海外版免费包\n * TCACA_code_002_AkiJS3ZHF5 CodeBuddy海外版Pro版本包-包月/CodeBuddy Pro Plan - Monthly:\n * TCACA_code_006_DbXS0lrypC CodeBuddy海外版一次性免费赠送2周的Pro版本包/CodeBuddy One-time Free 2-Week Pro Plan Trial\n * TCACA_code_007_nzdH5h4Nl0 CodeBuddy海外版运营裂变包/CodeBuddy Growth Plan\n * TCACA_code_003_FAnt7lcmRT CodeBuddy海外版Pro版本包-包年/CodeBuddy Pro Plan - Yearly\n * TCACA_code_008_cfWoLwvjU4 赠送月包\n */\nexport enum CommodityCode {\n free = 'TCACA_code_001_PqouKr6QWV', // free\n proMon = 'TCACA_code_002_AkiJS3ZHF5',\n // 国内月包(国际pro+)\n proMonPlus = 'TCACA_code_005_maRGyrHhw1',\n // 免费赠送2周\n gift = 'TCACA_code_006_DbXS0lrypC',\n activity = 'TCACA_code_007_nzdH5h4Nl0',\n proYear = 'TCACA_code_003_FAnt7lcmRT',\n // 国际(free 月包、国内试用包)\n freeMon = 'TCACA_code_008_cfWoLwvjU4', // free\n // 加量包\n extra = 'TCACA_code_009_0XmEQc2xOf',\n}\n\n/**\n * 套餐资源项(用于展示列表)\n */\nexport interface PlanResource {\n /** 资源 ID */\n id: string;\n /** 套餐名称(i18n key) */\n name: string;\n /** 套餐代码 */\n packageCode: CommodityCode;\n /** 是否是每日刷新的套餐 */\n isDaily: boolean;\n /** 总量 */\n total: number;\n /** 已用 */\n used: number;\n /** 剩余 */\n left: number;\n /** 到期时间戳 */\n expireAt?: number;\n /** 刷新时间戳 */\n refreshAt?: number;\n}\n\n/**\n * 账号套餐信息\n */\nexport interface AccountPlan {\n /** 是否是 Pro 版本 */\n isPro: boolean;\n // 是否是试用版本\n isTria?: boolean;\n /** 到期时间戳 */\n expireAt?: number;\n // 刷新时间(年套餐下、本周期结束日期)\n refreshAt?: number;\n /** 自动续费标志 0-关闭 1-开启 */\n renewFlag: 0 | 1;\n /** 套餐代码 */\n PackageCode?: CommodityCode;\n /** 套餐名称 */\n name: string;\n usageTotal?: string;\n usageUsed?: string;\n usageLeft?: string;\n /** 所有套餐资源列表 */\n resources?: PlanResource[];\n}\n\n/**\n * 账号信息\n */\nexport interface Account {\n /** 用户ID(唯一标识) */\n uid: string;\n /** 用户昵称 */\n nickname: string;\n /** 版本类型 */\n type: Edition;\n /** 版本展示类型(用于 UI 展示) */\n editionType: EditionDisplayType;\n /** 是否最后一次登录 */\n lastLogin: boolean;\n /** 企业ID */\n enterpriseId?: string;\n /** 企业名称 */\n enterpriseName?: string;\n /** 企业LOGO */\n enterpriseLogo?: string;\n /** 企业内用户名 */\n enterpriseUserName?: string;\n /** 插件是否启用 */\n pluginEnabled?: boolean;\n /** 部署状态 */\n deployStatus?: DeployStatus;\n /** 是否是 Pro 版本 */\n isPro?: boolean;\n /** 到期时间戳 */\n expireAt?: string | number;\n /** 刷新时间(年套餐下、本周期结束日期) */\n refreshAt?: number;\n /** 自动续费标志 0-关闭 1-开启 */\n renewFlag?: 0 | 1;\n /** 套餐代码 */\n PackageCode?: CommodityCode;\n /** 套餐名称 */\n name?: string;\n email?: string;\n /** 套餐总用量 */\n usageTotal?: string;\n /** 套餐已用量 */\n usageUsed?: string;\n /** 套餐剩余用量 */\n usageLeft?: string;\n /** 所有套餐资源列表(个人用户) */\n resources?: PlanResource[];\n}\n\n/** 账户状态 */\nexport enum AccountStatus {\n /** 有效 */\n valid = 0,\n /** 已退款 */\n refund = 1,\n /** 已过期 */\n expired = 2,\n /** 已用完 */\n usedUp = 3,\n}\n\n// https://iwiki.woa.com/p/1151041572#7%E3%80%81%E8%B4%B9%E7%94%A8%E4%B8%AD%E5%BF%83%E6%8E%A7%E5%88%B6%E5%8F%B0-%E8%B5%84%E6%BA%90%E5%8C%85%E7%AE%A1%E7%90%86%E6%9F%A5%E8%AF%A2%E6%8E%A5%E5%8F%A3\nexport interface UserResource {\n AccountId: number;\n ResourceId: string;\n // 账户类型: 1-RI 2-资源包 9 - 组合包\n // ResourceType: 1 | 2 | 9;\n // 总周期数量\n TotalCycles: 1 | 12;\n // 剩余周期数量\n RemainCycles: number;\n Status: AccountStatus;\n // 费用类型:免费1/付费2\n FeeType: 1 | 2;\n PackageCode: CommodityCode;\n PackageName: string;\n // // 周期容量(使用下方精确值)\n // CycleCapacitySize: number;\n // // 剩余周期数量\n // CycleCapacityRemain: number;\n SupportAutoRenew: 0 | 1;\n SupportManualRenew: 0 | 1;\n AutoRenewFlag: 0 | 1;\n // AutoRenewTimeUnit: string;\n // AutoRenewTimeSpan: 1;\n // ProductCode: string;\n // SubProductCode: string;\n // 单周期开始时间\n CycleStartTime: number; // 时间戳 单位为毫秒\n // 单周期结束时间\n CycleEndTime: number; // 时间戳 单位为毫秒\n // CapacityType: string;\n CreateTime: string;\n ExpiredTime: string; // 2025-11-27 14:36:00\n DeductionStartTime: number; // 时间戳 单位为毫秒\n // 抵扣结束时间戳 单位为毫秒\n DeductionEndTime: number; // 时间戳 单位为毫秒\n CapacityUsedPrecise: string;\n // 当前周期余量精确值\n CycleCapacityRemainPrecise: string;\n // 当前周期总量精确值\n CycleCapacitySizePrecise: string;\n // 剩余容量精确值\n CapacityRemainPrecise: string;\n // 用量精确值\n CapacitySizePrecise: string;\n}\n\n// ============================================================================\n// Model 相关类型\n// ============================================================================\n\n/**\n * 推理配置\n */\nexport interface ReasoningConfig {\n /** 推理努力程度 */\n effort: 'low' | 'medium' | 'high';\n /** 摘要模式 */\n summary: 'auto' | 'always' | 'never';\n}\n\n/**\n * 模型信息\n */\nexport interface ModelInfo {\n /** 模型ID */\n id: string;\n /** 模型名称 */\n name: string;\n /** 供应商 */\n vendor: string;\n /** 最大输出 token 数 */\n maxOutputTokens: number;\n /** 最大输入 token 数 */\n maxInputTokens: number;\n /** 是否支持工具调用 */\n supportsToolCall: boolean;\n /** 是否支持图像 */\n supportsImages: boolean;\n /** 是否禁用多模态 */\n disabledMultimodal: boolean;\n /** 最大允许大小 */\n maxAllowedSize: number;\n /** 是否支持推理 */\n supportsReasoning: boolean;\n /** 是否仅推理模式 */\n onlyReasoning: boolean;\n /** 温度参数 */\n temperature: number;\n /** 推理配置 */\n reasoning: ReasoningConfig;\n /** 英文描述 */\n descriptionEn: string;\n /** 中文描述 */\n descriptionZh: string;\n}\n\n// ============================================================================\n// UserConnector 相关类型\n// ============================================================================\n\n/**\n * 用户连接器信息\n */\nexport interface UserConnector {\n /** 用户ID */\n user_id: string;\n /** 连接器名称 */\n name: ConnectorType;\n displayName: string;\n /** 连接时间 */\n connect_at: string;\n /** 连接状态,0未连接,1已连接 */\n connectStatus: 0 | 1;\n /** 勾选仓库列表,逗号分隔 */\n repos: string;\n /** 激活状态,0未激活,1已激活 */\n activeStatus: 0 | 1;\n /** 提示词 */\n prompt: string;\n /** 跳转URL */\n url: string;\n /** 描述 */\n description: string;\n /** OAuth Client ID */\n oauthClientId: string;\n /** OAuth 重定向 URL */\n oauthRedirectUrl: string;\n}\n\n/**\n * 获取用户连接器列表响应\n */\nexport interface ListUserConnectorResponse {\n /** 连接器列表 */\n connectors: UserConnector[];\n}\n\n/**\n * 修改用户连接器连接状态请求\n */\nexport interface ModifyUserConnectorConnectStatusRequest {\n /** 连接器名称 */\n name: ConnectorType;\n /** 连接状态,0未连接,1已连接 */\n connectStatus: 0 | 1;\n /** 激活状态,0未激活,1已激活 */\n activeStatus?: 0 | 1;\n /** 勾选仓库列表 */\n repos?: string[];\n}\n\n/**\n * 修改用户连接器仓库请求\n */\nexport interface ModifyUserConnectorRepoRequest {\n /** 连接器名称 */\n name: ConnectorType;\n /** 仓库列表 */\n repo?: string[];\n}\n\n/**\n * 修改用户连接器激活状态请求\n */\nexport interface ModifyUserConnectorActiveStatusRequest {\n /** 连接器名称 */\n name: ConnectorType;\n /** 激活状态,0未激活,1已激活 */\n activeStatus: 0 | 1;\n}\n\n// 连接器类型\nexport type ConnectorType = 'github' | 'gongfeng' | 'cnb' | 'figma';\n\n// ============================================================================\n// TaskConnector 相关类型\n// ============================================================================\n\n/**\n * 任务连接器信息\n */\nexport interface TaskConnector {\n /** 用户ID */\n user_id: string;\n /** 任务ID */\n task_id: string;\n /** 连接器名称(唯一标识,如github/gongfeng/figma/cnb) */\n name: string;\n displayName: string;\n /** 勾选仓库列表,逗号分隔 */\n repos: string;\n /** 激活状态:0未激活 / 1已激活 */\n activeStatus: 0 | 1;\n /** 创建时间 */\n created_at: string;\n /** 更新时间 */\n updated_at: string;\n /** 提示词 */\n prompt: string;\n /** 跳转URL */\n url: string;\n /** 描述 */\n description: string;\n /** OAuth Client ID */\n oauthClientId?: string;\n /** OAuth 重定向 URL */\n oauthRedirectUrl?: string;\n /** 连接时间 */\n connect_at: string;\n /** 连接状态,0未连接,1已连接 */\n connectStatus: 0 | 1;\n}\n\n/**\n * 添加任务请求\n */\nexport interface AddTaskRequest {\n /** 任务ID */\n taskId: string;\n /** 连接器列表 */\n connectors?: Array<{\n name: string;\n repos: string;\n activeStatus?: 0 | 1;\n }>;\n}\n\n/**\n * 添加任务响应\n */\nexport interface AddTaskResponse {\n /** 任务ID */\n taskId: string;\n}\n\n/**\n * 获取任务连接器列表响应\n */\nexport interface ListTaskConnectorResponse {\n /** 连接器列表 */\n connectors: TaskConnector[];\n}\n\n/**\n * 修改任务连接器激活状态请求\n */\nexport interface ModifyTaskConnectorActiveStatusRequest {\n /** 任务ID */\n taskId: string;\n /** 连接器名称 */\n name: string;\n /** 激活状态,0未激活,1已激活 */\n activeStatus: 0 | 1;\n}\n\n/**\n * 修改任务连接器激活状态响应\n */\nexport interface ModifyTaskConnectorActiveStatusResponse {\n /** 任务ID */\n taskId: string;\n}\n\n/**\n * 修改任务连接器仓库请求\n */\nexport interface ModifyTaskConnectorRepoRequest {\n /** 任务ID */\n taskId: string;\n /** 连接器名称 */\n name: string;\n /** 仓库列表 */\n repo: string[];\n}\n\n/**\n * 修改任务连接器仓库响应\n */\nexport interface ModifyTaskConnectorRepoResponse {\n /** 任务ID */\n taskId: string;\n}\n\n/**\n * oauth回调请求参数\n */\nexport interface SaveOauthTokenRequest {\n /** 连接器名称 */\n name: ConnectorType;\n /** 第三方回调code */\n authorizationCode: string;\n}\n\n/**\n * 获取仓库列表请求参数\n */\nexport interface GetRepoListRequest {\n /** 连接器名称 */\n name: ConnectorType;\n}\n\n/**\n * 撤销所有OAuth连接请求参数\n */\nexport interface RevokeAllRequest {\n /** 连接器名称 */\n name: ConnectorType;\n /** 安装ID列表(可选,如果不传则撤销所有) */\n installationIds?: string[];\n}\n\n/**\n * GitHub 仓库所有者信息\n */\nexport interface OauthGitHubRepoOwner {\n login: string;\n avatar_url: string;\n url: string;\n html_url: string;\n followers_url: string;\n starred_url: string;\n repos_url: string;\n events_url: string;\n}\n\n/**\n * GitHub 仓库信息\n */\nexport interface OauthGitHubRepo {\n name: string;\n full_name: string;\n private: boolean;\n html_url: string;\n url: string;\n teams_url: string;\n hooks_url: string;\n events_url: string;\n branches_url: string;\n git_commits_url: string;\n merges_url: string;\n pulls_url: string;\n git_url: string;\n clone_url: string;\n svn_url: string;\n downloads_url: string;\n description: string;\n owner: OauthGitHubRepoOwner;\n forks_count: number;\n}\n/**\n * 工蜂仓库信息\n */\nexport interface OauthGongfengRepo {\n id: number; // 仓库 ID\n description: string; // 仓库描述\n public: boolean; // 是否公开\n archived: boolean; // 是否归档\n visibility_level: number; // 可见性级别\n public_visibility: number; // 公开可见性级别\n namespace: {\n name: string,\n description: string,\n path: string,\n }; // 命名空间信息\n owner: {\n username: string,\n web_url: string,\n name: string,\n state: string,\n avatar_url: string\n }; // 所有者信息\n name: string; // 仓库名称\n name_with_namespace: string; // 带命名空间的仓库名称\n path: string; // 仓库路径\n path_with_namespace: string; // 带命名空间的仓库路径\n type: string; // 仓库类型(如 \"GIT\")\n default_branch: string; // 默认分支\n ssh_url_to_repo: string; // SSH 克隆地址\n http_url_to_repo: string; // HTTP 克隆地址\n https_url_to_repo: string; // HTTPS 克隆地址\n web_url: string; // 仓库网页地址\n avatar_url: string;\n}\n\n/**\n * CNB仓库信息\n */\nexport interface OauthCnbRepo {\n name: string;\n [key: string]: any;\n}\n\n/**\n * 获取仓库列表响应\n */\nexport interface GetRepoListResponse {\n github_repos?: {\n [key: string | number]: OauthGitHubRepo[]\n }\n gongfeng_repos?: OauthGongfengRepo[]\n cnb_repos?: OauthCnbRepo[]\n}\n\n/**\n * OAuth 用户信息\n */\nexport interface OauthUserInfo {\n /** 头像 URL */\n avatarUrl: string;\n /** 用户名 */\n name: string;\n}\n\n/**\n * 获取 OAuth 用户信息请求参数\n */\nexport interface GetOauthUserRequest {\n /** 连接器名称 */\n name: ConnectorType;\n}\n\n/**\n * 获取 OAuth 用户信息响应\n */\nexport interface GetOauthUserResponse {\n /** 用户信息 */\n user: OauthUserInfo;\n}\n\n/**\n * Figma 文件信息\n */\nexport interface FigmaFileInfo {\n /** 文件名 */\n name: string;\n /** 角色 */\n role: string;\n /** 最后修改时间 */\n lastModified: string;\n /** 编辑器类型 */\n editorType: string;\n /** 缩略图 URL */\n thumbnailUrl: string;\n /** 版本 */\n version: string;\n /** 主文件 Key */\n mainFileKey: string;\n}\n\n/**\n * 获取文件信息请求参数\n */\nexport interface GetFileRequest {\n /** 连接器名称 */\n name: ConnectorType;\n /** 文件 URL */\n url: string;\n}\n\n/**\n * 获取文件信息响应\n */\nexport interface GetFileResponse {\n /** Figma 文件信息 */\n figmaFileInfo: FigmaFileInfo | null;\n}\n\n// ============================================================================\n// Backend Provider 配置\n// ============================================================================\n\n/**\n * Backend Provider 配置选项\n */\nexport interface BackendProviderConfig {\n /** API 基础 URL (例如: https://api.example.com) */\n baseUrl: string;\n /** 认证 Token */\n authToken?: string;\n}\n\n// ============================================================================\n// Backend Provider 接口\n// ============================================================================\n\n/**\n * 企业用户用量信息\n */\nexport interface EnterpriseUsage {\n /** 已使用 credit */\n credit: number;\n /** 周期开始时间 */\n cycleStartTime: string;\n /** 周期结束时间 */\n cycleEndTime: string;\n /** 周期重置时间 */\n cycleResetTime: string;\n /** 限额数量 */\n limitNum: number;\n}\n\n/**\n * IBackendProvider 接口\n *\n * 定义与后端 API 交互的抽象接口\n *\n * 注意:getAgents 和 getModels 方法已废弃并移除,\n * 请使用 IAgentAdapter 中的对应方法\n */\nexport interface IBackendProvider {\n /**\n * 获取当前账号信息\n * @returns Promise<Account | null> 账号信息,未登录时返回 null\n */\n getAccount(): Promise<Account | null>;\n\n /**\n * 获取用户连接器列表\n * @returns Promise<ListUserConnectorResponse | null> 用户连接器列表,失败时返回 null\n */\n getUserConnector(): Promise<ListUserConnectorResponse>;\n\n /**\n * 修改用户连接器连接状态\n * @param request 请求参数\n * @returns Promise<void>\n */\n modifyUserConnectorConnectStatus(request: ModifyUserConnectorConnectStatusRequest): Promise<void>;\n\n /**\n * 修改用户连接器仓库\n * @param request 请求参数\n * @returns Promise<void>\n */\n modifyUserConnectorRepo(request: ModifyUserConnectorRepoRequest): Promise<void>;\n\n /**\n * 修改用户连接器激活状态\n * @param request 请求参数\n * @returns Promise<void>\n */\n modifyUserConnectorActiveStatus(request: ModifyUserConnectorActiveStatusRequest): Promise<void>;\n\n /**\n * 删除用户连接器\n * @param name 连接器名称\n * @returns Promise<void>\n */\n deleteUserConnector(name: ConnectorType): Promise<void>;\n\n /**\n * 添加任务\n * @param request 请求参数\n * @returns Promise<AddTaskResponse>\n */\n addConnectorTask(request: AddTaskRequest): Promise<AddTaskResponse>;\n\n /**\n * 获取任务连接器列表\n * @param taskId 任务ID\n * @returns Promise<ListTaskConnectorResponse>\n */\n getTaskConnector(taskId: string): Promise<ListTaskConnectorResponse>;\n\n /**\n * 修改任务连接器激活状态\n * @param request 请求参数\n * @returns Promise<ModifyTaskConnectorActiveStatusResponse>\n */\n modifyTaskConnectorActiveStatus(request: ModifyTaskConnectorActiveStatusRequest): Promise<ModifyTaskConnectorActiveStatusResponse>;\n\n /**\n * 修改任务连接器仓库\n * @param request 请求参数\n * @returns Promise<ModifyTaskConnectorRepoResponse>\n */\n modifyTaskConnectorRepo(request: ModifyTaskConnectorRepoRequest): Promise<ModifyTaskConnectorRepoResponse>;\n\n /**\n * oauth回调,后端用第三方的code换token的\n */\n saveOauthToken(request: SaveOauthTokenRequest): Promise<void>;\n\n /**\n * 获取OAuth连接器的仓库列表\n * @param request 请求参数\n * @returns Promise<GetRepoListResponse> 仓库列表响应\n */\n getRepoList(request: GetRepoListRequest): Promise<GetRepoListResponse>;\n\n /**\n * 撤销OAuth连接器的所有连接\n * @param request 请求参数\n * @returns Promise<void>\n */\n revokeAll(request: RevokeAllRequest): Promise<void>;\n\n /**\n * 获取 OAuth 用户信息\n * @param request 请求参数\n * @returns Promise<GetOauthUserResponse> 用户信息响应\n */\n getOauthUser(request: GetOauthUserRequest): Promise<GetOauthUserResponse>;\n\n /**\n * 获取文件信息\n * @param request 请求参数\n * @returns Promise<GetFileResponse> 文件信息响应\n */\n getFile(request: GetFileRequest): Promise<GetFileResponse>;\n\n /**\n * 触发登录流程\n * - Web 环境: 跳转到登录页面\n * - IDE 环境: 通过 IPC 通知 IDE 打开登录流程\n */\n login(): Promise<void>;\n\n /**\n * 登出账号\n */\n logout(): Promise<void>;\n\n /**\n * 重新加载窗口(可选,仅 IPC 环境支持)\n * @param params 可选参数,如 locale\n */\n reloadWindow?(params?: { locale?: string }): Promise<void>;\n\n /**\n * 关闭 Agent Manager 面板(可选,仅 IPC 环境支持)\n * 用于 Local 模式下点击 Logo 返回 IDE\n */\n closeAgentManager?(): Promise<void>;\n\n /**\n * 监听事件(可选,用于 IPC 环境)\n * @param event 事件名称\n * @param callback 回调函数\n * @returns 取消订阅函数\n */\n on?(event: string, callback: (data?: unknown) => void): () => void;\n\n /** 获取企业用户用量信息\n * @param enterpriseId 企业 ID\n * @returns Promise<EnterpriseUsage | null> 企业用户用量信息\n */\n getEnterpriseUsage?(enterpriseId: string): Promise<EnterpriseUsage | null>;\n}\n\n// ============================================================================\n// Support Scenes 相关类型\n// ============================================================================\n\n/**\n * 场景中的插件信息\n * 用于描述场景关联的插件\n */\nexport interface ScenePlugin {\n /** 插件唯一标识 */\n id: number;\n /** 插件名称 */\n name: string;\n /** 插件市场名称 */\n marketplaceName: string;\n}\n\n/**\n * 支持的场景信息\n * 用于 Welcome 页面的 QuickActions 快捷操作\n * API 端点: GET /console/as/support/scenes (Web) 或 GET /v2/as/support/scenes (Local)\n */\nexport interface SupportScene {\n /** 场景唯一标识 */\n id: number;\n /** 场景显示名称 */\n name: string;\n /** 场景关联的插件列表 */\n plugins: ScenePlugin[];\n /** 场景对应的 prompt 列表 */\n prompts: string[];\n}\n\n/**\n * 获取支持场景列表响应\n */\nexport interface GetSupportScenesResponse {\n /** 场景列表 */\n scenes: SupportScene[];\n}\n\n// ============================================================================\n// Plugin Management 相关类型\n// ============================================================================\n\n/**\n * 插件作用域\n */\nexport type PluginScope = 'user' | 'project';\n\n/**\n * 插件操作类型\n */\nexport type PluginOperation = 'enable' | 'disable';\n\n/**\n * 批量插件操作项\n */\nexport interface BatchPluginOperationItem {\n /** 插件名称 */\n readonly pluginName: string;\n /** 市场名称 */\n readonly marketplaceName: string;\n /** 作用域 */\n readonly scope: PluginScope;\n /** 操作类型 */\n readonly operation: PluginOperation;\n}\n\n/**\n * 批量插件操作请求\n */\nexport interface BatchPluginOperationRequest {\n /** 操作项列表 */\n readonly items: BatchPluginOperationItem[];\n}\n\n/**\n * 批量插件操作失败项\n */\nexport interface BatchPluginOperationFailedItem extends BatchPluginOperationItem {\n /** 错误信息 */\n readonly error: string;\n}\n\n/**\n * 批量插件操作结果\n */\nexport interface BatchPluginOperationResult {\n /** 是否全部成功 */\n readonly success: boolean;\n /** 成功的插件列表 */\n readonly succeededPlugins: BatchPluginOperationItem[];\n /** 失败的插件列表 */\n readonly failedPlugins: BatchPluginOperationFailedItem[];\n}\n","/**\n * Backend Provider 实现\n *\n * 封装与后端 API 的 HTTP 通信\n */\nimport { accountService } from '../account';\nimport { httpService } from '../http';\nimport { Account, AccountPlan, AccountStatus,AddTaskRequest, AddTaskResponse, BackendProviderConfig, BatchPluginOperationRequest, BatchPluginOperationResult,CommodityCode, EditionDisplayType, EnterpriseUsage, GetFileRequest, GetFileResponse, GetOauthUserRequest, GetOauthUserResponse, GetRepoListRequest, GetRepoListResponse, IBackendProvider, ListTaskConnectorResponse, ListUserConnectorResponse, ModifyTaskConnectorActiveStatusRequest, ModifyTaskConnectorActiveStatusResponse, ModifyTaskConnectorRepoRequest, ModifyTaskConnectorRepoResponse, ModifyUserConnectorActiveStatusRequest, ModifyUserConnectorConnectStatusRequest, ModifyUserConnectorRepoRequest, RevokeAllRequest, SaveOauthTokenRequest,UserResource } from './types';\n\n/**\n * 判断当前是否在 SSO 域名下\n * SSO 域名格式: xxx.sso.copilot.tencent.com 或 xxx.sso.codebuddy.cn\n */\nconst isSSODomain = () => {\n const { hostname } = window.location;\n return hostname.includes('.sso.copilot') ||\n hostname.includes('sso.codebuddy.cn') ||\n hostname.includes('.sso.copilot-staging') ||\n hostname.includes('.staging-sso.codebuddy.cn');\n};\n\n/**\n * 获取登录页面 URL\n * - SSO 域名下需要跳转到对应的预发/生产域名\n * - 非 SSO 域名直接使用当前域名\n */\nconst getLoginUrl = () => {\n const { hostname, protocol } = window.location;\n\n // 如果是 SSO 域名,需要跳转到对应的预发/生产域名登录\n if (isSSODomain()) {\n const isCodebuddy = hostname.includes('codebuddy.cn');\n const isStaging = hostname.includes('staging');\n\n if (isCodebuddy) {\n // codebuddy.cn 域名\n return isStaging\n ? `${protocol}//staging.codebuddy.cn/login`\n : `${protocol}//www.codebuddy.cn/login`;\n } else {\n // copilot.tencent.com 域名\n return isStaging\n ? `${protocol}//staging-copilot.tencent.com/login`\n : `${protocol}//copilot.tencent.com/login`;\n }\n }\n\n // 非 SSO 域名,使用当前域名\n return `${window.location.origin}/login`;\n};\n\n/** 获取当前域名的账号选择页面 URL */\nconst getSelectAccountUrl = () => `${window.location.origin}/login/select`;\n\n/** localStorage 中存储选中账号 ID 的 key */\nexport const SELECTED_ACCOUNT_KEY = 'CODEBUDDY_IDE_SELECTED_ACCOUNT_ID';\n\n/**\n * 套餐代码对应的 i18n key 映射\n */\nconst CommodityCodeText: Record<CommodityCode, string> = {\n [CommodityCode.free]: 'plan.codebuddyFreePlan',\n [CommodityCode.proMon]: 'plan.codebuddyProPlanMonthly',\n [CommodityCode.proMonPlus]: 'plan.codebuddyProPlanMonthly',\n [CommodityCode.gift]: 'plan.codebuddyProPlanTrial',\n [CommodityCode.activity]: 'plan.codebuddyGrowthPlan',\n [CommodityCode.proYear]: 'plan.codebuddyProPlanYearly',\n [CommodityCode.freeMon]: 'plan.codebuddyProPlanDaily',\n [CommodityCode.extra]: 'plan.codebuddyCreditPackage',\n};\n\n/**\n * 获取套餐名称的 i18n key\n */\nconst getPackageName = (packageCode: CommodityCode): string => CommodityCodeText[packageCode] || '';\n\n/**\n * Backend Provider 实现类\n *\n * 职责:\n * - 触发登录/登出流程\n * - 获取 account 后自动同步到 accountService\n * - 使用单例 httpService 进行所有 HTTP 请求\n *\n * 注意:getAgents 和 getModels 方法已废弃并移除,\n * 请使用 IAgentAdapter 中的对应方法\n */\nexport class BackendProvider implements IBackendProvider {\n constructor(config: BackendProviderConfig) {\n // 设置 httpService 配置\n httpService.setBaseURL(config.baseUrl);\n\n if (config.authToken) {\n httpService.setAuthToken(config.authToken);\n }\n\n // 注册 401 回调,执行完整的登出流程\n httpService.onUnauthorized(() => {\n console.log('[BackendProvider] User unauthorized (401/403), triggering logout');\n this.logout().catch(error => {\n console.error('[BackendProvider] Logout failed in 401 handler:', error);\n });\n });\n }\n\n /**\n * 获取当前账号信息\n * API 端点: GET /console/accounts (返回账号列表)\n *\n * 逻辑:\n * 1. 从 localStorage 读取 CODEBUDDY_IDE_SELECTED_ACCOUNT_ID\n * 2. 根据 CODEBUDDY_IDE_SELECTED_ACCOUNT_ID 找到对应账号\n * - personal 类型: 用 uid 匹配\n * - 其他类型: 用 enterpriseId 匹配\n * 3. 如果没有选中的账号,跳转到账号选择页面\n * 4. 获取套餐信息并合并到账号中\n * 5. 同步到 accountService\n */\n async getAccount(): Promise<Account | null> {\n try {\n const result = await httpService.get<{ data: { accounts: Account[] } }>(\n '/console/accounts'\n );\n\n const accounts: Account[] = result.data?.accounts || [];\n if (!accounts || accounts.length === 0) {\n accountService.setAccount(null);\n return null;\n }\n\n // 从 localStorage 读取选中的账号 ID\n const selectedAccountId = localStorage.getItem(SELECTED_ACCOUNT_KEY);\n let selectedAccount: Account | undefined;\n\n if (selectedAccountId) {\n // 查找选中的账号\n selectedAccount = accounts.find(account => {\n // personal 类型用 uid 匹配,其他类型用 enterpriseId 匹配\n if (account.type === 'personal') {\n return account.uid === selectedAccountId;\n }\n return account.enterpriseId === selectedAccountId;\n });\n\n if (selectedAccount) {\n const account = await this.enrichAccountWithUsage(selectedAccount);\n accountService.setAccount(account);\n return account;\n }\n }\n\n // 如果只有一个账号,自动选中它\n if (accounts.length === 1) {\n selectedAccount = accounts[0];\n // 保存选中的账号 ID 到 localStorage\n const accountId = selectedAccount.type === 'personal'\n ? selectedAccount.uid\n : selectedAccount.enterpriseId;\n if (accountId) {\n localStorage.setItem(SELECTED_ACCOUNT_KEY, accountId);\n }\n const account = await this.enrichAccountWithUsage(selectedAccount);\n console.log('account (auto-selected)', account);\n accountService.setAccount(account);\n return account;\n }\n\n // 多个账号但没有选中的,跳转到账号选择页面\n const redirectUrl = encodeURIComponent(window.location.href);\n window.location.href = `${getSelectAccountUrl()}?platform=website&state=0&redirect_uri=${redirectUrl}`;\n accountService.setAccount(null);\n return null;\n } catch (error) {\n console.error('[BackendProvider] getAccount failed:', error);\n accountService.setAccount(null);\n return null;\n }\n }\n\n /**\n * 获取用户连接器列表\n * API 端点: GET /console/as/connector/user/\n */\n async getUserConnector(): Promise<ListUserConnectorResponse> {\n const result = await httpService.get<any>('/console/as/connector/user/');\n\n if (result.code === 0) {\n const { connectors } = result.data;\n return {\n connectors: connectors.map((connector: any) => ({\n ...connector,\n connectStatus: connector.connect_status,\n activeStatus: connector.active_status,\n displayName: connector.display_name,\n oauthClientId: connector.oauth_client_id,\n oauthRedirectUrl: connector.oauth_redirect_url,\n }))\n };\n }\n\n throw result;\n }\n\n /**\n * 修改用户连接器连接状态\n * API 端点: PATCH /console/as/connector/user/:name/connect_status\n */\n async modifyUserConnectorConnectStatus(request: ModifyUserConnectorConnectStatusRequest): Promise<void> {\n // 构造请求体,将驼峰命名转换为下划线命名\n const body: Record<string, any> = {\n name: request.name,\n connect_status: request.connectStatus,\n };\n\n if (request.activeStatus !== undefined) {\n body.active_status = request.activeStatus;\n }\n\n if (request.repos !== undefined) {\n body.repos = request.repos;\n }\n\n const result = await httpService.patch<any>(\n `/console/as/connector/user/${request.name}/connect_status`,\n body\n );\n\n if (result.code === 0) {\n return;\n }\n\n throw result;\n }\n\n /**\n * 修改用户连接器仓库\n * API 端点: PATCH /console/as/connector/user/:name/repo/\n */\n async modifyUserConnectorRepo(request: ModifyUserConnectorRepoRequest): Promise<void> {\n const body: Record<string, any> = {\n name: request.name,\n };\n\n if (request.repo !== undefined) {\n body.repo = request.repo;\n }\n\n const result = await httpService.patch<any>(\n `/console/as/connector/user/${request.name}/repo`,\n body\n );\n\n if (result.code === 0) {\n return;\n }\n\n throw result;\n }\n\n /**\n * 修改用户连接器激活状态\n * API 端点: PATCH /console/as/connector/user/:name/active_status\n */\n async modifyUserConnectorActiveStatus(request: ModifyUserConnectorActiveStatusRequest): Promise<void> {\n const body = {\n name: request.name,\n active_status: request.activeStatus,\n };\n\n const result = await httpService.patch<any>(\n `/console/as/connector/user/${request.name}/active_status`,\n body\n );\n\n if (result.code === 0) {\n return;\n }\n\n throw result;\n }\n\n /**\n * 删除用户连接器\n * API 端点: DELETE /console/as/connector/user/:name/\n */\n async deleteUserConnector(name: 'github' | 'gongfeng'): Promise<void> {\n const result = await httpService.delete<any>(\n `/console/as/connector/user/${name}/`\n );\n\n if (result.code === 0) {\n return;\n }\n\n throw result;\n }\n\n /**\n * 添加任务\n * API 端点: POST /console/as/connector/task/\n */\n async addConnectorTask(request: AddTaskRequest): Promise<AddTaskResponse> {\n // 构造请求体,将驼峰命名转换为下划线命名\n const body: Record<string, any> = {\n task_id: request.taskId,\n };\n\n if (request.connectors !== undefined) {\n body.connectors = request.connectors.map(c => ({\n name: c.name,\n repos: c.repos,\n active_status: c.activeStatus,\n }));\n }\n\n const result = await httpService.post<any>(\n '/console/as/connector/task/',\n body\n );\n\n if (result.code === 0) {\n return {\n taskId: result.data?.task_id || request.taskId,\n };\n }\n\n throw result;\n }\n\n /**\n * 获取任务连接器列表\n * API 端点: GET /console/as/connector/task/:taskid\n */\n async getTaskConnector(taskId: string): Promise<ListTaskConnectorResponse> {\n const result = await httpService.get<any>(\n `/console/as/connector/task/${taskId}`\n );\n\n if (result.code === 0) {\n const { connectors } = result.data;\n return {\n connectors: (connectors || []).map((connector: any) => ({\n ...connector,\n activeStatus: connector.active_status,\n connectStatus: connector.connect_status,\n displayName: connector.display_name,\n oauthClientId: connector.oauth_client_id,\n oauthRedirectUrl: connector.oauth_redirect_url,\n }))\n };\n }\n\n throw result;\n }\n\n /**\n * 修改任务连接器激活状态\n * API 端点: PATCH /console/as/connector/task/:taskid/active_status\n */\n async modifyTaskConnectorActiveStatus(request: ModifyTaskConnectorActiveStatusRequest): Promise<ModifyTaskConnectorActiveStatusResponse> {\n const body = {\n task_id: request.taskId,\n name: request.name,\n active_status: request.activeStatus,\n };\n\n const result = await httpService.patch<any>(\n `/console/as/connector/task/${request.taskId}/active_status`,\n body\n );\n\n if (result.code === 0) {\n return {\n taskId: result.data?.task_id || request.taskId,\n };\n }\n\n throw result;\n }\n\n /**\n * 修改任务连接器仓库\n * API 端点: PATCH /console/as/connector/task/:taskid/repo\n */\n async modifyTaskConnectorRepo(request: ModifyTaskConnectorRepoRequest): Promise<ModifyTaskConnectorRepoResponse> {\n const body = {\n task_id: request.taskId,\n name: request.name,\n repo: request.repo,\n };\n\n const result = await httpService.patch<any>(\n `/console/as/connector/task/${request.taskId}/repo`,\n body\n );\n\n if (result.code === 0) {\n return {\n taskId: result.data?.task_id || request.taskId,\n };\n }\n\n throw result;\n }\n\n /**\n * 根据账号类型获取用量信息并合并到账号中\n * - 企业用户:调用 getEnterpriseUsage 获取月度限额\n * - 个人用户:调用 getCurrentPlan 获取套餐信息\n */\n private async enrichAccountWithUsage(selectedAccount: Account): Promise<Account> {\n const isEnterpriseUser = !!(selectedAccount.enterpriseId && selectedAccount.enterpriseId !== '');\n\n try {\n if (isEnterpriseUser) {\n // 企业用户:获取企业用量\n const enterpriseUsage = await this.getEnterpriseUsage(selectedAccount.enterpriseId!);\n const editionType = this.getEditionDisplayType(selectedAccount.type, false);\n\n if (enterpriseUsage) {\n const usageLeft = (enterpriseUsage.limitNum - enterpriseUsage.credit).toString();\n const usageTotal = enterpriseUsage.limitNum.toString();\n return {\n ...selectedAccount,\n editionType,\n usageLeft,\n usageTotal,\n refreshAt: enterpriseUsage.cycleResetTime ? new Date(enterpriseUsage.cycleResetTime).getTime() : undefined,\n };\n }\n return { ...selectedAccount, editionType };\n } else {\n // 个人用户:获取套餐信息\n const plan = await this.getCurrentPlan();\n const editionType = this.getEditionDisplayType(selectedAccount.type, plan.isPro);\n console.log('account', { ...selectedAccount, ...plan, editionType });\n return { ...selectedAccount, ...plan, editionType };\n }\n } catch (error) {\n console.error('[BackendProvider] enrichAccountWithUsage failed:', error);\n return { ...selectedAccount };\n }\n }\n\n /**\n * 获取当前套餐信息\n * 从计量计费接口获取用户的套餐信息\n * API: POST /billing/meter/get-user-resource\n */\n private async getCurrentPlan(): Promise<AccountPlan> {\n // 默认套餐信息\n const defaultPlan: AccountPlan = {\n isPro: false,\n expireAt: 0,\n renewFlag: 0,\n PackageCode: undefined,\n name: '',\n };\n\n try {\n // 构造请求参数\n const now = new Date();\n const futureDate = new Date(now.getTime() + 101 * 365 * 24 * 60 * 60 * 1000);\n const formatDate = (d: Date) => {\n const pad = (n: number) => n.toString().padStart(2, '0');\n return `${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())} ${pad(d.getHours())}:${pad(d.getMinutes())}:${pad(d.getSeconds())}`;\n };\n\n const body = {\n PageNumber: 1,\n // PageSize、用户购买包的数量、暂定 100 上限。\n PageSize: 100,\n ProductCode: 'p_tcaca',\n Status: [AccountStatus.valid, AccountStatus.usedUp], // 0-有效, 3-已用完\n PackageEndTimeRangeBegin: formatDate(now),\n PackageEndTimeRangeEnd: formatDate(futureDate),\n };\n\n const result = await httpService.post<any>(\n '/billing/meter/get-user-resource',\n body\n );\n\n // 响应格式: { Response: { Data: { Accounts: [...] } } }\n const resources = (result?.data?.Response?.Data?.Accounts || []) as UserResource[];\n\n if (!resources || resources.length === 0) {\n return defaultPlan;\n }\n\n // 解析时间字符串或时间戳为毫秒时间戳\n const parseTime = (time: string | number | undefined): number => {\n if (!time) { return 0; }\n return new Date(time).getTime();\n };\n\n // 每日刷新的套餐类型\n const dailyCredits = [CommodityCode.free, CommodityCode.freeMon];\n\n // 转换所有套餐为 PlanResource 列表\n const planResources = resources.map(r => {\n const isDaily = dailyCredits.includes(r.PackageCode);\n const endTime = isDaily ? r.CycleEndTime : r.DeductionEndTime;\n\n return {\n id: r.ResourceId,\n name: isDaily ? 'plan.addonCredits' : getPackageName(r.PackageCode),\n packageCode: r.PackageCode,\n isDaily,\n total: Number(r.CycleCapacitySizePrecise) || 0,\n used: Math.max(0, Number(r.CycleCapacitySizePrecise) - Number(r.CycleCapacityRemainPrecise)) || 0,\n left: Number(r.CycleCapacityRemainPrecise) || 0,\n expireAt: parseTime(endTime),\n refreshAt: isDaily ? undefined : parseTime(r.CycleEndTime),\n };\n }).sort((a, b) => {\n const getPriority = (code: CommodityCode) => {\n // 付费套餐优先\n if ([CommodityCode.proMon, CommodityCode.proMonPlus, CommodityCode.proYear, CommodityCode.extra].includes(code)) {\n return 1;\n }\n // 赠送/试用套餐次之\n if ([CommodityCode.gift, CommodityCode.activity].includes(code)) {\n return 2;\n }\n // 免费版最后\n if ([CommodityCode.free, CommodityCode.freeMon].includes(code)) {\n return 3;\n }\n return 4;\n };\n return getPriority(a.packageCode) - getPriority(b.packageCode);\n });\n\n // 查找 Pro 套餐(proYear 或 proMon)\n const proPlan = resources.find((r: any) =>\n r.PackageCode === CommodityCode.proYear || r.PackageCode === CommodityCode.proMon || r.PackageCode === CommodityCode.proMonPlus\n );\n\n // 查找试用套餐\n const trialPlan = resources.find((r: any) => r.PackageCode === CommodityCode.gift || r.PackageCode === CommodityCode.freeMon);\n\n const activePlan = proPlan || trialPlan;\n\n // 计算所有套餐的总量、已用和剩余\n const totalUsageLeft = planResources.reduce((sum, r) => sum + r.left, 0);\n const totalUsageTotal = planResources.reduce((sum, r) => sum + r.total, 0);\n const totalUsageUsed = planResources.reduce((sum, r) => sum + r.used, 0);\n\n if (activePlan) {\n return {\n isPro: !!proPlan,\n // 试用套餐\n isTria: trialPlan ? [AccountStatus.valid, AccountStatus.usedUp].includes(trialPlan.Status) : false,\n expireAt: parseTime(activePlan.DeductionEndTime || activePlan.ExpiredTime || activePlan.CycleEndTime),\n refreshAt: parseTime(activePlan.CycleEndTime),\n renewFlag: Number(activePlan.AutoRenewFlag) === 1 ? 1 : 0,\n PackageCode: activePlan.PackageCode,\n name: getPackageName(activePlan.PackageCode),\n usageTotal: String(totalUsageTotal),\n usageUsed: String(totalUsageUsed),\n usageLeft: String(totalUsageLeft),\n resources: planResources,\n };\n }\n\n return {\n ...defaultPlan,\n usageTotal: String(totalUsageTotal),\n usageUsed: String(totalUsageUsed),\n usageLeft: String(totalUsageLeft),\n resources: planResources\n };\n } catch (error) {\n console.error('[BackendProvider] getCurrentPlan error:', error);\n return defaultPlan;\n }\n }\n\n /**\n * 通过回调code,换token\n * @param request\n * @returns\n */\n async saveOauthToken(request: SaveOauthTokenRequest) {\n const body = {\n authorization_code: request.authorizationCode,\n };\n\n const result = await httpService.post<any>(\n `/console/as/connector/oauth/${request.name}/connect`,\n body\n );\n\n if (result.code === 0) {\n return;\n }\n\n throw result;\n }\n\n /**\n * 获取OAuth连接器的仓库列表\n */\n async getRepoList(request: GetRepoListRequest): Promise<GetRepoListResponse> {\n const result = await httpService.get<any>(\n `/console/as/connector/oauth/${request.name}/repos`\n );\n\n if (result.code === 0) {\n return result.data;\n }\n\n throw result;\n }\n\n /**\n * 撤销OAuth连接器的所有连接\n */\n async revokeAll(request: RevokeAllRequest): Promise<void> {\n const installationIds = request?.installationIds ?? [];\n\n const body: { name: string; installation_ids?: string[] } = {\n name: request.name,\n };\n\n if (installationIds) {\n body.installation_ids = installationIds;\n }\n\n const result = await httpService.post<any>(\n `/console/as/connector/oauth/${request.name}/revokeall`,\n body\n );\n\n if (result.code === 0) {\n return;\n }\n\n throw result;\n }\n\n /**\n * 获取 OAuth 用户信息\n * API 端点: GET /console/as/connector/oauth/:name/oauthuser\n */\n async getOauthUser(request: GetOauthUserRequest): Promise<GetOauthUserResponse> {\n const result = await httpService.get<any>(\n `/console/as/connector/oauth/${request.name}/oauthuser`\n );\n\n if (result.code === 0) {\n const data = result.data || {};\n return {\n user: {\n avatarUrl: data.user?.avatar_url || '',\n name: data.user?.name || '',\n }\n };\n }\n\n throw result;\n }\n\n /**\n * 获取文件信息\n * API 端点: GET /console/as/connector/oauth/:name/file\n */\n async getFile(request: GetFileRequest): Promise<GetFileResponse> {\n const body: { url: string; } = {\n url: request.url,\n };\n\n const result = await httpService.post<any>(\n `/console/as/connector/oauth/${request.name}/file`,\n body\n );\n\n if (result.code === 0) {\n const data = result.data || {};\n const figmaFileInfo = data.figma_file_info;\n return {\n figmaFileInfo: figmaFileInfo ? {\n name: figmaFileInfo.name || '',\n role: figmaFileInfo.role || '',\n lastModified: figmaFileInfo.last_modified || '',\n editorType: figmaFileInfo.editor_type || '',\n thumbnailUrl: figmaFileInfo.thumbnail_url || '',\n version: figmaFileInfo.version || '',\n mainFileKey: figmaFileInfo.main_file_key || '',\n } : null\n };\n }\n\n throw result;\n }\n\n /**\n * 根据账号类型和 Pro 状态计算版本展示类型\n * - personal + isPro = 'pro'\n * - personal + !isPro = 'free'\n * - ultimate = 'ultimate' (旗舰版/团队版)\n * - exclusive = 'exclusive' (专享版/企业版)\n */\n private getEditionDisplayType(type: string, isPro: boolean): EditionDisplayType {\n if (type === 'personal') {\n return isPro ? 'pro' : 'free';\n }\n if (type === 'ultimate') {\n return 'ultimate';\n }\n if (type === 'exclusive') {\n return 'exclusive';\n }\n // 默认返回 free\n return 'free';\n }\n\n /**\n * 触发登录流程\n * Web 环境: 跳转到登录页面\n */\n async login(): Promise<void> {\n // 获取当前页面 URL 作为回调地址\n // todo 支持弹窗和跳转、弹窗优化点 别让主页面 reload\n const redirectUrl = encodeURIComponent(window.location.href);\n window.location.href = `${getLoginUrl()}?platform=website&state=0&redirect_uri=${redirectUrl}`;\n }\n\n /**\n * 登出账号\n * Web 环境: 通过 iframe 访问登出 URL 清除 cookie\n */\n async logout(): Promise<void> {\n const baseURL = httpService.getAxiosInstance().defaults.baseURL;\n const url = `${baseURL}/console/logout`;\n\n try {\n await new Promise<void>(resolve => {\n const iframe = document.createElement('iframe');\n iframe.style.cssText = 'position:fixed;top:-9999px;left:-9999px;width:1px;height:1px;border:none;';\n iframe.src = url;\n\n const timeout = setTimeout(() => {\n cleanup();\n resolve();\n }, 5000);\n\n const cleanup = () => {\n clearTimeout(timeout);\n if (iframe.parentNode) {\n iframe.parentNode.removeChild(iframe);\n }\n };\n\n iframe.onerror = () => {\n cleanup();\n resolve();\n };\n\n document.body.appendChild(iframe);\n });\n } catch (error) {\n console.error('[BackendProvider] logout failed:', error);\n }\n\n // 清除 localStorage 中的选中账号\n localStorage.removeItem(SELECTED_ACCOUNT_KEY);\n\n // 清空 accountService,会触发订阅者更新,UI 会自动切换到未登录状态\n accountService.clearAccount();\n }\n\n /**\n * 批量切换插件状态\n * Web 环境不支持此功能\n */\n async batchTogglePlugins(request: BatchPluginOperationRequest): Promise<BatchPluginOperationResult> {\n console.warn('[BackendProvider] batchTogglePlugins is not supported in web environment');\n return {\n success: false,\n succeededPlugins: [],\n failedPlugins: request.items.map(item => ({\n ...item,\n error: 'Plugin batch toggle is not supported in web environment'\n }))\n };\n }\n\n /**\n * 获取企业用户用量信息\n * API: POST /billing/meter/get-enterprise-user-usage\n */\n async getEnterpriseUsage(enterpriseId: string): Promise<EnterpriseUsage | null> {\n try {\n const result = await httpService.post<any>(\n '/billing/meter/get-enterprise-user-usage',\n {},\n {\n headers: {\n 'X-Enterprise-Id': enterpriseId,\n },\n }\n );\n\n // 响应格式: { code, data, msg } 或直接是 EnterpriseUsage\n const usageData = result?.data?.data || result?.data || result;\n\n if (usageData && typeof usageData.limitNum === 'number') {\n return usageData as EnterpriseUsage;\n }\n\n return null;\n } catch (error) {\n console.error('[BackendProvider] getEnterpriseUsage error:', error);\n return null;\n }\n }\n}\n\n/**\n * 创建 BackendProvider 实例\n */\nexport function createBackendProvider(config: BackendProviderConfig): BackendProvider {\n return new BackendProvider(config);\n}\n","/**\n * IPC Backend Provider 实现\n *\n * 通过 IWidgetChannel 与后端通信\n * 使用统一的消息格式: { type: 'backend', requestId, params: { type, params } }\n */\n\nimport { accountService } from '../account';\nimport { IWidgetChannel } from '../common';\nimport type { Account, AddTaskRequest, AddTaskResponse, BatchPluginOperationRequest, BatchPluginOperationResult, GetFileRequest, GetFileResponse, GetOauthUserRequest, GetOauthUserResponse, GetRepoListRequest, GetRepoListResponse, IBackendProvider, ListTaskConnectorResponse, ListUserConnectorResponse, ModifyTaskConnectorActiveStatusRequest, ModifyTaskConnectorActiveStatusResponse, ModifyTaskConnectorRepoRequest, ModifyTaskConnectorRepoResponse, ModifyUserConnectorActiveStatusRequest, ModifyUserConnectorConnectStatusRequest, ModifyUserConnectorRepoRequest, RevokeAllRequest, SaveOauthTokenRequest, SupportScene } from './types';\n\n/**\n * IPC Backend Provider 配置\n */\nexport interface IPCBackendProviderConfig {\n /** Widget Channel 接口 */\n channel: IWidgetChannel;\n /** 是否启用调试日志 */\n debug?: boolean;\n /** 请求超时时间(毫秒,默认 30000) */\n timeoutMs?: number;\n}\n\n/**\n * Backend 请求类型常量\n */\nconst BACKEND_REQUEST_TYPES = {\n LOGIN: 'backend:login',\n LOGOUT: 'backend:logout',\n GET_ACCOUNT: 'backend:get-account',\n GET_USER_CONNECTOR: 'backend:get-user-connector',\n MODIFY_USER_CONNECTOR_CONNECT_STATUS: 'backend:modify-user-connector-connect-status',\n MODIFY_USER_CONNECTOR_REPO: 'backend:modify-user-connector-repo',\n MODIFY_USER_CONNECTOR_ACTIVE_STATUS: 'backend:modify-user-connector-active-status',\n DELETE_USER_CONNECTOR: 'backend:delete-user-connector',\n ADD_CONNECTOR_TASK: 'backend:add-connector-task',\n GET_TASK_CONNECTOR: 'backend:get-task-connector',\n MODIFY_TASK_CONNECTOR_ACTIVE_STATUS: 'backend:modify-task-connector-active-status',\n MODIFY_TASK_CONNECTOR_REPO: 'backend:modify-task-connector-repo',\n GET_OAUTH_USER: 'backend:get-oauth-user',\n SAVE_OAUTH_TOKEN: 'backend:save-oauth-token',\n GET_REPO_LIST: 'backend:get-repo-list',\n REVOKE_ALL: 'backend:revoke-all',\n GET_FILE: 'backend:get-file',\n RELOAD_WINDOW: 'backend:reload-window',\n CLOSE_AGENT_MANAGER: 'backend:close-agent-manager',\n BATCH_TOGGLE_PLUGINS: 'backend:batch-toggle-plugins',\n GET_SUPPORT_SCENES: 'backend:get-support-scenes',\n} as const;\n\n/**\n * 生成唯一请求 ID\n */\nfunction generateRequestId(): string {\n return `req-${Date.now()}-${Math.random().toString(36).slice(2)}`;\n}\n\n/**\n * IPC Backend Provider 实现类\n *\n * 通过 IWidgetChannel 与后端通信\n */\nexport class IPCBackendProvider implements IBackendProvider {\n private readonly channel: IWidgetChannel;\n private readonly debug: boolean;\n private readonly timeoutMs: number;\n\n constructor(config: IPCBackendProviderConfig) {\n this.channel = config.channel;\n this.debug = config.debug ?? false;\n this.timeoutMs = config.timeoutMs ?? 30000;\n\n this.log('Initialized with IWidgetChannel');\n }\n\n /**\n * 发送统一格式的后端请求\n * @param requestType 请求类型\n * @param params 请求参数\n * @returns 响应数据\n */\n private async sendBackendRequest<T>(requestType: string, params?: unknown): Promise<T> {\n const message = {\n type: 'backend',\n requestId: generateRequestId(),\n params: {\n type: requestType,\n params: params,\n },\n };\n\n this.log('Sending backend request:', message);\n\n const response = await this.channel.callMethod('__backend__', message, this.timeoutMs);\n\n this.log('Received response:', response);\n\n // 检查响应中是否有错误\n if (response?.error) {\n throw new Error(response.error);\n }\n\n // 从响应的 data 字段中提取实际数据\n return (response?.data !== undefined ? response.data : response) as T;\n }\n\n /**\n * 获取当前账号信息\n * IDE 环境: 通过 IPC 获取账号信息,并同步到 accountService\n */\n async getAccount(): Promise<Account | null> {\n this.log('Getting account via IPC');\n\n try {\n const account = await this.sendBackendRequest<Account | null>(\n BACKEND_REQUEST_TYPES.GET_ACCOUNT\n );\n // 同步到 accountService\n accountService.setAccount(account);\n return account;\n } catch (error) {\n this.log('Get account failed:', error);\n accountService.setAccount(null);\n return null;\n }\n }\n\n /**\n * 获取用户连接器列表\n * IDE 环境: 通过 IPC 获取用户连接器列表\n */\n async getUserConnector(): Promise<ListUserConnectorResponse> {\n this.log('Getting user connector via IPC');\n\n try {\n return await this.sendBackendRequest<ListUserConnectorResponse>(\n BACKEND_REQUEST_TYPES.GET_USER_CONNECTOR\n );\n } catch (error) {\n this.log('Get user connector failed:', error);\n throw error;\n }\n }\n\n /**\n * 修改用户连接器连接状态\n * IDE 环境: 通过 IPC 修改用户连接器连接状态\n */\n async modifyUserConnectorConnectStatus(request: ModifyUserConnectorConnectStatusRequest): Promise<void> {\n this.log('Modifying user connector connect status via IPC:', request);\n\n try {\n await this.sendBackendRequest<void>(\n BACKEND_REQUEST_TYPES.MODIFY_USER_CONNECTOR_CONNECT_STATUS,\n request\n );\n } catch (error) {\n this.log('Modify user connector connect status failed:', error);\n throw error;\n }\n }\n\n /**\n * 修改用户连接器仓库\n * IDE 环境: 通过 IPC 修改用户连接器仓库\n */\n async modifyUserConnectorRepo(request: ModifyUserConnectorRepoRequest): Promise<void> {\n this.log('Modifying user connector repo via IPC:', request);\n\n try {\n await this.sendBackendRequest<void>(\n BACKEND_REQUEST_TYPES.MODIFY_USER_CONNECTOR_REPO,\n request\n );\n } catch (error) {\n this.log('Modify user connector repo failed:', error);\n throw error;\n }\n }\n\n /**\n * 修改用户连接器激活状态\n * IDE 环境: 通过 IPC 修改用户连接器激活状态\n */\n async modifyUserConnectorActiveStatus(request: ModifyUserConnectorActiveStatusRequest): Promise<void> {\n this.log('Modifying user connector active status via IPC:', request);\n\n try {\n await this.sendBackendRequest<void>(\n BACKEND_REQUEST_TYPES.MODIFY_USER_CONNECTOR_ACTIVE_STATUS,\n request\n );\n } catch (error) {\n this.log('Modify user connector active status failed:', error);\n throw error;\n }\n }\n\n /**\n * 删除用户连接器\n * IDE 环境: 通过 IPC 删除用户连接器\n */\n async deleteUserConnector(name: 'github' | 'gongfeng'): Promise<void> {\n this.log('Deleting user connector via IPC:', name);\n\n try {\n await this.sendBackendRequest<void>(\n BACKEND_REQUEST_TYPES.DELETE_USER_CONNECTOR,\n { name }\n );\n } catch (error) {\n this.log('Delete user connector failed:', error);\n throw error;\n }\n }\n\n /**\n * 添加任务\n * IDE 环境: 通过 IPC 添加任务\n */\n async addConnectorTask(request: AddTaskRequest): Promise<AddTaskResponse> {\n this.log('Adding connector task via IPC:', request);\n\n try {\n return await this.sendBackendRequest<AddTaskResponse>(\n BACKEND_REQUEST_TYPES.ADD_CONNECTOR_TASK,\n request\n );\n } catch (error) {\n this.log('Add task failed:', error);\n throw error;\n }\n }\n\n /**\n * 获取任务连接器列表\n * IDE 环境: 通过 IPC 获取任务连接器列表\n */\n async getTaskConnector(taskId: string): Promise<ListTaskConnectorResponse> {\n this.log('Getting task connector via IPC:', taskId);\n\n try {\n return await this.sendBackendRequest<ListTaskConnectorResponse>(\n BACKEND_REQUEST_TYPES.GET_TASK_CONNECTOR,\n { taskId }\n );\n } catch (error) {\n this.log('Get task connector failed:', error);\n throw error;\n }\n }\n\n /**\n * 修改任务连接器激活状态\n * IDE 环境: 通过 IPC 修改任务连接器激活状态\n */\n async modifyTaskConnectorActiveStatus(request: ModifyTaskConnectorActiveStatusRequest): Promise<ModifyTaskConnectorActiveStatusResponse> {\n this.log('Modifying task connector active status via IPC:', request);\n\n try {\n return await this.sendBackendRequest<ModifyTaskConnectorActiveStatusResponse>(\n BACKEND_REQUEST_TYPES.MODIFY_TASK_CONNECTOR_ACTIVE_STATUS,\n request\n );\n } catch (error) {\n this.log('Modify task connector active status failed:', error);\n throw error;\n }\n }\n\n /**\n * 修改任务连接器仓库\n * IDE 环境: 通过 IPC 修改任务连接器仓库\n */\n async modifyTaskConnectorRepo(request: ModifyTaskConnectorRepoRequest): Promise<ModifyTaskConnectorRepoResponse> {\n this.log('Modifying task connector repo via IPC:', request);\n\n try {\n return await this.sendBackendRequest<ModifyTaskConnectorRepoResponse>(\n BACKEND_REQUEST_TYPES.MODIFY_TASK_CONNECTOR_REPO,\n request\n );\n } catch (error) {\n this.log('Modify task connector repo failed:', error);\n throw error;\n }\n }\n\n /**\n * 获取 OAuth 用户信息\n * IDE 环境: 通过 IPC 获取 OAuth 用户信息\n */\n async getOauthUser(request: GetOauthUserRequest): Promise<GetOauthUserResponse> {\n this.log('Getting OAuth user via IPC:', request);\n\n try {\n return await this.sendBackendRequest<GetOauthUserResponse>(\n BACKEND_REQUEST_TYPES.GET_OAUTH_USER,\n request\n );\n } catch (error) {\n this.log('Get OAuth user failed:', error);\n throw error;\n }\n }\n\n /**\n * 通过回调code,换token\n * IDE 环境: 通过 IPC 保存 OAuth Token\n */\n async saveOauthToken(request: SaveOauthTokenRequest): Promise<void> {\n this.log('Saving OAuth token via IPC:', request);\n\n try {\n await this.sendBackendRequest<void>(\n BACKEND_REQUEST_TYPES.SAVE_OAUTH_TOKEN,\n request\n );\n } catch (error) {\n this.log('Save OAuth token failed:', error);\n throw error;\n }\n }\n\n /**\n * 获取OAuth连接器的仓库列表\n * IDE 环境: 通过 IPC 获取仓库列表\n */\n async getRepoList(request: GetRepoListRequest): Promise<GetRepoListResponse> {\n this.log('Getting repo list via IPC:', request);\n\n try {\n return await this.sendBackendRequest<GetRepoListResponse>(\n BACKEND_REQUEST_TYPES.GET_REPO_LIST,\n request\n );\n } catch (error) {\n this.log('Get repo list failed:', error);\n throw error;\n }\n }\n\n /**\n * 撤销OAuth连接器的所有连接\n * IDE 环境: 通过 IPC 撤销所有连接\n */\n async revokeAll(request: RevokeAllRequest): Promise<void> {\n this.log('Revoking all connections via IPC:', request);\n\n try {\n await this.sendBackendRequest<void>(\n BACKEND_REQUEST_TYPES.REVOKE_ALL,\n request\n );\n } catch (error) {\n this.log('Revoke all connections failed:', error);\n throw error;\n }\n }\n\n /**\n * 获取文件信息\n * IDE 环境: 通过 IPC 获取文件信息\n */\n async getFile(request: GetFileRequest): Promise<GetFileResponse> {\n this.log('Getting file via IPC:', request);\n\n try {\n return await this.sendBackendRequest<GetFileResponse>(\n BACKEND_REQUEST_TYPES.GET_FILE,\n request\n );\n } catch (error) {\n this.log('Get file failed:', error);\n throw error;\n }\n }\n\n /**\n * 触发登录流程\n * IDE 环境: 通过 IPC 通知 IDE 打开登录流程\n */\n async login(): Promise<void> {\n this.log('Triggering login via IPC');\n\n try {\n await this.sendBackendRequest<void>(BACKEND_REQUEST_TYPES.LOGIN);\n } catch (error) {\n this.log('Login request failed:', error);\n throw error;\n }\n }\n\n /**\n * 登出账号\n * IDE 环境: 通过 IPC 通知 IDE 登出\n */\n async logout(): Promise<void> {\n this.log('Triggering logout via IPC');\n\n try {\n await this.sendBackendRequest<void>(BACKEND_REQUEST_TYPES.LOGOUT);\n // 清空 accountService\n accountService.clearAccount();\n } catch (error) {\n this.log('Logout request failed:', error);\n throw error;\n }\n }\n\n /**\n * 重新加载窗口\n * IDE 环境: 通过 IPC 通知 IDE 重新加载窗口(用于应用语言设置等)\n * @param params 可选参数,如 locale\n */\n async reloadWindow(params?: { locale?: string }): Promise<void> {\n this.log('Triggering reload window via IPC', params);\n\n try {\n await this.sendBackendRequest<void>(BACKEND_REQUEST_TYPES.RELOAD_WINDOW, params);\n } catch (error) {\n this.log('Reload window request failed:', error);\n throw error;\n }\n }\n\n /**\n * 关闭 Agent Manager 面板\n * IDE 环境: 通过 IPC 通知 IDE 关闭 Agent Manager(用于返回 IDE)\n */\n async closeAgentManager(): Promise<void> {\n this.log('Triggering close agent manager via IPC');\n\n try {\n await this.sendBackendRequest<void>(BACKEND_REQUEST_TYPES.CLOSE_AGENT_MANAGER);\n } catch (error) {\n this.log('Close agent manager request failed:', error);\n throw error;\n }\n }\n\n /**\n * 批量切换插件状态\n * IDE 环境: 通过 IPC 调用 Extension Host 的 PluginService\n */\n async batchTogglePlugins(request: BatchPluginOperationRequest): Promise<BatchPluginOperationResult> {\n this.log('Batch toggling plugins via IPC:', request);\n\n try {\n return await this.sendBackendRequest<BatchPluginOperationResult>(\n BACKEND_REQUEST_TYPES.BATCH_TOGGLE_PLUGINS,\n request\n );\n } catch (error) {\n this.log('Batch toggle plugins failed:', error);\n throw error;\n }\n }\n\n /**\n * 获取支持的场景列表\n * IDE 环境: 通过 IPC 调用后端 API\n * 用于 Welcome 页面的 QuickActions 快捷操作\n *\n * 调用链:\n * 1. agent-ui: IPCBackendProvider.getSupportScenes()\n * 2. Extension Host: BackendBridgeService.handleGetSupportScenes()\n * 3. Backend API: GET /v2/as/support/scenes\n * 4. 返回 SupportScene[] 数据给 agent-ui\n */\n async getSupportScenes(): Promise<SupportScene[]> {\n this.log('Getting support scenes via IPC');\n\n try {\n const result = await this.sendBackendRequest<{ scenes: SupportScene[] }>(\n BACKEND_REQUEST_TYPES.GET_SUPPORT_SCENES,\n {}\n );\n return result?.scenes || [];\n } catch (error) {\n this.log('Get support scenes failed:', error);\n return [];\n }\n }\n\n /**\n * 调试日志\n */\n private log(...args: unknown[]): void {\n if (this.debug) {\n console.log('[IPCBackendProvider]', ...args);\n }\n }\n}\n\n/**\n * 创建 IPCBackendProvider 实例\n */\nexport function createIPCBackendProvider(config: IPCBackendProviderConfig): IPCBackendProvider {\n return new IPCBackendProvider(config);\n}\n"],"x_google_ignoreList":[13,14,15,16,17,18,19,20,21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40,41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60,61],"mappings":"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"}
|