@dev-anywhere/proxy 0.0.6 → 0.1.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +23 -14
- package/dist/chunk-CDKXSDAV.js +36 -0
- package/dist/chunk-CDKXSDAV.js.map +1 -0
- package/dist/{chunk-VHCL7NVJ.js → chunk-DFLQ3TFT.js} +152 -40
- package/dist/chunk-DFLQ3TFT.js.map +1 -0
- package/dist/{chunk-2XDZRLST.js → chunk-QXOARRC2.js} +2 -2
- package/dist/{chunk-XRTTHTM2.js → chunk-TG7JPHE5.js} +2 -2
- package/dist/index.js +46 -18
- package/dist/index.js.map +1 -1
- package/dist/serve.js +273 -135
- package/dist/serve.js.map +1 -1
- package/dist/session-worker.js +2 -2
- package/dist/session-worker.js.map +1 -1
- package/dist/{terminal-TVPPZQJW.js → terminal-GIU6MXOR.js} +14 -6
- package/dist/terminal-GIU6MXOR.js.map +1 -0
- package/package.json +4 -4
- package/dist/chunk-3XXFLKKG.js +0 -34
- package/dist/chunk-3XXFLKKG.js.map +0 -1
- package/dist/chunk-VHCL7NVJ.js.map +0 -1
- package/dist/terminal-TVPPZQJW.js.map +0 -1
- /package/dist/{chunk-2XDZRLST.js.map → chunk-QXOARRC2.js.map} +0 -0
- /package/dist/{chunk-XRTTHTM2.js.map → chunk-TG7JPHE5.js.map} +0 -0
package/dist/session-worker.js
CHANGED
|
@@ -2,7 +2,7 @@
|
|
|
2
2
|
import {
|
|
3
3
|
ControlRequestEventSchema,
|
|
4
4
|
SeqCounter
|
|
5
|
-
} from "./chunk-
|
|
5
|
+
} from "./chunk-QXOARRC2.js";
|
|
6
6
|
import {
|
|
7
7
|
CLAUDE_PROVIDER
|
|
8
8
|
} from "./chunk-6O6JTF24.js";
|
|
@@ -10,7 +10,7 @@ import {
|
|
|
10
10
|
LineBuffer,
|
|
11
11
|
createWorkerReader,
|
|
12
12
|
serializeWorkerMsg
|
|
13
|
-
} from "./chunk-
|
|
13
|
+
} from "./chunk-DFLQ3TFT.js";
|
|
14
14
|
|
|
15
15
|
// src/session-worker.ts
|
|
16
16
|
import { createServer } from "net";
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/session-worker.ts","../src/worker/json-session.ts"],"sourcesContent":["import { createServer, type Socket } from \"node:net\";\nimport { mkdirSync, unlinkSync, existsSync, chmodSync } from \"node:fs\";\nimport {\n JsonSession,\n ToolWhitelist,\n createRelayApprovalStrategy,\n type StreamJsonEvent,\n type ClaudePermissionMode,\n} from \"./worker/json-session.js\";\nimport { SeqCounter } from \"./common/seq-counter.js\";\nimport { createWorkerReader, serializeWorkerMsg, type WorkerMessage } from \"./ipc/ipc-protocol.js\";\nimport type { ProviderHookContext } from \"./providers/index.js\";\n\n// 参数格式: session-worker.ts <sessionId> <socketPath> [--cwd <dir>] [--resume <id>] [-- claude args...]\nconst sessionId = process.argv[2];\nconst sockPath = process.argv[3];\nconst separatorIdx = process.argv.indexOf(\"--\");\nconst claudeArgs = separatorIdx >= 0 ? process.argv.slice(separatorIdx + 1) : [];\n\n// 解析 -- 之前的可选参数\nconst preArgs = process.argv.slice(4, separatorIdx >= 0 ? separatorIdx : undefined);\nfunction getArg(name: string): string | undefined {\n const idx = preArgs.indexOf(name);\n return idx >= 0 && idx + 1 < preArgs.length ? preArgs[idx + 1] : undefined;\n}\nfunction hasFlag(name: string): boolean {\n return preArgs.includes(name);\n}\nconst workerCwd = getArg(\"--cwd\");\nconst workerResume = getArg(\"--resume\");\nconst workerPermissionMode = getArg(\"--permission-mode\") as ClaudePermissionMode | undefined;\nconst workerStreamDelta = hasFlag(\"--stream-delta\");\nconst workerHookUrl = getArg(\"--hook-url\");\nconst workerHookMarker = getArg(\"--hook-marker\");\nconst workerHookToken = process.env.DEV_ANYWHERE_HOOK_TOKEN;\nconst workerHookProvider = getArg(\"--hook-provider\") as ProviderHookContext[\"provider\"] | undefined;\n\nif (!sessionId || !sockPath) {\n console.error(\"Usage: session-worker <sessionId> <socketPath> [-- claudeArgs...]\");\n process.exit(1);\n}\n\nconst workerHook: ProviderHookContext | undefined =\n workerHookUrl && workerHookMarker && workerHookToken && workerHookProvider\n ? {\n provider: workerHookProvider,\n sessionId,\n hookUrl: workerHookUrl,\n marker: workerHookMarker,\n token: workerHookToken,\n }\n : undefined;\n\nlet serveSocket: Socket | null = null;\nconst seqCounter = new SeqCounter(sessionId);\nconst whitelist = new ToolWhitelist();\n\nconst pendingApprovals = new Map<\n string,\n {\n resolve: (decision: { behavior: \"allow\" | \"deny\"; message?: string }) => void;\n toolName: string;\n input: Record<string, unknown>;\n }\n>();\n\nfunction sendToServe(msg: WorkerMessage): void {\n if (serveSocket?.writable) {\n serveSocket.write(serializeWorkerMsg(msg));\n }\n}\n\n// 转发审批请求到 serve 进程,由 serve 进程通过 relay 转发到小程序\nconst forwardToRelay = async (\n toolName: string,\n input: Record<string, unknown>,\n): Promise<{ behavior: \"allow\" | \"deny\"; message?: string }> => {\n return new Promise((resolve) => {\n const requestId = `${sessionId}-${Date.now()}`;\n pendingApprovals.set(requestId, { resolve, toolName, input });\n sendToServe({\n type: \"worker_approval_request\",\n requestId,\n toolName,\n input,\n });\n });\n};\n\nconst session = new JsonSession({\n claudeArgs,\n cwd: workerCwd,\n resumeSessionId: workerResume,\n permissionMode: workerPermissionMode,\n includePartialMessages: workerStreamDelta,\n hook: workerHook,\n approvalStrategy: createRelayApprovalStrategy(whitelist, forwardToRelay),\n onEvent: (event: StreamJsonEvent) => {\n // 从 system 事件中捕获 Claude 会话 ID 并通知 serve\n if (event.type === \"system\" && typeof event.session_id === \"string\") {\n sendToServe({\n type: \"worker_claude_session_id\",\n sessionId: event.session_id,\n });\n }\n\n const seq = seqCounter.next();\n sendToServe({\n type: \"worker_event\",\n seq,\n event: event as Record<string, unknown>,\n });\n },\n onExit: (code: number) => {\n whitelist.clear();\n sendToServe({ type: \"worker_exit\", code });\n cleanup();\n process.exit(0);\n },\n});\n\nfunction handleServeConnection(socket: Socket): void {\n serveSocket = socket;\n\n for (const [requestId, pending] of pendingApprovals) {\n sendToServe({\n type: \"worker_approval_request\",\n requestId,\n toolName: pending.toolName,\n input: pending.input,\n });\n }\n\n createWorkerReader(socket, (msg: WorkerMessage) => {\n switch (msg.type) {\n case \"worker_input\":\n session.sendMessage(msg.content);\n break;\n case \"worker_stop\":\n session.stop();\n break;\n case \"worker_approval_response\": {\n const pending = pendingApprovals.get(msg.requestId);\n if (pending) {\n pending.resolve({ behavior: msg.behavior, message: msg.message });\n pendingApprovals.delete(msg.requestId);\n }\n break;\n }\n case \"worker_whitelist_add\":\n whitelist.add(msg.toolName);\n break;\n }\n });\n\n socket.on(\"close\", () => {\n serveSocket = null;\n });\n socket.on(\"error\", () => {\n serveSocket = null;\n });\n}\n\nconst sockDir = sockPath.substring(0, sockPath.lastIndexOf(\"/\"));\nmkdirSync(sockDir, { recursive: true });\n\nif (existsSync(sockPath)) {\n unlinkSync(sockPath);\n}\n\nconst server = createServer((socket) => {\n handleServeConnection(socket);\n});\n\nfunction cleanup(): void {\n server.close();\n try {\n unlinkSync(sockPath);\n } catch {\n // socket 文件可能已被删除\n }\n}\n\nprocess.on(\"SIGTERM\", () => {\n session.stop();\n});\n\nserver.listen(sockPath, () => {\n chmodSync(sockPath, 0o600);\n const pid = session.start();\n sendToServe({ type: \"worker_ready\", pid });\n});\n","import { spawn, type ChildProcess } from \"node:child_process\";\nimport type { z } from \"zod\";\nimport { LineBuffer } from \"../ipc/line-buffer.js\";\nimport { ControlRequestEventSchema } from \"../common/stream-json-schema.js\";\nimport {\n CLAUDE_PROVIDER,\n buildClaudeArgs,\n filterClaudeEnvVars,\n type ClaudePermissionMode,\n} from \"../providers/index.js\";\nimport type { ProviderHookContext } from \"../providers/index.js\";\n\nexport { buildClaudeArgs, filterClaudeEnvVars };\nexport type { ClaudePermissionMode };\n\n// stream-json 事件类型定义,基于 cc-connect 验证的结构\nexport type StreamJsonEventType =\n | \"system\"\n | \"assistant\"\n | \"user\"\n | \"result\"\n | \"control_request\"\n | \"control_cancel_request\"\n | \"stream_event\";\n\nexport interface StreamJsonEvent {\n type: StreamJsonEventType;\n [key: string]: unknown;\n}\n\nexport type ApprovalStrategy = (\n toolName: string,\n input: Record<string, unknown>,\n) => Promise<{ behavior: \"allow\" | \"deny\"; message?: string }>;\n\ninterface JsonSessionOptions {\n workDir?: string;\n claudeArgs?: string[];\n approvalStrategy?: ApprovalStrategy;\n onEvent?: (event: StreamJsonEvent) => void;\n onExit?: (code: number) => void;\n cwd?: string;\n resumeSessionId?: string;\n permissionMode?: ClaudePermissionMode;\n includePartialMessages?: boolean;\n hook?: ProviderHookContext;\n}\n\n// 默认拒绝所有工具调用,远程审批未配置前的安全兜底\nconst denyAllStrategy: ApprovalStrategy = async () => ({\n behavior: \"deny\" as const,\n message: \"Tool use denied by default policy. Remote approval not yet configured.\",\n});\n\n// 会话级别的工具白名单,用户点击\"全部允许\"后同名工具自动审批\nexport class ToolWhitelist {\n private allowed = new Set<string>();\n\n has(toolName: string): boolean {\n return this.allowed.has(toolName);\n }\n\n add(toolName: string): void {\n this.allowed.add(toolName);\n }\n\n clear(): void {\n this.allowed.clear();\n }\n}\n\n// 创建中继转发审批策略,先检查白名单再转发到 relay\nexport function createRelayApprovalStrategy(\n whitelist: ToolWhitelist,\n forwardToRelay: (\n toolName: string,\n input: Record<string, unknown>,\n ) => Promise<{ behavior: \"allow\" | \"deny\"; message?: string }>,\n): ApprovalStrategy {\n return async (toolName, input) => {\n if (whitelist.has(toolName)) {\n return { behavior: \"allow\", message: \"Auto-approved by session whitelist\" };\n }\n return forwardToRelay(toolName, input);\n };\n}\n\nexport class JsonSession {\n private child: ChildProcess | null = null;\n private stderrChunks: string[] = [];\n private writeQueue: Promise<void> = Promise.resolve();\n private claudeSessionId: string | null = null;\n private readonly workDir: string;\n private readonly claudeArgs: string[];\n private readonly approvalStrategy: ApprovalStrategy;\n private readonly onEvent?: (event: StreamJsonEvent) => void;\n private readonly onExitCb?: (code: number) => void;\n private readonly resumeSessionId?: string;\n private readonly permissionMode?: ClaudePermissionMode;\n private readonly includePartialMessages: boolean;\n private readonly hook?: ProviderHookContext;\n\n constructor(options: JsonSessionOptions = {}) {\n this.workDir = options.cwd ?? options.workDir ?? process.cwd();\n this.claudeArgs = options.claudeArgs ?? [];\n this.approvalStrategy = options.approvalStrategy ?? denyAllStrategy;\n this.onEvent = options.onEvent;\n this.onExitCb = options.onExit;\n this.resumeSessionId = options.resumeSessionId;\n this.permissionMode = options.permissionMode;\n this.includePartialMessages = options.includePartialMessages ?? false;\n this.hook = options.hook;\n }\n\n getClaudeSessionId(): string | null {\n return this.claudeSessionId;\n }\n\n start(): number {\n const command = CLAUDE_PROVIDER.buildJsonCommand(\n {\n extraArgs: this.claudeArgs,\n permissionMode: this.permissionMode,\n resumeSessionId: this.resumeSessionId,\n includePartialMessages: this.includePartialMessages,\n hook: this.hook,\n },\n process.env,\n );\n\n this.child = spawn(command.command, command.args, {\n cwd: this.workDir,\n stdio: [\"pipe\", \"pipe\", \"pipe\"],\n env: command.env,\n });\n\n this.setupStdoutParsing();\n this.setupStderrCollection();\n this.setupExitHandler();\n\n return this.child.pid!;\n }\n\n sendMessage(content: string): void {\n const message = {\n type: \"user\",\n message: { role: \"user\", content },\n };\n this.writeToStdin(JSON.stringify(message));\n }\n\n async stop(gracePeriodMs = 5000): Promise<void> {\n if (!this.child || !this.isAlive()) return;\n\n this.child.kill(\"SIGTERM\");\n\n const start = Date.now();\n while (Date.now() - start < gracePeriodMs) {\n if (!this.isAlive()) return;\n await new Promise((r) => setTimeout(r, 200));\n }\n\n if (this.isAlive()) {\n this.child.kill(\"SIGKILL\");\n }\n }\n\n isAlive(): boolean {\n if (!this.child || !this.child.pid) return false;\n try {\n process.kill(this.child.pid, 0);\n return true;\n } catch {\n return false;\n }\n }\n\n getStderr(): string {\n return this.stderrChunks.join(\"\");\n }\n\n private setupStdoutParsing(): void {\n if (!this.child?.stdout) return;\n\n const lineBuffer = new LineBuffer();\n this.child.stdout.pipe(lineBuffer);\n\n lineBuffer.on(\"data\", (line: Buffer | string) => {\n const str = typeof line === \"string\" ? line : line.toString();\n let event: StreamJsonEvent;\n try {\n event = JSON.parse(str) as StreamJsonEvent;\n } catch {\n // 非 JSON 行直接跳过,verbose 模式会输出调试日志\n return;\n }\n\n // 从 system 事件中捕获 Claude 会话 ID 用于后续 resume\n if (event.type === \"system\" && typeof event.session_id === \"string\") {\n this.claudeSessionId = event.session_id;\n }\n\n if (event.type === \"control_request\") {\n // schema parse 失败说明 CLI 协议漂移,调用方必须感知而不是静默吃掉\n const parsed = ControlRequestEventSchema.safeParse(event);\n if (!parsed.success) {\n console.error(\n \"[json-session] control_request shape mismatch; skipping approval\",\n parsed.error.issues.slice(0, 3),\n );\n return;\n }\n this.handleControlRequest(parsed.data);\n return;\n }\n\n this.onEvent?.(event);\n });\n }\n\n private setupStderrCollection(): void {\n if (!this.child?.stderr) return;\n this.child.stderr.on(\"data\", (chunk: Buffer | string) => {\n this.stderrChunks.push(typeof chunk === \"string\" ? chunk : chunk.toString());\n });\n }\n\n private setupExitHandler(): void {\n if (!this.child) return;\n this.child.on(\"exit\", (code: number | null) => {\n this.onExitCb?.(code ?? 1);\n });\n }\n\n private handleControlRequest(event: z.infer<typeof ControlRequestEventSchema>): void {\n const requestId = event.request_id;\n const request = event.request;\n\n this.approvalStrategy(request.tool_name, request.input).then((decision) => {\n const response =\n decision.behavior === \"deny\"\n ? {\n type: \"control_response\",\n response: {\n subtype: \"success\",\n request_id: requestId,\n response: {\n behavior: \"deny\",\n message: decision.message ?? \"Tool use denied by default policy.\",\n },\n },\n }\n : {\n type: \"control_response\",\n response: {\n subtype: \"success\",\n request_id: requestId,\n response: {\n behavior: \"allow\",\n updatedInput: {},\n },\n },\n };\n\n this.writeToStdin(JSON.stringify(response));\n });\n }\n\n private writeToStdin(data: string): void {\n this.writeQueue = this.writeQueue.then(() => {\n return new Promise<void>((resolve, reject) => {\n if (!this.child?.stdin?.writable) {\n reject(new Error(\"stdin not writable\"));\n return;\n }\n this.child.stdin.write(data + \"\\n\", (err) => {\n if (err) {\n reject(err);\n } else {\n resolve();\n }\n });\n });\n });\n }\n}\n"],"mappings":"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","names":["whitelist","forwardToRelay"]}
|
|
1
|
+
{"version":3,"sources":["../src/session-worker.ts","../src/worker/json-session.ts"],"sourcesContent":["import { createServer, type Socket } from \"node:net\";\nimport { mkdirSync, unlinkSync, existsSync, chmodSync } from \"node:fs\";\nimport {\n JsonSession,\n ToolWhitelist,\n createRelayApprovalStrategy,\n type StreamJsonEvent,\n type ClaudePermissionMode,\n} from \"./worker/json-session.js\";\nimport { SeqCounter } from \"./common/seq-counter.js\";\nimport { createWorkerReader, serializeWorkerMsg, type WorkerMessage } from \"./ipc/ipc-protocol.js\";\nimport type { ProviderHookContext } from \"./providers/index.js\";\n\n// 参数格式: session-worker.ts <sessionId> <socketPath> [--cwd <dir>] [--resume <id>] [-- claude args...]\nconst sessionId = process.argv[2];\nconst sockPath = process.argv[3];\nconst separatorIdx = process.argv.indexOf(\"--\");\nconst claudeArgs = separatorIdx >= 0 ? process.argv.slice(separatorIdx + 1) : [];\n\n// 解析 -- 之前的可选参数\nconst preArgs = process.argv.slice(4, separatorIdx >= 0 ? separatorIdx : undefined);\nfunction getArg(name: string): string | undefined {\n const idx = preArgs.indexOf(name);\n return idx >= 0 && idx + 1 < preArgs.length ? preArgs[idx + 1] : undefined;\n}\nfunction hasFlag(name: string): boolean {\n return preArgs.includes(name);\n}\nconst workerCwd = getArg(\"--cwd\");\nconst workerResume = getArg(\"--resume\");\nconst workerPermissionMode = getArg(\"--permission-mode\") as ClaudePermissionMode | undefined;\nconst workerStreamDelta = hasFlag(\"--stream-delta\");\nconst workerHookUrl = getArg(\"--hook-url\");\nconst workerHookMarker = getArg(\"--hook-marker\");\nconst workerHookToken = process.env.DEV_ANYWHERE_HOOK_TOKEN;\nconst workerHookProvider = getArg(\"--hook-provider\") as ProviderHookContext[\"provider\"] | undefined;\n\nif (!sessionId || !sockPath) {\n console.error(\"Usage: session-worker <sessionId> <socketPath> [-- claudeArgs...]\");\n process.exit(1);\n}\n\nconst workerHook: ProviderHookContext | undefined =\n workerHookUrl && workerHookMarker && workerHookToken && workerHookProvider\n ? {\n provider: workerHookProvider,\n sessionId,\n hookUrl: workerHookUrl,\n marker: workerHookMarker,\n token: workerHookToken,\n }\n : undefined;\n\nlet serveSocket: Socket | null = null;\nconst seqCounter = new SeqCounter(sessionId);\nconst whitelist = new ToolWhitelist();\n\nconst pendingApprovals = new Map<\n string,\n {\n resolve: (decision: { behavior: \"allow\" | \"deny\"; message?: string }) => void;\n toolName: string;\n input: Record<string, unknown>;\n }\n>();\n\nfunction sendToServe(msg: WorkerMessage): void {\n if (serveSocket?.writable) {\n serveSocket.write(serializeWorkerMsg(msg));\n }\n}\n\n// 转发审批请求到 serve 进程,由 serve 进程通过 relay 转发到小程序\nconst forwardToRelay = async (\n toolName: string,\n input: Record<string, unknown>,\n): Promise<{ behavior: \"allow\" | \"deny\"; message?: string }> => {\n return new Promise((resolve) => {\n const requestId = `${sessionId}-${Date.now()}`;\n pendingApprovals.set(requestId, { resolve, toolName, input });\n sendToServe({\n type: \"worker_approval_request\",\n requestId,\n toolName,\n input,\n });\n });\n};\n\nconst session = new JsonSession({\n claudeArgs,\n cwd: workerCwd,\n resumeSessionId: workerResume,\n permissionMode: workerPermissionMode,\n includePartialMessages: workerStreamDelta,\n hook: workerHook,\n approvalStrategy: createRelayApprovalStrategy(whitelist, forwardToRelay),\n onEvent: (event: StreamJsonEvent) => {\n // 从 system 事件中捕获 Claude 会话 ID 并通知 serve\n if (event.type === \"system\" && typeof event.session_id === \"string\") {\n sendToServe({\n type: \"worker_claude_session_id\",\n sessionId: event.session_id,\n });\n }\n\n const seq = seqCounter.next();\n sendToServe({\n type: \"worker_event\",\n seq,\n event: event as Record<string, unknown>,\n });\n },\n onExit: (code: number) => {\n whitelist.clear();\n sendToServe({ type: \"worker_exit\", code });\n cleanup();\n process.exit(0);\n },\n});\n\nfunction handleServeConnection(socket: Socket): void {\n serveSocket = socket;\n\n for (const [requestId, pending] of pendingApprovals) {\n sendToServe({\n type: \"worker_approval_request\",\n requestId,\n toolName: pending.toolName,\n input: pending.input,\n });\n }\n\n createWorkerReader(socket, (msg: WorkerMessage) => {\n switch (msg.type) {\n case \"worker_input\":\n session.sendMessage(msg.content);\n break;\n case \"worker_stop\":\n session.stop();\n break;\n case \"worker_approval_response\": {\n const pending = pendingApprovals.get(msg.requestId);\n if (pending) {\n pending.resolve({ behavior: msg.behavior, message: msg.message });\n pendingApprovals.delete(msg.requestId);\n }\n break;\n }\n case \"worker_whitelist_add\":\n whitelist.add(msg.toolName);\n break;\n }\n });\n\n socket.on(\"close\", () => {\n serveSocket = null;\n });\n socket.on(\"error\", () => {\n serveSocket = null;\n });\n}\n\nconst sockDir = sockPath.substring(0, sockPath.lastIndexOf(\"/\"));\nmkdirSync(sockDir, { recursive: true });\n\nif (existsSync(sockPath)) {\n unlinkSync(sockPath);\n}\n\nconst server = createServer((socket) => {\n handleServeConnection(socket);\n});\n\nfunction cleanup(): void {\n server.close();\n try {\n unlinkSync(sockPath);\n } catch {\n // socket 文件可能已被删除\n }\n}\n\nprocess.on(\"SIGTERM\", () => {\n session.stop();\n});\n\nserver.listen(sockPath, () => {\n chmodSync(sockPath, 0o600);\n const pid = session.start();\n sendToServe({ type: \"worker_ready\", pid });\n});\n","import { spawn, type ChildProcess } from \"node:child_process\";\nimport type { z } from \"zod\";\nimport { LineBuffer } from \"../ipc/line-buffer.js\";\nimport { ControlRequestEventSchema } from \"../common/stream-json-schema.js\";\nimport {\n CLAUDE_PROVIDER,\n buildClaudeArgs,\n filterClaudeEnvVars,\n type ClaudePermissionMode,\n} from \"../providers/index.js\";\nimport type { ProviderHookContext } from \"../providers/index.js\";\n\nexport { buildClaudeArgs, filterClaudeEnvVars };\nexport type { ClaudePermissionMode };\n\n// stream-json event types observed from provider output.\nexport type StreamJsonEventType =\n | \"system\"\n | \"assistant\"\n | \"user\"\n | \"result\"\n | \"control_request\"\n | \"control_cancel_request\"\n | \"stream_event\";\n\nexport interface StreamJsonEvent {\n type: StreamJsonEventType;\n [key: string]: unknown;\n}\n\nexport type ApprovalStrategy = (\n toolName: string,\n input: Record<string, unknown>,\n) => Promise<{ behavior: \"allow\" | \"deny\"; message?: string }>;\n\ninterface JsonSessionOptions {\n workDir?: string;\n claudeArgs?: string[];\n approvalStrategy?: ApprovalStrategy;\n onEvent?: (event: StreamJsonEvent) => void;\n onExit?: (code: number) => void;\n cwd?: string;\n resumeSessionId?: string;\n permissionMode?: ClaudePermissionMode;\n includePartialMessages?: boolean;\n hook?: ProviderHookContext;\n}\n\n// 默认拒绝所有工具调用,远程审批未配置前的安全兜底\nconst denyAllStrategy: ApprovalStrategy = async () => ({\n behavior: \"deny\" as const,\n message: \"Tool use denied by default policy. Remote approval not yet configured.\",\n});\n\n// 会话级别的工具白名单,用户点击\"全部允许\"后同名工具自动审批\nexport class ToolWhitelist {\n private allowed = new Set<string>();\n\n has(toolName: string): boolean {\n return this.allowed.has(toolName);\n }\n\n add(toolName: string): void {\n this.allowed.add(toolName);\n }\n\n clear(): void {\n this.allowed.clear();\n }\n}\n\n// 创建中继转发审批策略,先检查白名单再转发到 relay\nexport function createRelayApprovalStrategy(\n whitelist: ToolWhitelist,\n forwardToRelay: (\n toolName: string,\n input: Record<string, unknown>,\n ) => Promise<{ behavior: \"allow\" | \"deny\"; message?: string }>,\n): ApprovalStrategy {\n return async (toolName, input) => {\n if (whitelist.has(toolName)) {\n return { behavior: \"allow\", message: \"Auto-approved by session whitelist\" };\n }\n return forwardToRelay(toolName, input);\n };\n}\n\nexport class JsonSession {\n private child: ChildProcess | null = null;\n private stderrChunks: string[] = [];\n private writeQueue: Promise<void> = Promise.resolve();\n private claudeSessionId: string | null = null;\n private readonly workDir: string;\n private readonly claudeArgs: string[];\n private readonly approvalStrategy: ApprovalStrategy;\n private readonly onEvent?: (event: StreamJsonEvent) => void;\n private readonly onExitCb?: (code: number) => void;\n private readonly resumeSessionId?: string;\n private readonly permissionMode?: ClaudePermissionMode;\n private readonly includePartialMessages: boolean;\n private readonly hook?: ProviderHookContext;\n\n constructor(options: JsonSessionOptions = {}) {\n this.workDir = options.cwd ?? options.workDir ?? process.cwd();\n this.claudeArgs = options.claudeArgs ?? [];\n this.approvalStrategy = options.approvalStrategy ?? denyAllStrategy;\n this.onEvent = options.onEvent;\n this.onExitCb = options.onExit;\n this.resumeSessionId = options.resumeSessionId;\n this.permissionMode = options.permissionMode;\n this.includePartialMessages = options.includePartialMessages ?? false;\n this.hook = options.hook;\n }\n\n getClaudeSessionId(): string | null {\n return this.claudeSessionId;\n }\n\n start(): number {\n const command = CLAUDE_PROVIDER.buildJsonCommand(\n {\n extraArgs: this.claudeArgs,\n permissionMode: this.permissionMode,\n resumeSessionId: this.resumeSessionId,\n includePartialMessages: this.includePartialMessages,\n hook: this.hook,\n },\n process.env,\n );\n\n this.child = spawn(command.command, command.args, {\n cwd: this.workDir,\n stdio: [\"pipe\", \"pipe\", \"pipe\"],\n env: command.env,\n });\n\n this.setupStdoutParsing();\n this.setupStderrCollection();\n this.setupExitHandler();\n\n return this.child.pid!;\n }\n\n sendMessage(content: string): void {\n const message = {\n type: \"user\",\n message: { role: \"user\", content },\n };\n this.writeToStdin(JSON.stringify(message));\n }\n\n async stop(gracePeriodMs = 5000): Promise<void> {\n if (!this.child || !this.isAlive()) return;\n\n this.child.kill(\"SIGTERM\");\n\n const start = Date.now();\n while (Date.now() - start < gracePeriodMs) {\n if (!this.isAlive()) return;\n await new Promise((r) => setTimeout(r, 200));\n }\n\n if (this.isAlive()) {\n this.child.kill(\"SIGKILL\");\n }\n }\n\n isAlive(): boolean {\n if (!this.child || !this.child.pid) return false;\n try {\n process.kill(this.child.pid, 0);\n return true;\n } catch {\n return false;\n }\n }\n\n getStderr(): string {\n return this.stderrChunks.join(\"\");\n }\n\n private setupStdoutParsing(): void {\n if (!this.child?.stdout) return;\n\n const lineBuffer = new LineBuffer();\n this.child.stdout.pipe(lineBuffer);\n\n lineBuffer.on(\"data\", (line: Buffer | string) => {\n const str = typeof line === \"string\" ? line : line.toString();\n let event: StreamJsonEvent;\n try {\n event = JSON.parse(str) as StreamJsonEvent;\n } catch {\n // 非 JSON 行直接跳过,verbose 模式会输出调试日志\n return;\n }\n\n // 从 system 事件中捕获 Claude 会话 ID 用于后续 resume\n if (event.type === \"system\" && typeof event.session_id === \"string\") {\n this.claudeSessionId = event.session_id;\n }\n\n if (event.type === \"control_request\") {\n // schema parse 失败说明 CLI 协议漂移,调用方必须感知而不是静默吃掉\n const parsed = ControlRequestEventSchema.safeParse(event);\n if (!parsed.success) {\n console.error(\n \"[json-session] control_request shape mismatch; skipping approval\",\n parsed.error.issues.slice(0, 3),\n );\n return;\n }\n this.handleControlRequest(parsed.data);\n return;\n }\n\n this.onEvent?.(event);\n });\n }\n\n private setupStderrCollection(): void {\n if (!this.child?.stderr) return;\n this.child.stderr.on(\"data\", (chunk: Buffer | string) => {\n this.stderrChunks.push(typeof chunk === \"string\" ? chunk : chunk.toString());\n });\n }\n\n private setupExitHandler(): void {\n if (!this.child) return;\n this.child.on(\"exit\", (code: number | null) => {\n this.onExitCb?.(code ?? 1);\n });\n }\n\n private handleControlRequest(event: z.infer<typeof ControlRequestEventSchema>): void {\n const requestId = event.request_id;\n const request = event.request;\n\n this.approvalStrategy(request.tool_name, request.input).then((decision) => {\n const response =\n decision.behavior === \"deny\"\n ? {\n type: \"control_response\",\n response: {\n subtype: \"success\",\n request_id: requestId,\n response: {\n behavior: \"deny\",\n message: decision.message ?? \"Tool use denied by default policy.\",\n },\n },\n }\n : {\n type: \"control_response\",\n response: {\n subtype: \"success\",\n request_id: requestId,\n response: {\n behavior: \"allow\",\n updatedInput: {},\n },\n },\n };\n\n this.writeToStdin(JSON.stringify(response));\n });\n }\n\n private writeToStdin(data: string): void {\n this.writeQueue = this.writeQueue.then(() => {\n return new Promise<void>((resolve, reject) => {\n if (!this.child?.stdin?.writable) {\n reject(new Error(\"stdin not writable\"));\n return;\n }\n this.child.stdin.write(data + \"\\n\", (err) => {\n if (err) {\n reject(err);\n } else {\n resolve();\n }\n });\n });\n });\n }\n}\n"],"mappings":"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","names":["whitelist","forwardToRelay"]}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
2
|
import {
|
|
3
|
-
|
|
4
|
-
} from "./chunk-
|
|
3
|
+
daemonRelayArgs
|
|
4
|
+
} from "./chunk-CDKXSDAV.js";
|
|
5
5
|
import {
|
|
6
6
|
createFSM,
|
|
7
7
|
extractOscSequences,
|
|
@@ -9,7 +9,7 @@ import {
|
|
|
9
9
|
shouldReleaseApprovalWait,
|
|
10
10
|
stateAfterApprovalRelease,
|
|
11
11
|
terminalLogger
|
|
12
|
-
} from "./chunk-
|
|
12
|
+
} from "./chunk-TG7JPHE5.js";
|
|
13
13
|
import {
|
|
14
14
|
spawnScript
|
|
15
15
|
} from "./chunk-ZUWAB67J.js";
|
|
@@ -18,6 +18,7 @@ import {
|
|
|
18
18
|
CODEX_PROVIDER
|
|
19
19
|
} from "./chunk-6O6JTF24.js";
|
|
20
20
|
import {
|
|
21
|
+
PROFILE_NAME,
|
|
21
22
|
SERVICE_LOG_PATH,
|
|
22
23
|
SOCK_PATH,
|
|
23
24
|
STOPPED_PATH,
|
|
@@ -25,7 +26,7 @@ import {
|
|
|
25
26
|
encodeBinaryIpcFrame,
|
|
26
27
|
serializeIpc,
|
|
27
28
|
tildify
|
|
28
|
-
} from "./chunk-
|
|
29
|
+
} from "./chunk-DFLQ3TFT.js";
|
|
29
30
|
|
|
30
31
|
// src/terminal.ts
|
|
31
32
|
import { connect } from "net";
|
|
@@ -258,7 +259,14 @@ async function ensureService(autoStart = true) {
|
|
|
258
259
|
if (!autoStart) throw new Error("Service is not running");
|
|
259
260
|
if (existsSync(STOPPED_PATH)) unlinkSync(STOPPED_PATH);
|
|
260
261
|
terminalLogger.info("Auto-starting serve daemon");
|
|
261
|
-
const child = spawnScript(
|
|
262
|
+
const child = spawnScript(
|
|
263
|
+
new URL("./serve", import.meta.url),
|
|
264
|
+
["--profile", PROFILE_NAME, ...daemonRelayArgs()],
|
|
265
|
+
{
|
|
266
|
+
env: { ...process.env },
|
|
267
|
+
logger: terminalLogger
|
|
268
|
+
}
|
|
269
|
+
);
|
|
262
270
|
let childFailed = false;
|
|
263
271
|
let exitCode = null;
|
|
264
272
|
let exitSignal = null;
|
|
@@ -668,4 +676,4 @@ async function startTerminal(providerArgs, providerId = providerFromEnv()) {
|
|
|
668
676
|
export {
|
|
669
677
|
startTerminal
|
|
670
678
|
};
|
|
671
|
-
//# sourceMappingURL=terminal-
|
|
679
|
+
//# sourceMappingURL=terminal-GIU6MXOR.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/terminal.ts","../src/terminal/tty.ts","../src/terminal/pty-manager.ts","../src/terminal/cwd.ts","../src/terminal/state.ts"],"sourcesContent":["import { connect, type Socket } from \"node:net\";\nimport { existsSync, unlinkSync } from \"node:fs\";\nimport { setTimeout as sleep } from \"node:timers/promises\";\nimport { readTtySize, notifyUser } from \"./terminal/tty.js\";\nimport { PtyManager } from \"./terminal/pty-manager.js\";\nimport { resolveTerminalCwd } from \"./terminal/cwd.js\";\nimport pkg from \"@xterm/headless\";\nconst { Terminal: HeadlessTerminal } = pkg;\nimport { SerializeAddon } from \"@xterm/addon-serialize\";\nimport { UnicodeGraphemesAddon } from \"@xterm/addon-unicode-graphemes\";\nimport {\n extractOscSequences,\n extractOscSignals,\n type PtySemanticState,\n} from \"./common/osc-extractor.js\";\nimport {\n shouldReleaseApprovalWait,\n stateAfterApprovalRelease,\n} from \"./common/pty-approval-state.js\";\nimport { TerminalState, TERMINAL_TRANSITIONS, createExitHandler } from \"./terminal/state.js\";\nimport {\n SOCK_PATH,\n STOPPED_PATH,\n SERVICE_LOG_PATH,\n PROFILE_NAME,\n tildify,\n} from \"./common/paths.js\";\nimport { spawnScript } from \"./common/env.js\";\nimport { daemonRelayArgs } from \"./common/daemon-env.js\";\nimport {\n createIpcReader,\n serializeIpc,\n encodeBinaryIpcFrame,\n type IpcMessage,\n} from \"./ipc/ipc-protocol.js\";\nimport { terminalLogger as log } from \"./common/logger.js\";\nimport { createFSM } from \"./common/state-machine.js\";\nimport {\n CLAUDE_PROVIDER,\n CODEX_PROVIDER,\n type ProviderAdapter,\n type ProviderHookContext,\n type ProviderId,\n} from \"./providers/index.js\";\n\n// serve daemon 自动拉起的连接重试参数\nconst ENSURE_SERVICE_MAX_RETRIES = 20;\nconst ENSURE_SERVICE_INITIAL_DELAY_MS = 100;\nconst ENSURE_SERVICE_MAX_DELAY_MS = 2_000;\n\n// 等待特定类型 IPC 消息的默认超时\nconst WAIT_FOR_MESSAGE_TIMEOUT_MS = 10_000;\n\n// idle 检测:超过 IDLE_THRESHOLD_MS 无输出则翻转 working -> turn_complete\nconst IDLE_CHECK_INTERVAL_MS = 3_000;\nconst IDLE_THRESHOLD_MS = 3_000;\n\n// serve 连接断开后的重连重试参数\nconst RECONNECT_INITIAL_DELAY_MS = 1_000;\nconst RECONNECT_MAX_DELAY_MS = 5_000;\n// 连续 spawn 失败到达阈值后停止自动 spawn,降为被动 tryConnect 轮询。\n// 作用:环境异常(端口占用、依赖缺失、权限不足)时避免反复拉起短命子进程把日志刷爆。\nconst SPAWN_FAILURE_THRESHOLD = 3;\n\nconst PROVIDERS: Record<ProviderId, ProviderAdapter> = {\n claude: CLAUDE_PROVIDER,\n codex: CODEX_PROVIDER,\n};\n\nfunction tryConnect(sockPath: string): Promise<Socket | null> {\n return new Promise((resolve) => {\n const s = connect(sockPath);\n s.on(\"connect\", () => resolve(s));\n s.on(\"error\", () => resolve(null));\n });\n}\n\nasync function ensureService(autoStart = true): Promise<Socket> {\n const existing = await tryConnect(SOCK_PATH);\n if (existing) {\n log.info(\"Connected to existing service\");\n return existing;\n }\n\n if (!autoStart) throw new Error(\"Service is not running\");\n\n if (existsSync(STOPPED_PATH)) unlinkSync(STOPPED_PATH);\n\n log.info(\"Auto-starting serve daemon\");\n const child = spawnScript(\n new URL(\"./serve\", import.meta.url),\n [\"--profile\", PROFILE_NAME, ...daemonRelayArgs()],\n {\n env: { ...process.env },\n logger: log,\n },\n );\n\n // 监听 daemon 失败信号,让下面的 tryConnect 轮询能在 daemon 启动时就崩的场景下立刻抛诊断。\n // - 'exit':进程启动成功后又退出(配置错误、端口占用、内部崩溃),带 code/signal。\n // - 'error':spawn 本身失败(ENOENT 找不到 tsx/node 等),Node 文档说此时 'exit' may or may not 跟着 fire,\n // 所以显式监听补完备性。spawnScript 内部另装了一对只管日志的监听器,跟这里互不影响。\n let childFailed = false;\n let exitCode: number | null = null;\n let exitSignal: NodeJS.Signals | null = null;\n let spawnError: Error | null = null;\n child.once(\"exit\", (code, signal) => {\n childFailed = true;\n exitCode = code;\n exitSignal = signal;\n });\n child.once(\"error\", (err) => {\n childFailed = true;\n spawnError = err;\n });\n\n for (let i = 0; i < ENSURE_SERVICE_MAX_RETRIES; i++) {\n const delay = Math.min(ENSURE_SERVICE_INITIAL_DELAY_MS * (i + 1), ENSURE_SERVICE_MAX_DELAY_MS);\n await sleep(delay);\n\n if (childFailed) {\n log.error(\n { code: exitCode, signal: exitSignal, err: spawnError && String(spawnError) },\n \"Serve daemon failed to start\",\n );\n const detail = spawnError\n ? `spawn error=${String(spawnError)}`\n : `code=${exitCode}, signal=${exitSignal}`;\n throw new Error(\n `Serve daemon failed to start (${detail}). Check ${SERVICE_LOG_PATH} for details.`,\n );\n }\n\n const socket = await tryConnect(SOCK_PATH);\n if (socket) {\n log.info({ attempt: i + 1 }, \"Connected to service after retry\");\n return socket;\n }\n }\n\n log.error({ maxRetries: ENSURE_SERVICE_MAX_RETRIES }, \"Failed to connect to service\");\n throw new Error(\n `Failed to connect to dev-anywhere service after ${ENSURE_SERVICE_MAX_RETRIES} retries. Check ${SERVICE_LOG_PATH} for details.`,\n );\n}\n\nfunction waitForMessage<T extends IpcMessage[\"type\"]>(\n socket: Socket,\n messageType: T,\n): Promise<Extract<IpcMessage, { type: T }>> {\n return new Promise((resolve, reject) => {\n let timeout: NodeJS.Timeout | null = null;\n const dispose = createIpcReader(socket, (msg: IpcMessage) => {\n if (msg.type === messageType) {\n if (timeout) clearTimeout(timeout);\n dispose();\n resolve(msg as Extract<IpcMessage, { type: T }>);\n }\n });\n timeout = setTimeout(() => {\n dispose();\n reject(new Error(`Timeout waiting for ${messageType}`));\n }, WAIT_FOR_MESSAGE_TIMEOUT_MS);\n });\n}\n\nclass TerminalSession {\n private readonly fsm = createFSM<TerminalState>({\n initial: TerminalState.INIT,\n transitions: TERMINAL_TRANSITIONS,\n onTransition: (from, to) => log.info({ from, to }, \"Terminal state transition\"),\n });\n private readonly sessionCwd = resolveTerminalCwd();\n // socket 在 run() 中连上 serve 后首次赋值;reconnect 会重新赋值为新实例\n private socket!: Socket;\n private sessionId: string | null = null;\n private hookContext: ProviderHookContext | null = null;\n private ptyManager: PtyManager | null = null;\n private lastOutputTime = 0;\n private idleCheckTimer: NodeJS.Timeout | null = null;\n private currentPtyState: PtySemanticState = \"turn_complete\";\n // headless terminal 在本进程维护,用于按需 serialize() 给远程 client\n private headlessTerminal: InstanceType<typeof HeadlessTerminal> | null = null;\n private serializeAddon: SerializeAddon | null = null;\n private outputSeq = 0;\n private remoteDetached = false;\n // 记录上次 bridge 状态避免重连抖动导致 banner 连刷;初值 null 让首次状态(无论真假)都打,启动时提示 remote viewing 是否就绪\n private lastBridgeConnected: boolean | null = null;\n // 收尾函数在 run() 里创建一次,PTY 退出与 SIGTERM 共用;内部通过 fsm EXITED 检查短路\n private cleanupAndExit!: (code: number) => void;\n\n constructor(\n private readonly provider: ProviderAdapter,\n private readonly providerArgs: string[],\n ) {}\n\n async run(): Promise<void> {\n log.info(\"Terminal starting\");\n this.fsm.transitionTo(TerminalState.CONNECTING_SERVICE);\n this.socket = await ensureService();\n\n await this.createSession();\n this.initHeadlessTerminal();\n this.cleanupAndExit = createExitHandler({\n fsm: this.fsm,\n getSocket: () => this.socket,\n getSessionId: () => this.sessionId,\n getIdleCheckTimer: () => this.idleCheckTimer,\n });\n\n this.setupSocketHandlers();\n this.startPtyManager();\n\n this.socket.write(\n serializeIpc({ type: \"pty_register\", sessionId: this.sessionId!, pid: process.pid }),\n );\n this.replayCurrentPtyState();\n this.fsm.transitionTo(TerminalState.RUNNING);\n this.setupIdleCheck();\n\n process.on(\"SIGTERM\", () => {\n log.info({ sessionId: this.sessionId }, \"SIGTERM received, shutting down\");\n this.cleanupAndExit(143);\n });\n }\n\n private async createSession(): Promise<void> {\n this.fsm.transitionTo(TerminalState.CREATING_SESSION);\n const responsePromise = waitForMessage(this.socket, \"session_create_response\");\n this.socket.write(\n serializeIpc({\n type: \"session_create_request\",\n mode: \"pty\",\n provider: this.provider.id,\n cwd: this.sessionCwd,\n name: tildify(this.sessionCwd),\n pid: process.pid,\n }),\n );\n const response = await responsePromise;\n if (response.error) {\n throw new Error(`Failed to create session: ${response.error}`);\n }\n this.sessionId = response.sessionId;\n this.hookContext = response.hook ?? null;\n }\n\n private initHeadlessTerminal(): void {\n const { cols, rows } = readTtySize(process.stdout);\n log.info(\n { sessionId: this.sessionId, cols, rows },\n \"Session created, initializing headless terminal\",\n );\n this.headlessTerminal = new HeadlessTerminal({\n cols,\n rows,\n scrollback: 5000,\n allowProposedApi: true,\n });\n this.serializeAddon = new SerializeAddon();\n // UnicodeGraphemesAddon activate() 里会设置 activeVersion = '15-graphemes'\n this.headlessTerminal.loadAddon(this.serializeAddon);\n this.headlessTerminal.loadAddon(new UnicodeGraphemesAddon());\n }\n\n private startPtyManager(): void {\n this.ptyManager = new PtyManager({\n provider: this.provider,\n providerArgs: this.providerArgs,\n cwd: this.sessionCwd,\n hook: this.hookContext ?? undefined,\n tap: (data) => this.handlePtyData(data),\n stdin: process.stdin,\n stdout: process.stdout,\n onResize: (newCols, newRows) => {\n if (this.headlessTerminal) this.headlessTerminal.resize(newCols, newRows);\n if (this.socket.writable && this.sessionId) {\n this.socket.write(\n serializeIpc({\n type: \"pty_resize\",\n sessionId: this.sessionId,\n cols: newCols,\n rows: newRows,\n }),\n );\n }\n },\n onSessionExit: (code: number) => {\n log.info({ sessionId: this.sessionId, exitCode: code }, \"PTY exited, cleaning up\");\n this.cleanupAndExit(code);\n },\n });\n this.ptyManager.start();\n log.info({ sessionId: this.sessionId }, \"PTY started with headless terminal\");\n }\n\n // PTY 的每一帧输出都要:追到 headless terminal 状态、推 binary IPC、提取 provider 语义事件\n private handlePtyData(data: string): void {\n this.lastOutputTime = Date.now();\n this.outputSeq += 1;\n\n if (this.headlessTerminal) this.headlessTerminal.write(data);\n\n if (!this.remoteDetached && this.socket.writable && this.sessionId) {\n this.socket.write(\n encodeBinaryIpcFrame(this.sessionId, Buffer.from(data, \"utf-8\"), this.outputSeq),\n );\n }\n\n const oscSequences = extractOscSequences(data);\n const signal = extractOscSignals(data, this.provider.id);\n if (oscSequences.length > 0) {\n log.debug(\n {\n sessionId: this.sessionId,\n oscSequences,\n signal,\n },\n \"PTY OSC sequences parsed\",\n );\n }\n if (signal?.title) {\n this.sendTerminalTitle(signal.title);\n }\n if (signal?.state === \"approval_wait\") {\n this.currentPtyState = \"approval_wait\";\n this.sendPtyState(\"approval_wait\", { title: signal?.title, tool: signal?.tool });\n return;\n }\n if (\n shouldReleaseApprovalWait({\n currentState: this.currentPtyState,\n signalState: signal?.state,\n })\n ) {\n const nextState = stateAfterApprovalRelease(signal?.state);\n this.currentPtyState = nextState;\n this.sendPtyState(nextState, { title: signal?.title, tool: signal?.tool });\n return;\n }\n if (this.currentPtyState === \"approval_wait\" && signal?.state !== \"turn_complete\") {\n this.sendPtyState(\"approval_wait\", { title: signal?.title, tool: signal?.tool });\n return;\n }\n if (signal && signal.state !== \"working\") {\n this.currentPtyState = signal.state;\n this.sendPtyState(signal.state, { title: signal.title, tool: signal.tool });\n return;\n }\n if (this.currentPtyState !== \"working\") {\n this.currentPtyState = \"working\";\n this.sendPtyState(\"working\");\n }\n }\n\n private sendTerminalTitle(title: string): void {\n if (this.remoteDetached || !this.socket.writable || !this.sessionId) return;\n this.socket.write(\n serializeIpc({\n type: \"pty_title_change\",\n sessionId: this.sessionId,\n title,\n }),\n );\n }\n\n private sendPtyState(state: PtySemanticState, meta?: { title?: string; tool?: string }): void {\n if (this.remoteDetached || !this.socket.writable || !this.sessionId) return;\n this.socket.write(\n serializeIpc({\n type: \"pty_semantic_event\",\n sessionId: this.sessionId,\n state,\n ...(meta?.title !== undefined ? { title: meta.title } : {}),\n ...(meta?.tool !== undefined ? { tool: meta.tool } : {}),\n }),\n );\n log.info(\n { sessionId: this.sessionId, state, title: meta?.title, tool: meta?.tool },\n \"PTY semantic event pushed\",\n );\n }\n\n private replayCurrentPtyState(): void {\n if (this.currentPtyState === \"turn_complete\") return;\n this.sendPtyState(this.currentPtyState);\n }\n\n private handleBridgeStatus(connected: boolean): void {\n if (this.remoteDetached) return;\n if (this.lastBridgeConnected === connected) return;\n this.lastBridgeConnected = connected;\n log.info({ connected }, \"Bridge status changed, notifying user\");\n notifyUser(connected ? \"relay online\" : \"relay offline — remote viewing unavailable\");\n }\n\n private setupSocketHandlers(): void {\n createIpcReader(this.socket, (msg: IpcMessage) => {\n if (msg.type === \"pty_input\" && msg.sessionId === this.sessionId) {\n log.debug({ sessionId: this.sessionId, bytes: msg.data.length }, \"Remote input received\");\n this.ptyManager?.write(msg.data);\n } else if (msg.type === \"pty_detach\" && msg.sessionId === this.sessionId) {\n this.detachRemoteView();\n } else if (msg.type === \"bridge_status\") {\n this.handleBridgeStatus(msg.connected);\n } else if (msg.type === \"pty_subscribe\" && msg.sessionId === this.sessionId) {\n if (this.serializeAddon && this.headlessTerminal) {\n const data = this.serializeAddon.serialize();\n this.socket.write(\n serializeIpc({\n type: \"pty_snapshot\",\n sessionId: msg.sessionId,\n cols: this.headlessTerminal.cols,\n rows: this.headlessTerminal.rows,\n data,\n outputSeq: this.outputSeq,\n requestId: msg.requestId,\n }),\n );\n log.info(\n {\n sessionId: this.sessionId,\n cols: this.headlessTerminal.cols,\n rows: this.headlessTerminal.rows,\n bytes: data.length,\n },\n \"Snapshot sent via IPC\",\n );\n }\n }\n });\n\n this.socket.on(\"close\", () => {\n log.info(\"Serve socket closed\");\n if (this.remoteDetached) {\n log.info(\"Remote view detached, skipping serve reconnect\");\n return;\n }\n if (!this.fsm.isIn([TerminalState.RECONNECTING, TerminalState.EXITED])) {\n this.fsm.transitionTo(TerminalState.RECONNECTING);\n this.reconnectToServe();\n }\n });\n\n // socket error 通常和 close 成对出现;这里只记 warn 避免静默吞错,重连仍由 close handler 触发\n this.socket.on(\"error\", (err) => {\n log.warn({ err: err.message }, \"Serve socket error\");\n });\n }\n\n // 超过 IDLE_THRESHOLD_MS 无 PTY 输出则从 working 翻回 turn_complete\n private setupIdleCheck(): void {\n if (this.idleCheckTimer) clearInterval(this.idleCheckTimer);\n this.idleCheckTimer = setInterval(() => {\n if (this.lastOutputTime > 0 && Date.now() - this.lastOutputTime > IDLE_THRESHOLD_MS) {\n this.lastOutputTime = 0;\n if (this.currentPtyState === \"working\") {\n this.currentPtyState = \"turn_complete\";\n this.sendPtyState(\"turn_complete\");\n }\n }\n }, IDLE_CHECK_INTERVAL_MS);\n }\n\n private async reconnectToServe(): Promise<void> {\n log.info(\"Serve connection lost, starting reconnection\");\n\n // 两条路径都不该再继续 spawn daemon:\n // - STOPPED=true:用户主动 dev-anywhere stop,不要对抗用户意图。\n // - consecutiveSpawnFailures 跨过阈值:说明环境有持续性问题,spawn 再多也白搭。\n // 进入 passive 后仅做 tryConnect 等待,daemon 起来或用户 dev-anywhere start 后自动恢复。\n let consecutiveSpawnFailures = 0;\n\n for (let i = 0; ; i++) {\n if (this.remoteDetached) return;\n await sleep(Math.min(RECONNECT_INITIAL_DELAY_MS * (i + 1), RECONNECT_MAX_DELAY_MS));\n\n const stopped = existsSync(STOPPED_PATH);\n const degraded = consecutiveSpawnFailures >= SPAWN_FAILURE_THRESHOLD;\n const passive = stopped || degraded;\n\n try {\n log.debug({ attempt: i + 1, stopped, degraded }, \"Reconnect attempt\");\n const newSocket = passive ? await tryConnect(SOCK_PATH) : await ensureService();\n if (!newSocket) continue;\n\n if (degraded) notifyUser(\"serve daemon reachable, reconnected\");\n consecutiveSpawnFailures = 0;\n\n this.socket = newSocket;\n log.info({ attempt: i + 1, sessionId: this.sessionId }, \"Reconnected to serve\");\n\n this.setupSocketHandlers();\n\n if (this.sessionId) {\n this.fsm.transitionTo(TerminalState.CREATING_SESSION);\n this.socket.write(\n serializeIpc({\n type: \"session_create_request\",\n mode: \"pty\",\n provider: this.provider.id,\n cwd: this.sessionCwd,\n name: tildify(this.sessionCwd),\n pid: process.pid,\n sessionId: this.sessionId,\n }),\n );\n const resp = await waitForMessage(this.socket, \"session_create_response\");\n if (!resp.error) {\n this.sessionId = resp.sessionId;\n this.socket.write(\n serializeIpc({ type: \"pty_register\", sessionId: this.sessionId, pid: process.pid }),\n );\n this.replayCurrentPtyState();\n this.fsm.transitionTo(TerminalState.RUNNING);\n log.info({ sessionId: this.sessionId }, \"Session re-registered after reconnect\");\n }\n } else {\n this.fsm.transitionTo(TerminalState.RUNNING);\n }\n\n return;\n } catch (err) {\n // passive 模式走 tryConnect,失败返回 null 不抛;这里只可能是 ensureService spawn 失败\n if (!passive) {\n consecutiveSpawnFailures++;\n if (consecutiveSpawnFailures === SPAWN_FAILURE_THRESHOLD) {\n notifyUser(\n `serve daemon spawn failed ${SPAWN_FAILURE_THRESHOLD}x — auto-spawn disabled; check environment or run 'dev-anywhere start'`,\n );\n }\n }\n log.debug(\n { err: err instanceof Error ? err.message : err, attempt: i + 1, degraded },\n \"Reconnect attempt failed\",\n );\n }\n }\n }\n\n private detachRemoteView(): void {\n const sessionId = this.sessionId;\n if (!sessionId) return;\n this.remoteDetached = true;\n this.sessionId = null;\n this.hookContext = null;\n this.currentPtyState = \"turn_complete\";\n log.info({ sessionId }, \"Remote view detached; local PTY keeps running\");\n notifyUser(\"remote viewing detached\");\n if (this.socket.writable) this.socket.end();\n }\n}\n\nfunction providerFromEnv(): ProviderId {\n return process.env.DEV_ANYWHERE_PROVIDER === \"codex\" ? \"codex\" : \"claude\";\n}\n\nexport async function startTerminal(\n providerArgs: string[],\n providerId: ProviderId = providerFromEnv(),\n): Promise<void> {\n await new TerminalSession(PROVIDERS[providerId], providerArgs).run();\n}\n","// 读 stdout cols/rows,非 TTY 抛错。\nexport function readTtySize(stream: NodeJS.WriteStream): { cols: number; rows: number } {\n const { columns, rows } = stream;\n if (columns === undefined || rows === undefined) {\n throw new Error(\n \"stdout is not an interactive TTY (columns/rows undefined); dev-anywhere requires running in a real terminal\",\n );\n }\n return { cols: columns, rows };\n}\n\n// 发一条 OSC 9 iTerm2-style 系统通知 + 响铃。iTerm2 / kitty / wezterm 等会弹出带 message\n// 的系统通知;不认 OSC 9 的终端会忽略转义序列只剩下 BEL 响铃。\n// 用此而非 stderr banner 的原因:dev-anywhere 对 Claude PTY 画面保持透明是硬约束,\n// banner 会挤掉 Claude 的渲染行,OSC 9 不占画面,BEL 是纯听觉信号。\nexport function notifyUser(message: string): void {\n process.stderr.write(`\\x1b]9;${message}\\x07`);\n}\n\n// Provider TUI 可能开启 bracketed paste、application cursor/keypad、mouse tracking、\n// xterm modifyOtherKeys 或 kitty keyboard protocol。若 provider 被远程终止或异常退出,\n// 这些模式可能来不及自行恢复,外层 shell 会把 Ctrl-C 显示成 \";5;99~\" 一类残留序列。\nexport function restoreHostTerminalModes(stream: NodeJS.WriteStream): void {\n if (!stream.isTTY) return;\n const restoreSequences = [\n \"\\x1b[?1l\", // application cursor keys off\n \"\\x1b>\", // application keypad off\n \"\\x1b[?1000l\",\n \"\\x1b[?1002l\",\n \"\\x1b[?1003l\",\n \"\\x1b[?1004l\",\n \"\\x1b[?1006l\",\n \"\\x1b[?1015l\",\n \"\\x1b[?2004l\", // bracketed paste off\n \"\\x1b[>4;0m\", // xterm modifyOtherKeys off\n \"\\x1b[<u\", // kitty keyboard protocol off\n ].join(\"\");\n stream.write(restoreSequences);\n}\n","import * as pty from \"node-pty\";\nimport type { IPty } from \"node-pty\";\nimport type { ProviderAdapter, ProviderHookContext } from \"../providers/index.js\";\nimport { readTtySize, restoreHostTerminalModes } from \"./tty.js\";\n\ninterface PtyManagerOptions {\n provider: ProviderAdapter;\n providerArgs: string[];\n cwd: string;\n hook?: ProviderHookContext;\n tap: (data: string) => void;\n stdin: NodeJS.ReadStream;\n stdout: NodeJS.WriteStream;\n onSessionExit?: (code: number) => void | Promise<void>;\n onResize?: (cols: number, rows: number) => void;\n}\n\nexport class PtyManager {\n private child: IPty | null = null;\n private readonly provider: ProviderAdapter;\n private readonly providerArgs: string[];\n private readonly cwd: string;\n private readonly hook?: ProviderHookContext;\n private readonly tap: (data: string) => void;\n private readonly stdin: NodeJS.ReadStream;\n private readonly stdout: NodeJS.WriteStream;\n private readonly onSessionExit?: (code: number) => void;\n private readonly onResize?: (cols: number, rows: number) => void;\n\n constructor(options: PtyManagerOptions) {\n this.provider = options.provider;\n this.providerArgs = options.providerArgs;\n this.cwd = options.cwd;\n this.hook = options.hook;\n this.tap = options.tap;\n this.stdin = options.stdin;\n this.stdout = options.stdout;\n this.onSessionExit = options.onSessionExit;\n this.onResize = options.onResize;\n }\n\n start(): void {\n const { cols, rows } = readTtySize(this.stdout);\n\n const command = this.provider.buildTerminalCommand(\n { args: this.providerArgs, hook: this.hook },\n process.env,\n );\n const child = pty.spawn(command.command, command.args, {\n name: process.env.TERM ?? \"xterm-256color\",\n cols,\n rows,\n cwd: this.cwd,\n env: command.env as Record<string, string>,\n });\n this.child = child;\n\n // raw mode 仅在 stdin 为 TTY 时开启\n const isInteractive = this.stdin.isTTY === true;\n if (isInteractive) {\n this.stdin.setRawMode(true);\n }\n this.stdin.resume();\n\n // stdin -> PTY\n this.stdin.on(\"data\", (data: Buffer) => {\n child.write(data.toString());\n });\n\n // PTY -> stdout + tap\n child.onData((data: string) => this.handleData(data));\n\n // resize 防抖,50ms 窗口合并快速连续的尺寸变化\n let resizeTimer: ReturnType<typeof setTimeout> | null = null;\n this.stdout.on(\"resize\", () => {\n if (resizeTimer) clearTimeout(resizeTimer);\n resizeTimer = setTimeout(() => {\n const { cols: newCols, rows: newRows } = readTtySize(this.stdout);\n child.resize(newCols, newRows);\n this.onResize?.(newCols, newRows);\n }, 50);\n });\n\n // 子进程退出,按 Unix 惯例处理信号退出码,通过回调通知调用方\n child.onExit(({ exitCode, signal }) => {\n if (isInteractive) {\n try {\n this.stdin.setRawMode(false);\n } catch {\n // stdin 可能已关闭\n }\n }\n restoreHostTerminalModes(this.stdout);\n const code = signal ? 128 + signal : exitCode;\n this.onSessionExit?.(code);\n });\n\n // stdin 结束时写入 EOF 控制字符到 PTY\n this.stdin.on(\"end\", () => {\n child.write(\"\\x04\");\n });\n }\n\n /**\n * PTY 数据到达时的统一处理:OSC 9 修复 + 输出到终端 + 传给 tap\n */\n private handleData(data: string): void {\n // PTY 的 onlcr 会把 OSC 序列里的 \\n 转成 \\r\\n,还原为 \\n\n const fixed = data.replace(\n // eslint-disable-next-line no-control-regex\n /\\x1b\\]9;([\\s\\S]*?)\\x07/g,\n (_, content: string) => `\\x1b]9;${content.replace(/\\r\\n/g, \"\\n\")}\\x07`,\n );\n this.stdout.write(fixed);\n this.tap(data);\n }\n\n // 向 PTY 子进程写入数据,用于远程输入注入\n write(data: string): void {\n this.child?.write(data);\n }\n\n cleanup(exitCode: number): void {\n if (this.stdin.isTTY) {\n try {\n this.stdin.setRawMode(false);\n } catch {\n // stdin 可能已关闭\n }\n }\n restoreHostTerminalModes(this.stdout);\n if (this.child) {\n try {\n this.child.kill();\n } catch {\n // 子进程可能已退出\n }\n }\n this.onSessionExit?.(exitCode);\n }\n}\n","import { statSync } from \"node:fs\";\n\nfunction isDirectory(path: string | undefined): path is string {\n if (!path) return false;\n try {\n return statSync(path).isDirectory();\n } catch {\n return false;\n }\n}\n\nexport function resolveTerminalCwd(env: NodeJS.ProcessEnv = process.env): string {\n const candidates = [env.DEV_ANYWHERE_CWD, env.INIT_CWD, env.PWD, process.cwd()];\n return candidates.find(isDirectory) ?? process.cwd();\n}\n","import type { Socket } from \"node:net\";\nimport { createFSM } from \"../common/state-machine.js\";\nimport { serializeIpc } from \"../ipc/ipc-protocol.js\";\n\n// terminal 进程生命周期状态\nexport const TerminalState = {\n INIT: \"init\",\n CONNECTING_SERVICE: \"connecting_service\",\n CREATING_SESSION: \"creating_session\",\n RUNNING: \"running\",\n RECONNECTING: \"reconnecting\",\n EXITED: \"exited\",\n} as const;\nexport type TerminalState = (typeof TerminalState)[keyof typeof TerminalState];\n\n// 允许的状态转换。CREATING_SESSION/RUNNING 下可被 socket close 打断进入 RECONNECTING;\n// 任意非终态都可能被 PTY 退出或 SIGTERM 打断进入 EXITED。\nexport const TERMINAL_TRANSITIONS: Record<TerminalState, readonly TerminalState[]> = {\n init: [\"connecting_service\"],\n connecting_service: [\"creating_session\", \"exited\"],\n creating_session: [\"running\", \"reconnecting\", \"exited\"],\n running: [\"reconnecting\", \"exited\"],\n reconnecting: [\"creating_session\", \"running\", \"exited\"],\n exited: [],\n};\n\n// 下面三个依赖是 getter 而非值:因为它们在 terminal.ts 里是 let 变量,在 handler 创建之后还会变——\n// socket 在 reconnect 时被重新赋值为新实例,sessionId 在 session_create 成功后才有值,\n// idleCheckTimer 在 setupIdleCheck 跑完才赋值。直接传值只会记录 handler 构造那一刻的旧值。\ninterface ExitHandlerDeps {\n fsm: ReturnType<typeof createFSM<TerminalState>>;\n getSocket: () => Socket;\n getSessionId: () => string | null;\n getIdleCheckTimer: () => NodeJS.Timeout | null;\n // 测试注入点,production 默认 process.exit\n exit?: (code: number) => void;\n}\n\n// 构造统一的收尾函数:转 EXITED → 停 idle 定时器 → 给 serve 发 pty_deregister → 退进程。\n// onSessionExit 与 SIGTERM handler 共享同一实例;Ctrl+C 两连击或 PTY 退出与 SIGTERM 竞争时,\n// 第二次调用通过 fsm EXITED 检查直接短路。\nexport function createExitHandler(deps: ExitHandlerDeps): (code: number) => void {\n const exit = deps.exit ?? ((code: number) => process.exit(code));\n return (code: number) => {\n if (deps.fsm.is(TerminalState.EXITED)) return;\n deps.fsm.transitionTo(TerminalState.EXITED);\n const timer = deps.getIdleCheckTimer();\n if (timer) clearInterval(timer);\n const socket = deps.getSocket();\n const sessionId = deps.getSessionId();\n if (socket.writable && sessionId) {\n socket.end(serializeIpc({ type: \"pty_deregister\", sessionId }), () => exit(code));\n } else {\n exit(code);\n }\n };\n}\n"],"mappings":"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","names":[]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@dev-anywhere/proxy",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.1.1",
|
|
4
4
|
"description": "Transparent local proxy for AI coding CLIs that bridges local sessions to a web/PWA client via a relay server.",
|
|
5
5
|
"keywords": [
|
|
6
6
|
"claude-code",
|
|
@@ -12,7 +12,7 @@
|
|
|
12
12
|
"websocket"
|
|
13
13
|
],
|
|
14
14
|
"license": "MIT",
|
|
15
|
-
"author": "catli
|
|
15
|
+
"author": "catli",
|
|
16
16
|
"homepage": "https://github.com/lichenxicatapple-blip/dev-anywhere",
|
|
17
17
|
"repository": {
|
|
18
18
|
"type": "git",
|
|
@@ -52,8 +52,8 @@
|
|
|
52
52
|
"@types/node": "^25.5.2",
|
|
53
53
|
"@types/ws": "^8.18.1",
|
|
54
54
|
"vitest": "^4.1.2",
|
|
55
|
-
"@dev-anywhere/shared": "0.
|
|
56
|
-
"@dev-anywhere/relay": "0.
|
|
55
|
+
"@dev-anywhere/shared": "0.1.1",
|
|
56
|
+
"@dev-anywhere/relay": "0.1.1"
|
|
57
57
|
},
|
|
58
58
|
"scripts": {
|
|
59
59
|
"build": "tsup",
|
package/dist/chunk-3XXFLKKG.js
DELETED
|
@@ -1,34 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
import {
|
|
3
|
-
DESIRED_ENV_PATH
|
|
4
|
-
} from "./chunk-VHCL7NVJ.js";
|
|
5
|
-
|
|
6
|
-
// src/common/daemon-env.ts
|
|
7
|
-
import { existsSync, readFileSync, unlinkSync, writeFileSync } from "fs";
|
|
8
|
-
function setDesiredDaemonEnv(envName) {
|
|
9
|
-
const normalized = envName?.trim();
|
|
10
|
-
if (normalized) {
|
|
11
|
-
writeFileSync(DESIRED_ENV_PATH, `${normalized}
|
|
12
|
-
`);
|
|
13
|
-
return;
|
|
14
|
-
}
|
|
15
|
-
try {
|
|
16
|
-
unlinkSync(DESIRED_ENV_PATH);
|
|
17
|
-
} catch {
|
|
18
|
-
}
|
|
19
|
-
}
|
|
20
|
-
function readDesiredDaemonEnv() {
|
|
21
|
-
if (!existsSync(DESIRED_ENV_PATH)) return void 0;
|
|
22
|
-
const value = readFileSync(DESIRED_ENV_PATH, "utf-8").trim();
|
|
23
|
-
return value || void 0;
|
|
24
|
-
}
|
|
25
|
-
function daemonEnvArgs(envName) {
|
|
26
|
-
const selected = envName?.trim() || readDesiredDaemonEnv();
|
|
27
|
-
return selected ? ["--env", selected] : [];
|
|
28
|
-
}
|
|
29
|
-
|
|
30
|
-
export {
|
|
31
|
-
setDesiredDaemonEnv,
|
|
32
|
-
daemonEnvArgs
|
|
33
|
-
};
|
|
34
|
-
//# sourceMappingURL=chunk-3XXFLKKG.js.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/common/daemon-env.ts"],"sourcesContent":["import { existsSync, readFileSync, unlinkSync, writeFileSync } from \"node:fs\";\nimport { DESIRED_ENV_PATH } from \"./paths.js\";\n\nexport function setDesiredDaemonEnv(envName: string | undefined): void {\n const normalized = envName?.trim();\n if (normalized) {\n writeFileSync(DESIRED_ENV_PATH, `${normalized}\\n`);\n return;\n }\n try {\n unlinkSync(DESIRED_ENV_PATH);\n } catch {\n // Missing desired-env file means \"use config default\".\n }\n}\n\nfunction readDesiredDaemonEnv(): string | undefined {\n if (!existsSync(DESIRED_ENV_PATH)) return undefined;\n const value = readFileSync(DESIRED_ENV_PATH, \"utf-8\").trim();\n return value || undefined;\n}\n\nexport function daemonEnvArgs(envName?: string): string[] {\n const selected = envName?.trim() || readDesiredDaemonEnv();\n return selected ? [\"--env\", selected] : [];\n}\n"],"mappings":";;;;;;AAAA,SAAS,YAAY,cAAc,YAAY,qBAAqB;AAG7D,SAAS,oBAAoB,SAAmC;AACrE,QAAM,aAAa,SAAS,KAAK;AACjC,MAAI,YAAY;AACd,kBAAc,kBAAkB,GAAG,UAAU;AAAA,CAAI;AACjD;AAAA,EACF;AACA,MAAI;AACF,eAAW,gBAAgB;AAAA,EAC7B,QAAQ;AAAA,EAER;AACF;AAEA,SAAS,uBAA2C;AAClD,MAAI,CAAC,WAAW,gBAAgB,EAAG,QAAO;AAC1C,QAAM,QAAQ,aAAa,kBAAkB,OAAO,EAAE,KAAK;AAC3D,SAAO,SAAS;AAClB;AAEO,SAAS,cAAc,SAA4B;AACxD,QAAM,WAAW,SAAS,KAAK,KAAK,qBAAqB;AACzD,SAAO,WAAW,CAAC,SAAS,QAAQ,IAAI,CAAC;AAC3C;","names":[]}
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/common/paths.ts","../../../packages/shared/src/schemas/envelope.ts","../../../packages/shared/src/schemas/chat.ts","../../../packages/shared/src/schemas/tool.ts","../../../packages/shared/src/schemas/session.ts","../../../packages/shared/src/schemas/system.ts","../../../packages/shared/src/builders/index.ts","../../../packages/shared/src/schemas/relay-control.ts","../../../packages/shared/src/constants/relay-errors.ts","../../../packages/shared/src/constants/control-errors.ts","../../../packages/shared/src/constants/session.ts","../../../packages/shared/src/logger.ts","../src/ipc/ipc-protocol.ts","../src/ipc/line-buffer.ts"],"sourcesContent":["import { cpSync, existsSync, mkdirSync, writeFileSync } from \"node:fs\";\nimport { homedir } from \"node:os\";\nimport { dirname, resolve } from \"node:path\";\nimport { fileURLToPath } from \"node:url\";\n\n// 所有 dev-anywhere 文件路径的集中定义\n// 使用 os.homedir():POSIX 走 HOME,Windows 走 USERPROFILE;未设置时回退到 getpwuid。\n// 相比 process.env.HOME,不会在缺失环境变量时构造出 \"undefined/.dev-anywhere\"。\nconst HOME = homedir();\nconst APP_DIR = `${HOME}/.dev-anywhere`;\nconst MODULE_DIR = dirname(fileURLToPath(import.meta.url));\nconst SOURCE_FONT_ASSETS_DIR = resolve(MODULE_DIR, \"../../assets/fonts\");\nconst DIST_FONT_ASSETS_DIR = resolve(MODULE_DIR, \"../assets/fonts\");\nconst DEFAULT_FONT_FAMILY = \"sarasa-fixed-sc\";\n\n// 把 cwd 前缀替换为 ~,HOME 为空时原样返回(避免 replace(\"\", \"~\") 把 ~ 前缀到所有路径)\nexport function tildify(cwd: string): string {\n return HOME ? cwd.replace(HOME, \"~\") : cwd;\n}\nexport const CONFIG_PATH = `${APP_DIR}/config.json`;\n\n// 运行时文件\nexport const RUN_DIR = `${APP_DIR}/run`;\nexport const SOCK_PATH = `${RUN_DIR}/dev-anywhere.sock`;\nexport const PID_PATH = `${RUN_DIR}/dev-anywhere.pid`;\n// 停机标记文件。用户执行 `dev-anywhere stop` 时创建,其它时候不存在。文件内容无意义。\n// 作用:告诉 terminal 不要在此期间自动重启 daemon。\n//\n// 背景:terminal 在与 serve 的连接断开时,默认会 spawn 新 daemon 把连接修复。\n// 这与用户执行 stop 的诉求冲突——stop 刚结束 daemon,terminal 会立即把它重新拉起。\n// 解决办法是 stop 落下此标记,terminal 重连逻辑先检查标记:存在则仅 tryConnect,不 spawn。\nexport const STOPPED_PATH = `${RUN_DIR}/stopped`;\nexport const DESIRED_ENV_PATH = `${RUN_DIR}/desired-env`;\n\n// 持久化状态\nconst STATE_DIR = `${APP_DIR}/state`;\nexport const SESSIONS_PATH = `${STATE_DIR}/sessions.json`;\nexport const HOOK_REGISTRY_PATH = `${STATE_DIR}/hooks.json`;\n\n// 会话数据\nexport const DATA_DIR = `${APP_DIR}/data`;\nconst RELAY_DATA_DIR = `${APP_DIR}/relay-data`;\nconst FONT_DIR = `${RELAY_DATA_DIR}/fonts`;\n\n// 日志\nexport const LOG_DIR = `${APP_DIR}/logs`;\nexport const SERVICE_LOG_PATH = `${LOG_DIR}/service.log`;\n\nfunction sessionDir(sessionId: string): string {\n return `${DATA_DIR}/${sessionId}`;\n}\n\nexport function sessionPaths(sessionId: string) {\n const dir = sessionDir(sessionId);\n return {\n dir,\n workerSock: `${dir}/worker.sock`,\n };\n}\n\nexport function isInitialized(): boolean {\n return existsSync(CONFIG_PATH);\n}\n\nconst DEFAULT_CONFIG = `{\n \"defaultEnv\": \"local\",\n \"envs\": {\n \"local\": {\n \"relayUrl\": \"ws://localhost:3100\"\n },\n \"cloud\": {\n \"relayUrl\": \"wss://dev-anywhere.vita-tools.top\",\n \"relayToken\": \"\"\n }\n }\n}\n`;\n\ntype FontAssetSource = {\n dir: string;\n family?: string;\n};\n\nfunction copyFontFamilyIfMissing(targetFontsDir: string, source: FontAssetSource): boolean {\n const family = source.family ?? DEFAULT_FONT_FAMILY;\n const sourceFamilyDir = `${source.dir}/${family}`;\n const targetFamilyDir = `${targetFontsDir}/${family}`;\n if (existsSync(targetFamilyDir) || !existsSync(sourceFamilyDir)) return false;\n mkdirSync(targetFontsDir, { recursive: true });\n cpSync(sourceFamilyDir, targetFamilyDir, { recursive: true });\n return true;\n}\n\nexport function installFontAssetsFromSources(\n targetFontsDir: string,\n sources: FontAssetSource[],\n): boolean {\n for (const source of sources) {\n if (copyFontFamilyIfMissing(targetFontsDir, source)) return true;\n }\n return false;\n}\n\nfunction installFontAssets(): void {\n installFontAssetsFromSources(FONT_DIR, [\n { dir: SOURCE_FONT_ASSETS_DIR, family: DEFAULT_FONT_FAMILY },\n { dir: DIST_FONT_ASSETS_DIR, family: DEFAULT_FONT_FAMILY },\n ]);\n}\n\nexport function initWorkspace(): void {\n mkdirSync(RUN_DIR, { recursive: true });\n mkdirSync(STATE_DIR, { recursive: true });\n mkdirSync(DATA_DIR, { recursive: true });\n mkdirSync(RELAY_DATA_DIR, { recursive: true });\n mkdirSync(LOG_DIR, { recursive: true });\n installFontAssets();\n if (!existsSync(CONFIG_PATH)) {\n writeFileSync(CONFIG_PATH, DEFAULT_CONFIG);\n }\n}\n","import { z } from \"zod\";\nimport {\n UserInputPayloadSchema,\n AssistantMessagePayloadSchema,\n ThinkingPayloadSchema,\n} from \"./chat.js\";\nimport { ToolUseRequestPayloadSchema, ToolResultPayloadSchema } from \"./tool.js\";\nimport {\n SessionCreatePayloadSchema,\n SessionListPayloadSchema,\n SessionSwitchPayloadSchema,\n SessionTerminatePayloadSchema,\n SessionStatusPayloadSchema,\n} from \"./session.js\";\nimport {\n HeartbeatPayloadSchema,\n AuthPayloadSchema,\n SyncRequestPayloadSchema,\n SyncResponsePayloadSchema,\n} from \"./system.js\";\n\n// 信封基础字段:序列号、会话ID、时间戳、来源、协议版本\nconst BaseEnvelopeFields = {\n seq: z.number().int().nonnegative(),\n sessionId: z.string(),\n timestamp: z.number(),\n source: z.enum([\"proxy\", \"client\"]),\n version: z.string(),\n};\n\n// 按 type 字段区分的 discriminatedUnion 信封\nexport const MessageEnvelopeSchema = z.discriminatedUnion(\"type\", [\n // chat (3)\n z.object({\n ...BaseEnvelopeFields,\n type: z.literal(\"user_input\"),\n payload: UserInputPayloadSchema,\n }),\n z.object({\n ...BaseEnvelopeFields,\n type: z.literal(\"assistant_message\"),\n payload: AssistantMessagePayloadSchema,\n }),\n z.object({\n ...BaseEnvelopeFields,\n type: z.literal(\"thinking\"),\n payload: ThinkingPayloadSchema,\n }),\n // tool (4): 工具审批决策属于 relay control,不进入会话消息信封。\n // tool_use_request: 审批流请求(proxy → client),toolId 是 approval requestId\n z.object({\n ...BaseEnvelopeFields,\n type: z.literal(\"tool_use_request\"),\n payload: ToolUseRequestPayloadSchema,\n }),\n // tool_result: 工具执行结果(proxy → client),toolId 对应 assistant_tool_use / tool_use_request 的 toolId\n z.object({\n ...BaseEnvelopeFields,\n type: z.literal(\"tool_result\"),\n payload: ToolResultPayloadSchema,\n }),\n // assistant_tool_use: 纯展示型工具调用(proxy → client),区别于 tool_use_request 无审批语义\n // payload 结构复用 ToolUseRequestPayloadSchema;toolId 是 Claude 分配的 tool_use id\n z.object({\n ...BaseEnvelopeFields,\n type: z.literal(\"assistant_tool_use\"),\n payload: ToolUseRequestPayloadSchema,\n }),\n // session (5)\n z.object({\n ...BaseEnvelopeFields,\n type: z.literal(\"session_create\"),\n payload: SessionCreatePayloadSchema,\n }),\n z.object({\n ...BaseEnvelopeFields,\n type: z.literal(\"session_list\"),\n payload: SessionListPayloadSchema,\n }),\n z.object({\n ...BaseEnvelopeFields,\n type: z.literal(\"session_switch\"),\n payload: SessionSwitchPayloadSchema,\n }),\n z.object({\n ...BaseEnvelopeFields,\n type: z.literal(\"session_terminate\"),\n payload: SessionTerminatePayloadSchema,\n }),\n z.object({\n ...BaseEnvelopeFields,\n type: z.literal(\"session_status\"),\n payload: SessionStatusPayloadSchema,\n }),\n // system (5)\n z.object({\n ...BaseEnvelopeFields,\n type: z.literal(\"heartbeat\"),\n payload: HeartbeatPayloadSchema,\n }),\n z.object({\n ...BaseEnvelopeFields,\n type: z.literal(\"auth\"),\n payload: AuthPayloadSchema,\n }),\n z.object({\n ...BaseEnvelopeFields,\n type: z.literal(\"sync_request\"),\n payload: SyncRequestPayloadSchema,\n }),\n z.object({\n ...BaseEnvelopeFields,\n type: z.literal(\"sync_response\"),\n payload: SyncResponsePayloadSchema,\n }),\n]);\n\nexport type MessageEnvelope = z.infer<typeof MessageEnvelopeSchema>;\n\nexport type MessageType = MessageEnvelope[\"type\"];\n\nexport type MessageSource = MessageEnvelope[\"source\"];\n","import { z } from \"zod\";\n\n// 用户输入消息\nexport const UserInputPayloadSchema = z.object({\n text: z.string().min(1),\n messageId: z.string().min(1).optional(),\n});\n\nexport type UserInputPayload = z.infer<typeof UserInputPayloadSchema>;\n\n// 助手回复消息,isPartial 标识是否为流式中间结果\nexport const AssistantMessagePayloadSchema = z.object({\n text: z.string(),\n isPartial: z.boolean(),\n});\n\nexport type AssistantMessagePayload = z.infer<typeof AssistantMessagePayloadSchema>;\n\n// 思考过程消息\nexport const ThinkingPayloadSchema = z.object({\n text: z.string(),\n});\n\nexport type ThinkingPayload = z.infer<typeof ThinkingPayloadSchema>;\n","import { z } from \"zod\";\n\n// 工具调用请求\nexport const ToolUseRequestPayloadSchema = z.object({\n toolName: z.string(),\n toolId: z.string(),\n parameters: z.record(z.string(), z.unknown()),\n});\n\nexport type ToolUseRequestPayload = z.infer<typeof ToolUseRequestPayloadSchema>;\n\n// 工具调用批准,whitelistTool 为 true 时将该工具加入会话级白名单自动审批\nexport const ToolApprovePayloadSchema = z.object({\n toolId: z.string(),\n whitelistTool: z.boolean().optional(),\n});\n\nexport type ToolApprovePayload = z.infer<typeof ToolApprovePayloadSchema>;\n\n// 工具调用拒绝\nexport const ToolDenyPayloadSchema = z.object({\n toolId: z.string(),\n reason: z.string().optional(),\n});\n\nexport type ToolDenyPayload = z.infer<typeof ToolDenyPayloadSchema>;\n\n// 工具调用结果\nexport const ToolResultPayloadSchema = z.object({\n toolId: z.string(),\n result: z.unknown(),\n isError: z.boolean(),\n});\n\nexport type ToolResultPayload = z.infer<typeof ToolResultPayloadSchema>;\n","import { z } from \"zod\";\n\nconst sessionStateValues = [\"idle\", \"working\", \"waiting_approval\", \"error\", \"terminated\"] as const;\nconst providerValues = [\"claude\", \"codex\"] as const;\nconst ptyOwnerValues = [\"local-terminal\", \"proxy-hosted\"] as const;\nconst agentStatusPhaseValues = [\n \"idle\",\n \"thinking\",\n \"tool_use\",\n \"outputting\",\n \"waiting_permission\",\n \"error\",\n] as const;\n\n// 会话信息,用于会话列表展示\n// lastActive: 最近一次状态变更/消息时间戳 (ms), 用于列表\"N 分钟前\"显示, 可选\nexport const SessionInfoSchema = z.object({\n sessionId: z.string(),\n name: z.string().optional(),\n state: z.enum(sessionStateValues),\n mode: z.enum([\"pty\", \"json\"]).optional(),\n provider: z.enum(providerValues),\n // PTY 尺寸所有权:\n // - local-terminal: 本地 terminal 进程持有真实 PTY,Web 只按原始 cols/rows 展示\n // - proxy-hosted: serve 内托管 PTY,Web 可按视口请求 resize\n ptyOwner: z.enum(ptyOwnerValues).optional(),\n lastActive: z.number().optional(),\n});\nexport type SessionInfo = z.infer<typeof SessionInfoSchema>;\n\n// 创建会话\n// streamDelta: client 端系统设置\"逐字流式\"toggle,true 时 proxy spawn 带 --include-partial-messages\nexport const SessionCreatePayloadSchema = z.object({\n name: z.string().optional(),\n cwd: z.string().optional(),\n streamDelta: z.boolean().optional(),\n});\n\nexport type SessionCreatePayload = z.infer<typeof SessionCreatePayloadSchema>;\n\n// 会话列表\nexport const SessionListPayloadSchema = z.object({\n sessions: z.array(SessionInfoSchema),\n});\n\nexport type SessionListPayload = z.infer<typeof SessionListPayloadSchema>;\n\n// 切换会话\nexport const SessionSwitchPayloadSchema = z.object({\n sessionId: z.string(),\n});\n\nexport type SessionSwitchPayload = z.infer<typeof SessionSwitchPayloadSchema>;\n\n// 终止会话\nexport const SessionTerminatePayloadSchema = z.object({\n sessionId: z.string(),\n});\n\nexport type SessionTerminatePayload = z.infer<typeof SessionTerminatePayloadSchema>;\n\n// 会话状态变更\n// lastActive: 触发本次状态迁移的时间戳 (ms),用于列表相对时间显示。\nexport const SessionStatusPayloadSchema = z.object({\n sessionId: z.string(),\n state: z.enum(sessionStateValues),\n lastActive: z.number(),\n});\n\nexport type SessionStatusPayload = z.infer<typeof SessionStatusPayloadSchema>;\n\n// PTY 语义状态事件,描述当前 PTY 处于何种状态\nexport const PtyStatePayloadSchema = z.object({\n state: z.enum([\"working\", \"turn_complete\", \"approval_wait\", \"mid_pause\"]),\n title: z.string().optional(),\n tool: z.string().optional(),\n});\nexport type PtyStatePayload = z.infer<typeof PtyStatePayloadSchema>;\n\nexport const AgentStatusPayloadSchema = z.object({\n provider: z.enum(providerValues),\n phase: z.enum(agentStatusPhaseValues),\n seq: z.number().int().nonnegative(),\n updatedAt: z.number(),\n toolName: z.string().optional(),\n toolInput: z.record(z.string(), z.unknown()).optional(),\n permissionRequest: z\n .object({\n requestId: z.string(),\n toolName: z.string(),\n input: z.record(z.string(), z.unknown()),\n })\n .optional(),\n permissionResolution: z\n .object({\n requestId: z.string(),\n outcome: z.enum([\"allow\", \"deny\"]),\n })\n .optional(),\n summary: z.string().optional(),\n});\nexport type AgentStatusPayload = z.infer<typeof AgentStatusPayloadSchema>;\n","import { z } from \"zod\";\n\n// 心跳消息,空 payload\nexport const HeartbeatPayloadSchema = z.object({});\n\nexport type HeartbeatPayload = z.infer<typeof HeartbeatPayloadSchema>;\n\n// 认证消息,支持配对码和 token 两种方式\nexport const AuthPayloadSchema = z.object({\n pairingCode: z.string().optional(),\n token: z.string().optional(),\n});\n\nexport type AuthPayload = z.infer<typeof AuthPayloadSchema>;\n\n// 同步请求,客户端发送已收到的最大序列号\nexport const SyncRequestPayloadSchema = z.object({\n lastSeq: z.number().int().nonnegative(),\n});\n\nexport type SyncRequestPayload = z.infer<typeof SyncRequestPayloadSchema>;\n\n// 同步响应,使用 z.unknown 数组避免循环引用;恢复协议稳定后再收紧类型\nexport const SyncResponsePayloadSchema = z.object({\n messages: z.array(z.record(z.string(), z.unknown())),\n});\n\nexport type SyncResponsePayload = z.infer<typeof SyncResponsePayloadSchema>;\n","import type { MessageEnvelope } from \"../schemas/envelope.js\";\nimport { MessageEnvelopeSchema } from \"../schemas/envelope.js\";\n\n// 构建经过 schema 验证的消息信封\n// seq 由调用方提供,必须与 EventStore per-session seq 一致,保证 proxy 和 relay 对账可靠\nexport function buildMessage<T extends MessageEnvelope[\"type\"]>(\n type: T,\n sessionId: string,\n seq: number,\n payload: Extract<MessageEnvelope, { type: T }>[\"payload\"],\n source: \"proxy\" | \"client\",\n): Extract<MessageEnvelope, { type: T }> {\n const envelope = {\n seq,\n sessionId,\n type,\n payload,\n timestamp: Date.now(),\n source,\n version: \"1.0\",\n };\n return MessageEnvelopeSchema.parse(envelope) as Extract<MessageEnvelope, { type: T }>;\n}\n","import { z } from \"zod\";\nimport { AgentStatusPayloadSchema, PtyStatePayloadSchema } from \"./session.js\";\nimport { ToolApprovePayloadSchema, ToolDenyPayloadSchema } from \"./tool.js\";\nimport { RelayErrorCode } from \"../constants/relay-errors.js\";\nimport { ControlErrorCode } from \"../constants/control-errors.js\";\n\n// 控制消息中复用的子类型\nexport const ProxyInfoSchema = z.object({\n proxyId: z.string(),\n name: z.string().optional(),\n online: z.boolean(),\n sessions: z.array(z.string()).optional(),\n});\nexport type ProxyInfo = z.infer<typeof ProxyInfoSchema>;\n\nexport const AgentCliAvailabilitySchema = z.object({\n available: z.boolean(),\n command: z.string().optional(),\n error: z.string().optional(),\n suggestions: z.array(z.string()).optional(),\n});\nexport type AgentCliAvailability = z.infer<typeof AgentCliAvailabilitySchema>;\n\nexport const AgentCliStatusSchema = z.object({\n claude: AgentCliAvailabilitySchema,\n codex: AgentCliAvailabilitySchema,\n});\nexport type AgentCliStatus = z.infer<typeof AgentCliStatusSchema>;\n\nexport const DirEntrySchema = z.object({ name: z.string(), isDir: z.boolean() });\nexport type DirEntry = z.infer<typeof DirEntrySchema>;\n\nexport const FileTreeGroupSchema = z.object({\n path: z.string(),\n entries: z.array(DirEntrySchema),\n});\nexport type FileTreeGroup = z.infer<typeof FileTreeGroupSchema>;\n\nexport const CommandEntrySchema = z.object({\n name: z.string(),\n description: z.string(),\n argumentHint: z.string().optional(),\n source: z.string(),\n});\nexport type CommandEntry = z.infer<typeof CommandEntrySchema>;\n\nexport const HistorySessionSchema = z.object({\n id: z.string(),\n title: z.string(),\n projectDir: z.string(),\n updatedAt: z.number(),\n provider: z.enum([\"claude\", \"codex\"]).optional(),\n});\nexport type HistorySession = z.infer<typeof HistorySessionSchema>;\n\nconst SessionHistoryMessageSchema = z.object({\n role: z.enum([\"user\", \"assistant\"]),\n text: z.string(),\n timestamp: z.number().optional(),\n cursor: z.string().optional(),\n});\n\ntype RelayControlDirection = \"proxy_to_client\" | \"client_to_proxy\";\ntype EmptyShape = Record<never, never>;\nconst RequestIdShape = { requestId: z.string().min(1).optional() };\nconst ControlErrorCodeSchema = z.enum(\n Object.values(ControlErrorCode) as [ControlErrorCode, ...ControlErrorCode[]],\n);\nconst RequestErrorShape = {\n error: z.string().optional(),\n errorCode: ControlErrorCodeSchema.optional(),\n};\n\ntype ControlDefinition<T extends string, S extends z.ZodRawShape> = {\n type: T;\n directions: ReadonlySet<RelayControlDirection>;\n schema: z.ZodObject<{ type: z.ZodLiteral<T> } & S>;\n};\n\nfunction control<T extends string>(type: T): ControlDefinition<T, EmptyShape>;\nfunction control<T extends string>(\n type: T,\n shape: undefined,\n directions: RelayControlDirection | RelayControlDirection[],\n): ControlDefinition<T, EmptyShape>;\nfunction control<T extends string, S extends z.ZodRawShape>(\n type: T,\n shape: S,\n directions?: RelayControlDirection | RelayControlDirection[],\n): ControlDefinition<T, S>;\nfunction control<T extends string, S extends z.ZodRawShape>(\n type: T,\n shape?: S,\n directions?: RelayControlDirection | RelayControlDirection[],\n): ControlDefinition<T, S | EmptyShape> {\n return {\n type,\n directions: new Set(Array.isArray(directions) ? directions : directions ? [directions] : []),\n schema: z.object({\n type: z.literal(type),\n ...(shape ?? {}),\n }) as z.ZodObject<{ type: z.ZodLiteral<T> } & (S | EmptyShape)>,\n };\n}\n\n// 中转服务器控制消息,独立于 MessageEnvelope 的传输层协议\nconst relayControlDefinitions = [\n control(\"proxy_register\", {\n proxyId: z.string().min(1),\n name: z.string().optional(),\n }),\n control(\"proxy_register_response\", {\n status: z.enum([\"new\", \"reconnected\"]),\n }),\n control(\"proxy_list_request\", RequestIdShape),\n control(\"proxy_list_response\", {\n ...RequestIdShape,\n proxies: z.array(ProxyInfoSchema),\n }),\n control(\"proxy_select\", { ...RequestIdShape, proxyId: z.string().min(1) }),\n control(\"proxy_select_response\", {\n ...RequestIdShape,\n success: z.boolean(),\n proxyId: z.string().optional(),\n ...RequestErrorShape,\n }),\n control(\"relay_error\", {\n code: z.enum(Object.values(RelayErrorCode) as [RelayErrorCode, ...RelayErrorCode[]]),\n message: z.string(),\n }),\n\n // 客户端注册协议\n control(\"client_register\", {\n clientId: z.string().min(1),\n }),\n control(\"client_register_response\", {\n status: z.enum([\"restored\", \"proxy_offline\", \"new\"]),\n proxyId: z.string().optional(),\n }),\n\n // Proxy 离线通知\n control(\"proxy_offline\", {\n proxyId: z.string(),\n }),\n\n // Proxy 主动断开,relay 立即清理资源\n control(\"proxy_disconnect\", {\n proxyId: z.string().min(1),\n }),\n\n // Proxy 重连后通知 client 恢复\n control(\"proxy_online\", {\n proxyId: z.string().min(1),\n }),\n\n // 目录列表请求与响应\n control(\n \"dir_list_request\",\n {\n proxyId: z.string().min(1).optional(),\n ...RequestIdShape,\n path: z.string(),\n },\n \"client_to_proxy\",\n ),\n control(\n \"dir_list_response\",\n { ...RequestIdShape, ...RequestErrorShape, entries: z.array(DirEntrySchema), path: z.string() },\n \"proxy_to_client\",\n ),\n\n // 目录创建请求与响应\n control(\"dir_create_request\", { ...RequestIdShape, path: z.string() }, \"client_to_proxy\"),\n control(\n \"dir_create_response\",\n {\n ...RequestIdShape,\n ...RequestErrorShape,\n path: z.string(),\n success: z.boolean(),\n },\n \"proxy_to_client\",\n ),\n\n // 命令列表推送,proxy 将可用命令列表推给 client\n control(\"command_list_push\", { commands: z.array(CommandEntrySchema) }, \"proxy_to_client\"),\n\n // 文件树推送: 按目录分组, 首组 path 即为 session cwd\n // 前端直接把每组写入 tree[path], 与 dir_list_response 共享 cache slot\n control(\n \"file_tree_push\",\n {\n groups: z.array(FileTreeGroupSchema),\n },\n \"proxy_to_client\",\n ),\n\n // 会话列表请求与权限模式变更\n control(\"session_list\", undefined, [\"client_to_proxy\", \"proxy_to_client\"]),\n control(\n \"permission_mode_change\",\n {\n mode: z.enum([\"default\", \"auto_accept\", \"plan\"]),\n // sessionId 可选:传入时 proxy 按该会话的 mode 分叉(PTY 发 Tab ANSI),未传走全局日志行为\n sessionId: z.string().optional(),\n },\n \"client_to_proxy\",\n ),\n\n // 会话历史浏览\n control(\"session_history_request\", RequestIdShape, \"client_to_proxy\"),\n control(\n \"session_history_response\",\n { ...RequestIdShape, sessions: z.array(HistorySessionSchema) },\n \"proxy_to_client\",\n ),\n\n // PTY 语义状态,从 Envelope 迁移到 Control 层\n control(\n \"pty_state\",\n { sessionId: z.string(), payload: PtyStatePayloadSchema },\n \"proxy_to_client\",\n ),\n\n // Provider 语义状态,来自 Claude/Codex hook 等结构化事件,不从 PTY 字节推断\n control(\n \"agent_status\",\n { sessionId: z.string(), payload: AgentStatusPayloadSchema },\n \"proxy_to_client\",\n ),\n\n // 终端标题变化,proxy -> client\n control(\"terminal_title\", { sessionId: z.string(), title: z.string() }, \"proxy_to_client\"),\n\n // 终端尺寸变化,proxy -> client\n control(\n \"terminal_resize\",\n { sessionId: z.string(), cols: z.number().int().positive(), rows: z.number().int().positive() },\n \"proxy_to_client\",\n ),\n control(\n \"terminal_resize_request\",\n { sessionId: z.string(), cols: z.number().int().positive(), rows: z.number().int().positive() },\n \"client_to_proxy\",\n ),\n\n // 远程终止 JSON 会话,client -> proxy\n control(\"session_terminate\", { sessionId: z.string() }, \"client_to_proxy\"),\n\n // 中断当前 turn,client -> proxy,SIGINT 到 worker 进程让 claude CLI abort 当前流\n control(\"session_worker_abort\", { sessionId: z.string() }, \"client_to_proxy\"),\n\n // turn 完成信号,proxy -> client,对应 claude stream-json 的 result 事件\n control(\n \"turn_result\",\n {\n sessionId: z.string(),\n success: z.boolean(),\n isError: z.boolean(),\n // stream-json result.result 是本轮最终文本。assistant_message 流丢失或 CLI 未发增量时,\n // Web 用它作为 JSON 模式兜底展示,避免 turn 已结束但界面空白。\n result: z.string().optional(),\n },\n \"proxy_to_client\",\n ),\n\n // 客户端发送到 PTY 的原始字节(ANSI 序列),不追加换行\n control(\n \"remote_input_raw\",\n { sessionId: z.string().min(1), data: z.string() },\n \"client_to_proxy\",\n ),\n\n // 客户端询问 proxy 的环境信息 (home 路径等), client -> proxy -> response\n // FilePathPicker 用 homePath 作为 select 模式下的默认起点, 新建会话时打开即可浏览\n control(\"proxy_info_request\", RequestIdShape, \"client_to_proxy\"),\n control(\n \"proxy_info\",\n { ...RequestIdShape, homePath: z.string(), agentCli: AgentCliStatusSchema },\n \"proxy_to_client\",\n ),\n control(\n \"agent_cli_config_update\",\n { ...RequestIdShape, provider: z.enum([\"claude\", \"codex\"]), path: z.string().min(1) },\n \"client_to_proxy\",\n ),\n control(\n \"agent_cli_config_update_response\",\n {\n ...RequestIdShape,\n provider: z.enum([\"claude\", \"codex\"]),\n agentCli: AgentCliStatusSchema.optional(),\n ...RequestErrorShape,\n },\n \"proxy_to_client\",\n ),\n\n // 远程创建 JSON 会话,client -> proxy -> response\n control(\n \"session_create\",\n {\n ...RequestIdShape,\n cwd: z.string(),\n provider: z.enum([\"claude\", \"codex\"]),\n mode: z.enum([\"json\", \"pty\"]).optional(),\n resumeSessionId: z.string().optional(),\n // 透传给 claude CLI 的 --permission-mode, undefined 时 proxy 兜底为 \"default\"\n permissionMode: z\n .enum([\"default\", \"auto\", \"acceptEdits\", \"plan\", \"bypassPermissions\", \"dontAsk\"])\n .optional(),\n },\n \"client_to_proxy\",\n ),\n control(\n \"session_create_response\",\n {\n ...RequestIdShape,\n sessionId: z.string(),\n mode: z.enum([\"json\", \"pty\"]).optional(),\n provider: z.enum([\"claude\", \"codex\"]).optional(),\n ptyOwner: z.enum([\"local-terminal\", \"proxy-hosted\"]).optional(),\n ...RequestErrorShape,\n },\n \"proxy_to_client\",\n ),\n\n // 客户端请求会话历史消息,client -> proxy\n control(\n \"session_messages_request\",\n {\n ...RequestIdShape,\n sessionId: z.string(),\n limit: z.number().int().min(1).max(200).optional(),\n before: z.string().optional(),\n },\n \"client_to_proxy\",\n ),\n\n // 客户端请求会话资源(命令列表 + 文件树),client -> proxy\n control(\n \"session_resources_request\",\n { ...RequestIdShape, sessionId: z.string() },\n \"client_to_proxy\",\n ),\n control(\n \"session_resources_response\",\n {\n ...RequestIdShape,\n ...RequestErrorShape,\n sessionId: z.string(),\n commands: z.array(CommandEntrySchema),\n groups: z.array(FileTreeGroupSchema),\n },\n \"proxy_to_client\",\n ),\n\n // 客户端请求当前 provider 语义状态;不经 relay 缓存,由 proxy 返回当前值\n control(\n \"agent_status_request\",\n { ...RequestIdShape, sessionId: z.string().optional() },\n \"client_to_proxy\",\n ),\n control(\n \"agent_status_response\",\n {\n ...RequestIdShape,\n statuses: z.array(z.object({ sessionId: z.string(), payload: AgentStatusPayloadSchema })),\n },\n \"proxy_to_client\",\n ),\n\n // 客户端确认已收到审批请求;proxy 只记录送达状态,不把它当成用户决策\n control(\n \"permission_request_delivered\",\n { sessionId: z.string(), requestId: z.string() },\n \"client_to_proxy\",\n ),\n control(\n \"tool_approve\",\n { sessionId: z.string(), payload: ToolApprovePayloadSchema },\n \"client_to_proxy\",\n ),\n control(\n \"tool_deny\",\n { sessionId: z.string(), payload: ToolDenyPayloadSchema },\n \"client_to_proxy\",\n ),\n\n // proxy 确认用户决策已进入 provider/worker 路径;web 用它更新审批卡片状态\n control(\n \"permission_decision_result\",\n {\n sessionId: z.string(),\n requestId: z.string(),\n outcome: z.enum([\"allow\", \"deny\"]),\n delivered: z.boolean(),\n message: z.string().optional(),\n },\n \"proxy_to_client\",\n ),\n\n // proxy 推送当前 pending 的工具审批列表,client 据此恢复审批卡片\n control(\n \"pending_approvals_push\",\n {\n sessionId: z.string(),\n approvals: z.array(\n z.object({\n requestId: z.string(),\n toolName: z.string(),\n input: z.record(z.string(), z.unknown()),\n }),\n ),\n },\n \"proxy_to_client\",\n ),\n\n // 恢复会话时推送历史消息,proxy -> client\n control(\n \"session_history_messages\",\n {\n ...RequestIdShape,\n sessionId: z.string(),\n before: z.string().optional(),\n messages: z.array(SessionHistoryMessageSchema),\n hasMore: z.boolean().optional(),\n nextBefore: z.string().optional(),\n },\n \"proxy_to_client\",\n ),\n\n // proxy 重连后同步活跃 session 列表给 relay\n control(\"session_sync\", {\n sessions: z.array(\n z.object({\n id: z.string(),\n mode: z.enum([\"pty\", \"json\"]),\n provider: z.enum([\"claude\", \"codex\"]),\n ptyOwner: z.enum([\"local-terminal\", \"proxy-hosted\"]).optional(),\n state: z.string(),\n }),\n ),\n }),\n\n // PTY 会话订阅,client -> proxy,触发 terminal serialize() 返回当前状态\n control(\n \"session_subscribe\",\n { sessionId: z.string(), requestId: z.string().optional() },\n \"client_to_proxy\",\n ),\n\n // PTY 会话快照,proxy -> client,serialize() 的全量终端状态\n control(\n \"session_snapshot\",\n {\n sessionId: z.string(),\n cols: z.number().int().positive(),\n rows: z.number().int().positive(),\n data: z.string(),\n outputSeq: z.number().int().nonnegative(),\n requestId: z.string().optional(),\n },\n \"proxy_to_client\",\n ),\n] as const;\n\nconst relayControlSchemas = relayControlDefinitions.map((definition) => definition.schema) as [\n (typeof relayControlDefinitions)[number][\"schema\"],\n ...Array<(typeof relayControlDefinitions)[number][\"schema\"]>,\n];\n\nexport const RelayControlSchema = z.discriminatedUnion(\"type\", relayControlSchemas);\n\nexport type RelayControlMessage = z.infer<typeof RelayControlSchema>;\nexport type RelayControlType = RelayControlMessage[\"type\"];\n\nexport const ProxyToClientRelayControlTypes = new Set(\n relayControlDefinitions\n .filter((definition) => definition.directions.has(\"proxy_to_client\"))\n .map((definition) => definition.type),\n);\n\nexport function isProxyToClientRelayControlType(type: RelayControlType): boolean {\n return ProxyToClientRelayControlTypes.has(type);\n}\n\nexport const ClientToProxyRelayControlTypes = new Set(\n relayControlDefinitions\n .filter((definition) => definition.directions.has(\"client_to_proxy\"))\n .map((definition) => definition.type),\n);\n\nexport function isClientToProxyRelayControlType(type: RelayControlType): boolean {\n return ClientToProxyRelayControlTypes.has(type);\n}\n","// relay 实际发出的 6 个错误码。schema 侧 RelayControlSchema.relay_error.code 用 z.enum 收紧,\n// handler 侧用这个常量引用避免裸字面量拼错。\nexport const RelayErrorCode = {\n NOT_REGISTERED: \"NOT_REGISTERED\",\n NOT_BOUND: \"NOT_BOUND\",\n PROXY_OFFLINE: \"PROXY_OFFLINE\",\n INVALID_MESSAGE: \"INVALID_MESSAGE\",\n UNSUPPORTED: \"UNSUPPORTED\",\n INVALID_RANGE: \"INVALID_RANGE\",\n} as const;\n\nexport type RelayErrorCode = (typeof RelayErrorCode)[keyof typeof RelayErrorCode];\n","export const ControlErrorCode = {\n INVALID_PATH: \"INVALID_PATH\",\n PATH_NOT_FOUND: \"PATH_NOT_FOUND\",\n PATH_NOT_DIRECTORY: \"PATH_NOT_DIRECTORY\",\n PATH_ACCESS_DENIED: \"PATH_ACCESS_DENIED\",\n PROXY_OFFLINE: \"PROXY_OFFLINE\",\n SESSION_NOT_FOUND: \"SESSION_NOT_FOUND\",\n PROVIDER_UNSUPPORTED: \"PROVIDER_UNSUPPORTED\",\n WORKER_START_FAILED: \"WORKER_START_FAILED\",\n PROCESS_START_FAILED: \"PROCESS_START_FAILED\",\n UNKNOWN: \"UNKNOWN\",\n} as const;\n\nexport type ControlErrorCode = (typeof ControlErrorCode)[keyof typeof ControlErrorCode];\n","// 会话状态枚举\nexport const SessionState = {\n IDLE: \"idle\",\n WORKING: \"working\",\n WAITING_APPROVAL: \"waiting_approval\",\n ERROR: \"error\",\n TERMINATED: \"terminated\",\n} as const;\n\nexport type SessionState = (typeof SessionState)[keyof typeof SessionState];\n","import {\n lstatSync,\n mkdirSync,\n readdirSync,\n renameSync,\n statSync,\n symlinkSync,\n unlinkSync,\n} from \"node:fs\";\nimport { homedir } from \"node:os\";\nimport { basename, join } from \"node:path\";\nimport pino from \"pino\";\n\nexport type { Logger } from \"pino\";\n\nexport interface CreateLoggerOptions {\n name: string;\n level?: string;\n logDir?: string;\n stdout?: boolean;\n silent?: boolean;\n}\n\nconst DEFAULT_LOG_DIR = `${homedir()}/.dev-anywhere/logs`;\nconst DEFAULT_LOG_RETENTION = 50;\n\nconst PROCESS_LOG_RUN_ID = sanitizeRunId(\n process.env.DEV_ANYWHERE_LOG_RUN_ID ??\n `${new Date().toISOString().replace(/[:.]/g, \"-\")}-${process.pid}`,\n);\n\nfunction sanitizeRunId(runId: string): string {\n return runId.replace(/[^a-zA-Z0-9._-]/g, \"_\");\n}\n\nfunction linkLatestLog(logDir: string, name: string, filePath: string, runId: string): void {\n const latestPath = join(logDir, `${name}.log`);\n\n try {\n const stat = lstatSync(latestPath);\n if (stat.isSymbolicLink()) {\n unlinkSync(latestPath);\n } else {\n renameSync(latestPath, join(logDir, `${name}-legacy-${runId}.log`));\n }\n } catch (err) {\n const code = (err as NodeJS.ErrnoException).code;\n if (code !== \"ENOENT\") return;\n }\n\n try {\n symlinkSync(basename(filePath), latestPath);\n } catch {\n // 日志本体仍然写入 run-specific 文件;latest 链接失败不应阻塞服务启动。\n }\n}\n\nfunction resolveRetention(): number {\n const raw = process.env.DEV_ANYWHERE_LOG_RETENTION;\n if (!raw) return DEFAULT_LOG_RETENTION;\n const parsed = Number.parseInt(raw, 10);\n return Number.isFinite(parsed) && parsed >= 0 ? parsed : DEFAULT_LOG_RETENTION;\n}\n\nfunction pruneOldLogs(logDir: string, name: string, currentFilePath: string): void {\n const keep = resolveRetention();\n if (keep === 0) return;\n\n const currentFileName = basename(currentFilePath);\n const prefix = `${name}-`;\n const candidates = readdirSync(logDir)\n .filter(\n (entry) => entry.startsWith(prefix) && entry.endsWith(\".log\") && entry !== currentFileName,\n )\n .map((entry) => {\n const path = join(logDir, entry);\n try {\n return { path, mtimeMs: statSync(path).mtimeMs };\n } catch {\n return null;\n }\n })\n .filter((entry): entry is { path: string; mtimeMs: number } => entry !== null)\n .sort((a, b) => b.mtimeMs - a.mtimeMs);\n\n for (const stale of candidates.slice(Math.max(0, keep - 1))) {\n try {\n unlinkSync(stale.path);\n } catch {\n // 日志清理失败不能影响主进程启动。\n }\n }\n}\n\nexport function createLogger(options: CreateLoggerOptions): pino.Logger {\n const {\n name,\n level = \"info\",\n logDir = DEFAULT_LOG_DIR,\n stdout = false,\n silent = false,\n } = options;\n\n if (silent) {\n return pino({ level: \"silent\" });\n }\n\n mkdirSync(logDir, { recursive: true });\n\n const runId = PROCESS_LOG_RUN_ID;\n const filePath = join(logDir, `${name}-${runId}.log`);\n linkLatestLog(logDir, name, filePath, runId);\n pruneOldLogs(logDir, name, filePath);\n const streams: pino.StreamEntry[] = [{ stream: pino.destination(filePath) }];\n\n if (stdout) {\n streams.unshift({ stream: process.stdout });\n }\n\n return pino({ level }, pino.multistream(streams));\n}\n","import { z } from \"zod\";\nimport { SessionState } from \"@dev-anywhere/shared\";\nimport { LineBuffer } from \"./line-buffer.js\";\n\n// IPC binary 帧标记字节,0x00 不可能是 JSON 行的首字节(JSON 以 '{' 开头)\nexport const IPC_BINARY_MARKER = 0x00;\n\n// 编码 binary PTY 数据帧用于 IPC 传输\n// 格式: [1B 0x00 marker][4B payload_len uint32LE][1B sessionId_len][sessionId UTF-8][4B outputSeq uint32LE][PTY data]\nexport function encodeBinaryIpcFrame(sessionId: string, data: Buffer, outputSeq: number): Buffer {\n const sessionIdBuf = Buffer.from(sessionId, \"utf-8\");\n const payloadLen = 1 + sessionIdBuf.length + 4 + data.length;\n const frame = Buffer.alloc(1 + 4 + payloadLen);\n let offset = 0;\n frame[offset] = IPC_BINARY_MARKER;\n offset += 1;\n frame.writeUInt32LE(payloadLen, offset);\n offset += 4;\n frame[offset] = sessionIdBuf.length;\n offset += 1;\n sessionIdBuf.copy(frame, offset);\n offset += sessionIdBuf.length;\n frame.writeUInt32LE(outputSeq, offset);\n offset += 4;\n data.copy(frame, offset);\n return frame;\n}\n\nconst sessionStateValues = Object.values(SessionState) as [SessionState, ...SessionState[]];\n\nconst ProviderHookContextSchema = z.object({\n provider: z.enum([\"claude\", \"codex\"]),\n sessionId: z.string(),\n hookUrl: z.string(),\n marker: z.string(),\n token: z.string(),\n});\n\n// IPC 消息 schema,客户端与服务端通过 Unix domain socket 使用 NDJSON 通信\nexport const IpcMessageSchema = z.discriminatedUnion(\"type\", [\n // 客户端请求创建新会话,sessionId 可选用于重连时复用\n z.object({\n type: z.literal(\"session_create_request\"),\n name: z.string().optional(),\n mode: z.enum([\"pty\", \"json\"]),\n provider: z.enum([\"claude\", \"codex\"]),\n cwd: z.string(),\n pid: z.number(),\n sessionId: z.string().optional(),\n }),\n\n // 服务端响应创建会话\n z.object({\n type: z.literal(\"session_create_response\"),\n sessionId: z.string(),\n error: z.string().optional(),\n hook: ProviderHookContextSchema.optional(),\n }),\n\n // 客户端请求终止会话\n z.object({\n type: z.literal(\"session_terminate_request\"),\n sessionId: z.string(),\n }),\n\n // 服务端响应终止会话\n z.object({\n type: z.literal(\"session_terminate_response\"),\n sessionId: z.string(),\n success: z.boolean(),\n }),\n\n // 客户端向服务端注册 PTY 会话\n z.object({\n type: z.literal(\"pty_register\"),\n sessionId: z.string(),\n pid: z.number(),\n }),\n\n // 客户端取消注册 PTY 会话\n z.object({\n type: z.literal(\"pty_deregister\"),\n sessionId: z.string(),\n }),\n\n // 输入,从服务端转发到客户端的 PTY stdin(手机远程输入注入)\n z.object({\n type: z.literal(\"pty_input\"),\n sessionId: z.string(),\n data: z.string(),\n }),\n\n // serve → terminal:Web 端移除本地终端会话时,只断开远程视图,不杀本地 CLI。\n z.object({\n type: z.literal(\"pty_detach\"),\n sessionId: z.string(),\n }),\n\n // 服务端广播会话状态变更\n z.object({\n type: z.literal(\"session_status_update\"),\n sessionId: z.string(),\n state: z.enum(sessionStateValues),\n }),\n\n // 错误响应\n z.object({\n type: z.literal(\"error\"),\n message: z.string(),\n code: z.string().optional(),\n }),\n\n // 客户端请求服务状态(含 relay 连接信息和 worker 状态)\n z.object({\n type: z.literal(\"service_status_request\"),\n }),\n\n // 服务端响应增强版服务状态\n z.object({\n type: z.literal(\"service_status_response\"),\n config: z.object({\n envName: z.string().optional(),\n envNameSource: z.enum([\"cli\", \"file\", \"single\", \"default\", \"none\"]),\n relayUrl: z.string().optional(),\n relayUrlSource: z.enum([\"env\", \"file\", \"none\"]),\n relayTokenSource: z.enum([\"env\", \"file\", \"none\"]),\n hookPort: z.number(),\n hookPortSource: z.enum([\"env\", \"file\", \"default\"]),\n }),\n relay: z\n .object({\n connected: z.boolean(),\n proxyId: z.string(),\n reconnectAttempt: z.number(),\n queueDepth: z.number(),\n })\n .nullable(),\n sessions: z.array(\n z.object({\n id: z.string(),\n mode: z.enum([\"pty\", \"json\"]),\n state: z.enum(sessionStateValues),\n createdAt: z.string(),\n name: z.string().optional(),\n hasWorker: z.boolean(),\n }),\n ),\n }),\n\n // terminal → serve:终端标题变化,由 xterm onTitleChange 触发\n z.object({\n type: z.literal(\"pty_title_change\"),\n sessionId: z.string(),\n title: z.string(),\n }),\n\n // terminal → serve:local runtime 观察到的 PTY 语义事件。\n // 下面 state 枚举必须与 src/common/osc-extractor.ts 的 PtySemanticState 保持一致。\n z.object({\n type: z.literal(\"pty_semantic_event\"),\n sessionId: z.string(),\n state: z.enum([\"working\", \"turn_complete\", \"approval_wait\", \"mid_pause\"]),\n title: z.string().optional(),\n tool: z.string().optional(),\n }),\n\n // terminal → serve:终端尺寸变化\n z.object({\n type: z.literal(\"pty_resize\"),\n sessionId: z.string(),\n cols: z.number(),\n rows: z.number(),\n }),\n\n // serve → terminal:请求 HeadlessTerminal serialize() 快照\n z.object({\n type: z.literal(\"pty_subscribe\"),\n sessionId: z.string(),\n requestId: z.string().optional(),\n }),\n\n // terminal → serve:serialize() 结果\n z.object({\n type: z.literal(\"pty_snapshot\"),\n sessionId: z.string(),\n cols: z.number(),\n rows: z.number(),\n data: z.string(),\n outputSeq: z.number().int().nonnegative(),\n requestId: z.string().optional(),\n }),\n\n // serve → terminal:relay 连接状态变更,供终端给用户显示 remote viewing 是否通畅\n z.object({\n type: z.literal(\"bridge_status\"),\n connected: z.boolean(),\n }),\n]);\n\n// serve 与 session-worker 之间的通信协议\nexport const WorkerMessageSchema = z.discriminatedUnion(\"type\", [\n // serve → worker: 发送用户输入给 claude\n z.object({\n type: z.literal(\"worker_input\"),\n content: z.string(),\n }),\n\n // serve → worker: 停止 claude 进程\n z.object({\n type: z.literal(\"worker_stop\"),\n }),\n\n // serve → worker: 工具审批响应\n z.object({\n type: z.literal(\"worker_approval_response\"),\n requestId: z.string(),\n behavior: z.enum([\"allow\", \"deny\"]),\n message: z.string().optional(),\n }),\n\n // worker → serve: claude 输出事件(带序列号)\n z.object({\n type: z.literal(\"worker_event\"),\n seq: z.number(),\n event: z.record(z.string(), z.unknown()),\n }),\n\n // worker → serve: claude 进程退出\n z.object({\n type: z.literal(\"worker_exit\"),\n code: z.number(),\n }),\n\n // worker → serve: 工具审批请求\n z.object({\n type: z.literal(\"worker_approval_request\"),\n requestId: z.string(),\n toolName: z.string(),\n input: z.record(z.string(), z.unknown()),\n }),\n\n // worker → serve: worker 就绪,claude 已启动\n z.object({\n type: z.literal(\"worker_ready\"),\n pid: z.number(),\n }),\n\n // worker → serve: 从 stream-json 的 system.init 事件捕获 Claude CLI 侧的 session ID\n // proxy 拿它来读 ~/.claude/projects/.../<id>.jsonl 历史或后续 --resume\n z.object({\n type: z.literal(\"worker_claude_session_id\"),\n sessionId: z.string(),\n }),\n\n // serve → worker: 将指定工具加入会话白名单,后续同名工具自动审批\n z.object({\n type: z.literal(\"worker_whitelist_add\"),\n toolName: z.string(),\n }),\n]);\n\nexport type WorkerMessage = z.infer<typeof WorkerMessageSchema>;\n\nexport function serializeWorkerMsg(msg: WorkerMessage): string {\n return JSON.stringify(msg) + \"\\n\";\n}\n\nexport function createWorkerReader(\n stream: NodeJS.ReadableStream,\n onMessage: (msg: WorkerMessage) => void,\n): void {\n const lineBuffer = new LineBuffer();\n lineBuffer.on(\"data\", (line: Buffer | string) => {\n const str = typeof line === \"string\" ? line : line.toString();\n if (str.length === 0) return;\n try {\n const raw = JSON.parse(str);\n const result = WorkerMessageSchema.safeParse(raw);\n if (result.success) {\n onMessage(result.data);\n } else {\n stream.emit(\n \"error\",\n new Error(`Worker message validation failed: ${result.error.message}`),\n );\n }\n } catch (err) {\n stream.emit(\"error\", new Error(\"Worker message parse error\", { cause: err }));\n }\n });\n (stream as NodeJS.ReadableStream).pipe(lineBuffer);\n}\n\nexport type IpcMessage = z.infer<typeof IpcMessageSchema>;\n\n// 将 IPC 消息序列化为 NDJSON 格式的字符串\nexport function serializeIpc(msg: IpcMessage): string {\n return JSON.stringify(msg) + \"\\n\";\n}\n\n// 混合协议 IPC 读取器,支持 NDJSON 控制消息和 binary PTY 帧。\n// binary 帧以 0x00 开头,NDJSON 行以 '{' 开头,通过首字节区分。\n// 返回 dispose 函数用于摘掉 'data' 监听,长连接可以忽略,一次性等待(如 waitForMessage)必须调用避免累积 listener 重复解析每条消息。\nexport function createIpcReader(\n stream: NodeJS.ReadableStream,\n onMessage: (msg: IpcMessage) => void,\n onBinaryFrame?: (sessionId: string, data: Buffer, outputSeq: number) => void,\n): () => void {\n let buf = Buffer.alloc(0);\n let disposed = false;\n\n // 解析状态机:不断消费 buf 中的完整消息\n function drain(): void {\n while (buf.length > 0) {\n if (buf[0] === IPC_BINARY_MARKER) {\n // binary 帧: [1B marker][4B payload_len LE][payload]\n // 需要至少 5 字节才能读取 header\n if (buf.length < 5) return;\n const payloadLen = buf.readUInt32LE(1);\n const totalFrameLen = 1 + 4 + payloadLen;\n if (buf.length < totalFrameLen) return;\n\n // 解析 payload: [1B sessionId_len][sessionId][4B outputSeq][pty data]\n const payloadStart = 5;\n const sessionIdLen = buf[payloadStart];\n const sessionId = buf\n .subarray(payloadStart + 1, payloadStart + 1 + sessionIdLen)\n .toString(\"utf-8\");\n const seqOffset = payloadStart + 1 + sessionIdLen;\n const outputSeq = buf.readUInt32LE(seqOffset);\n const ptyData = Buffer.from(buf.subarray(seqOffset + 4, totalFrameLen));\n\n if (onBinaryFrame) {\n onBinaryFrame(sessionId, ptyData, outputSeq);\n }\n\n buf = buf.subarray(totalFrameLen);\n } else {\n // NDJSON 行: 找 \\n 分隔符\n const newlineIdx = buf.indexOf(0x0a); // '\\n'\n if (newlineIdx === -1) return;\n\n const line = buf.subarray(0, newlineIdx).toString(\"utf-8\");\n buf = buf.subarray(newlineIdx + 1);\n\n if (line.length === 0) continue;\n\n try {\n const raw = JSON.parse(line);\n const result = IpcMessageSchema.safeParse(raw);\n if (result.success) {\n onMessage(result.data);\n } else {\n stream.emit(\n \"error\",\n new Error(`IPC message validation failed: ${result.error.message}`),\n );\n }\n } catch (err) {\n stream.emit(\"error\", new Error(\"IPC message parse error\", { cause: err }));\n }\n }\n }\n }\n\n function onData(chunk: Buffer | string): void {\n if (disposed) return;\n const incoming = typeof chunk === \"string\" ? Buffer.from(chunk) : chunk;\n buf = Buffer.concat([buf, incoming]);\n drain();\n }\n\n stream.on(\"data\", onData);\n\n return () => {\n disposed = true;\n stream.off(\"data\", onData);\n };\n}\n","import { Transform, type TransformCallback } from \"node:stream\";\n\n// 将任意 data 事件分割为完整的 \\n 分隔行\n// Node.js data 事件不保证按行边界分割,此 Transform 保证每次 push 一个完整行\nexport class LineBuffer extends Transform {\n private buffer = \"\";\n\n _transform(chunk: Buffer | string, _encoding: BufferEncoding, callback: TransformCallback): void {\n this.buffer += typeof chunk === \"string\" ? chunk : chunk.toString();\n const segments = this.buffer.split(\"\\n\");\n // 最后一段可能是不完整行,保留到 buffer\n this.buffer = segments.pop()!;\n\n for (const segment of segments) {\n if (segment.length > 0) {\n this.push(segment);\n }\n }\n callback();\n }\n\n _flush(callback: TransformCallback): void {\n if (this.buffer.length > 0) {\n this.push(this.buffer);\n this.buffer = \"\";\n }\n callback();\n }\n}\n"],"mappings":";;;AAAA,SAAS,QAAQ,YAAY,WAAW,qBAAqB;AAC7D,SAAS,eAAe;AACxB,SAAS,SAAS,eAAe;AACjC,SAAS,qBAAqB;AAK9B,IAAM,OAAO,QAAQ;AACrB,IAAM,UAAU,GAAG,IAAI;AACvB,IAAM,aAAa,QAAQ,cAAc,YAAY,GAAG,CAAC;AACzD,IAAM,yBAAyB,QAAQ,YAAY,oBAAoB;AACvE,IAAM,uBAAuB,QAAQ,YAAY,iBAAiB;AAClE,IAAM,sBAAsB;AAGrB,SAAS,QAAQ,KAAqB;AAC3C,SAAO,OAAO,IAAI,QAAQ,MAAM,GAAG,IAAI;AACzC;AACO,IAAM,cAAc,GAAG,OAAO;AAG9B,IAAM,UAAU,GAAG,OAAO;AAC1B,IAAM,YAAY,GAAG,OAAO;AAC5B,IAAM,WAAW,GAAG,OAAO;AAO3B,IAAM,eAAe,GAAG,OAAO;AAC/B,IAAM,mBAAmB,GAAG,OAAO;AAG1C,IAAM,YAAY,GAAG,OAAO;AACrB,IAAM,gBAAgB,GAAG,SAAS;AAClC,IAAM,qBAAqB,GAAG,SAAS;AAGvC,IAAM,WAAW,GAAG,OAAO;AAClC,IAAM,iBAAiB,GAAG,OAAO;AACjC,IAAM,WAAW,GAAG,cAAc;AAG3B,IAAM,UAAU,GAAG,OAAO;AAC1B,IAAM,mBAAmB,GAAG,OAAO;AAE1C,SAAS,WAAW,WAA2B;AAC7C,SAAO,GAAG,QAAQ,IAAI,SAAS;AACjC;AAEO,SAAS,aAAa,WAAmB;AAC9C,QAAM,MAAM,WAAW,SAAS;AAChC,SAAO;AAAA,IACL;AAAA,IACA,YAAY,GAAG,GAAG;AAAA,EACpB;AACF;AAEO,SAAS,gBAAyB;AACvC,SAAO,WAAW,WAAW;AAC/B;AAEA,IAAM,iBAAiB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAmBvB,SAAS,wBAAwB,gBAAwB,QAAkC;AACzF,QAAM,SAAS,OAAO,UAAU;AAChC,QAAM,kBAAkB,GAAG,OAAO,GAAG,IAAI,MAAM;AAC/C,QAAM,kBAAkB,GAAG,cAAc,IAAI,MAAM;AACnD,MAAI,WAAW,eAAe,KAAK,CAAC,WAAW,eAAe,EAAG,QAAO;AACxE,YAAU,gBAAgB,EAAE,WAAW,KAAK,CAAC;AAC7C,SAAO,iBAAiB,iBAAiB,EAAE,WAAW,KAAK,CAAC;AAC5D,SAAO;AACT;AAEO,SAAS,6BACd,gBACA,SACS;AACT,aAAW,UAAU,SAAS;AAC5B,QAAI,wBAAwB,gBAAgB,MAAM,EAAG,QAAO;AAAA,EAC9D;AACA,SAAO;AACT;AAEA,SAAS,oBAA0B;AACjC,+BAA6B,UAAU;AAAA,IACrC,EAAE,KAAK,wBAAwB,QAAQ,oBAAoB;AAAA,IAC3D,EAAE,KAAK,sBAAsB,QAAQ,oBAAoB;AAAA,EAC3D,CAAC;AACH;AAEO,SAAS,gBAAsB;AACpC,YAAU,SAAS,EAAE,WAAW,KAAK,CAAC;AACtC,YAAU,WAAW,EAAE,WAAW,KAAK,CAAC;AACxC,YAAU,UAAU,EAAE,WAAW,KAAK,CAAC;AACvC,YAAU,gBAAgB,EAAE,WAAW,KAAK,CAAC;AAC7C,YAAU,SAAS,EAAE,WAAW,KAAK,CAAC;AACtC,oBAAkB;AAClB,MAAI,CAAC,WAAW,WAAW,GAAG;AAC5B,kBAAc,aAAa,cAAc;AAAA,EAC3C;AACF;;;ACxHA,SAAS,KAAAA,UAAS;;;ACAlB,SAAS,SAAS;AAGX,IAAM,yBAAyB,EAAE,OAAO;EAC7C,MAAM,EAAE,OAAM,EAAG,IAAI,CAAC;EACtB,WAAW,EAAE,OAAM,EAAG,IAAI,CAAC,EAAE,SAAQ;CACtC;AAKM,IAAM,gCAAgC,EAAE,OAAO;EACpD,MAAM,EAAE,OAAM;EACd,WAAW,EAAE,QAAO;CACrB;AAKM,IAAM,wBAAwB,EAAE,OAAO;EAC5C,MAAM,EAAE,OAAM;CACf;;;ACrBD,SAAS,KAAAC,UAAS;AAGX,IAAM,8BAA8BA,GAAE,OAAO;EAClD,UAAUA,GAAE,OAAM;EAClB,QAAQA,GAAE,OAAM;EAChB,YAAYA,GAAE,OAAOA,GAAE,OAAM,GAAIA,GAAE,QAAO,CAAE;CAC7C;AAKM,IAAM,2BAA2BA,GAAE,OAAO;EAC/C,QAAQA,GAAE,OAAM;EAChB,eAAeA,GAAE,QAAO,EAAG,SAAQ;CACpC;AAKM,IAAM,wBAAwBA,GAAE,OAAO;EAC5C,QAAQA,GAAE,OAAM;EAChB,QAAQA,GAAE,OAAM,EAAG,SAAQ;CAC5B;AAKM,IAAM,0BAA0BA,GAAE,OAAO;EAC9C,QAAQA,GAAE,OAAM;EAChB,QAAQA,GAAE,QAAO;EACjB,SAASA,GAAE,QAAO;CACnB;;;AChCD,SAAS,KAAAC,UAAS;AAElB,IAAM,qBAAqB,CAAC,QAAQ,WAAW,oBAAoB,SAAS,YAAY;AACxF,IAAM,iBAAiB,CAAC,UAAU,OAAO;AACzC,IAAM,iBAAiB,CAAC,kBAAkB,cAAc;AACxD,IAAM,yBAAyB;EAC7B;EACA;EACA;EACA;EACA;EACA;;AAKK,IAAM,oBAAoBA,GAAE,OAAO;EACxC,WAAWA,GAAE,OAAM;EACnB,MAAMA,GAAE,OAAM,EAAG,SAAQ;EACzB,OAAOA,GAAE,KAAK,kBAAkB;EAChC,MAAMA,GAAE,KAAK,CAAC,OAAO,MAAM,CAAC,EAAE,SAAQ;EACtC,UAAUA,GAAE,KAAK,cAAc;;;;EAI/B,UAAUA,GAAE,KAAK,cAAc,EAAE,SAAQ;EACzC,YAAYA,GAAE,OAAM,EAAG,SAAQ;CAChC;AAKM,IAAM,6BAA6BA,GAAE,OAAO;EACjD,MAAMA,GAAE,OAAM,EAAG,SAAQ;EACzB,KAAKA,GAAE,OAAM,EAAG,SAAQ;EACxB,aAAaA,GAAE,QAAO,EAAG,SAAQ;CAClC;AAKM,IAAM,2BAA2BA,GAAE,OAAO;EAC/C,UAAUA,GAAE,MAAM,iBAAiB;CACpC;AAKM,IAAM,6BAA6BA,GAAE,OAAO;EACjD,WAAWA,GAAE,OAAM;CACpB;AAKM,IAAM,gCAAgCA,GAAE,OAAO;EACpD,WAAWA,GAAE,OAAM;CACpB;AAMM,IAAM,6BAA6BA,GAAE,OAAO;EACjD,WAAWA,GAAE,OAAM;EACnB,OAAOA,GAAE,KAAK,kBAAkB;EAChC,YAAYA,GAAE,OAAM;CACrB;AAKM,IAAM,wBAAwBA,GAAE,OAAO;EAC5C,OAAOA,GAAE,KAAK,CAAC,WAAW,iBAAiB,iBAAiB,WAAW,CAAC;EACxE,OAAOA,GAAE,OAAM,EAAG,SAAQ;EAC1B,MAAMA,GAAE,OAAM,EAAG,SAAQ;CAC1B;AAGM,IAAM,2BAA2BA,GAAE,OAAO;EAC/C,UAAUA,GAAE,KAAK,cAAc;EAC/B,OAAOA,GAAE,KAAK,sBAAsB;EACpC,KAAKA,GAAE,OAAM,EAAG,IAAG,EAAG,YAAW;EACjC,WAAWA,GAAE,OAAM;EACnB,UAAUA,GAAE,OAAM,EAAG,SAAQ;EAC7B,WAAWA,GAAE,OAAOA,GAAE,OAAM,GAAIA,GAAE,QAAO,CAAE,EAAE,SAAQ;EACrD,mBAAmBA,GAChB,OAAO;IACN,WAAWA,GAAE,OAAM;IACnB,UAAUA,GAAE,OAAM;IAClB,OAAOA,GAAE,OAAOA,GAAE,OAAM,GAAIA,GAAE,QAAO,CAAE;GACxC,EACA,SAAQ;EACX,sBAAsBA,GACnB,OAAO;IACN,WAAWA,GAAE,OAAM;IACnB,SAASA,GAAE,KAAK,CAAC,SAAS,MAAM,CAAC;GAClC,EACA,SAAQ;EACX,SAASA,GAAE,OAAM,EAAG,SAAQ;CAC7B;;;ACpGD,SAAS,KAAAC,UAAS;AAGX,IAAM,yBAAyBA,GAAE,OAAO,CAAA,CAAE;AAK1C,IAAM,oBAAoBA,GAAE,OAAO;EACxC,aAAaA,GAAE,OAAM,EAAG,SAAQ;EAChC,OAAOA,GAAE,OAAM,EAAG,SAAQ;CAC3B;AAKM,IAAM,2BAA2BA,GAAE,OAAO;EAC/C,SAASA,GAAE,OAAM,EAAG,IAAG,EAAG,YAAW;CACtC;AAKM,IAAM,4BAA4BA,GAAE,OAAO;EAChD,UAAUA,GAAE,MAAMA,GAAE,OAAOA,GAAE,OAAM,GAAIA,GAAE,QAAO,CAAE,CAAC;CACpD;;;AJHD,IAAM,qBAAqB;EACzB,KAAKC,GAAE,OAAM,EAAG,IAAG,EAAG,YAAW;EACjC,WAAWA,GAAE,OAAM;EACnB,WAAWA,GAAE,OAAM;EACnB,QAAQA,GAAE,KAAK,CAAC,SAAS,QAAQ,CAAC;EAClC,SAASA,GAAE,OAAM;;AAIZ,IAAM,wBAAwBA,GAAE,mBAAmB,QAAQ;;EAEhEA,GAAE,OAAO;IACP,GAAG;IACH,MAAMA,GAAE,QAAQ,YAAY;IAC5B,SAAS;GACV;EACDA,GAAE,OAAO;IACP,GAAG;IACH,MAAMA,GAAE,QAAQ,mBAAmB;IACnC,SAAS;GACV;EACDA,GAAE,OAAO;IACP,GAAG;IACH,MAAMA,GAAE,QAAQ,UAAU;IAC1B,SAAS;GACV;;;EAGDA,GAAE,OAAO;IACP,GAAG;IACH,MAAMA,GAAE,QAAQ,kBAAkB;IAClC,SAAS;GACV;;EAEDA,GAAE,OAAO;IACP,GAAG;IACH,MAAMA,GAAE,QAAQ,aAAa;IAC7B,SAAS;GACV;;;EAGDA,GAAE,OAAO;IACP,GAAG;IACH,MAAMA,GAAE,QAAQ,oBAAoB;IACpC,SAAS;GACV;;EAEDA,GAAE,OAAO;IACP,GAAG;IACH,MAAMA,GAAE,QAAQ,gBAAgB;IAChC,SAAS;GACV;EACDA,GAAE,OAAO;IACP,GAAG;IACH,MAAMA,GAAE,QAAQ,cAAc;IAC9B,SAAS;GACV;EACDA,GAAE,OAAO;IACP,GAAG;IACH,MAAMA,GAAE,QAAQ,gBAAgB;IAChC,SAAS;GACV;EACDA,GAAE,OAAO;IACP,GAAG;IACH,MAAMA,GAAE,QAAQ,mBAAmB;IACnC,SAAS;GACV;EACDA,GAAE,OAAO;IACP,GAAG;IACH,MAAMA,GAAE,QAAQ,gBAAgB;IAChC,SAAS;GACV;;EAEDA,GAAE,OAAO;IACP,GAAG;IACH,MAAMA,GAAE,QAAQ,WAAW;IAC3B,SAAS;GACV;EACDA,GAAE,OAAO;IACP,GAAG;IACH,MAAMA,GAAE,QAAQ,MAAM;IACtB,SAAS;GACV;EACDA,GAAE,OAAO;IACP,GAAG;IACH,MAAMA,GAAE,QAAQ,cAAc;IAC9B,SAAS;GACV;EACDA,GAAE,OAAO;IACP,GAAG;IACH,MAAMA,GAAE,QAAQ,eAAe;IAC/B,SAAS;GACV;CACF;;;AK9GK,SAAU,aACd,MACA,WACA,KACA,SACA,QAA0B;AAE1B,QAAM,WAAW;IACf;IACA;IACA;IACA;IACA,WAAW,KAAK,IAAG;IACnB;IACA,SAAS;;AAEX,SAAO,sBAAsB,MAAM,QAAQ;AAC7C;;;ACtBA,SAAS,KAAAC,UAAS;;;ACEX,IAAM,iBAAiB;EAC5B,gBAAgB;EAChB,WAAW;EACX,eAAe;EACf,iBAAiB;EACjB,aAAa;EACb,eAAe;;;;ACRV,IAAM,mBAAmB;EAC9B,cAAc;EACd,gBAAgB;EAChB,oBAAoB;EACpB,oBAAoB;EACpB,eAAe;EACf,mBAAmB;EACnB,sBAAsB;EACtB,qBAAqB;EACrB,sBAAsB;EACtB,SAAS;;;;AFHJ,IAAM,kBAAkBC,GAAE,OAAO;EACtC,SAASA,GAAE,OAAM;EACjB,MAAMA,GAAE,OAAM,EAAG,SAAQ;EACzB,QAAQA,GAAE,QAAO;EACjB,UAAUA,GAAE,MAAMA,GAAE,OAAM,CAAE,EAAE,SAAQ;CACvC;AAGM,IAAM,6BAA6BA,GAAE,OAAO;EACjD,WAAWA,GAAE,QAAO;EACpB,SAASA,GAAE,OAAM,EAAG,SAAQ;EAC5B,OAAOA,GAAE,OAAM,EAAG,SAAQ;EAC1B,aAAaA,GAAE,MAAMA,GAAE,OAAM,CAAE,EAAE,SAAQ;CAC1C;AAGM,IAAM,uBAAuBA,GAAE,OAAO;EAC3C,QAAQ;EACR,OAAO;CACR;AAGM,IAAM,iBAAiBA,GAAE,OAAO,EAAE,MAAMA,GAAE,OAAM,GAAI,OAAOA,GAAE,QAAO,EAAE,CAAE;AAGxE,IAAM,sBAAsBA,GAAE,OAAO;EAC1C,MAAMA,GAAE,OAAM;EACd,SAASA,GAAE,MAAM,cAAc;CAChC;AAGM,IAAM,qBAAqBA,GAAE,OAAO;EACzC,MAAMA,GAAE,OAAM;EACd,aAAaA,GAAE,OAAM;EACrB,cAAcA,GAAE,OAAM,EAAG,SAAQ;EACjC,QAAQA,GAAE,OAAM;CACjB;AAGM,IAAM,uBAAuBA,GAAE,OAAO;EAC3C,IAAIA,GAAE,OAAM;EACZ,OAAOA,GAAE,OAAM;EACf,YAAYA,GAAE,OAAM;EACpB,WAAWA,GAAE,OAAM;EACnB,UAAUA,GAAE,KAAK,CAAC,UAAU,OAAO,CAAC,EAAE,SAAQ;CAC/C;AAGD,IAAM,8BAA8BA,GAAE,OAAO;EAC3C,MAAMA,GAAE,KAAK,CAAC,QAAQ,WAAW,CAAC;EAClC,MAAMA,GAAE,OAAM;EACd,WAAWA,GAAE,OAAM,EAAG,SAAQ;EAC9B,QAAQA,GAAE,OAAM,EAAG,SAAQ;CAC5B;AAID,IAAM,iBAAiB,EAAE,WAAWA,GAAE,OAAM,EAAG,IAAI,CAAC,EAAE,SAAQ,EAAE;AAChE,IAAM,yBAAyBA,GAAE,KAC/B,OAAO,OAAO,gBAAgB,CAA8C;AAE9E,IAAM,oBAAoB;EACxB,OAAOA,GAAE,OAAM,EAAG,SAAQ;EAC1B,WAAW,uBAAuB,SAAQ;;AAoB5C,SAAS,QACP,MACA,OACA,YAA4D;AAE5D,SAAO;IACL;IACA,YAAY,IAAI,IAAI,MAAM,QAAQ,UAAU,IAAI,aAAa,aAAa,CAAC,UAAU,IAAI,CAAA,CAAE;IAC3F,QAAQA,GAAE,OAAO;MACf,MAAMA,GAAE,QAAQ,IAAI;MACpB,GAAI,SAAS,CAAA;KACd;;AAEL;AAGA,IAAM,0BAA0B;EAC9B,QAAQ,kBAAkB;IACxB,SAASA,GAAE,OAAM,EAAG,IAAI,CAAC;IACzB,MAAMA,GAAE,OAAM,EAAG,SAAQ;GAC1B;EACD,QAAQ,2BAA2B;IACjC,QAAQA,GAAE,KAAK,CAAC,OAAO,aAAa,CAAC;GACtC;EACD,QAAQ,sBAAsB,cAAc;EAC5C,QAAQ,uBAAuB;IAC7B,GAAG;IACH,SAASA,GAAE,MAAM,eAAe;GACjC;EACD,QAAQ,gBAAgB,EAAE,GAAG,gBAAgB,SAASA,GAAE,OAAM,EAAG,IAAI,CAAC,EAAC,CAAE;EACzE,QAAQ,yBAAyB;IAC/B,GAAG;IACH,SAASA,GAAE,QAAO;IAClB,SAASA,GAAE,OAAM,EAAG,SAAQ;IAC5B,GAAG;GACJ;EACD,QAAQ,eAAe;IACrB,MAAMA,GAAE,KAAK,OAAO,OAAO,cAAc,CAA0C;IACnF,SAASA,GAAE,OAAM;GAClB;;EAGD,QAAQ,mBAAmB;IACzB,UAAUA,GAAE,OAAM,EAAG,IAAI,CAAC;GAC3B;EACD,QAAQ,4BAA4B;IAClC,QAAQA,GAAE,KAAK,CAAC,YAAY,iBAAiB,KAAK,CAAC;IACnD,SAASA,GAAE,OAAM,EAAG,SAAQ;GAC7B;;EAGD,QAAQ,iBAAiB;IACvB,SAASA,GAAE,OAAM;GAClB;;EAGD,QAAQ,oBAAoB;IAC1B,SAASA,GAAE,OAAM,EAAG,IAAI,CAAC;GAC1B;;EAGD,QAAQ,gBAAgB;IACtB,SAASA,GAAE,OAAM,EAAG,IAAI,CAAC;GAC1B;;EAGD,QACE,oBACA;IACE,SAASA,GAAE,OAAM,EAAG,IAAI,CAAC,EAAE,SAAQ;IACnC,GAAG;IACH,MAAMA,GAAE,OAAM;KAEhB,iBAAiB;EAEnB,QACE,qBACA,EAAE,GAAG,gBAAgB,GAAG,mBAAmB,SAASA,GAAE,MAAM,cAAc,GAAG,MAAMA,GAAE,OAAM,EAAE,GAC7F,iBAAiB;;EAInB,QAAQ,sBAAsB,EAAE,GAAG,gBAAgB,MAAMA,GAAE,OAAM,EAAE,GAAI,iBAAiB;EACxF,QACE,uBACA;IACE,GAAG;IACH,GAAG;IACH,MAAMA,GAAE,OAAM;IACd,SAASA,GAAE,QAAO;KAEpB,iBAAiB;;EAInB,QAAQ,qBAAqB,EAAE,UAAUA,GAAE,MAAM,kBAAkB,EAAC,GAAI,iBAAiB;;;EAIzF,QACE,kBACA;IACE,QAAQA,GAAE,MAAM,mBAAmB;KAErC,iBAAiB;;EAInB,QAAQ,gBAAgB,QAAW,CAAC,mBAAmB,iBAAiB,CAAC;EACzE,QACE,0BACA;IACE,MAAMA,GAAE,KAAK,CAAC,WAAW,eAAe,MAAM,CAAC;;IAE/C,WAAWA,GAAE,OAAM,EAAG,SAAQ;KAEhC,iBAAiB;;EAInB,QAAQ,2BAA2B,gBAAgB,iBAAiB;EACpE,QACE,4BACA,EAAE,GAAG,gBAAgB,UAAUA,GAAE,MAAM,oBAAoB,EAAC,GAC5D,iBAAiB;;EAInB,QACE,aACA,EAAE,WAAWA,GAAE,OAAM,GAAI,SAAS,sBAAqB,GACvD,iBAAiB;;EAInB,QACE,gBACA,EAAE,WAAWA,GAAE,OAAM,GAAI,SAAS,yBAAwB,GAC1D,iBAAiB;;EAInB,QAAQ,kBAAkB,EAAE,WAAWA,GAAE,OAAM,GAAI,OAAOA,GAAE,OAAM,EAAE,GAAI,iBAAiB;;EAGzF,QACE,mBACA,EAAE,WAAWA,GAAE,OAAM,GAAI,MAAMA,GAAE,OAAM,EAAG,IAAG,EAAG,SAAQ,GAAI,MAAMA,GAAE,OAAM,EAAG,IAAG,EAAG,SAAQ,EAAE,GAC7F,iBAAiB;EAEnB,QACE,2BACA,EAAE,WAAWA,GAAE,OAAM,GAAI,MAAMA,GAAE,OAAM,EAAG,IAAG,EAAG,SAAQ,GAAI,MAAMA,GAAE,OAAM,EAAG,IAAG,EAAG,SAAQ,EAAE,GAC7F,iBAAiB;;EAInB,QAAQ,qBAAqB,EAAE,WAAWA,GAAE,OAAM,EAAE,GAAI,iBAAiB;;EAGzE,QAAQ,wBAAwB,EAAE,WAAWA,GAAE,OAAM,EAAE,GAAI,iBAAiB;;EAG5E,QACE,eACA;IACE,WAAWA,GAAE,OAAM;IACnB,SAASA,GAAE,QAAO;IAClB,SAASA,GAAE,QAAO;;;IAGlB,QAAQA,GAAE,OAAM,EAAG,SAAQ;KAE7B,iBAAiB;;EAInB,QACE,oBACA,EAAE,WAAWA,GAAE,OAAM,EAAG,IAAI,CAAC,GAAG,MAAMA,GAAE,OAAM,EAAE,GAChD,iBAAiB;;;EAKnB,QAAQ,sBAAsB,gBAAgB,iBAAiB;EAC/D,QACE,cACA,EAAE,GAAG,gBAAgB,UAAUA,GAAE,OAAM,GAAI,UAAU,qBAAoB,GACzE,iBAAiB;EAEnB,QACE,2BACA,EAAE,GAAG,gBAAgB,UAAUA,GAAE,KAAK,CAAC,UAAU,OAAO,CAAC,GAAG,MAAMA,GAAE,OAAM,EAAG,IAAI,CAAC,EAAC,GACnF,iBAAiB;EAEnB,QACE,oCACA;IACE,GAAG;IACH,UAAUA,GAAE,KAAK,CAAC,UAAU,OAAO,CAAC;IACpC,UAAU,qBAAqB,SAAQ;IACvC,GAAG;KAEL,iBAAiB;;EAInB,QACE,kBACA;IACE,GAAG;IACH,KAAKA,GAAE,OAAM;IACb,UAAUA,GAAE,KAAK,CAAC,UAAU,OAAO,CAAC;IACpC,MAAMA,GAAE,KAAK,CAAC,QAAQ,KAAK,CAAC,EAAE,SAAQ;IACtC,iBAAiBA,GAAE,OAAM,EAAG,SAAQ;;IAEpC,gBAAgBA,GACb,KAAK,CAAC,WAAW,QAAQ,eAAe,QAAQ,qBAAqB,SAAS,CAAC,EAC/E,SAAQ;KAEb,iBAAiB;EAEnB,QACE,2BACA;IACE,GAAG;IACH,WAAWA,GAAE,OAAM;IACnB,MAAMA,GAAE,KAAK,CAAC,QAAQ,KAAK,CAAC,EAAE,SAAQ;IACtC,UAAUA,GAAE,KAAK,CAAC,UAAU,OAAO,CAAC,EAAE,SAAQ;IAC9C,UAAUA,GAAE,KAAK,CAAC,kBAAkB,cAAc,CAAC,EAAE,SAAQ;IAC7D,GAAG;KAEL,iBAAiB;;EAInB,QACE,4BACA;IACE,GAAG;IACH,WAAWA,GAAE,OAAM;IACnB,OAAOA,GAAE,OAAM,EAAG,IAAG,EAAG,IAAI,CAAC,EAAE,IAAI,GAAG,EAAE,SAAQ;IAChD,QAAQA,GAAE,OAAM,EAAG,SAAQ;KAE7B,iBAAiB;;EAInB,QACE,6BACA,EAAE,GAAG,gBAAgB,WAAWA,GAAE,OAAM,EAAE,GAC1C,iBAAiB;EAEnB,QACE,8BACA;IACE,GAAG;IACH,GAAG;IACH,WAAWA,GAAE,OAAM;IACnB,UAAUA,GAAE,MAAM,kBAAkB;IACpC,QAAQA,GAAE,MAAM,mBAAmB;KAErC,iBAAiB;;EAInB,QACE,wBACA,EAAE,GAAG,gBAAgB,WAAWA,GAAE,OAAM,EAAG,SAAQ,EAAE,GACrD,iBAAiB;EAEnB,QACE,yBACA;IACE,GAAG;IACH,UAAUA,GAAE,MAAMA,GAAE,OAAO,EAAE,WAAWA,GAAE,OAAM,GAAI,SAAS,yBAAwB,CAAE,CAAC;KAE1F,iBAAiB;;EAInB,QACE,gCACA,EAAE,WAAWA,GAAE,OAAM,GAAI,WAAWA,GAAE,OAAM,EAAE,GAC9C,iBAAiB;EAEnB,QACE,gBACA,EAAE,WAAWA,GAAE,OAAM,GAAI,SAAS,yBAAwB,GAC1D,iBAAiB;EAEnB,QACE,aACA,EAAE,WAAWA,GAAE,OAAM,GAAI,SAAS,sBAAqB,GACvD,iBAAiB;;EAInB,QACE,8BACA;IACE,WAAWA,GAAE,OAAM;IACnB,WAAWA,GAAE,OAAM;IACnB,SAASA,GAAE,KAAK,CAAC,SAAS,MAAM,CAAC;IACjC,WAAWA,GAAE,QAAO;IACpB,SAASA,GAAE,OAAM,EAAG,SAAQ;KAE9B,iBAAiB;;EAInB,QACE,0BACA;IACE,WAAWA,GAAE,OAAM;IACnB,WAAWA,GAAE,MACXA,GAAE,OAAO;MACP,WAAWA,GAAE,OAAM;MACnB,UAAUA,GAAE,OAAM;MAClB,OAAOA,GAAE,OAAOA,GAAE,OAAM,GAAIA,GAAE,QAAO,CAAE;KACxC,CAAC;KAGN,iBAAiB;;EAInB,QACE,4BACA;IACE,GAAG;IACH,WAAWA,GAAE,OAAM;IACnB,QAAQA,GAAE,OAAM,EAAG,SAAQ;IAC3B,UAAUA,GAAE,MAAM,2BAA2B;IAC7C,SAASA,GAAE,QAAO,EAAG,SAAQ;IAC7B,YAAYA,GAAE,OAAM,EAAG,SAAQ;KAEjC,iBAAiB;;EAInB,QAAQ,gBAAgB;IACtB,UAAUA,GAAE,MACVA,GAAE,OAAO;MACP,IAAIA,GAAE,OAAM;MACZ,MAAMA,GAAE,KAAK,CAAC,OAAO,MAAM,CAAC;MAC5B,UAAUA,GAAE,KAAK,CAAC,UAAU,OAAO,CAAC;MACpC,UAAUA,GAAE,KAAK,CAAC,kBAAkB,cAAc,CAAC,EAAE,SAAQ;MAC7D,OAAOA,GAAE,OAAM;KAChB,CAAC;GAEL;;EAGD,QACE,qBACA,EAAE,WAAWA,GAAE,OAAM,GAAI,WAAWA,GAAE,OAAM,EAAG,SAAQ,EAAE,GACzD,iBAAiB;;EAInB,QACE,oBACA;IACE,WAAWA,GAAE,OAAM;IACnB,MAAMA,GAAE,OAAM,EAAG,IAAG,EAAG,SAAQ;IAC/B,MAAMA,GAAE,OAAM,EAAG,IAAG,EAAG,SAAQ;IAC/B,MAAMA,GAAE,OAAM;IACd,WAAWA,GAAE,OAAM,EAAG,IAAG,EAAG,YAAW;IACvC,WAAWA,GAAE,OAAM,EAAG,SAAQ;KAEhC,iBAAiB;;AAIrB,IAAM,sBAAsB,wBAAwB,IAAI,CAAC,eAAe,WAAW,MAAM;AAKlF,IAAM,qBAAqBA,GAAE,mBAAmB,QAAQ,mBAAmB;AAK3E,IAAM,iCAAiC,IAAI,IAChD,wBACG,OAAO,CAAC,eAAe,WAAW,WAAW,IAAI,iBAAiB,CAAC,EACnE,IAAI,CAAC,eAAe,WAAW,IAAI,CAAC;AAOlC,IAAM,iCAAiC,IAAI,IAChD,wBACG,OAAO,CAAC,eAAe,WAAW,WAAW,IAAI,iBAAiB,CAAC,EACnE,IAAI,CAAC,eAAe,WAAW,IAAI,CAAC;;;AGxelC,IAAM,eAAe;EAC1B,MAAM;EACN,SAAS;EACT,kBAAkB;EAClB,OAAO;EACP,YAAY;;;;ACNd,SACE,WACA,aAAAC,YACA,aACA,YACA,UACA,aACA,kBACK;AACP,SAAS,WAAAC,gBAAe;AACxB,SAAS,UAAU,YAAY;AAC/B,OAAO,UAAU;AAYjB,IAAM,kBAAkB,GAAGA,SAAO,CAAE;AACpC,IAAM,wBAAwB;AAE9B,IAAM,qBAAqB,cACzB,QAAQ,IAAI,2BACV,IAAG,oBAAI,KAAI,GAAG,YAAW,EAAG,QAAQ,SAAS,GAAG,CAAC,IAAI,QAAQ,GAAG,EAAE;AAGtE,SAAS,cAAc,OAAa;AAClC,SAAO,MAAM,QAAQ,oBAAoB,GAAG;AAC9C;AAEA,SAAS,cAAc,QAAgB,MAAc,UAAkB,OAAa;AAClF,QAAM,aAAa,KAAK,QAAQ,GAAG,IAAI,MAAM;AAE7C,MAAI;AACF,UAAM,OAAO,UAAU,UAAU;AACjC,QAAI,KAAK,eAAc,GAAI;AACzB,iBAAW,UAAU;IACvB,OAAO;AACL,iBAAW,YAAY,KAAK,QAAQ,GAAG,IAAI,WAAW,KAAK,MAAM,CAAC;IACpE;EACF,SAAS,KAAK;AACZ,UAAM,OAAQ,IAA8B;AAC5C,QAAI,SAAS;AAAU;EACzB;AAEA,MAAI;AACF,gBAAY,SAAS,QAAQ,GAAG,UAAU;EAC5C,QAAQ;EAER;AACF;AAEA,SAAS,mBAAgB;AACvB,QAAM,MAAM,QAAQ,IAAI;AACxB,MAAI,CAAC;AAAK,WAAO;AACjB,QAAM,SAAS,OAAO,SAAS,KAAK,EAAE;AACtC,SAAO,OAAO,SAAS,MAAM,KAAK,UAAU,IAAI,SAAS;AAC3D;AAEA,SAAS,aAAa,QAAgB,MAAc,iBAAuB;AACzE,QAAM,OAAO,iBAAgB;AAC7B,MAAI,SAAS;AAAG;AAEhB,QAAM,kBAAkB,SAAS,eAAe;AAChD,QAAM,SAAS,GAAG,IAAI;AACtB,QAAM,aAAa,YAAY,MAAM,EAClC,OACC,CAAC,UAAU,MAAM,WAAW,MAAM,KAAK,MAAM,SAAS,MAAM,KAAK,UAAU,eAAe,EAE3F,IAAI,CAAC,UAAS;AACb,UAAM,OAAO,KAAK,QAAQ,KAAK;AAC/B,QAAI;AACF,aAAO,EAAE,MAAM,SAAS,SAAS,IAAI,EAAE,QAAO;IAChD,QAAQ;AACN,aAAO;IACT;EACF,CAAC,EACA,OAAO,CAAC,UAAsD,UAAU,IAAI,EAC5E,KAAK,CAAC,GAAG,MAAM,EAAE,UAAU,EAAE,OAAO;AAEvC,aAAW,SAAS,WAAW,MAAM,KAAK,IAAI,GAAG,OAAO,CAAC,CAAC,GAAG;AAC3D,QAAI;AACF,iBAAW,MAAM,IAAI;IACvB,QAAQ;IAER;EACF;AACF;AAEM,SAAU,aAAa,SAA4B;AACvD,QAAM,EACJ,MACA,QAAQ,QACR,SAAS,iBACT,SAAS,OACT,SAAS,MAAK,IACZ;AAEJ,MAAI,QAAQ;AACV,WAAO,KAAK,EAAE,OAAO,SAAQ,CAAE;EACjC;AAEA,EAAAD,WAAU,QAAQ,EAAE,WAAW,KAAI,CAAE;AAErC,QAAM,QAAQ;AACd,QAAM,WAAW,KAAK,QAAQ,GAAG,IAAI,IAAI,KAAK,MAAM;AACpD,gBAAc,QAAQ,MAAM,UAAU,KAAK;AAC3C,eAAa,QAAQ,MAAM,QAAQ;AACnC,QAAM,UAA8B,CAAC,EAAE,QAAQ,KAAK,YAAY,QAAQ,EAAC,CAAE;AAE3E,MAAI,QAAQ;AACV,YAAQ,QAAQ,EAAE,QAAQ,QAAQ,OAAM,CAAE;EAC5C;AAEA,SAAO,KAAK,EAAE,MAAK,GAAI,KAAK,YAAY,OAAO,CAAC;AAClD;;;ACxHA,SAAS,KAAAE,UAAS;;;ACAlB,SAAS,iBAAyC;AAI3C,IAAM,aAAN,cAAyB,UAAU;AAAA,EAChC,SAAS;AAAA,EAEjB,WAAW,OAAwB,WAA2B,UAAmC;AAC/F,SAAK,UAAU,OAAO,UAAU,WAAW,QAAQ,MAAM,SAAS;AAClE,UAAM,WAAW,KAAK,OAAO,MAAM,IAAI;AAEvC,SAAK,SAAS,SAAS,IAAI;AAE3B,eAAW,WAAW,UAAU;AAC9B,UAAI,QAAQ,SAAS,GAAG;AACtB,aAAK,KAAK,OAAO;AAAA,MACnB;AAAA,IACF;AACA,aAAS;AAAA,EACX;AAAA,EAEA,OAAO,UAAmC;AACxC,QAAI,KAAK,OAAO,SAAS,GAAG;AAC1B,WAAK,KAAK,KAAK,MAAM;AACrB,WAAK,SAAS;AAAA,IAChB;AACA,aAAS;AAAA,EACX;AACF;;;ADvBO,IAAM,oBAAoB;AAI1B,SAAS,qBAAqB,WAAmB,MAAc,WAA2B;AAC/F,QAAM,eAAe,OAAO,KAAK,WAAW,OAAO;AACnD,QAAM,aAAa,IAAI,aAAa,SAAS,IAAI,KAAK;AACtD,QAAM,QAAQ,OAAO,MAAM,IAAI,IAAI,UAAU;AAC7C,MAAI,SAAS;AACb,QAAM,MAAM,IAAI;AAChB,YAAU;AACV,QAAM,cAAc,YAAY,MAAM;AACtC,YAAU;AACV,QAAM,MAAM,IAAI,aAAa;AAC7B,YAAU;AACV,eAAa,KAAK,OAAO,MAAM;AAC/B,YAAU,aAAa;AACvB,QAAM,cAAc,WAAW,MAAM;AACrC,YAAU;AACV,OAAK,KAAK,OAAO,MAAM;AACvB,SAAO;AACT;AAEA,IAAMC,sBAAqB,OAAO,OAAO,YAAY;AAErD,IAAM,4BAA4BC,GAAE,OAAO;AAAA,EACzC,UAAUA,GAAE,KAAK,CAAC,UAAU,OAAO,CAAC;AAAA,EACpC,WAAWA,GAAE,OAAO;AAAA,EACpB,SAASA,GAAE,OAAO;AAAA,EAClB,QAAQA,GAAE,OAAO;AAAA,EACjB,OAAOA,GAAE,OAAO;AAClB,CAAC;AAGM,IAAM,mBAAmBA,GAAE,mBAAmB,QAAQ;AAAA;AAAA,EAE3DA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,wBAAwB;AAAA,IACxC,MAAMA,GAAE,OAAO,EAAE,SAAS;AAAA,IAC1B,MAAMA,GAAE,KAAK,CAAC,OAAO,MAAM,CAAC;AAAA,IAC5B,UAAUA,GAAE,KAAK,CAAC,UAAU,OAAO,CAAC;AAAA,IACpC,KAAKA,GAAE,OAAO;AAAA,IACd,KAAKA,GAAE,OAAO;AAAA,IACd,WAAWA,GAAE,OAAO,EAAE,SAAS;AAAA,EACjC,CAAC;AAAA;AAAA,EAGDA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,yBAAyB;AAAA,IACzC,WAAWA,GAAE,OAAO;AAAA,IACpB,OAAOA,GAAE,OAAO,EAAE,SAAS;AAAA,IAC3B,MAAM,0BAA0B,SAAS;AAAA,EAC3C,CAAC;AAAA;AAAA,EAGDA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,2BAA2B;AAAA,IAC3C,WAAWA,GAAE,OAAO;AAAA,EACtB,CAAC;AAAA;AAAA,EAGDA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,4BAA4B;AAAA,IAC5C,WAAWA,GAAE,OAAO;AAAA,IACpB,SAASA,GAAE,QAAQ;AAAA,EACrB,CAAC;AAAA;AAAA,EAGDA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,cAAc;AAAA,IAC9B,WAAWA,GAAE,OAAO;AAAA,IACpB,KAAKA,GAAE,OAAO;AAAA,EAChB,CAAC;AAAA;AAAA,EAGDA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,gBAAgB;AAAA,IAChC,WAAWA,GAAE,OAAO;AAAA,EACtB,CAAC;AAAA;AAAA,EAGDA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,WAAW;AAAA,IAC3B,WAAWA,GAAE,OAAO;AAAA,IACpB,MAAMA,GAAE,OAAO;AAAA,EACjB,CAAC;AAAA;AAAA,EAGDA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,YAAY;AAAA,IAC5B,WAAWA,GAAE,OAAO;AAAA,EACtB,CAAC;AAAA;AAAA,EAGDA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,uBAAuB;AAAA,IACvC,WAAWA,GAAE,OAAO;AAAA,IACpB,OAAOA,GAAE,KAAKD,mBAAkB;AAAA,EAClC,CAAC;AAAA;AAAA,EAGDC,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,OAAO;AAAA,IACvB,SAASA,GAAE,OAAO;AAAA,IAClB,MAAMA,GAAE,OAAO,EAAE,SAAS;AAAA,EAC5B,CAAC;AAAA;AAAA,EAGDA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,wBAAwB;AAAA,EAC1C,CAAC;AAAA;AAAA,EAGDA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,yBAAyB;AAAA,IACzC,QAAQA,GAAE,OAAO;AAAA,MACf,SAASA,GAAE,OAAO,EAAE,SAAS;AAAA,MAC7B,eAAeA,GAAE,KAAK,CAAC,OAAO,QAAQ,UAAU,WAAW,MAAM,CAAC;AAAA,MAClE,UAAUA,GAAE,OAAO,EAAE,SAAS;AAAA,MAC9B,gBAAgBA,GAAE,KAAK,CAAC,OAAO,QAAQ,MAAM,CAAC;AAAA,MAC9C,kBAAkBA,GAAE,KAAK,CAAC,OAAO,QAAQ,MAAM,CAAC;AAAA,MAChD,UAAUA,GAAE,OAAO;AAAA,MACnB,gBAAgBA,GAAE,KAAK,CAAC,OAAO,QAAQ,SAAS,CAAC;AAAA,IACnD,CAAC;AAAA,IACD,OAAOA,GACJ,OAAO;AAAA,MACN,WAAWA,GAAE,QAAQ;AAAA,MACrB,SAASA,GAAE,OAAO;AAAA,MAClB,kBAAkBA,GAAE,OAAO;AAAA,MAC3B,YAAYA,GAAE,OAAO;AAAA,IACvB,CAAC,EACA,SAAS;AAAA,IACZ,UAAUA,GAAE;AAAA,MACVA,GAAE,OAAO;AAAA,QACP,IAAIA,GAAE,OAAO;AAAA,QACb,MAAMA,GAAE,KAAK,CAAC,OAAO,MAAM,CAAC;AAAA,QAC5B,OAAOA,GAAE,KAAKD,mBAAkB;AAAA,QAChC,WAAWC,GAAE,OAAO;AAAA,QACpB,MAAMA,GAAE,OAAO,EAAE,SAAS;AAAA,QAC1B,WAAWA,GAAE,QAAQ;AAAA,MACvB,CAAC;AAAA,IACH;AAAA,EACF,CAAC;AAAA;AAAA,EAGDA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,kBAAkB;AAAA,IAClC,WAAWA,GAAE,OAAO;AAAA,IACpB,OAAOA,GAAE,OAAO;AAAA,EAClB,CAAC;AAAA;AAAA;AAAA,EAIDA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,oBAAoB;AAAA,IACpC,WAAWA,GAAE,OAAO;AAAA,IACpB,OAAOA,GAAE,KAAK,CAAC,WAAW,iBAAiB,iBAAiB,WAAW,CAAC;AAAA,IACxE,OAAOA,GAAE,OAAO,EAAE,SAAS;AAAA,IAC3B,MAAMA,GAAE,OAAO,EAAE,SAAS;AAAA,EAC5B,CAAC;AAAA;AAAA,EAGDA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,YAAY;AAAA,IAC5B,WAAWA,GAAE,OAAO;AAAA,IACpB,MAAMA,GAAE,OAAO;AAAA,IACf,MAAMA,GAAE,OAAO;AAAA,EACjB,CAAC;AAAA;AAAA,EAGDA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,eAAe;AAAA,IAC/B,WAAWA,GAAE,OAAO;AAAA,IACpB,WAAWA,GAAE,OAAO,EAAE,SAAS;AAAA,EACjC,CAAC;AAAA;AAAA,EAGDA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,cAAc;AAAA,IAC9B,WAAWA,GAAE,OAAO;AAAA,IACpB,MAAMA,GAAE,OAAO;AAAA,IACf,MAAMA,GAAE,OAAO;AAAA,IACf,MAAMA,GAAE,OAAO;AAAA,IACf,WAAWA,GAAE,OAAO,EAAE,IAAI,EAAE,YAAY;AAAA,IACxC,WAAWA,GAAE,OAAO,EAAE,SAAS;AAAA,EACjC,CAAC;AAAA;AAAA,EAGDA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,eAAe;AAAA,IAC/B,WAAWA,GAAE,QAAQ;AAAA,EACvB,CAAC;AACH,CAAC;AAGM,IAAM,sBAAsBA,GAAE,mBAAmB,QAAQ;AAAA;AAAA,EAE9DA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,cAAc;AAAA,IAC9B,SAASA,GAAE,OAAO;AAAA,EACpB,CAAC;AAAA;AAAA,EAGDA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,aAAa;AAAA,EAC/B,CAAC;AAAA;AAAA,EAGDA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,0BAA0B;AAAA,IAC1C,WAAWA,GAAE,OAAO;AAAA,IACpB,UAAUA,GAAE,KAAK,CAAC,SAAS,MAAM,CAAC;AAAA,IAClC,SAASA,GAAE,OAAO,EAAE,SAAS;AAAA,EAC/B,CAAC;AAAA;AAAA,EAGDA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,cAAc;AAAA,IAC9B,KAAKA,GAAE,OAAO;AAAA,IACd,OAAOA,GAAE,OAAOA,GAAE,OAAO,GAAGA,GAAE,QAAQ,CAAC;AAAA,EACzC,CAAC;AAAA;AAAA,EAGDA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,aAAa;AAAA,IAC7B,MAAMA,GAAE,OAAO;AAAA,EACjB,CAAC;AAAA;AAAA,EAGDA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,yBAAyB;AAAA,IACzC,WAAWA,GAAE,OAAO;AAAA,IACpB,UAAUA,GAAE,OAAO;AAAA,IACnB,OAAOA,GAAE,OAAOA,GAAE,OAAO,GAAGA,GAAE,QAAQ,CAAC;AAAA,EACzC,CAAC;AAAA;AAAA,EAGDA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,cAAc;AAAA,IAC9B,KAAKA,GAAE,OAAO;AAAA,EAChB,CAAC;AAAA;AAAA;AAAA,EAIDA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,0BAA0B;AAAA,IAC1C,WAAWA,GAAE,OAAO;AAAA,EACtB,CAAC;AAAA;AAAA,EAGDA,GAAE,OAAO;AAAA,IACP,MAAMA,GAAE,QAAQ,sBAAsB;AAAA,IACtC,UAAUA,GAAE,OAAO;AAAA,EACrB,CAAC;AACH,CAAC;AAIM,SAAS,mBAAmB,KAA4B;AAC7D,SAAO,KAAK,UAAU,GAAG,IAAI;AAC/B;AAEO,SAAS,mBACd,QACA,WACM;AACN,QAAM,aAAa,IAAI,WAAW;AAClC,aAAW,GAAG,QAAQ,CAAC,SAA0B;AAC/C,UAAM,MAAM,OAAO,SAAS,WAAW,OAAO,KAAK,SAAS;AAC5D,QAAI,IAAI,WAAW,EAAG;AACtB,QAAI;AACF,YAAM,MAAM,KAAK,MAAM,GAAG;AAC1B,YAAM,SAAS,oBAAoB,UAAU,GAAG;AAChD,UAAI,OAAO,SAAS;AAClB,kBAAU,OAAO,IAAI;AAAA,MACvB,OAAO;AACL,eAAO;AAAA,UACL;AAAA,UACA,IAAI,MAAM,qCAAqC,OAAO,MAAM,OAAO,EAAE;AAAA,QACvE;AAAA,MACF;AAAA,IACF,SAAS,KAAK;AACZ,aAAO,KAAK,SAAS,IAAI,MAAM,8BAA8B,EAAE,OAAO,IAAI,CAAC,CAAC;AAAA,IAC9E;AAAA,EACF,CAAC;AACD,EAAC,OAAiC,KAAK,UAAU;AACnD;AAKO,SAAS,aAAa,KAAyB;AACpD,SAAO,KAAK,UAAU,GAAG,IAAI;AAC/B;AAKO,SAAS,gBACd,QACA,WACA,eACY;AACZ,MAAI,MAAM,OAAO,MAAM,CAAC;AACxB,MAAI,WAAW;AAGf,WAAS,QAAc;AACrB,WAAO,IAAI,SAAS,GAAG;AACrB,UAAI,IAAI,CAAC,MAAM,mBAAmB;AAGhC,YAAI,IAAI,SAAS,EAAG;AACpB,cAAM,aAAa,IAAI,aAAa,CAAC;AACrC,cAAM,gBAAgB,IAAI,IAAI;AAC9B,YAAI,IAAI,SAAS,cAAe;AAGhC,cAAM,eAAe;AACrB,cAAM,eAAe,IAAI,YAAY;AACrC,cAAM,YAAY,IACf,SAAS,eAAe,GAAG,eAAe,IAAI,YAAY,EAC1D,SAAS,OAAO;AACnB,cAAM,YAAY,eAAe,IAAI;AACrC,cAAM,YAAY,IAAI,aAAa,SAAS;AAC5C,cAAM,UAAU,OAAO,KAAK,IAAI,SAAS,YAAY,GAAG,aAAa,CAAC;AAEtE,YAAI,eAAe;AACjB,wBAAc,WAAW,SAAS,SAAS;AAAA,QAC7C;AAEA,cAAM,IAAI,SAAS,aAAa;AAAA,MAClC,OAAO;AAEL,cAAM,aAAa,IAAI,QAAQ,EAAI;AACnC,YAAI,eAAe,GAAI;AAEvB,cAAM,OAAO,IAAI,SAAS,GAAG,UAAU,EAAE,SAAS,OAAO;AACzD,cAAM,IAAI,SAAS,aAAa,CAAC;AAEjC,YAAI,KAAK,WAAW,EAAG;AAEvB,YAAI;AACF,gBAAM,MAAM,KAAK,MAAM,IAAI;AAC3B,gBAAM,SAAS,iBAAiB,UAAU,GAAG;AAC7C,cAAI,OAAO,SAAS;AAClB,sBAAU,OAAO,IAAI;AAAA,UACvB,OAAO;AACL,mBAAO;AAAA,cACL;AAAA,cACA,IAAI,MAAM,kCAAkC,OAAO,MAAM,OAAO,EAAE;AAAA,YACpE;AAAA,UACF;AAAA,QACF,SAAS,KAAK;AACZ,iBAAO,KAAK,SAAS,IAAI,MAAM,2BAA2B,EAAE,OAAO,IAAI,CAAC,CAAC;AAAA,QAC3E;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAEA,WAAS,OAAO,OAA8B;AAC5C,QAAI,SAAU;AACd,UAAM,WAAW,OAAO,UAAU,WAAW,OAAO,KAAK,KAAK,IAAI;AAClE,UAAM,OAAO,OAAO,CAAC,KAAK,QAAQ,CAAC;AACnC,UAAM;AAAA,EACR;AAEA,SAAO,GAAG,QAAQ,MAAM;AAExB,SAAO,MAAM;AACX,eAAW;AACX,WAAO,IAAI,QAAQ,MAAM;AAAA,EAC3B;AACF;","names":["z","z","z","z","z","z","z","mkdirSync","homedir","z","sessionStateValues","z"]}
|